mirror of
https://github.com/koush/scrypted.git
synced 2026-02-12 18:12:04 +00:00
190 lines
5.9 KiB
JavaScript
190 lines
5.9 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
import fs from 'fs';
|
|
import path from 'path';
|
|
import axios, { AxiosRequestConfig } from 'axios';
|
|
import readline from 'readline-sync';
|
|
import https from 'https';
|
|
import mkdirp from 'mkdirp';
|
|
import { installServe, serveMain } from './service';
|
|
import { connectScryptedClient } from '../../client/src/index';
|
|
import semver from 'semver';
|
|
|
|
if (!semver.gte(process.version, '16.0.0')) {
|
|
throw new Error('"node" version out of date. Please update node to v16 or higher.')
|
|
}
|
|
|
|
function getUserHome() {
|
|
const ret = process.env[(process.platform == 'win32') ? 'USERPROFILE' : 'HOME'];
|
|
if (!ret)
|
|
throw new Error('Neither USERPROFILE or HOME are defined.');
|
|
return ret;
|
|
}
|
|
|
|
const scryptedHome = path.join(getUserHome(), '.scrypted');
|
|
const loginPath = path.join(scryptedHome, 'login.json');
|
|
|
|
function toIpAndPort(ip: string) {
|
|
if (ip.indexOf(':') === -1)
|
|
ip += ':10443'
|
|
return ip;
|
|
}
|
|
|
|
async function doLogin(host: string) {
|
|
host = toIpAndPort(host);
|
|
|
|
const username = readline.question('username: ');
|
|
const password = readline.question('password: ', {
|
|
hideEchoBack: true,
|
|
});
|
|
|
|
const url = `https://${host}/login`;
|
|
const response = await axios(Object.assign({
|
|
method: 'GET',
|
|
auth: {
|
|
username,
|
|
password,
|
|
},
|
|
url,
|
|
}, axiosConfig));
|
|
|
|
mkdirp.sync(scryptedHome);
|
|
let login: any;
|
|
try {
|
|
login = JSON.parse(fs.readFileSync(loginPath).toString());
|
|
}
|
|
catch (e) {
|
|
login = {};
|
|
}
|
|
if (typeof login !== 'object')
|
|
login = {};
|
|
login = login || {};
|
|
|
|
login[host] = response.data;
|
|
fs.writeFileSync(loginPath, JSON.stringify(login));
|
|
return response.data.token;
|
|
}
|
|
|
|
async function getOrDoLogin(host: string): Promise<{
|
|
username: string,
|
|
token: string,
|
|
}> {
|
|
let login: any;
|
|
try {
|
|
login = JSON.parse(fs.readFileSync(loginPath).toString());
|
|
if (typeof login !== 'object')
|
|
login = {};
|
|
|
|
if (!login[host].username || !login[host].token)
|
|
throw new Error();
|
|
}
|
|
catch (e) {
|
|
login = await doLogin(host);
|
|
}
|
|
return login[host];
|
|
}
|
|
|
|
const axiosConfig: AxiosRequestConfig = {
|
|
httpsAgent: new https.Agent({
|
|
rejectUnauthorized: false
|
|
})
|
|
}
|
|
|
|
async function main() {
|
|
if (process.argv[2] === 'serve') {
|
|
await serveMain(false);
|
|
}
|
|
else if (process.argv[2] === 'serve@latest') {
|
|
await serveMain(true);
|
|
}
|
|
else if (process.argv[2] === 'install-server') {
|
|
const installDir = await installServe();
|
|
console.log('server installation successful:', installDir);
|
|
}
|
|
else if (process.argv[2] === 'login') {
|
|
const ip = process.argv[3] || '127.0.0.1';
|
|
const token = await doLogin(ip);
|
|
console.log('login successful. token:', token);
|
|
}
|
|
else if (process.argv[2] === 'command') {
|
|
const [idOrName, optionalHost] = process.argv[3].split('@');
|
|
const host = toIpAndPort(optionalHost || '127.0.0.1');
|
|
|
|
const login = await getOrDoLogin(host);
|
|
|
|
const sdk = await connectScryptedClient({
|
|
baseUrl: `https://${host}`,
|
|
pluginId: '@scrypted/core',
|
|
username: login.username,
|
|
password: login.token,
|
|
});
|
|
|
|
const device: any = sdk.systemManager.getDeviceById(idOrName) || sdk.systemManager.getDeviceByName(idOrName);
|
|
if (!device)
|
|
throw new Error('device not found: ' + idOrName);
|
|
const method = process.argv[4];
|
|
const args = process.argv.slice(5).map(arg => () => {
|
|
try {
|
|
return JSON.parse(arg);
|
|
}
|
|
catch (e) {
|
|
}
|
|
return arg;
|
|
});
|
|
await device[method](...args);
|
|
sdk.disconnect();
|
|
}
|
|
else if (process.argv[2] === 'create-cert-json' && process.argv.length === 5) {
|
|
const key = fs.readFileSync(process.argv[3]).toString();
|
|
const cert = fs.readFileSync(process.argv[4]).toString();
|
|
const json = JSON.stringify({
|
|
key,
|
|
cert,
|
|
}, null, 2);
|
|
|
|
fs.writeFileSync('cert.json', json);
|
|
console.log('Saved cert.json.');
|
|
console.log();
|
|
console.log('Start the Scrypted server with the following environment variable:');
|
|
console.log(' SCRYPTED_HTTPS_OPTIONS_FILE=/path/to/cert.json')
|
|
console.log();
|
|
console.log('Docker users will need to mount the cert in a volume and use the following docker run arguments:');
|
|
console.log(' -e SCRYPTED_HTTPS_OPTIONS_FILE=/path/to/cert.json')
|
|
}
|
|
else if (process.argv[2] === 'install') {
|
|
const ip = toIpAndPort(process.argv[4] || '127.0.0.1');
|
|
const pkg = process.argv[3];
|
|
|
|
if (!pkg) {
|
|
console.log('usage: npx scrypted install npm-package-name [ip]');
|
|
process.exit(1);
|
|
}
|
|
|
|
const login = await getOrDoLogin(ip);
|
|
const url = `https://${ip}/web/component/script/install/${pkg}`;
|
|
const response = await axios(Object.assign({
|
|
method: 'POST',
|
|
auth: {
|
|
username: login.username,
|
|
password: login.token,
|
|
},
|
|
url,
|
|
}, axiosConfig));
|
|
|
|
console.log('install successful. id:', response.data.id);
|
|
}
|
|
else {
|
|
console.log('usage:');
|
|
console.log(' npx scrypted install npm-package-name [127.0.0.1[:10443]]');
|
|
console.log(' npx scrypted install npm-package-name[/0.0.1] [127.0.0.1[:10443]]');
|
|
console.log(' npx scrypted login [127.0.0.1[:10443]]');
|
|
console.log(' npx scrypted serve');
|
|
console.log(' npx scrypted serve@latest');
|
|
console.log(' npx scrypted command name-or-id[@127.0.0.1[:10443]] method-name [...method-arguments]');
|
|
console.log(' npx scrypted create-cert-json /path/to/key.pem /path/to/cert.pem');
|
|
process.exit(1);
|
|
}
|
|
}
|
|
|
|
main();
|