@alwaysai/device-agent 0.0.1-2.1-beta-provision
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/application-control/backup.d.ts +8 -0
- package/lib/application-control/backup.d.ts.map +1 -0
- package/lib/application-control/backup.js +37 -0
- package/lib/application-control/backup.js.map +1 -0
- package/lib/application-control/config.d.ts +17 -0
- package/lib/application-control/config.d.ts.map +1 -0
- package/lib/application-control/config.js +62 -0
- package/lib/application-control/config.js.map +1 -0
- package/lib/application-control/environment-variables.d.ts +9 -0
- package/lib/application-control/environment-variables.d.ts.map +1 -0
- package/lib/application-control/environment-variables.js +73 -0
- package/lib/application-control/environment-variables.js.map +1 -0
- package/lib/application-control/index.d.ts +10 -0
- package/lib/application-control/index.d.ts.map +1 -0
- package/lib/application-control/index.js +32 -0
- package/lib/application-control/index.js.map +1 -0
- package/lib/application-control/install.d.ts +22 -0
- package/lib/application-control/install.d.ts.map +1 -0
- package/lib/application-control/install.js +156 -0
- package/lib/application-control/install.js.map +1 -0
- package/lib/application-control/models.d.ts +23 -0
- package/lib/application-control/models.d.ts.map +1 -0
- package/lib/application-control/models.js +154 -0
- package/lib/application-control/models.js.map +1 -0
- package/lib/application-control/status.d.ts +27 -0
- package/lib/application-control/status.d.ts.map +1 -0
- package/lib/application-control/status.js +153 -0
- package/lib/application-control/status.js.map +1 -0
- package/lib/application-control/types.d.ts +5 -0
- package/lib/application-control/types.d.ts.map +1 -0
- package/lib/application-control/types.js +3 -0
- package/lib/application-control/types.js.map +1 -0
- package/lib/application-control/utils.d.ts +14 -0
- package/lib/application-control/utils.d.ts.map +1 -0
- package/lib/application-control/utils.js +82 -0
- package/lib/application-control/utils.js.map +1 -0
- package/lib/cloud-connection/device-agent-cloud-connection.d.ts +51 -0
- package/lib/cloud-connection/device-agent-cloud-connection.d.ts.map +1 -0
- package/lib/cloud-connection/device-agent-cloud-connection.js +490 -0
- package/lib/cloud-connection/device-agent-cloud-connection.js.map +1 -0
- package/lib/cloud-connection/device-agent.d.ts +21 -0
- package/lib/cloud-connection/device-agent.d.ts.map +1 -0
- package/lib/cloud-connection/device-agent.js +65 -0
- package/lib/cloud-connection/device-agent.js.map +1 -0
- package/lib/device-control/device-control.d.ts +4 -0
- package/lib/device-control/device-control.d.ts.map +1 -0
- package/lib/device-control/device-control.js +20 -0
- package/lib/device-control/device-control.js.map +1 -0
- package/lib/docker/docker-cmd.d.ts +4 -0
- package/lib/docker/docker-cmd.d.ts.map +1 -0
- package/lib/docker/docker-cmd.js +16 -0
- package/lib/docker/docker-cmd.js.map +1 -0
- package/lib/docker/docker-compose-cmd.d.ts +5 -0
- package/lib/docker/docker-compose-cmd.d.ts.map +1 -0
- package/lib/docker/docker-compose-cmd.js +16 -0
- package/lib/docker/docker-compose-cmd.js.map +1 -0
- package/lib/endpoints.d.ts +3 -0
- package/lib/endpoints.d.ts.map +1 -0
- package/lib/endpoints.js +28 -0
- package/lib/endpoints.js.map +1 -0
- package/lib/environment.d.ts +7 -0
- package/lib/environment.d.ts.map +1 -0
- package/lib/environment.js +24 -0
- package/lib/environment.js.map +1 -0
- package/lib/index.d.ts +3 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +26 -0
- package/lib/index.js.map +1 -0
- package/lib/infrastructure/agent-config.d.ts +73 -0
- package/lib/infrastructure/agent-config.d.ts.map +1 -0
- package/lib/infrastructure/agent-config.js +175 -0
- package/lib/infrastructure/agent-config.js.map +1 -0
- package/lib/infrastructure/agent-config.test.d.ts +2 -0
- package/lib/infrastructure/agent-config.test.d.ts.map +1 -0
- package/lib/infrastructure/agent-config.test.js +182 -0
- package/lib/infrastructure/agent-config.test.js.map +1 -0
- package/lib/infrastructure/certificates-and-tokens.d.ts +6 -0
- package/lib/infrastructure/certificates-and-tokens.d.ts.map +1 -0
- package/lib/infrastructure/certificates-and-tokens.js +37 -0
- package/lib/infrastructure/certificates-and-tokens.js.map +1 -0
- package/lib/infrastructure/urls.d.ts +2 -0
- package/lib/infrastructure/urls.d.ts.map +1 -0
- package/lib/infrastructure/urls.js +25 -0
- package/lib/infrastructure/urls.js.map +1 -0
- package/lib/root.d.ts +2 -0
- package/lib/root.d.ts.map +1 -0
- package/lib/root.js +12 -0
- package/lib/root.js.map +1 -0
- package/lib/subcommands/app/app.d.ts +59 -0
- package/lib/subcommands/app/app.d.ts.map +1 -0
- package/lib/subcommands/app/app.js +317 -0
- package/lib/subcommands/app/app.js.map +1 -0
- package/lib/subcommands/app/index.d.ts +2 -0
- package/lib/subcommands/app/index.d.ts.map +1 -0
- package/lib/subcommands/app/index.js +30 -0
- package/lib/subcommands/app/index.js.map +1 -0
- package/lib/subcommands/device/clean.d.ts +2 -0
- package/lib/subcommands/device/clean.d.ts.map +1 -0
- package/lib/subcommands/device/clean.js +20 -0
- package/lib/subcommands/device/clean.js.map +1 -0
- package/lib/subcommands/device/device.d.ts +6 -0
- package/lib/subcommands/device/device.d.ts.map +1 -0
- package/lib/subcommands/device/device.js +92 -0
- package/lib/subcommands/device/device.js.map +1 -0
- package/lib/subcommands/device/index.d.ts +2 -0
- package/lib/subcommands/device/index.d.ts.map +1 -0
- package/lib/subcommands/device/index.js +12 -0
- package/lib/subcommands/device/index.js.map +1 -0
- package/lib/subcommands/get-model-package.d.ts +5 -0
- package/lib/subcommands/get-model-package.d.ts.map +1 -0
- package/lib/subcommands/get-model-package.js +35 -0
- package/lib/subcommands/get-model-package.js.map +1 -0
- package/lib/subcommands/index.d.ts +9 -0
- package/lib/subcommands/index.d.ts.map +1 -0
- package/lib/subcommands/index.js +14 -0
- package/lib/subcommands/index.js.map +1 -0
- package/lib/subcommands/login.d.ts +6 -0
- package/lib/subcommands/login.d.ts.map +1 -0
- package/lib/subcommands/login.js +35 -0
- package/lib/subcommands/login.js.map +1 -0
- package/lib/util/clean-certs.d.ts +2 -0
- package/lib/util/clean-certs.d.ts.map +1 -0
- package/lib/util/clean-certs.js +16 -0
- package/lib/util/clean-certs.js.map +1 -0
- package/lib/util/copy-dir.d.ts +5 -0
- package/lib/util/copy-dir.d.ts.map +1 -0
- package/lib/util/copy-dir.js +16 -0
- package/lib/util/copy-dir.js.map +1 -0
- package/lib/util/directories.d.ts +23 -0
- package/lib/util/directories.d.ts.map +1 -0
- package/lib/util/directories.js +50 -0
- package/lib/util/directories.js.map +1 -0
- package/lib/util/get-device-id.d.ts +2 -0
- package/lib/util/get-device-id.d.ts.map +1 -0
- package/lib/util/get-device-id.js +24 -0
- package/lib/util/get-device-id.js.map +1 -0
- package/lib/util/http-client.d.ts +3 -0
- package/lib/util/http-client.d.ts.map +1 -0
- package/lib/util/http-client.js +30 -0
- package/lib/util/http-client.js.map +1 -0
- package/lib/util/logger.d.ts +4 -0
- package/lib/util/logger.d.ts.map +1 -0
- package/lib/util/logger.js +24 -0
- package/lib/util/logger.js.map +1 -0
- package/lib/util/run-in-dir.d.ts +2 -0
- package/lib/util/run-in-dir.d.ts.map +1 -0
- package/lib/util/run-in-dir.js +17 -0
- package/lib/util/run-in-dir.js.map +1 -0
- package/lib/util/sleep.d.ts +2 -0
- package/lib/util/sleep.d.ts.map +1 -0
- package/lib/util/sleep.js +9 -0
- package/lib/util/sleep.js.map +1 -0
- package/package.json +98 -0
- package/readme.md +219 -0
- package/src/application-control/backup.ts +36 -0
- package/src/application-control/config.ts +61 -0
- package/src/application-control/environment-variables.ts +74 -0
- package/src/application-control/index.ts +45 -0
- package/src/application-control/install.ts +206 -0
- package/src/application-control/models.ts +194 -0
- package/src/application-control/status.ts +187 -0
- package/src/application-control/types.ts +1 -0
- package/src/application-control/utils.ts +95 -0
- package/src/cloud-connection/device-agent-cloud-connection.ts +673 -0
- package/src/cloud-connection/device-agent.ts +120 -0
- package/src/device-control/device-control.ts +16 -0
- package/src/docker/docker-cmd.ts +12 -0
- package/src/docker/docker-compose-cmd.ts +12 -0
- package/src/endpoints.ts +24 -0
- package/src/environment.ts +28 -0
- package/src/index.ts +26 -0
- package/src/infrastructure/agent-config.test.ts +199 -0
- package/src/infrastructure/agent-config.ts +208 -0
- package/src/infrastructure/certificates-and-tokens.ts +47 -0
- package/src/infrastructure/urls.ts +21 -0
- package/src/root.ts +11 -0
- package/src/subcommands/app/app.ts +337 -0
- package/src/subcommands/app/index.ts +46 -0
- package/src/subcommands/device/clean.ts +16 -0
- package/src/subcommands/device/device.ts +126 -0
- package/src/subcommands/device/index.ts +9 -0
- package/src/subcommands/get-model-package.ts +33 -0
- package/src/subcommands/index.ts +11 -0
- package/src/subcommands/login.ts +33 -0
- package/src/util/clean-certs.ts +12 -0
- package/src/util/copy-dir.ts +12 -0
- package/src/util/directories.ts +82 -0
- package/src/util/get-device-id.ts +22 -0
- package/src/util/http-client.ts +35 -0
- package/src/util/logger.ts +28 -0
- package/src/util/run-in-dir.ts +15 -0
- package/src/util/sleep.ts +5 -0
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
const awsIot = require("aws-iot-device-sdk");
|
|
2
|
+
import {
|
|
3
|
+
DEVICE_PRIVATE_KEY_FILE_NAME,
|
|
4
|
+
CERTIFICATE_OWNERSHIP_TOKEN_FILE_NAME,
|
|
5
|
+
DEVICE_CLAIM_ID_FILE_NAME,
|
|
6
|
+
DEVICE_CERTIFICATE_FILE_NAME,
|
|
7
|
+
} from "../util/directories";
|
|
8
|
+
import { getTargetHardwareUuid } from "../infrastructure/certificates-and-tokens";
|
|
9
|
+
import { getDeviceId } from "../util/get-device-id";
|
|
10
|
+
import { LOCAL_CERT_AND_KEY_DIR } from "alwaysai/lib/constants";
|
|
11
|
+
import { JsSpawner } from "alwaysai/lib/util";
|
|
12
|
+
import { logger } from "../util/logger";
|
|
13
|
+
import { v4 as uuidv4 } from "uuid";
|
|
14
|
+
import { join } from "path";
|
|
15
|
+
const process = require("process");
|
|
16
|
+
|
|
17
|
+
interface DeviceAgentConfigType {
|
|
18
|
+
keyPath: string;
|
|
19
|
+
certPath: string;
|
|
20
|
+
caPath: string;
|
|
21
|
+
clientId: string;
|
|
22
|
+
host: string;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
interface FleetProvisionTemplateMessageType {
|
|
26
|
+
certificateOwnershipToken: String;
|
|
27
|
+
parameters: {
|
|
28
|
+
hardwareId: String;
|
|
29
|
+
deviceUuid: String;
|
|
30
|
+
certificateId: String;
|
|
31
|
+
deviceType: String;
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export class DeviceAgent {
|
|
36
|
+
constructor(config: DeviceAgentConfigType) {
|
|
37
|
+
this.device = awsIot.device(config);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
public deviceType = "aai-device";
|
|
41
|
+
public device = awsIot.device;
|
|
42
|
+
public hardwareId = async () =>
|
|
43
|
+
await getTargetHardwareUuid(
|
|
44
|
+
JsSpawner(),
|
|
45
|
+
join(LOCAL_CERT_AND_KEY_DIR, "hardware-id.txt")
|
|
46
|
+
);
|
|
47
|
+
public deviceId = getDeviceId();
|
|
48
|
+
|
|
49
|
+
public publishMessage(topic: string, message: string) {
|
|
50
|
+
this.device.publish(topic, message);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
export class BootstrapAgent extends DeviceAgent {
|
|
55
|
+
public async subscribeToAllTopics() {
|
|
56
|
+
const AWS_CERTIFICATE_ACCEPT_TOPIC =
|
|
57
|
+
"$aws/certificates/create/json/accepted";
|
|
58
|
+
const PROVISIONING_ACCEPTED_TOPIC =
|
|
59
|
+
"$aws/provisioning-templates/FleetProvisionTemplate/provision/json/accepted";
|
|
60
|
+
|
|
61
|
+
const topics = [AWS_CERTIFICATE_ACCEPT_TOPIC, PROVISIONING_ACCEPTED_TOPIC];
|
|
62
|
+
const resp = this.device.subscribe(
|
|
63
|
+
topics,
|
|
64
|
+
function (err: any, granted: { topic: string; qos: number }[]) {
|
|
65
|
+
logger.debug(granted);
|
|
66
|
+
}
|
|
67
|
+
);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
public async handleAwsCertificateTopics(topic: string, payload: any) {
|
|
71
|
+
switch (topic) {
|
|
72
|
+
case "$aws/certificates/create/json/accepted": {
|
|
73
|
+
const {
|
|
74
|
+
certificateId,
|
|
75
|
+
certificatePem,
|
|
76
|
+
privateKey,
|
|
77
|
+
certificateOwnershipToken,
|
|
78
|
+
} = JSON.parse(payload);
|
|
79
|
+
|
|
80
|
+
const certSpawner = JsSpawner({ path: LOCAL_CERT_AND_KEY_DIR });
|
|
81
|
+
|
|
82
|
+
await certSpawner.writeFile(
|
|
83
|
+
DEVICE_CERTIFICATE_FILE_NAME,
|
|
84
|
+
certificatePem
|
|
85
|
+
);
|
|
86
|
+
|
|
87
|
+
await certSpawner.writeFile(DEVICE_PRIVATE_KEY_FILE_NAME, privateKey);
|
|
88
|
+
|
|
89
|
+
await certSpawner.writeFile(DEVICE_CLAIM_ID_FILE_NAME, certificateId);
|
|
90
|
+
|
|
91
|
+
await certSpawner.writeFile(
|
|
92
|
+
CERTIFICATE_OWNERSHIP_TOKEN_FILE_NAME,
|
|
93
|
+
certificateOwnershipToken
|
|
94
|
+
);
|
|
95
|
+
|
|
96
|
+
const content: FleetProvisionTemplateMessageType = {
|
|
97
|
+
certificateOwnershipToken,
|
|
98
|
+
parameters: {
|
|
99
|
+
hardwareId: await this.hardwareId(),
|
|
100
|
+
deviceUuid: this.deviceId,
|
|
101
|
+
certificateId,
|
|
102
|
+
deviceType: this.deviceType,
|
|
103
|
+
},
|
|
104
|
+
};
|
|
105
|
+
logger.debug(JSON.stringify(content));
|
|
106
|
+
|
|
107
|
+
this.publishMessage(
|
|
108
|
+
"$aws/provisioning-templates/FleetProvisionTemplate/provision/json",
|
|
109
|
+
JSON.stringify(content)
|
|
110
|
+
);
|
|
111
|
+
|
|
112
|
+
break;
|
|
113
|
+
}
|
|
114
|
+
case "$aws/provisioning-templates/FleetProvisionTemplate/provision/json/accepted": {
|
|
115
|
+
logger.info("Device agent provisioning: success");
|
|
116
|
+
process.exit();
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
const osu = require('node-os-utils');
|
|
2
|
+
|
|
3
|
+
export async function getCpuUtil(): Promise<number> {
|
|
4
|
+
const cpuUsage = await osu.cpu.usage();
|
|
5
|
+
return cpuUsage;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
export async function getDiskUtil(): Promise<number> {
|
|
9
|
+
const driveInfo = await osu.drive.info();
|
|
10
|
+
return parseFloat(driveInfo.usedPercentage);
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export async function getMemUtil(): Promise<number> {
|
|
14
|
+
const memInfo = await osu.mem.info();
|
|
15
|
+
return 100.0 - memInfo.freeMemPercentage;
|
|
16
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { JsSpawner } from 'alwaysai/lib/util';
|
|
2
|
+
|
|
3
|
+
export async function runDockerLogin(props: { token: string }) {
|
|
4
|
+
const { token } = props;
|
|
5
|
+
const spawner = JsSpawner();
|
|
6
|
+
const server = '994534263224.dkr.ecr.us-west-2.amazonaws.com';
|
|
7
|
+
const output = await spawner.run({
|
|
8
|
+
exe: 'docker',
|
|
9
|
+
args: ['login', '--username', 'AWS', '--password', token, server],
|
|
10
|
+
});
|
|
11
|
+
return output;
|
|
12
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { JsSpawner } from 'alwaysai/lib/util';
|
|
2
|
+
|
|
3
|
+
export async function runDockerComposeCmd(props: { args: string[]; dir: string }) {
|
|
4
|
+
const { args, dir } = props;
|
|
5
|
+
const spawner = JsSpawner();
|
|
6
|
+
const output = await spawner.run({
|
|
7
|
+
exe: 'docker-compose',
|
|
8
|
+
args,
|
|
9
|
+
cwd: dir,
|
|
10
|
+
});
|
|
11
|
+
return output;
|
|
12
|
+
}
|
package/src/endpoints.ts
ADDED
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { getSystemId } from "alwaysai/lib/infrastructure";
|
|
2
|
+
|
|
3
|
+
export const getSecondLevelDomain = () => {
|
|
4
|
+
let domain = "";
|
|
5
|
+
switch (getSystemId()) {
|
|
6
|
+
case "development":
|
|
7
|
+
domain = "a6i0.net";
|
|
8
|
+
break;
|
|
9
|
+
case "qa":
|
|
10
|
+
domain = "a6i1.net";
|
|
11
|
+
break;
|
|
12
|
+
case "production":
|
|
13
|
+
domain = "alwaysai.co";
|
|
14
|
+
break;
|
|
15
|
+
default:
|
|
16
|
+
domain = "alwaysai.co";
|
|
17
|
+
break;
|
|
18
|
+
}
|
|
19
|
+
return domain;
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
export const serviceEndpointBuilder = (service: string, path: string) => {
|
|
23
|
+
return `https://${service}.${getSecondLevelDomain()}/${path}`;
|
|
24
|
+
};
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { platform } from "os";
|
|
2
|
+
|
|
3
|
+
export const ALWAYSAI_OS_PLATFORM = parseOsPlatform(
|
|
4
|
+
process.env.ALWAYSAI_OS_PLATFORM
|
|
5
|
+
);
|
|
6
|
+
export const ALWAYSAI_SHOW_HIDDEN = parseBoolean(
|
|
7
|
+
process.env.ALWAYSAI_SHOW_HIDDEN
|
|
8
|
+
);
|
|
9
|
+
export const ALWAYSAI_DEVICE_AGENT_MODE =
|
|
10
|
+
process.env.ALWAYSAI_DEVICE_AGENT_MODE;
|
|
11
|
+
export const ALWAYSAI_LOG_LEVEL = process.env.AAI_LOG_LEVEL;
|
|
12
|
+
export const ALWAYSAI_LOG_TO_CONSOLE = process.env.ALWAYSAI_LOG_TO_CONSOLE;
|
|
13
|
+
|
|
14
|
+
function parseOsPlatform(str: string | undefined): NodeJS.Platform {
|
|
15
|
+
switch (str) {
|
|
16
|
+
case "win32":
|
|
17
|
+
case "darwin":
|
|
18
|
+
case "linux": {
|
|
19
|
+
return str;
|
|
20
|
+
}
|
|
21
|
+
default:
|
|
22
|
+
return platform();
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function parseBoolean(str: string | undefined) {
|
|
27
|
+
return str === "1";
|
|
28
|
+
}
|
package/src/index.ts
ADDED
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
/*
|
|
4
|
+
Though this file is not executable, the above "shebang" line is necessary as an
|
|
5
|
+
indicator to npm that this file is a Node.js script, not a shell script e.g.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { runCliAndExit } from "@alwaysai/alwayscli";
|
|
9
|
+
import { root } from "./root";
|
|
10
|
+
import { runDeviceAgentCloudInterface } from "./cloud-connection/device-agent-cloud-connection";
|
|
11
|
+
import { AgentConfigFile } from "./infrastructure/agent-config";
|
|
12
|
+
import { ALWAYSAI_DEVICE_AGENT_MODE } from "./environment";
|
|
13
|
+
import { logger } from "./util/logger";
|
|
14
|
+
|
|
15
|
+
if (module === require.main) {
|
|
16
|
+
logger.info("Starting alwaysAI Device Agent");
|
|
17
|
+
if (!AgentConfigFile().exists()) {
|
|
18
|
+
AgentConfigFile().initialize();
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
if (ALWAYSAI_DEVICE_AGENT_MODE === "cloud") {
|
|
22
|
+
runDeviceAgentCloudInterface();
|
|
23
|
+
} else {
|
|
24
|
+
runCliAndExit(root, {});
|
|
25
|
+
}
|
|
26
|
+
}
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
import * as tempy from 'tempy';
|
|
2
|
+
import { AgentConfigFile } from './agent-config';
|
|
3
|
+
|
|
4
|
+
const mockAgentModeGetter = jest.fn().mockReturnValue(undefined);
|
|
5
|
+
jest.mock('../environment', () => ({
|
|
6
|
+
get ALWAYSAI_DEVICE_AGENT_MODE() {
|
|
7
|
+
return mockAgentModeGetter();
|
|
8
|
+
},
|
|
9
|
+
}));
|
|
10
|
+
|
|
11
|
+
const configFile = AgentConfigFile(tempy.directory());
|
|
12
|
+
|
|
13
|
+
describe('Test Agent Config', () => {
|
|
14
|
+
beforeEach(() => {
|
|
15
|
+
configFile.remove();
|
|
16
|
+
configFile.initialize();
|
|
17
|
+
});
|
|
18
|
+
describe('Test applications', () => {
|
|
19
|
+
test('Commands with no apps', async () => {
|
|
20
|
+
const apps = await configFile.getApps();
|
|
21
|
+
expect(apps).toEqual([]);
|
|
22
|
+
const app = await configFile.getApp({ projectId: 'wrong-id' });
|
|
23
|
+
expect(app).toBeNull();
|
|
24
|
+
const present = await configFile.isAppPresent({ projectId: 'another-id' });
|
|
25
|
+
expect(present).toBeFalsy();
|
|
26
|
+
const ready = await configFile.isAppReady({ projectId: 'yet-another-id' });
|
|
27
|
+
expect(ready).toBeFalsy();
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
test('Indicate app installation in progress', async () => {
|
|
31
|
+
const projectId = 'add-me';
|
|
32
|
+
const version = 'im-a-version';
|
|
33
|
+
|
|
34
|
+
await configFile.setAppInstalling({ projectId, version });
|
|
35
|
+
const apps = await configFile.getApps();
|
|
36
|
+
expect(apps).toEqual([
|
|
37
|
+
{
|
|
38
|
+
projectId,
|
|
39
|
+
version,
|
|
40
|
+
ready: false,
|
|
41
|
+
},
|
|
42
|
+
]);
|
|
43
|
+
const readyApps = await configFile.getReadyApps();
|
|
44
|
+
expect(readyApps).toEqual([]);
|
|
45
|
+
const app = await configFile.getApp({ projectId });
|
|
46
|
+
expect(app).toEqual({
|
|
47
|
+
projectId,
|
|
48
|
+
version,
|
|
49
|
+
ready: false,
|
|
50
|
+
});
|
|
51
|
+
const present = await configFile.isAppPresent({ projectId });
|
|
52
|
+
expect(present).toBeTruthy();
|
|
53
|
+
const ready = await configFile.isAppReady({ projectId });
|
|
54
|
+
expect(ready).toBeFalsy();
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
test('Indicate app installation complete', async () => {
|
|
58
|
+
const projectId = 'add-me';
|
|
59
|
+
const version = 'im-a-version';
|
|
60
|
+
|
|
61
|
+
await configFile.setAppInstalling({ projectId, version });
|
|
62
|
+
await configFile.setAppInstalled({ projectId, version });
|
|
63
|
+
const apps = await configFile.getApps();
|
|
64
|
+
expect(apps).toEqual([
|
|
65
|
+
{
|
|
66
|
+
projectId,
|
|
67
|
+
version,
|
|
68
|
+
ready: true,
|
|
69
|
+
},
|
|
70
|
+
]);
|
|
71
|
+
const readyApps = await configFile.getReadyApps();
|
|
72
|
+
expect(readyApps).toEqual([
|
|
73
|
+
{
|
|
74
|
+
projectId,
|
|
75
|
+
version,
|
|
76
|
+
ready: true,
|
|
77
|
+
},
|
|
78
|
+
]);
|
|
79
|
+
const app = await configFile.getApp({ projectId });
|
|
80
|
+
expect(app).toEqual({
|
|
81
|
+
projectId,
|
|
82
|
+
version,
|
|
83
|
+
ready: true,
|
|
84
|
+
});
|
|
85
|
+
const appVersion = await configFile.getAppVersion({ projectId });
|
|
86
|
+
expect(appVersion).toEqual(version);
|
|
87
|
+
const present = await configFile.isAppPresent({ projectId });
|
|
88
|
+
expect(present).toBeTruthy();
|
|
89
|
+
const ready = await configFile.isAppReady({ projectId });
|
|
90
|
+
expect(ready).toBeTruthy();
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
test('Indicate app uninstalled', async () => {
|
|
94
|
+
const projectId = 'add-me';
|
|
95
|
+
const version = 'im-a-version';
|
|
96
|
+
|
|
97
|
+
await configFile.setAppInstalling({ projectId, version });
|
|
98
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version);
|
|
99
|
+
await configFile.setAppInstalled({ projectId, version });
|
|
100
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version);
|
|
101
|
+
await configFile.setAppUninstalled({ projectId });
|
|
102
|
+
const apps = await configFile.getApps();
|
|
103
|
+
expect(apps).toEqual([]);
|
|
104
|
+
const app = await configFile.getApp({ projectId: 'wrong-id' });
|
|
105
|
+
expect(app).toBeNull();
|
|
106
|
+
const present = await configFile.isAppPresent({ projectId: 'another-id' });
|
|
107
|
+
expect(present).toBeFalsy();
|
|
108
|
+
const ready = await configFile.isAppReady({ projectId: 'yet-another-id' });
|
|
109
|
+
expect(ready).toBeFalsy();
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
test('Set and get backup', async () => {
|
|
113
|
+
const projectId = 'add-me';
|
|
114
|
+
const version = 'im-a-version';
|
|
115
|
+
const version2 = 'newer-version';
|
|
116
|
+
|
|
117
|
+
// first installation, we use the version
|
|
118
|
+
await configFile.setAppInstalling({ projectId, version });
|
|
119
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version);
|
|
120
|
+
await configFile.setAppInstalled({ projectId, version });
|
|
121
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version);
|
|
122
|
+
const backup1 = await configFile.getAppBackup({ projectId });
|
|
123
|
+
expect(backup1).toBeNull();
|
|
124
|
+
await configFile.setAppBackup({ projectId });
|
|
125
|
+
const apps = await configFile.getApps();
|
|
126
|
+
expect(apps).toEqual([
|
|
127
|
+
{
|
|
128
|
+
projectId,
|
|
129
|
+
version,
|
|
130
|
+
ready: true,
|
|
131
|
+
backup: {
|
|
132
|
+
version,
|
|
133
|
+
},
|
|
134
|
+
},
|
|
135
|
+
]);
|
|
136
|
+
const app = await configFile.getApp({ projectId });
|
|
137
|
+
expect(app).toEqual({
|
|
138
|
+
projectId,
|
|
139
|
+
version,
|
|
140
|
+
ready: true,
|
|
141
|
+
backup: {
|
|
142
|
+
version,
|
|
143
|
+
},
|
|
144
|
+
});
|
|
145
|
+
const backup2 = await configFile.getAppBackup({ projectId });
|
|
146
|
+
expect(backup2).toEqual({ version });
|
|
147
|
+
|
|
148
|
+
await configFile.setAppInstalling({ projectId, version: version2 });
|
|
149
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version);
|
|
150
|
+
await configFile.setAppInstalled({ projectId, version: version2 });
|
|
151
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version2);
|
|
152
|
+
const backup3 = await configFile.getAppBackup({ projectId });
|
|
153
|
+
expect(backup3).toEqual({ version });
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
test('Test app rollback', async () => {
|
|
157
|
+
const projectId = 'add-me';
|
|
158
|
+
const version1 = 'original-version';
|
|
159
|
+
const version2 = 'newer-version';
|
|
160
|
+
await configFile.setAppInstalling({ projectId, version: version1 });
|
|
161
|
+
await configFile.setAppInstalled({ projectId, version: version1 });
|
|
162
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version1);
|
|
163
|
+
expect(await configFile.getAppBackup({ projectId })).toBeNull();
|
|
164
|
+
|
|
165
|
+
// next installation - verify app is present and ready
|
|
166
|
+
expect(await configFile.isAppPresent({ projectId })).toBe(true);
|
|
167
|
+
expect(await configFile.isAppReady({ projectId })).toBe(true);
|
|
168
|
+
|
|
169
|
+
// test to see if app has backup
|
|
170
|
+
await configFile.setAppBackup({ projectId });
|
|
171
|
+
const expectedBackupVersion = (await configFile.getAppBackup({ projectId }))
|
|
172
|
+
?.version;
|
|
173
|
+
expect(expectedBackupVersion).toEqual(version1);
|
|
174
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version1);
|
|
175
|
+
|
|
176
|
+
// simulate second version install
|
|
177
|
+
await configFile.setAppInstalling({ projectId, version: version2 });
|
|
178
|
+
const newExpectedBackupVersion = (await configFile.getAppBackup({ projectId }))
|
|
179
|
+
?.version;
|
|
180
|
+
expect(newExpectedBackupVersion).toEqual(version1);
|
|
181
|
+
await configFile.setAppInstalled({ projectId, version: version2 });
|
|
182
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version2);
|
|
183
|
+
|
|
184
|
+
// simulate the rollback step: set the version that we will roll back to
|
|
185
|
+
await configFile.setAppInstalling({ projectId, version: version1 });
|
|
186
|
+
|
|
187
|
+
// test state during rollback
|
|
188
|
+
expect(await configFile.isAppReady({ projectId })).toEqual(false);
|
|
189
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version2);
|
|
190
|
+
|
|
191
|
+
// test final state of rolled-back app
|
|
192
|
+
await configFile.setAppInstalled({ projectId, version: version1 });
|
|
193
|
+
const finalBackupVersion = (await configFile.getAppBackup({ projectId }))?.version;
|
|
194
|
+
expect(finalBackupVersion).toEqual(version1);
|
|
195
|
+
expect(await configFile.isAppReady({ projectId })).toEqual(true);
|
|
196
|
+
expect(await configFile.getAppVersion({ projectId })).toEqual(version1);
|
|
197
|
+
});
|
|
198
|
+
});
|
|
199
|
+
});
|
|
@@ -0,0 +1,208 @@
|
|
|
1
|
+
import { ConfigFileSchema } from '@alwaysai/config-nodejs';
|
|
2
|
+
import { JSONSchemaType } from 'ajv';
|
|
3
|
+
import { homedir } from 'os';
|
|
4
|
+
import { join } from 'path';
|
|
5
|
+
|
|
6
|
+
export interface AppBackupConfig {
|
|
7
|
+
version: string;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export interface InstalledAppConfig {
|
|
11
|
+
projectId: string;
|
|
12
|
+
version: string;
|
|
13
|
+
ready: boolean;
|
|
14
|
+
backup?: AppBackupConfig;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export interface AgentConfig {
|
|
18
|
+
applications: InstalledAppConfig[];
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
const schema: JSONSchemaType<AgentConfig> = {
|
|
22
|
+
type: 'object',
|
|
23
|
+
properties: {
|
|
24
|
+
applications: {
|
|
25
|
+
type: 'array',
|
|
26
|
+
items: {
|
|
27
|
+
type: 'object',
|
|
28
|
+
properties: {
|
|
29
|
+
projectId: { type: 'string' },
|
|
30
|
+
version: { type: 'string' },
|
|
31
|
+
ready: { type: 'boolean' },
|
|
32
|
+
backup: {
|
|
33
|
+
type: 'object',
|
|
34
|
+
nullable: true,
|
|
35
|
+
properties: {
|
|
36
|
+
version: { type: 'string' },
|
|
37
|
+
},
|
|
38
|
+
required: ['version'],
|
|
39
|
+
additionalProperties: true,
|
|
40
|
+
},
|
|
41
|
+
},
|
|
42
|
+
required: ['projectId', 'version', 'ready'],
|
|
43
|
+
additionalProperties: true,
|
|
44
|
+
},
|
|
45
|
+
},
|
|
46
|
+
},
|
|
47
|
+
required: ['applications'],
|
|
48
|
+
additionalProperties: true,
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
const ALWAYSAI_CONFIG_DIR = join(homedir(), '.config', 'alwaysai');
|
|
52
|
+
|
|
53
|
+
const AGENT_CONFIG_FILE_NAME = 'alwaysai.agent.json';
|
|
54
|
+
|
|
55
|
+
export function AgentConfigFile(dir = ALWAYSAI_CONFIG_DIR) {
|
|
56
|
+
const path = join(dir, AGENT_CONFIG_FILE_NAME);
|
|
57
|
+
const initialValue: AgentConfig = {
|
|
58
|
+
applications: [],
|
|
59
|
+
};
|
|
60
|
+
const configFile = ConfigFileSchema({
|
|
61
|
+
path,
|
|
62
|
+
schema,
|
|
63
|
+
initialValue,
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
return {
|
|
67
|
+
...configFile,
|
|
68
|
+
name: AGENT_CONFIG_FILE_NAME,
|
|
69
|
+
getApps,
|
|
70
|
+
getReadyApps,
|
|
71
|
+
getApp,
|
|
72
|
+
isAppPresent,
|
|
73
|
+
isAppReady,
|
|
74
|
+
removeApp,
|
|
75
|
+
setAppInstalling,
|
|
76
|
+
setAppInstalled,
|
|
77
|
+
setAppUninstalled,
|
|
78
|
+
getAppVersion,
|
|
79
|
+
setAppBackup,
|
|
80
|
+
getAppBackup,
|
|
81
|
+
};
|
|
82
|
+
|
|
83
|
+
async function getApps() {
|
|
84
|
+
const config = configFile.read();
|
|
85
|
+
return config.applications;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
async function getReadyApps() {
|
|
89
|
+
const config = configFile.read();
|
|
90
|
+
return config.applications.filter((app) => {
|
|
91
|
+
return app.ready;
|
|
92
|
+
});
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
async function getApp(props: { projectId: string }) {
|
|
96
|
+
const { projectId } = props;
|
|
97
|
+
const apps = await getApps();
|
|
98
|
+
for (const app of apps) {
|
|
99
|
+
if (app.projectId === projectId) {
|
|
100
|
+
return app;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
return null;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
async function isAppPresent(props: { projectId: string }): Promise<boolean> {
|
|
107
|
+
const { projectId } = props;
|
|
108
|
+
const app = await getApp({ projectId });
|
|
109
|
+
if (app) {
|
|
110
|
+
return true;
|
|
111
|
+
}
|
|
112
|
+
return false;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
async function isAppReady(props: { projectId: string }): Promise<boolean> {
|
|
116
|
+
const { projectId } = props;
|
|
117
|
+
const app = await getApp({ projectId });
|
|
118
|
+
if (app && app.ready) {
|
|
119
|
+
return true;
|
|
120
|
+
}
|
|
121
|
+
return false;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
async function removeApp(props: { projectId: string }) {
|
|
125
|
+
const { projectId } = props;
|
|
126
|
+
const config = configFile.read();
|
|
127
|
+
const newList = config.applications.filter((app) => {
|
|
128
|
+
return app.projectId !== projectId;
|
|
129
|
+
});
|
|
130
|
+
config.applications = newList;
|
|
131
|
+
configFile.write(config);
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
async function setAppInstalling(props: { projectId: string; version: string }) {
|
|
135
|
+
const { projectId, version } = props;
|
|
136
|
+
|
|
137
|
+
const app = await getApp({ projectId });
|
|
138
|
+
if (app) {
|
|
139
|
+
await removeApp({ projectId });
|
|
140
|
+
const config = configFile.read();
|
|
141
|
+
// NOTE: do not update the version for an existing app until it is installed
|
|
142
|
+
config.applications.push({ ...app, ...{ ready: false } });
|
|
143
|
+
configFile.write(config);
|
|
144
|
+
} else {
|
|
145
|
+
// NOTE: for a brand-new app, we need to specify the version
|
|
146
|
+
const config = configFile.read();
|
|
147
|
+
config.applications.push({
|
|
148
|
+
projectId,
|
|
149
|
+
version,
|
|
150
|
+
ready: false,
|
|
151
|
+
});
|
|
152
|
+
configFile.write(config);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
async function setAppInstalled(props: { projectId: string; version: string }) {
|
|
157
|
+
const { projectId, version } = props;
|
|
158
|
+
const app = await getApp({ projectId });
|
|
159
|
+
if (app) {
|
|
160
|
+
await removeApp({ projectId });
|
|
161
|
+
const config = configFile.read();
|
|
162
|
+
config.applications.push({ ...app, ...{ version, ready: true } });
|
|
163
|
+
configFile.write(config);
|
|
164
|
+
} else {
|
|
165
|
+
// NOTE: we should never be setting an app as installed
|
|
166
|
+
// if it doesn't exist (setAppInstalling was never called)
|
|
167
|
+
throw new Error(
|
|
168
|
+
`App ${projectId} was not previously configured and could not be set to installed!`,
|
|
169
|
+
);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
async function setAppUninstalled(props: { projectId: string }) {
|
|
174
|
+
const { projectId } = props;
|
|
175
|
+
await removeApp({ projectId });
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
async function getAppVersion(props: { projectId: string }) {
|
|
179
|
+
const { projectId } = props;
|
|
180
|
+
const app = await getApp({ projectId });
|
|
181
|
+
if (!app) {
|
|
182
|
+
throw new Error(`App ${projectId} not installed`);
|
|
183
|
+
}
|
|
184
|
+
return app.version;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
async function setAppBackup(props: { projectId: string }) {
|
|
188
|
+
const { projectId } = props;
|
|
189
|
+
const app = await getApp({ projectId });
|
|
190
|
+
if (!app) {
|
|
191
|
+
throw new Error(`Project ${projectId} not installed!`);
|
|
192
|
+
}
|
|
193
|
+
await removeApp({ projectId });
|
|
194
|
+
const config = configFile.read();
|
|
195
|
+
app.backup = { version: app.version };
|
|
196
|
+
config.applications.push(app);
|
|
197
|
+
configFile.write(config);
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
async function getAppBackup(props: { projectId: string }) {
|
|
201
|
+
const { projectId } = props;
|
|
202
|
+
const app = await getApp({ projectId });
|
|
203
|
+
if (app && app.backup) {
|
|
204
|
+
return app.backup;
|
|
205
|
+
}
|
|
206
|
+
return null;
|
|
207
|
+
}
|
|
208
|
+
}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import {
|
|
2
|
+
DeviceTokens,
|
|
3
|
+
writeOrUpdateDeviceCfgFile,
|
|
4
|
+
} from "alwaysai/lib/components/device";
|
|
5
|
+
import { checkUserIsLoggedInComponent } from "alwaysai/lib/components/user";
|
|
6
|
+
import {
|
|
7
|
+
LOCAL_AAI_CFG_DIR,
|
|
8
|
+
LOCAL_CERT_AND_KEY_DIR,
|
|
9
|
+
} from "alwaysai/lib/constants";
|
|
10
|
+
import { checkPaidPlan } from "alwaysai/lib/core/project";
|
|
11
|
+
import {
|
|
12
|
+
JsSpawner,
|
|
13
|
+
Spawner,
|
|
14
|
+
writeCertificate,
|
|
15
|
+
writeTokens,
|
|
16
|
+
} from "alwaysai/lib/util";
|
|
17
|
+
import { logger } from "../util/logger";
|
|
18
|
+
|
|
19
|
+
// NOTE: This is reimplemented from the CLI to work for local checks
|
|
20
|
+
export async function getTargetHardwareUuid(spawner: Spawner, path: string) {
|
|
21
|
+
return await spawner.run({
|
|
22
|
+
exe: "cat",
|
|
23
|
+
args: [path],
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
// NOTE: This closely follows the flow of deviceCheckAndUpdateComponent in the CLI
|
|
28
|
+
export async function writeCertificateAndToken(props: { deviceUuid: string }) {
|
|
29
|
+
const { deviceUuid } = props;
|
|
30
|
+
await checkUserIsLoggedInComponent({ yes: true });
|
|
31
|
+
if (!(await checkPaidPlan())) {
|
|
32
|
+
throw new Error(
|
|
33
|
+
`This action only supported for Enterprise alwaysAI accounts!`
|
|
34
|
+
);
|
|
35
|
+
}
|
|
36
|
+
const tokens: DeviceTokens = {
|
|
37
|
+
deviceId: deviceUuid,
|
|
38
|
+
accessToken: "",
|
|
39
|
+
refreshToken: "",
|
|
40
|
+
idToken: "",
|
|
41
|
+
};
|
|
42
|
+
const tokenSpawner = JsSpawner({ path: LOCAL_AAI_CFG_DIR });
|
|
43
|
+
await writeTokens({ spawner: tokenSpawner, tokens });
|
|
44
|
+
await writeOrUpdateDeviceCfgFile({ spawner: tokenSpawner });
|
|
45
|
+
|
|
46
|
+
logger.info("Updated tokens and certificate");
|
|
47
|
+
}
|