@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.
Files changed (70) hide show
  1. package/CHANGELOG.md +28 -0
  2. package/client.d.mts +2 -1
  3. package/client.d.mts.map +1 -1
  4. package/client.d.ts +2 -1
  5. package/client.d.ts.map +1 -1
  6. package/client.js +2 -2
  7. package/client.js.map +1 -1
  8. package/client.mjs +1 -1
  9. package/client.mjs.map +1 -1
  10. package/index.d.mts +1 -0
  11. package/index.d.mts.map +1 -1
  12. package/index.d.ts +1 -0
  13. package/index.d.ts.map +1 -1
  14. package/index.js +3 -1
  15. package/index.js.map +1 -1
  16. package/index.mjs +1 -0
  17. package/index.mjs.map +1 -1
  18. package/instance-client.d.mts +68 -0
  19. package/instance-client.d.mts.map +1 -0
  20. package/instance-client.d.ts +68 -0
  21. package/instance-client.d.ts.map +1 -0
  22. package/instance-client.js +214 -0
  23. package/instance-client.js.map +1 -0
  24. package/instance-client.mjs +211 -0
  25. package/instance-client.mjs.map +1 -0
  26. package/internal/utils/values.js +3 -3
  27. package/internal/utils/values.js.map +1 -1
  28. package/internal/utils/values.mjs +3 -3
  29. package/internal/utils/values.mjs.map +1 -1
  30. package/package.json +21 -1
  31. package/resources/android-instances-helpers.d.mts +5 -56
  32. package/resources/android-instances-helpers.d.mts.map +1 -1
  33. package/resources/android-instances-helpers.d.ts +5 -56
  34. package/resources/android-instances-helpers.d.ts.map +1 -1
  35. package/resources/android-instances-helpers.js +19 -217
  36. package/resources/android-instances-helpers.js.map +1 -1
  37. package/resources/android-instances-helpers.mjs +17 -216
  38. package/resources/android-instances-helpers.mjs.map +1 -1
  39. package/resources/index.d.mts +2 -1
  40. package/resources/index.d.mts.map +1 -1
  41. package/resources/index.d.ts +2 -1
  42. package/resources/index.d.ts.map +1 -1
  43. package/resources/index.js +2 -2
  44. package/resources/index.js.map +1 -1
  45. package/resources/index.mjs +1 -1
  46. package/resources/index.mjs.map +1 -1
  47. package/src/client.ts +1 -1
  48. package/src/index.ts +1 -0
  49. package/src/instance-client.ts +345 -0
  50. package/src/internal/utils/values.ts +3 -3
  51. package/src/resources/android-instances-helpers.ts +24 -347
  52. package/src/resources/index.ts +2 -1
  53. package/src/version.ts +1 -1
  54. package/tunnel.d.mts.map +1 -0
  55. package/tunnel.d.ts.map +1 -0
  56. package/{resources/tunnel.js → tunnel.js} +1 -1
  57. package/tunnel.js.map +1 -0
  58. package/tunnel.mjs.map +1 -0
  59. package/version.d.mts +1 -1
  60. package/version.d.ts +1 -1
  61. package/version.js +1 -1
  62. package/version.mjs +1 -1
  63. package/resources/tunnel.d.mts.map +0 -1
  64. package/resources/tunnel.d.ts.map +0 -1
  65. package/resources/tunnel.js.map +0 -1
  66. package/resources/tunnel.mjs.map +0 -1
  67. /package/src/{resources/tunnel.ts → tunnel.ts} +0 -0
  68. /package/{resources/tunnel.d.mts → tunnel.d.mts} +0 -0
  69. /package/{resources/tunnel.d.ts → tunnel.d.ts} +0 -0
  70. /package/{resources/tunnel.mjs → tunnel.mjs} +0 -0
@@ -1,4 +1,4 @@
1
1
  // File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details.
2
- export { AndroidInstances, } from "./android-instances.mjs";
2
+ export { AndroidInstances } from "./android-instances-helpers.mjs";
3
3
  export { Assets } from "./assets-helpers.mjs";
4
4
  //# sourceMappingURL=index.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.mjs","sourceRoot":"","sources":["../src/resources/index.ts"],"names":[],"mappings":"AAAA,sFAAsF;OAE/E,EACL,gBAAgB,GAKjB;OAUM,EAAE,MAAM,EAAoD"}
1
+ {"version":3,"file":"index.mjs","sourceRoot":"","sources":["../src/resources/index.ts"],"names":[],"mappings":"AAAA,sFAAsF;OAiB/E,EAAE,gBAAgB,EAAE;OAEpB,EAAE,MAAM,EAAoD"}
package/src/client.ts CHANGED
@@ -21,7 +21,6 @@ import {
21
21
  AndroidInstanceCreateParams,
22
22
  AndroidInstanceListParams,
23
23
  AndroidInstanceListResponse,
24
- AndroidInstances,
25
24
  } from './resources/android-instances';
26
25
  import {
27
26
  Asset,
@@ -31,6 +30,7 @@ import {
31
30
  AssetListResponse,
32
31
  AssetGetParams,
33
32
  } from './resources/assets';
33
+ import { AndroidInstances } from './resources/android-instances-helpers';
34
34
  import { Assets, AssetGetOrUploadParams, AssetGetOrUploadResponse } from './resources/assets-helpers';
35
35
  import { type Fetch } from './internal/builtin-types';
36
36
  import { HeadersLike, NullableHeaders, buildHeaders } from './internal/headers';
package/src/index.ts CHANGED
@@ -5,6 +5,7 @@ export { Limrun as default } from './client';
5
5
  export { type Uploadable, toFile } from './core/uploads';
6
6
  export { APIPromise } from './core/api-promise';
7
7
  export { Limrun, type ClientOptions } from './client';
8
+ export { createInstanceClient } from './instance-client';
8
9
  export {
9
10
  LimrunError,
10
11
  APIError,
@@ -0,0 +1,345 @@
1
+ import { WebSocket, Data } from 'ws';
2
+ import { exec } from 'node:child_process';
3
+
4
+ import { startTcpTunnel } from './tunnel';
5
+ import type { Tunnel } from './tunnel';
6
+
7
+ /**
8
+ * A client for interacting with a Limbar instance
9
+ */
10
+ export type InstanceClient = {
11
+ /**
12
+ * Take a screenshot of the current screen
13
+ * @returns A promise that resolves to the screenshot data
14
+ */
15
+ screenshot: () => Promise<ScreenshotData>;
16
+ /**
17
+ * Disconnect from the Limbar instance
18
+ */
19
+ disconnect: () => void;
20
+
21
+ /**
22
+ * Establish an ADB tunnel to the instance.
23
+ * Returns the local TCP port and a cleanup function.
24
+ */
25
+ startAdbTunnel: () => Promise<Tunnel>;
26
+ /**
27
+ * Send an asset URL to the instance. The instance will download the asset
28
+ * and process it (currently APK install is supported). Resolves on success,
29
+ * rejects with an Error on failure.
30
+ */
31
+ sendAsset: (url: string) => Promise<void>;
32
+ };
33
+
34
+ /**
35
+ * Controls the verbosity of logging in the client
36
+ */
37
+ export type LogLevel = 'none' | 'error' | 'warn' | 'info' | 'debug';
38
+
39
+ /**
40
+ * Configuration options for creating an Instance API client
41
+ */
42
+ export type InstanceClientOptions = {
43
+ /**
44
+ * The URL of the ADB WebSocket endpoint.
45
+ */
46
+ adbUrl: string;
47
+ /**
48
+ * The URL of the main endpoint WebSocket.
49
+ */
50
+ endpointUrl: string;
51
+ /**
52
+ * The token to use for the WebSocket connections.
53
+ */
54
+ token: string;
55
+ /**
56
+ * Path to the ADB executable.
57
+ * @default 'adb'
58
+ */
59
+ adbPath?: string;
60
+ /**
61
+ * Controls logging verbosity
62
+ * @default 'info'
63
+ */
64
+ logLevel?: LogLevel;
65
+ };
66
+
67
+ type ScreenshotRequest = {
68
+ type: 'screenshot';
69
+ id: string;
70
+ };
71
+
72
+ type ScreenshotResponse = {
73
+ type: 'screenshot';
74
+ dataUri: string;
75
+ id: string;
76
+ };
77
+
78
+ type ScreenshotData = {
79
+ dataUri: string;
80
+ };
81
+
82
+ type ScreenshotErrorResponse = {
83
+ type: 'screenshotError';
84
+ message: string;
85
+ id: string;
86
+ };
87
+
88
+ type AssetRequest = {
89
+ type: 'asset';
90
+ url: string;
91
+ };
92
+
93
+ type AssetResultResponse = {
94
+ type: 'assetResult';
95
+ result: 'success' | 'failure' | string;
96
+ url: string;
97
+ message?: string;
98
+ };
99
+
100
+ type ServerMessage =
101
+ | ScreenshotResponse
102
+ | ScreenshotErrorResponse
103
+ | AssetResultResponse
104
+ | { type: string; [key: string]: unknown };
105
+
106
+ /**
107
+ * Creates a client for interacting with a Limbar instance
108
+ * @param options Configuration options including webrtcUrl, token and log level
109
+ * @returns An InstanceClient for controlling the instance
110
+ */
111
+ export async function createInstanceClient(options: InstanceClientOptions): Promise<InstanceClient> {
112
+ const serverAddress = `${options.endpointUrl}?token=${options.token}`;
113
+ const logLevel = options.logLevel ?? 'info';
114
+ let ws: WebSocket | undefined = undefined;
115
+
116
+ const screenshotRequests: Map<
117
+ string,
118
+ {
119
+ resolver: (value: ScreenshotData | PromiseLike<ScreenshotData>) => void;
120
+ rejecter: (reason?: any) => void;
121
+ }
122
+ > = new Map();
123
+
124
+ const assetRequests: Map<
125
+ string,
126
+ {
127
+ resolver: (value: void | PromiseLike<void>) => void;
128
+ rejecter: (reason?: any) => void;
129
+ }
130
+ > = new Map();
131
+
132
+ // Logger functions
133
+ const logger = {
134
+ debug: (...args: any[]) => {
135
+ if (logLevel === 'debug') console.log(...args);
136
+ },
137
+ info: (...args: any[]) => {
138
+ if (logLevel === 'info' || logLevel === 'debug') console.log(...args);
139
+ },
140
+ warn: (...args: any[]) => {
141
+ if (logLevel === 'warn' || logLevel === 'info' || logLevel === 'debug') console.warn(...args);
142
+ },
143
+ error: (...args: any[]) => {
144
+ if (logLevel !== 'none') console.error(...args);
145
+ },
146
+ };
147
+
148
+ return new Promise<InstanceClient>((resolveConnection, rejectConnection) => {
149
+ logger.debug(`Attempting to connect to WebSocket server at ${serverAddress}...`);
150
+ ws = new WebSocket(serverAddress);
151
+ ws.on('message', (data: Data) => {
152
+ let message: ServerMessage;
153
+ try {
154
+ message = JSON.parse(data.toString());
155
+ } catch (e) {
156
+ logger.error({ data, error: e }, 'Failed to parse JSON message');
157
+ return;
158
+ }
159
+
160
+ switch (message.type) {
161
+ case 'screenshot': {
162
+ if (!('dataUri' in message) || typeof message.dataUri !== 'string' || !('id' in message)) {
163
+ logger.warn('Received invalid screenshot message:', message);
164
+ break;
165
+ }
166
+
167
+ const screenshotMessage = message as ScreenshotResponse;
168
+ const request = screenshotRequests.get(screenshotMessage.id);
169
+
170
+ if (!request) {
171
+ logger.warn(
172
+ `Received screenshot data for unknown or already handled session: ${screenshotMessage.id}`,
173
+ );
174
+ break;
175
+ }
176
+
177
+ logger.debug(`Received screenshot data URI for session ${screenshotMessage.id}.`);
178
+ request.resolver({ dataUri: screenshotMessage.dataUri });
179
+ screenshotRequests.delete(screenshotMessage.id);
180
+ break;
181
+ }
182
+ case 'screenshotError': {
183
+ if (!('message' in message) || !('id' in message)) {
184
+ logger.warn('Received invalid screenshot error message:', message);
185
+ break;
186
+ }
187
+
188
+ const errorMessage = message as ScreenshotErrorResponse;
189
+ const request = screenshotRequests.get(errorMessage.id);
190
+
191
+ if (!request) {
192
+ logger.warn(
193
+ `Received screenshot error for unknown or already handled session: ${errorMessage.id}`,
194
+ );
195
+ break;
196
+ }
197
+
198
+ logger.error(
199
+ `Server reported an error capturing screenshot for session ${errorMessage.id}:`,
200
+ errorMessage.message,
201
+ );
202
+ request.rejecter(new Error(errorMessage.message));
203
+ screenshotRequests.delete(errorMessage.id);
204
+ break;
205
+ }
206
+ case 'assetResult': {
207
+ logger.debug('Received assetResult:', message);
208
+ const request = assetRequests.get(message.url as string);
209
+ if (!request) {
210
+ logger.warn(`Received assetResult for unknown or already handled url: ${message.url}`);
211
+ break;
212
+ }
213
+ if (message.result === 'success') {
214
+ logger.debug('Asset result is success');
215
+ request.resolver();
216
+ assetRequests.delete(message.url as string);
217
+ break;
218
+ }
219
+ const errorMessage =
220
+ typeof message.message === 'string' && message.message ?
221
+ message.message
222
+ : `Asset processing failed: ${JSON.stringify(message)}`;
223
+ logger.debug('Asset result is failure', errorMessage);
224
+ request.rejecter(new Error(errorMessage));
225
+ assetRequests.delete(message.url as string);
226
+ break;
227
+ }
228
+ default:
229
+ logger.warn(`Received unexpected message type: ${message.type}`);
230
+ break;
231
+ }
232
+ });
233
+
234
+ ws.on('error', (err: Error) => {
235
+ logger.error('WebSocket error:', err.message);
236
+ if (ws && (ws.readyState === WebSocket.CONNECTING || ws.readyState === WebSocket.OPEN)) {
237
+ rejectConnection(err);
238
+ }
239
+ screenshotRequests.forEach((request) => request.rejecter(err));
240
+ });
241
+
242
+ ws.on('close', () => {
243
+ logger.debug('Disconnected from server.');
244
+ screenshotRequests.forEach((request) => request.rejecter('Disconnected from server'));
245
+ });
246
+
247
+ const screenshot = async (): Promise<ScreenshotData> => {
248
+ if (!ws || ws.readyState !== WebSocket.OPEN) {
249
+ return Promise.reject(new Error('WebSocket is not connected or connection is not open.'));
250
+ }
251
+
252
+ const id = 'ts-client-' + Date.now();
253
+ const screenshotRequest: ScreenshotRequest = {
254
+ type: 'screenshot',
255
+ id,
256
+ };
257
+
258
+ return new Promise<ScreenshotData>((resolve, reject) => {
259
+ logger.debug('Sending screenshot request:', screenshotRequest);
260
+ ws!.send(JSON.stringify(screenshotRequest), (err?: Error) => {
261
+ if (err) {
262
+ logger.error('Failed to send screenshot request:', err);
263
+ reject(err);
264
+ }
265
+ });
266
+
267
+ const timeout = setTimeout(() => {
268
+ if (screenshotRequests.has(id)) {
269
+ logger.error(`Screenshot request timed out for session ${id}`);
270
+ screenshotRequests.get(id)?.rejecter(new Error('Screenshot request timed out'));
271
+ screenshotRequests.delete(id);
272
+ }
273
+ }, 30000);
274
+ screenshotRequests.set(id, {
275
+ resolver: (value: ScreenshotData | PromiseLike<ScreenshotData>) => {
276
+ clearTimeout(timeout);
277
+ resolve(value);
278
+ screenshotRequests.delete(id);
279
+ },
280
+ rejecter: (reason?: any) => {
281
+ clearTimeout(timeout);
282
+ reject(reason);
283
+ screenshotRequests.delete(id);
284
+ },
285
+ });
286
+ });
287
+ };
288
+
289
+ const disconnect = (): void => {
290
+ if (ws) {
291
+ logger.debug('Closing WebSocket connection.');
292
+ ws.close();
293
+ }
294
+ screenshotRequests.forEach((request) => request.rejecter('Websocket connection closed'));
295
+ };
296
+
297
+ /**
298
+ * Opens a WebSocket TCP proxy for the ADB port and connects the local adb
299
+ * client to it.
300
+ */
301
+ const startAdbTunnel = async (): Promise<Tunnel> => {
302
+ const { address, close } = await startTcpTunnel(options.adbUrl, options.token, '127.0.0.1', 0);
303
+ try {
304
+ await new Promise<void>((resolve, reject) => {
305
+ exec(`${options.adbPath ?? 'adb'} connect ${address.address}:${address.port}`, (err) => {
306
+ if (err) return reject(err);
307
+ resolve();
308
+ });
309
+ });
310
+ logger.debug(`ADB connected on ${address.address}`);
311
+ } catch (err) {
312
+ close();
313
+ throw err;
314
+ }
315
+ return { address, close };
316
+ };
317
+
318
+ const sendAsset = async (url: string): Promise<void> => {
319
+ if (!ws || ws.readyState !== WebSocket.OPEN) {
320
+ return Promise.reject(new Error('WebSocket is not connected or connection is not open.'));
321
+ }
322
+ const assetRequest: AssetRequest = {
323
+ type: 'asset',
324
+ url,
325
+ };
326
+ ws.send(JSON.stringify(assetRequest), (err?: Error) => {
327
+ if (err) {
328
+ logger.error('Failed to send asset request:', err);
329
+ }
330
+ });
331
+ return new Promise<void>((resolve, reject) => {
332
+ assetRequests.set(url, { resolver: resolve, rejecter: reject });
333
+ });
334
+ };
335
+ ws.on('open', () => {
336
+ logger.debug(`Connected to ${serverAddress}`);
337
+ resolveConnection({
338
+ screenshot,
339
+ disconnect,
340
+ startAdbTunnel,
341
+ sendAsset,
342
+ });
343
+ });
344
+ });
345
+ }
@@ -76,21 +76,21 @@ export const coerceBoolean = (value: unknown): boolean => {
76
76
  };
77
77
 
78
78
  export const maybeCoerceInteger = (value: unknown): number | undefined => {
79
- if (value === undefined) {
79
+ if (value == null) {
80
80
  return undefined;
81
81
  }
82
82
  return coerceInteger(value);
83
83
  };
84
84
 
85
85
  export const maybeCoerceFloat = (value: unknown): number | undefined => {
86
- if (value === undefined) {
86
+ if (value == null) {
87
87
  return undefined;
88
88
  }
89
89
  return coerceFloat(value);
90
90
  };
91
91
 
92
92
  export const maybeCoerceBoolean = (value: unknown): boolean | undefined => {
93
- if (value === undefined) {
93
+ if (value == null) {
94
94
  return undefined;
95
95
  }
96
96
  return coerceBoolean(value);