@limrun/api 0.6.2 → 0.8.0
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/CHANGELOG.md +28 -0
- package/client.d.mts +2 -1
- package/client.d.mts.map +1 -1
- package/client.d.ts +2 -1
- package/client.d.ts.map +1 -1
- package/client.js +2 -2
- package/client.js.map +1 -1
- package/client.mjs +1 -1
- package/client.mjs.map +1 -1
- package/index.d.mts +1 -0
- package/index.d.mts.map +1 -1
- package/index.d.ts +1 -0
- package/index.d.ts.map +1 -1
- package/index.js +3 -1
- package/index.js.map +1 -1
- package/index.mjs +1 -0
- package/index.mjs.map +1 -1
- package/instance-client.d.mts +68 -0
- package/instance-client.d.mts.map +1 -0
- package/instance-client.d.ts +68 -0
- package/instance-client.d.ts.map +1 -0
- package/instance-client.js +214 -0
- package/instance-client.js.map +1 -0
- package/instance-client.mjs +211 -0
- package/instance-client.mjs.map +1 -0
- package/internal/utils/values.js +3 -3
- package/internal/utils/values.js.map +1 -1
- package/internal/utils/values.mjs +3 -3
- package/internal/utils/values.mjs.map +1 -1
- package/package.json +21 -1
- package/resources/android-instances-helpers.d.mts +5 -56
- package/resources/android-instances-helpers.d.mts.map +1 -1
- package/resources/android-instances-helpers.d.ts +5 -56
- package/resources/android-instances-helpers.d.ts.map +1 -1
- package/resources/android-instances-helpers.js +19 -217
- package/resources/android-instances-helpers.js.map +1 -1
- package/resources/android-instances-helpers.mjs +17 -216
- package/resources/android-instances-helpers.mjs.map +1 -1
- package/resources/index.d.mts +2 -1
- package/resources/index.d.mts.map +1 -1
- package/resources/index.d.ts +2 -1
- package/resources/index.d.ts.map +1 -1
- package/resources/index.js +2 -2
- package/resources/index.js.map +1 -1
- package/resources/index.mjs +1 -1
- package/resources/index.mjs.map +1 -1
- package/src/client.ts +1 -1
- package/src/index.ts +1 -0
- package/src/instance-client.ts +345 -0
- package/src/internal/utils/values.ts +3 -3
- package/src/resources/android-instances-helpers.ts +24 -347
- package/src/resources/index.ts +2 -1
- package/src/version.ts +1 -1
- package/tunnel.d.mts.map +1 -0
- package/tunnel.d.ts.map +1 -0
- package/{resources/tunnel.js → tunnel.js} +1 -1
- package/tunnel.js.map +1 -0
- package/tunnel.mjs.map +1 -0
- package/version.d.mts +1 -1
- package/version.d.ts +1 -1
- package/version.js +1 -1
- package/version.mjs +1 -1
- package/resources/tunnel.d.mts.map +0 -1
- package/resources/tunnel.d.ts.map +0 -1
- package/resources/tunnel.js.map +0 -1
- package/resources/tunnel.mjs.map +0 -1
- /package/src/{resources/tunnel.ts → tunnel.ts} +0 -0
- /package/{resources/tunnel.d.mts → tunnel.d.mts} +0 -0
- /package/{resources/tunnel.d.ts → tunnel.d.ts} +0 -0
- /package/{resources/tunnel.mjs → tunnel.mjs} +0 -0
|
@@ -1,57 +1,6 @@
|
|
|
1
|
-
import
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
*/
|
|
7
|
-
export type InstanceClient = {
|
|
8
|
-
/**
|
|
9
|
-
* Take a screenshot of the current screen
|
|
10
|
-
* @returns A promise that resolves to the screenshot data
|
|
11
|
-
*/
|
|
12
|
-
screenshot: () => Promise<ScreenshotData>;
|
|
13
|
-
/**
|
|
14
|
-
* Disconnect from the Limbar instance
|
|
15
|
-
*/
|
|
16
|
-
disconnect: () => void;
|
|
17
|
-
/**
|
|
18
|
-
* Establish an ADB tunnel to the instance.
|
|
19
|
-
* Returns the local TCP port and a cleanup function.
|
|
20
|
-
*/
|
|
21
|
-
startAdbTunnel: () => Promise<Tunnel>;
|
|
22
|
-
/**
|
|
23
|
-
* Send an asset URL to the instance. The instance will download the asset
|
|
24
|
-
* and process it (currently APK install is supported). Resolves on success,
|
|
25
|
-
* rejects with an Error on failure.
|
|
26
|
-
*/
|
|
27
|
-
sendAsset: (url: string) => Promise<void>;
|
|
28
|
-
};
|
|
29
|
-
/**
|
|
30
|
-
* Controls the verbosity of logging in the client
|
|
31
|
-
*/
|
|
32
|
-
export type LogLevel = 'none' | 'error' | 'warn' | 'info' | 'debug';
|
|
33
|
-
/**
|
|
34
|
-
* Configuration options for creating an Instance API client
|
|
35
|
-
*/
|
|
36
|
-
export type InstanceClientOptions = {
|
|
37
|
-
/**
|
|
38
|
-
* Path to the ADB executable.
|
|
39
|
-
* @default 'adb'
|
|
40
|
-
*/
|
|
41
|
-
adbPath?: string;
|
|
42
|
-
/**
|
|
43
|
-
* Controls logging verbosity
|
|
44
|
-
* @default 'info'
|
|
45
|
-
*/
|
|
46
|
-
logLevel?: LogLevel;
|
|
47
|
-
};
|
|
48
|
-
type ScreenshotData = {
|
|
49
|
-
dataUri: string;
|
|
50
|
-
};
|
|
51
|
-
/**
|
|
52
|
-
* Creates a client for interacting with a Limbar instance
|
|
53
|
-
* @param options Configuration options including webrtcUrl, token and log level
|
|
54
|
-
* @returns An InstanceClient for controlling the instance
|
|
55
|
-
*/
|
|
56
|
-
export declare function createInstanceClient(androidInstance: AndroidInstance, options?: InstanceClientOptions): Promise<InstanceClient>;
|
|
1
|
+
import { RequestOptions } from "../internal/request-options.js";
|
|
2
|
+
import { AndroidInstance, AndroidInstanceCreateParams, AndroidInstances as GeneratedAndroidInstances } from "./android-instances.js";
|
|
3
|
+
export declare class AndroidInstances extends GeneratedAndroidInstances {
|
|
4
|
+
getOrCreate(params: AndroidInstanceCreateParams, options?: RequestOptions): Promise<AndroidInstance>;
|
|
5
|
+
}
|
|
57
6
|
//# sourceMappingURL=android-instances-helpers.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"android-instances-helpers.d.ts","sourceRoot":"","sources":["../src/resources/android-instances-helpers.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"android-instances-helpers.d.ts","sourceRoot":"","sources":["../src/resources/android-instances-helpers.ts"],"names":[],"mappings":"OAAO,EAAE,cAAc,EAAE;OAClB,EACL,eAAe,EACf,2BAA2B,EAC3B,gBAAgB,IAAI,yBAAyB,EAC9C;AAED,qBAAa,gBAAiB,SAAQ,yBAAyB;IACvD,WAAW,CAAC,MAAM,EAAE,2BAA2B,EAAE,OAAO,CAAC,EAAE,cAAc,GAAG,OAAO,CAAC,eAAe,CAAC;CAkB3G"}
|
|
@@ -1,221 +1,23 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.
|
|
4
|
-
const
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
})
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
const assetRequests = new Map();
|
|
22
|
-
// Logger functions
|
|
23
|
-
const logger = {
|
|
24
|
-
debug: (...args) => {
|
|
25
|
-
if (logLevel === 'debug')
|
|
26
|
-
console.log(...args);
|
|
27
|
-
},
|
|
28
|
-
info: (...args) => {
|
|
29
|
-
if (logLevel === 'info' || logLevel === 'debug')
|
|
30
|
-
console.log(...args);
|
|
31
|
-
},
|
|
32
|
-
warn: (...args) => {
|
|
33
|
-
if (logLevel === 'warn' || logLevel === 'info' || logLevel === 'debug')
|
|
34
|
-
console.warn(...args);
|
|
35
|
-
},
|
|
36
|
-
error: (...args) => {
|
|
37
|
-
if (logLevel !== 'none')
|
|
38
|
-
console.error(...args);
|
|
39
|
-
},
|
|
40
|
-
};
|
|
41
|
-
return new Promise((resolveConnection, rejectConnection) => {
|
|
42
|
-
logger.debug(`Attempting to connect to WebSocket server at ${serverAddress}...`);
|
|
43
|
-
ws = new ws_1.WebSocket(serverAddress);
|
|
44
|
-
ws.on('message', (data) => {
|
|
45
|
-
let message;
|
|
46
|
-
try {
|
|
47
|
-
message = JSON.parse(data.toString());
|
|
48
|
-
}
|
|
49
|
-
catch (e) {
|
|
50
|
-
logger.error({ data, error: e }, 'Failed to parse JSON message');
|
|
51
|
-
return;
|
|
52
|
-
}
|
|
53
|
-
switch (message.type) {
|
|
54
|
-
case 'screenshot': {
|
|
55
|
-
if (!('dataUri' in message) || typeof message.dataUri !== 'string' || !('id' in message)) {
|
|
56
|
-
logger.warn('Received invalid screenshot message:', message);
|
|
57
|
-
break;
|
|
58
|
-
}
|
|
59
|
-
const screenshotMessage = message;
|
|
60
|
-
const request = screenshotRequests.get(screenshotMessage.id);
|
|
61
|
-
if (!request) {
|
|
62
|
-
logger.warn(`Received screenshot data for unknown or already handled session: ${screenshotMessage.id}`);
|
|
63
|
-
break;
|
|
64
|
-
}
|
|
65
|
-
logger.debug(`Received screenshot data URI for session ${screenshotMessage.id}.`);
|
|
66
|
-
request.resolver({ dataUri: screenshotMessage.dataUri });
|
|
67
|
-
screenshotRequests.delete(screenshotMessage.id);
|
|
68
|
-
break;
|
|
69
|
-
}
|
|
70
|
-
case 'screenshotError': {
|
|
71
|
-
if (!('message' in message) || !('id' in message)) {
|
|
72
|
-
logger.warn('Received invalid screenshot error message:', message);
|
|
73
|
-
break;
|
|
74
|
-
}
|
|
75
|
-
const errorMessage = message;
|
|
76
|
-
const request = screenshotRequests.get(errorMessage.id);
|
|
77
|
-
if (!request) {
|
|
78
|
-
logger.warn(`Received screenshot error for unknown or already handled session: ${errorMessage.id}`);
|
|
79
|
-
break;
|
|
80
|
-
}
|
|
81
|
-
logger.error(`Server reported an error capturing screenshot for session ${errorMessage.id}:`, errorMessage.message);
|
|
82
|
-
request.rejecter(new Error(errorMessage.message));
|
|
83
|
-
screenshotRequests.delete(errorMessage.id);
|
|
84
|
-
break;
|
|
85
|
-
}
|
|
86
|
-
case 'assetResult': {
|
|
87
|
-
logger.debug('Received assetResult:', message);
|
|
88
|
-
const request = assetRequests.get(message.url);
|
|
89
|
-
if (!request) {
|
|
90
|
-
logger.warn(`Received assetResult for unknown or already handled url: ${message.url}`);
|
|
91
|
-
break;
|
|
92
|
-
}
|
|
93
|
-
if (message.result === 'success') {
|
|
94
|
-
logger.debug('Asset result is success');
|
|
95
|
-
request.resolver();
|
|
96
|
-
assetRequests.delete(message.url);
|
|
97
|
-
break;
|
|
98
|
-
}
|
|
99
|
-
const errorMessage = typeof message.message === 'string' && message.message ?
|
|
100
|
-
message.message
|
|
101
|
-
: `Asset processing failed: ${JSON.stringify(message)}`;
|
|
102
|
-
logger.debug('Asset result is failure', errorMessage);
|
|
103
|
-
request.rejecter(new Error(errorMessage));
|
|
104
|
-
assetRequests.delete(message.url);
|
|
105
|
-
break;
|
|
106
|
-
}
|
|
107
|
-
default:
|
|
108
|
-
logger.warn(`Received unexpected message type: ${message.type}`);
|
|
109
|
-
break;
|
|
110
|
-
}
|
|
111
|
-
});
|
|
112
|
-
ws.on('error', (err) => {
|
|
113
|
-
logger.error('WebSocket error:', err.message);
|
|
114
|
-
if (ws && (ws.readyState === ws_1.WebSocket.CONNECTING || ws.readyState === ws_1.WebSocket.OPEN)) {
|
|
115
|
-
rejectConnection(err);
|
|
116
|
-
}
|
|
117
|
-
screenshotRequests.forEach((request) => request.rejecter(err));
|
|
118
|
-
});
|
|
119
|
-
ws.on('close', () => {
|
|
120
|
-
logger.debug('Disconnected from server.');
|
|
121
|
-
screenshotRequests.forEach((request) => request.rejecter('Disconnected from server'));
|
|
122
|
-
});
|
|
123
|
-
const screenshot = async () => {
|
|
124
|
-
if (!ws || ws.readyState !== ws_1.WebSocket.OPEN) {
|
|
125
|
-
return Promise.reject(new Error('WebSocket is not connected or connection is not open.'));
|
|
126
|
-
}
|
|
127
|
-
const id = 'ts-client-' + Date.now();
|
|
128
|
-
const screenshotRequest = {
|
|
129
|
-
type: 'screenshot',
|
|
130
|
-
id,
|
|
131
|
-
};
|
|
132
|
-
return new Promise((resolve, reject) => {
|
|
133
|
-
logger.debug('Sending screenshot request:', screenshotRequest);
|
|
134
|
-
ws.send(JSON.stringify(screenshotRequest), (err) => {
|
|
135
|
-
if (err) {
|
|
136
|
-
logger.error('Failed to send screenshot request:', err);
|
|
137
|
-
reject(err);
|
|
138
|
-
}
|
|
139
|
-
});
|
|
140
|
-
const timeout = setTimeout(() => {
|
|
141
|
-
if (screenshotRequests.has(id)) {
|
|
142
|
-
logger.error(`Screenshot request timed out for session ${id}`);
|
|
143
|
-
screenshotRequests.get(id)?.rejecter(new Error('Screenshot request timed out'));
|
|
144
|
-
screenshotRequests.delete(id);
|
|
145
|
-
}
|
|
146
|
-
}, 30000);
|
|
147
|
-
screenshotRequests.set(id, {
|
|
148
|
-
resolver: (value) => {
|
|
149
|
-
clearTimeout(timeout);
|
|
150
|
-
resolve(value);
|
|
151
|
-
screenshotRequests.delete(id);
|
|
152
|
-
},
|
|
153
|
-
rejecter: (reason) => {
|
|
154
|
-
clearTimeout(timeout);
|
|
155
|
-
reject(reason);
|
|
156
|
-
screenshotRequests.delete(id);
|
|
157
|
-
},
|
|
158
|
-
});
|
|
159
|
-
});
|
|
160
|
-
};
|
|
161
|
-
const disconnect = () => {
|
|
162
|
-
if (ws) {
|
|
163
|
-
logger.debug('Closing WebSocket connection.');
|
|
164
|
-
ws.close();
|
|
165
|
-
}
|
|
166
|
-
screenshotRequests.forEach((request) => request.rejecter('Websocket connection closed'));
|
|
167
|
-
};
|
|
168
|
-
/**
|
|
169
|
-
* Opens a WebSocket TCP proxy for the ADB port and connects the local adb
|
|
170
|
-
* client to it.
|
|
171
|
-
*/
|
|
172
|
-
const startAdbTunnel = async () => {
|
|
173
|
-
if (!androidInstance.status.adbWebSocketUrl) {
|
|
174
|
-
return Promise.reject(new Error('ADB WebSocket URL is not set'));
|
|
175
|
-
}
|
|
176
|
-
const { address, close } = await (0, tunnel_js_1.startTcpTunnel)(androidInstance.status.adbWebSocketUrl, token, '127.0.0.1', 0);
|
|
177
|
-
try {
|
|
178
|
-
await new Promise((resolve, reject) => {
|
|
179
|
-
(0, node_child_process_1.exec)(`${options.adbPath ?? 'adb'} connect ${address.address}:${address.port}`, (err) => {
|
|
180
|
-
if (err)
|
|
181
|
-
return reject(err);
|
|
182
|
-
resolve();
|
|
183
|
-
});
|
|
184
|
-
});
|
|
185
|
-
logger.debug(`ADB connected on ${address.address}`);
|
|
186
|
-
}
|
|
187
|
-
catch (err) {
|
|
188
|
-
close();
|
|
189
|
-
throw err;
|
|
190
|
-
}
|
|
191
|
-
return { address, close };
|
|
192
|
-
};
|
|
193
|
-
const sendAsset = async (url) => {
|
|
194
|
-
if (!ws || ws.readyState !== ws_1.WebSocket.OPEN) {
|
|
195
|
-
return Promise.reject(new Error('WebSocket is not connected or connection is not open.'));
|
|
196
|
-
}
|
|
197
|
-
const assetRequest = {
|
|
198
|
-
type: 'asset',
|
|
199
|
-
url,
|
|
200
|
-
};
|
|
201
|
-
ws.send(JSON.stringify(assetRequest), (err) => {
|
|
202
|
-
if (err) {
|
|
203
|
-
logger.error('Failed to send asset request:', err);
|
|
204
|
-
}
|
|
205
|
-
});
|
|
206
|
-
return new Promise((resolve, reject) => {
|
|
207
|
-
assetRequests.set(url, { resolver: resolve, rejecter: reject });
|
|
208
|
-
});
|
|
209
|
-
};
|
|
210
|
-
ws.on('open', () => {
|
|
211
|
-
logger.debug(`Connected to ${serverAddress}`);
|
|
212
|
-
resolveConnection({
|
|
213
|
-
screenshot,
|
|
214
|
-
disconnect,
|
|
215
|
-
startAdbTunnel,
|
|
216
|
-
sendAsset,
|
|
217
|
-
});
|
|
218
|
-
});
|
|
219
|
-
});
|
|
3
|
+
exports.AndroidInstances = void 0;
|
|
4
|
+
const android_instances_1 = require("./android-instances.js");
|
|
5
|
+
class AndroidInstances extends android_instances_1.AndroidInstances {
|
|
6
|
+
async getOrCreate(params, options) {
|
|
7
|
+
if (!params.metadata || !params.metadata.labels || Object.keys(params.metadata.labels).length === 0) {
|
|
8
|
+
return Promise.reject(new Error('At least one label is required for getOrCreate operation'));
|
|
9
|
+
}
|
|
10
|
+
const instances = await super.list({
|
|
11
|
+
labelSelector: Object.entries(params.metadata.labels)
|
|
12
|
+
.map(([key, value]) => `${key}=${value}`)
|
|
13
|
+
.join(','),
|
|
14
|
+
state: 'ready',
|
|
15
|
+
}, options);
|
|
16
|
+
if (instances && instances.length > 0) {
|
|
17
|
+
return instances[0];
|
|
18
|
+
}
|
|
19
|
+
return super.create(params, options);
|
|
20
|
+
}
|
|
220
21
|
}
|
|
22
|
+
exports.AndroidInstances = AndroidInstances;
|
|
221
23
|
//# sourceMappingURL=android-instances-helpers.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"android-instances-helpers.js","sourceRoot":"","sources":["../src/resources/android-instances-helpers.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"android-instances-helpers.js","sourceRoot":"","sources":["../src/resources/android-instances-helpers.ts"],"names":[],"mappings":";;;AACA,8DAI6B;AAE7B,MAAa,gBAAiB,SAAQ,oCAAyB;IAC7D,KAAK,CAAC,WAAW,CAAC,MAAmC,EAAE,OAAwB;QAC7E,IAAI,CAAC,MAAM,CAAC,QAAQ,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACpG,OAAO,OAAO,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC,CAAC;QAC/F,CAAC;QACD,MAAM,SAAS,GAAG,MAAM,KAAK,CAAC,IAAI,CAChC;YACE,aAAa,EAAE,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC;iBAClD,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,IAAI,KAAK,EAAE,CAAC;iBACxC,IAAI,CAAC,GAAG,CAAC;YACZ,KAAK,EAAE,OAAO;SACf,EACD,OAAO,CACR,CAAC;QACF,IAAI,SAAS,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtC,OAAO,SAAS,CAAC,CAAC,CAAE,CAAC;QACvB,CAAC;QACD,OAAO,KAAK,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACvC,CAAC;CACF;AAnBD,4CAmBC"}
|
|
@@ -1,218 +1,19 @@
|
|
|
1
|
-
import {
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
})
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
const assetRequests = new Map();
|
|
19
|
-
// Logger functions
|
|
20
|
-
const logger = {
|
|
21
|
-
debug: (...args) => {
|
|
22
|
-
if (logLevel === 'debug')
|
|
23
|
-
console.log(...args);
|
|
24
|
-
},
|
|
25
|
-
info: (...args) => {
|
|
26
|
-
if (logLevel === 'info' || logLevel === 'debug')
|
|
27
|
-
console.log(...args);
|
|
28
|
-
},
|
|
29
|
-
warn: (...args) => {
|
|
30
|
-
if (logLevel === 'warn' || logLevel === 'info' || logLevel === 'debug')
|
|
31
|
-
console.warn(...args);
|
|
32
|
-
},
|
|
33
|
-
error: (...args) => {
|
|
34
|
-
if (logLevel !== 'none')
|
|
35
|
-
console.error(...args);
|
|
36
|
-
},
|
|
37
|
-
};
|
|
38
|
-
return new Promise((resolveConnection, rejectConnection) => {
|
|
39
|
-
logger.debug(`Attempting to connect to WebSocket server at ${serverAddress}...`);
|
|
40
|
-
ws = new WebSocket(serverAddress);
|
|
41
|
-
ws.on('message', (data) => {
|
|
42
|
-
let message;
|
|
43
|
-
try {
|
|
44
|
-
message = JSON.parse(data.toString());
|
|
45
|
-
}
|
|
46
|
-
catch (e) {
|
|
47
|
-
logger.error({ data, error: e }, 'Failed to parse JSON message');
|
|
48
|
-
return;
|
|
49
|
-
}
|
|
50
|
-
switch (message.type) {
|
|
51
|
-
case 'screenshot': {
|
|
52
|
-
if (!('dataUri' in message) || typeof message.dataUri !== 'string' || !('id' in message)) {
|
|
53
|
-
logger.warn('Received invalid screenshot message:', message);
|
|
54
|
-
break;
|
|
55
|
-
}
|
|
56
|
-
const screenshotMessage = message;
|
|
57
|
-
const request = screenshotRequests.get(screenshotMessage.id);
|
|
58
|
-
if (!request) {
|
|
59
|
-
logger.warn(`Received screenshot data for unknown or already handled session: ${screenshotMessage.id}`);
|
|
60
|
-
break;
|
|
61
|
-
}
|
|
62
|
-
logger.debug(`Received screenshot data URI for session ${screenshotMessage.id}.`);
|
|
63
|
-
request.resolver({ dataUri: screenshotMessage.dataUri });
|
|
64
|
-
screenshotRequests.delete(screenshotMessage.id);
|
|
65
|
-
break;
|
|
66
|
-
}
|
|
67
|
-
case 'screenshotError': {
|
|
68
|
-
if (!('message' in message) || !('id' in message)) {
|
|
69
|
-
logger.warn('Received invalid screenshot error message:', message);
|
|
70
|
-
break;
|
|
71
|
-
}
|
|
72
|
-
const errorMessage = message;
|
|
73
|
-
const request = screenshotRequests.get(errorMessage.id);
|
|
74
|
-
if (!request) {
|
|
75
|
-
logger.warn(`Received screenshot error for unknown or already handled session: ${errorMessage.id}`);
|
|
76
|
-
break;
|
|
77
|
-
}
|
|
78
|
-
logger.error(`Server reported an error capturing screenshot for session ${errorMessage.id}:`, errorMessage.message);
|
|
79
|
-
request.rejecter(new Error(errorMessage.message));
|
|
80
|
-
screenshotRequests.delete(errorMessage.id);
|
|
81
|
-
break;
|
|
82
|
-
}
|
|
83
|
-
case 'assetResult': {
|
|
84
|
-
logger.debug('Received assetResult:', message);
|
|
85
|
-
const request = assetRequests.get(message.url);
|
|
86
|
-
if (!request) {
|
|
87
|
-
logger.warn(`Received assetResult for unknown or already handled url: ${message.url}`);
|
|
88
|
-
break;
|
|
89
|
-
}
|
|
90
|
-
if (message.result === 'success') {
|
|
91
|
-
logger.debug('Asset result is success');
|
|
92
|
-
request.resolver();
|
|
93
|
-
assetRequests.delete(message.url);
|
|
94
|
-
break;
|
|
95
|
-
}
|
|
96
|
-
const errorMessage = typeof message.message === 'string' && message.message ?
|
|
97
|
-
message.message
|
|
98
|
-
: `Asset processing failed: ${JSON.stringify(message)}`;
|
|
99
|
-
logger.debug('Asset result is failure', errorMessage);
|
|
100
|
-
request.rejecter(new Error(errorMessage));
|
|
101
|
-
assetRequests.delete(message.url);
|
|
102
|
-
break;
|
|
103
|
-
}
|
|
104
|
-
default:
|
|
105
|
-
logger.warn(`Received unexpected message type: ${message.type}`);
|
|
106
|
-
break;
|
|
107
|
-
}
|
|
108
|
-
});
|
|
109
|
-
ws.on('error', (err) => {
|
|
110
|
-
logger.error('WebSocket error:', err.message);
|
|
111
|
-
if (ws && (ws.readyState === WebSocket.CONNECTING || ws.readyState === WebSocket.OPEN)) {
|
|
112
|
-
rejectConnection(err);
|
|
113
|
-
}
|
|
114
|
-
screenshotRequests.forEach((request) => request.rejecter(err));
|
|
115
|
-
});
|
|
116
|
-
ws.on('close', () => {
|
|
117
|
-
logger.debug('Disconnected from server.');
|
|
118
|
-
screenshotRequests.forEach((request) => request.rejecter('Disconnected from server'));
|
|
119
|
-
});
|
|
120
|
-
const screenshot = async () => {
|
|
121
|
-
if (!ws || ws.readyState !== WebSocket.OPEN) {
|
|
122
|
-
return Promise.reject(new Error('WebSocket is not connected or connection is not open.'));
|
|
123
|
-
}
|
|
124
|
-
const id = 'ts-client-' + Date.now();
|
|
125
|
-
const screenshotRequest = {
|
|
126
|
-
type: 'screenshot',
|
|
127
|
-
id,
|
|
128
|
-
};
|
|
129
|
-
return new Promise((resolve, reject) => {
|
|
130
|
-
logger.debug('Sending screenshot request:', screenshotRequest);
|
|
131
|
-
ws.send(JSON.stringify(screenshotRequest), (err) => {
|
|
132
|
-
if (err) {
|
|
133
|
-
logger.error('Failed to send screenshot request:', err);
|
|
134
|
-
reject(err);
|
|
135
|
-
}
|
|
136
|
-
});
|
|
137
|
-
const timeout = setTimeout(() => {
|
|
138
|
-
if (screenshotRequests.has(id)) {
|
|
139
|
-
logger.error(`Screenshot request timed out for session ${id}`);
|
|
140
|
-
screenshotRequests.get(id)?.rejecter(new Error('Screenshot request timed out'));
|
|
141
|
-
screenshotRequests.delete(id);
|
|
142
|
-
}
|
|
143
|
-
}, 30000);
|
|
144
|
-
screenshotRequests.set(id, {
|
|
145
|
-
resolver: (value) => {
|
|
146
|
-
clearTimeout(timeout);
|
|
147
|
-
resolve(value);
|
|
148
|
-
screenshotRequests.delete(id);
|
|
149
|
-
},
|
|
150
|
-
rejecter: (reason) => {
|
|
151
|
-
clearTimeout(timeout);
|
|
152
|
-
reject(reason);
|
|
153
|
-
screenshotRequests.delete(id);
|
|
154
|
-
},
|
|
155
|
-
});
|
|
156
|
-
});
|
|
157
|
-
};
|
|
158
|
-
const disconnect = () => {
|
|
159
|
-
if (ws) {
|
|
160
|
-
logger.debug('Closing WebSocket connection.');
|
|
161
|
-
ws.close();
|
|
162
|
-
}
|
|
163
|
-
screenshotRequests.forEach((request) => request.rejecter('Websocket connection closed'));
|
|
164
|
-
};
|
|
165
|
-
/**
|
|
166
|
-
* Opens a WebSocket TCP proxy for the ADB port and connects the local adb
|
|
167
|
-
* client to it.
|
|
168
|
-
*/
|
|
169
|
-
const startAdbTunnel = async () => {
|
|
170
|
-
if (!androidInstance.status.adbWebSocketUrl) {
|
|
171
|
-
return Promise.reject(new Error('ADB WebSocket URL is not set'));
|
|
172
|
-
}
|
|
173
|
-
const { address, close } = await startTcpTunnel(androidInstance.status.adbWebSocketUrl, token, '127.0.0.1', 0);
|
|
174
|
-
try {
|
|
175
|
-
await new Promise((resolve, reject) => {
|
|
176
|
-
exec(`${options.adbPath ?? 'adb'} connect ${address.address}:${address.port}`, (err) => {
|
|
177
|
-
if (err)
|
|
178
|
-
return reject(err);
|
|
179
|
-
resolve();
|
|
180
|
-
});
|
|
181
|
-
});
|
|
182
|
-
logger.debug(`ADB connected on ${address.address}`);
|
|
183
|
-
}
|
|
184
|
-
catch (err) {
|
|
185
|
-
close();
|
|
186
|
-
throw err;
|
|
187
|
-
}
|
|
188
|
-
return { address, close };
|
|
189
|
-
};
|
|
190
|
-
const sendAsset = async (url) => {
|
|
191
|
-
if (!ws || ws.readyState !== WebSocket.OPEN) {
|
|
192
|
-
return Promise.reject(new Error('WebSocket is not connected or connection is not open.'));
|
|
193
|
-
}
|
|
194
|
-
const assetRequest = {
|
|
195
|
-
type: 'asset',
|
|
196
|
-
url,
|
|
197
|
-
};
|
|
198
|
-
ws.send(JSON.stringify(assetRequest), (err) => {
|
|
199
|
-
if (err) {
|
|
200
|
-
logger.error('Failed to send asset request:', err);
|
|
201
|
-
}
|
|
202
|
-
});
|
|
203
|
-
return new Promise((resolve, reject) => {
|
|
204
|
-
assetRequests.set(url, { resolver: resolve, rejecter: reject });
|
|
205
|
-
});
|
|
206
|
-
};
|
|
207
|
-
ws.on('open', () => {
|
|
208
|
-
logger.debug(`Connected to ${serverAddress}`);
|
|
209
|
-
resolveConnection({
|
|
210
|
-
screenshot,
|
|
211
|
-
disconnect,
|
|
212
|
-
startAdbTunnel,
|
|
213
|
-
sendAsset,
|
|
214
|
-
});
|
|
215
|
-
});
|
|
216
|
-
});
|
|
1
|
+
import { AndroidInstances as GeneratedAndroidInstances, } from "./android-instances.mjs";
|
|
2
|
+
export class AndroidInstances extends GeneratedAndroidInstances {
|
|
3
|
+
async getOrCreate(params, options) {
|
|
4
|
+
if (!params.metadata || !params.metadata.labels || Object.keys(params.metadata.labels).length === 0) {
|
|
5
|
+
return Promise.reject(new Error('At least one label is required for getOrCreate operation'));
|
|
6
|
+
}
|
|
7
|
+
const instances = await super.list({
|
|
8
|
+
labelSelector: Object.entries(params.metadata.labels)
|
|
9
|
+
.map(([key, value]) => `${key}=${value}`)
|
|
10
|
+
.join(','),
|
|
11
|
+
state: 'ready',
|
|
12
|
+
}, options);
|
|
13
|
+
if (instances && instances.length > 0) {
|
|
14
|
+
return instances[0];
|
|
15
|
+
}
|
|
16
|
+
return super.create(params, options);
|
|
17
|
+
}
|
|
217
18
|
}
|
|
218
19
|
//# sourceMappingURL=android-instances-helpers.mjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"android-instances-helpers.mjs","sourceRoot":"","sources":["../src/resources/android-instances-helpers.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"android-instances-helpers.mjs","sourceRoot":"","sources":["../src/resources/android-instances-helpers.ts"],"names":[],"mappings":"OACO,EAGL,gBAAgB,IAAI,yBAAyB,GAC9C;AAED,MAAM,OAAO,gBAAiB,SAAQ,yBAAyB;IAC7D,KAAK,CAAC,WAAW,CAAC,MAAmC,EAAE,OAAwB;QAC7E,IAAI,CAAC,MAAM,CAAC,QAAQ,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACpG,OAAO,OAAO,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC,CAAC;QAC/F,CAAC;QACD,MAAM,SAAS,GAAG,MAAM,KAAK,CAAC,IAAI,CAChC;YACE,aAAa,EAAE,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC;iBAClD,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,IAAI,KAAK,EAAE,CAAC;iBACxC,IAAI,CAAC,GAAG,CAAC;YACZ,KAAK,EAAE,OAAO;SACf,EACD,OAAO,CACR,CAAC;QACF,IAAI,SAAS,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtC,OAAO,SAAS,CAAC,CAAC,CAAE,CAAC;QACvB,CAAC;QACD,OAAO,KAAK,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACvC,CAAC;CACF"}
|
package/resources/index.d.mts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
export {
|
|
1
|
+
export { type AndroidInstance, type AndroidInstanceListResponse, type AndroidInstanceCreateParams, type AndroidInstanceListParams, } from "./android-instances.mjs";
|
|
2
2
|
export { type Asset, type AssetListResponse, type AssetGetOrCreateResponse, type AssetListParams, type AssetGetParams, type AssetGetOrCreateParams, } from "./assets.mjs";
|
|
3
|
+
export { AndroidInstances } from "./android-instances-helpers.mjs";
|
|
3
4
|
export { Assets, AssetGetOrUploadParams, AssetGetOrUploadResponse } from "./assets-helpers.mjs";
|
|
4
5
|
//# sourceMappingURL=index.d.mts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.mts","sourceRoot":"","sources":["../src/resources/index.ts"],"names":[],"mappings":"OAEO,EACL,
|
|
1
|
+
{"version":3,"file":"index.d.mts","sourceRoot":"","sources":["../src/resources/index.ts"],"names":[],"mappings":"OAEO,EACL,KAAK,eAAe,EACpB,KAAK,2BAA2B,EAChC,KAAK,2BAA2B,EAChC,KAAK,yBAAyB,GAC/B;OACM,EACL,KAAK,KAAK,EACV,KAAK,iBAAiB,EACtB,KAAK,wBAAwB,EAC7B,KAAK,eAAe,EACpB,KAAK,cAAc,EACnB,KAAK,sBAAsB,GAC5B;OAEM,EAAE,gBAAgB,EAAE;OAEpB,EAAE,MAAM,EAAE,sBAAsB,EAAE,wBAAwB,EAAE"}
|
package/resources/index.d.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
export {
|
|
1
|
+
export { type AndroidInstance, type AndroidInstanceListResponse, type AndroidInstanceCreateParams, type AndroidInstanceListParams, } from "./android-instances.js";
|
|
2
2
|
export { type Asset, type AssetListResponse, type AssetGetOrCreateResponse, type AssetListParams, type AssetGetParams, type AssetGetOrCreateParams, } from "./assets.js";
|
|
3
|
+
export { AndroidInstances } from "./android-instances-helpers.js";
|
|
3
4
|
export { Assets, AssetGetOrUploadParams, AssetGetOrUploadResponse } from "./assets-helpers.js";
|
|
4
5
|
//# sourceMappingURL=index.d.ts.map
|
package/resources/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/resources/index.ts"],"names":[],"mappings":"OAEO,EACL,
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/resources/index.ts"],"names":[],"mappings":"OAEO,EACL,KAAK,eAAe,EACpB,KAAK,2BAA2B,EAChC,KAAK,2BAA2B,EAChC,KAAK,yBAAyB,GAC/B;OACM,EACL,KAAK,KAAK,EACV,KAAK,iBAAiB,EACtB,KAAK,wBAAwB,EAC7B,KAAK,eAAe,EACpB,KAAK,cAAc,EACnB,KAAK,sBAAsB,GAC5B;OAEM,EAAE,gBAAgB,EAAE;OAEpB,EAAE,MAAM,EAAE,sBAAsB,EAAE,wBAAwB,EAAE"}
|
package/resources/index.js
CHANGED
|
@@ -2,8 +2,8 @@
|
|
|
2
2
|
// File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details.
|
|
3
3
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
4
|
exports.Assets = exports.AndroidInstances = void 0;
|
|
5
|
-
var
|
|
6
|
-
Object.defineProperty(exports, "AndroidInstances", { enumerable: true, get: function () { return
|
|
5
|
+
var android_instances_helpers_1 = require("./android-instances-helpers.js");
|
|
6
|
+
Object.defineProperty(exports, "AndroidInstances", { enumerable: true, get: function () { return android_instances_helpers_1.AndroidInstances; } });
|
|
7
7
|
var assets_helpers_1 = require("./assets-helpers.js");
|
|
8
8
|
Object.defineProperty(exports, "Assets", { enumerable: true, get: function () { return assets_helpers_1.Assets; } });
|
|
9
9
|
//# sourceMappingURL=index.js.map
|
package/resources/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/resources/index.ts"],"names":[],"mappings":";AAAA,sFAAsF;;;
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/resources/index.ts"],"names":[],"mappings":";AAAA,sFAAsF;;;AAiBtF,4EAA+D;AAAtD,6HAAA,gBAAgB,OAAA;AAEzB,sDAA4F;AAAnF,wGAAA,MAAM,OAAA"}
|