@intuned/runtime-dev 1.3.18-interface.10 → 1.3.18-interface.12

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.
@@ -0,0 +1,2094 @@
1
+ #!/usr/bin/env node
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __commonJS = (cb, mod) => function __require() {
9
+ return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
10
+ };
11
+ var __export = (target, all) => {
12
+ for (var name in all)
13
+ __defProp(target, name, { get: all[name], enumerable: true });
14
+ };
15
+ var __copyProps = (to, from, except, desc) => {
16
+ if (from && typeof from === "object" || typeof from === "function") {
17
+ for (let key of __getOwnPropNames(from))
18
+ if (!__hasOwnProp.call(to, key) && key !== except)
19
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
20
+ }
21
+ return to;
22
+ };
23
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
24
+ // If the importer is in node compatibility mode or this is not an ESM
25
+ // file that has been converted to a CommonJS file using a Babel-
26
+ // compatible transform (i.e. "__esModule" has not been set), then set
27
+ // "default" to the CommonJS "module.exports" for node compatibility.
28
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
29
+ mod
30
+ ));
31
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
32
+
33
+ // ../packages/runtime-interface/dist/errors.js
34
+ var require_errors = __commonJS({
35
+ "../packages/runtime-interface/dist/errors.js"(exports2) {
36
+ "use strict";
37
+ Object.defineProperty(exports2, "__esModule", { value: true });
38
+ exports2.ResultTooBigError = exports2.InternalInvalidInputError = exports2.AutomationError = exports2.MaxLevelsExceededError = exports2.AuthCheckFailedError = exports2.AuthCheckNotFoundError = exports2.AuthRequiredError = exports2.AbortedError = exports2.InvalidCheckError = exports2.InvalidApiError = exports2.ApiNotFoundError = exports2.RunAutomationError = exports2.runAutomationErrorCodes = exports2.resultTooBigErrorCode = exports2.internalInvalidInputErrorCode = exports2.automationError = exports2.maxLevelsExceededErrorCode = exports2.authCheckFailedErrorCode = exports2.authCheckNotFoundErrorCode = exports2.authRequiredErrorCode = exports2.abortedErrorCode = exports2.invalidCheckErrorCode = exports2.invalidApiErrorCode = exports2.apiNotFoundErrorCode = void 0;
39
+ exports2.apiNotFoundErrorCode = "APINotFoundError";
40
+ exports2.invalidApiErrorCode = "InvalidAPIError";
41
+ exports2.invalidCheckErrorCode = "InvalidCheckError";
42
+ exports2.abortedErrorCode = "AbortedError";
43
+ exports2.authRequiredErrorCode = "AuthRequiredError";
44
+ exports2.authCheckNotFoundErrorCode = "AuthCheckNotFoundError";
45
+ exports2.authCheckFailedErrorCode = "AuthCheckFailedError";
46
+ exports2.maxLevelsExceededErrorCode = "MaxLevelsExceededError";
47
+ exports2.automationError = "AutomationError";
48
+ exports2.internalInvalidInputErrorCode = "InternalInvalidInputError";
49
+ exports2.resultTooBigErrorCode = "ResultTooBigError";
50
+ exports2.runAutomationErrorCodes = [
51
+ exports2.apiNotFoundErrorCode,
52
+ exports2.invalidApiErrorCode,
53
+ exports2.invalidCheckErrorCode,
54
+ exports2.abortedErrorCode,
55
+ exports2.authRequiredErrorCode,
56
+ exports2.authCheckNotFoundErrorCode,
57
+ exports2.authCheckFailedErrorCode,
58
+ exports2.maxLevelsExceededErrorCode,
59
+ exports2.automationError,
60
+ exports2.internalInvalidInputErrorCode,
61
+ exports2.resultTooBigErrorCode
62
+ ];
63
+ var RunAutomationError4 = class {
64
+ constructor(code, message) {
65
+ this.code = code;
66
+ this.message = message;
67
+ }
68
+ get json() {
69
+ return {
70
+ code: this.code,
71
+ details: this.details,
72
+ cause: this.cause?.json
73
+ };
74
+ }
75
+ };
76
+ exports2.RunAutomationError = RunAutomationError4;
77
+ var ApiNotFoundError3 = class extends RunAutomationError4 {
78
+ constructor(apiName) {
79
+ super(exports2.apiNotFoundErrorCode, `API ${apiName} not found`);
80
+ }
81
+ };
82
+ exports2.ApiNotFoundError = ApiNotFoundError3;
83
+ var InvalidApiError2 = class extends RunAutomationError4 {
84
+ constructor(message) {
85
+ super(exports2.invalidApiErrorCode, `API is invalid: ${message}`);
86
+ this.details = {
87
+ message
88
+ };
89
+ }
90
+ };
91
+ exports2.InvalidApiError = InvalidApiError2;
92
+ var InvalidCheckError = class extends RunAutomationError4 {
93
+ constructor(message, cause) {
94
+ super(exports2.invalidCheckErrorCode, message);
95
+ this.cause = cause;
96
+ }
97
+ };
98
+ exports2.InvalidCheckError = InvalidCheckError;
99
+ var AbortedError2 = class extends RunAutomationError4 {
100
+ constructor() {
101
+ super(exports2.abortedErrorCode, "Operation was aborted");
102
+ }
103
+ };
104
+ exports2.AbortedError = AbortedError2;
105
+ var AuthRequiredError2 = class extends RunAutomationError4 {
106
+ constructor() {
107
+ super(exports2.authRequiredErrorCode, "AuthSessions are required");
108
+ }
109
+ };
110
+ exports2.AuthRequiredError = AuthRequiredError2;
111
+ var AuthCheckNotFoundError = class extends RunAutomationError4 {
112
+ constructor() {
113
+ super(exports2.authCheckNotFoundErrorCode, "AuthSession check not found");
114
+ }
115
+ };
116
+ exports2.AuthCheckNotFoundError = AuthCheckNotFoundError;
117
+ var AuthCheckFailedError = class extends RunAutomationError4 {
118
+ constructor() {
119
+ super(exports2.authCheckFailedErrorCode, "AuthSession check failed");
120
+ }
121
+ };
122
+ exports2.AuthCheckFailedError = AuthCheckFailedError;
123
+ var MaxLevelsExceededError = class extends RunAutomationError4 {
124
+ constructor(levels) {
125
+ super(exports2.maxLevelsExceededErrorCode, `Max levels exceeded. Only ${levels} levels are supported`);
126
+ this.details = { levels };
127
+ }
128
+ };
129
+ exports2.MaxLevelsExceededError = MaxLevelsExceededError;
130
+ var AutomationError5 = class extends RunAutomationError4 {
131
+ constructor(error) {
132
+ super(exports2.automationError, `[${error?.name ?? error}] ${error?.message}`);
133
+ this.details = {
134
+ ...error
135
+ };
136
+ }
137
+ };
138
+ exports2.AutomationError = AutomationError5;
139
+ var InternalInvalidInputError2 = class extends RunAutomationError4 {
140
+ constructor(message, details) {
141
+ super(exports2.internalInvalidInputErrorCode, message);
142
+ this.details = details;
143
+ }
144
+ };
145
+ exports2.InternalInvalidInputError = InternalInvalidInputError2;
146
+ var ResultTooBigError2 = class extends RunAutomationError4 {
147
+ constructor(sizeInBytes, maxSizeInBytes) {
148
+ super(exports2.resultTooBigErrorCode, `Automation result is too big. Size: ${Math.round(sizeInBytes / 1024 / 1024 * 100) / 100}MB, Max allowed: ${Math.round(maxSizeInBytes / 1024 / 1024 * 100) / 100}MB`);
149
+ this.details = { sizeInBytes, maxSizeInBytes };
150
+ }
151
+ };
152
+ exports2.ResultTooBigError = ResultTooBigError2;
153
+ }
154
+ });
155
+
156
+ // ../packages/runtime-interface/dist/types.js
157
+ var require_types = __commonJS({
158
+ "../packages/runtime-interface/dist/types.js"(exports2) {
159
+ "use strict";
160
+ var __importDefault = exports2 && exports2.__importDefault || function(mod) {
161
+ return mod && mod.__esModule ? mod : { "default": mod };
162
+ };
163
+ Object.defineProperty(exports2, "__esModule", { value: true });
164
+ exports2.outputRunApiSchema = exports2.pongMessageSchema = exports2.doneMessageSchema = exports2.extendMessageSchema = exports2.runApiInputSchema = exports2.pingSchema = exports2.tokenUpdateSchema = exports2.abortRunApiSchema = exports2.startRunApiSchema = exports2.startRunApiParametersSchema = exports2.runApiParametersSchema = exports2.runApiRunOptionsSchema = exports2.runApiCdpRunOptionsSchema = exports2.runApiStandaloneRunOptionsSchema = exports2.runApiAuthSchema = exports2.runApiTracingSchema = exports2.runApiAutomationFunctionSchema = exports2.runApiSessionSchema = exports2.runApiStorageStateSchema = void 0;
165
+ exports2.runApiResultOkSchema = runApiResultOkSchema;
166
+ exports2.runApiResultWithSessionOkSchema = runApiResultWithSessionOkSchema;
167
+ var zod_1 = __importDefault(require("zod"));
168
+ exports2.runApiStorageStateSchema = zod_1.default.object({
169
+ cookies: zod_1.default.array(zod_1.default.object({
170
+ name: zod_1.default.string(),
171
+ value: zod_1.default.string(),
172
+ domain: zod_1.default.string(),
173
+ path: zod_1.default.string(),
174
+ expires: zod_1.default.number(),
175
+ httpOnly: zod_1.default.boolean(),
176
+ secure: zod_1.default.boolean(),
177
+ sameSite: zod_1.default.enum(["Strict", "Lax", "None"])
178
+ })),
179
+ origins: zod_1.default.array(zod_1.default.object({
180
+ origin: zod_1.default.string(),
181
+ localStorage: zod_1.default.array(zod_1.default.object({
182
+ name: zod_1.default.string(),
183
+ value: zod_1.default.string()
184
+ }))
185
+ })),
186
+ sessionStorage: zod_1.default.array(zod_1.default.object({
187
+ origin: zod_1.default.string(),
188
+ sessionStorage: zod_1.default.array(zod_1.default.object({
189
+ name: zod_1.default.string(),
190
+ value: zod_1.default.string()
191
+ }))
192
+ })).optional()
193
+ });
194
+ exports2.runApiSessionSchema = zod_1.default.discriminatedUnion("type", [
195
+ zod_1.default.object({
196
+ type: zod_1.default.literal("file"),
197
+ path: zod_1.default.string()
198
+ }),
199
+ zod_1.default.object({
200
+ type: zod_1.default.literal("state"),
201
+ state: exports2.runApiStorageStateSchema.nullable().optional()
202
+ })
203
+ ]);
204
+ exports2.runApiAutomationFunctionSchema = zod_1.default.object({
205
+ name: zod_1.default.string(),
206
+ params: zod_1.default.any().optional()
207
+ });
208
+ exports2.runApiTracingSchema = zod_1.default.discriminatedUnion("enabled", [
209
+ zod_1.default.object({ enabled: zod_1.default.literal(false) }),
210
+ zod_1.default.object({ enabled: zod_1.default.literal(true), filePath: zod_1.default.string() })
211
+ ]).optional().default({ enabled: false });
212
+ exports2.runApiAuthSchema = zod_1.default.object({
213
+ session: exports2.runApiSessionSchema,
214
+ parameters: zod_1.default.record(zod_1.default.any()).optional()
215
+ }).optional();
216
+ exports2.runApiStandaloneRunOptionsSchema = zod_1.default.object({
217
+ environment: zod_1.default.literal("standalone"),
218
+ headless: zod_1.default.boolean().default(true),
219
+ proxy: zod_1.default.object({
220
+ server: zod_1.default.string(),
221
+ username: zod_1.default.string(),
222
+ password: zod_1.default.string()
223
+ }).optional()
224
+ });
225
+ exports2.runApiCdpRunOptionsSchema = zod_1.default.object({
226
+ environment: zod_1.default.literal("cdp"),
227
+ cdpAddress: zod_1.default.string(),
228
+ cdpTargetId: zod_1.default.string().optional()
229
+ });
230
+ exports2.runApiRunOptionsSchema = zod_1.default.discriminatedUnion("environment", [
231
+ exports2.runApiStandaloneRunOptionsSchema,
232
+ exports2.runApiCdpRunOptionsSchema
233
+ ]).default({ environment: "standalone", headless: true });
234
+ exports2.runApiParametersSchema = zod_1.default.object({
235
+ automationFunction: exports2.runApiAutomationFunctionSchema,
236
+ tracing: exports2.runApiTracingSchema,
237
+ auth: exports2.runApiAuthSchema,
238
+ runOptions: exports2.runApiRunOptionsSchema,
239
+ retrieveSession: zod_1.default.boolean().default(false)
240
+ });
241
+ function runApiResultOkSchema(resultSchema) {
242
+ return zod_1.default.object({
243
+ result: resultSchema,
244
+ extendedPayloads: zod_1.default.array(zod_1.default.object({
245
+ api: zod_1.default.string(),
246
+ parameters: zod_1.default.record(zod_1.default.any())
247
+ })).optional()
248
+ });
249
+ }
250
+ function runApiResultWithSessionOkSchema(resultSchema) {
251
+ return runApiResultOkSchema(resultSchema).extend({
252
+ session: exports2.runApiStorageStateSchema
253
+ });
254
+ }
255
+ exports2.startRunApiParametersSchema = exports2.runApiParametersSchema.extend({
256
+ retrieveSession: zod_1.default.boolean(),
257
+ functionsToken: zod_1.default.string().optional(),
258
+ context: zod_1.default.object({
259
+ jobId: zod_1.default.string().optional(),
260
+ jobRunId: zod_1.default.string().optional(),
261
+ runId: zod_1.default.string().optional(),
262
+ queueId: zod_1.default.string().optional(),
263
+ authSessionId: zod_1.default.string().optional()
264
+ }).optional()
265
+ });
266
+ exports2.startRunApiSchema = zod_1.default.object({
267
+ type: zod_1.default.literal("start"),
268
+ parameters: exports2.startRunApiParametersSchema
269
+ });
270
+ exports2.abortRunApiSchema = zod_1.default.object({
271
+ type: zod_1.default.literal("abort"),
272
+ parameters: zod_1.default.object({}).optional()
273
+ });
274
+ exports2.tokenUpdateSchema = zod_1.default.object({
275
+ type: zod_1.default.literal("tokenUpdate"),
276
+ parameters: zod_1.default.object({
277
+ functionsToken: zod_1.default.string()
278
+ })
279
+ });
280
+ exports2.pingSchema = zod_1.default.object({
281
+ type: zod_1.default.literal("ping"),
282
+ parameters: zod_1.default.object({}).optional()
283
+ });
284
+ exports2.runApiInputSchema = zod_1.default.union([
285
+ exports2.startRunApiSchema,
286
+ exports2.abortRunApiSchema,
287
+ exports2.tokenUpdateSchema,
288
+ exports2.pingSchema
289
+ ]);
290
+ exports2.extendMessageSchema = zod_1.default.object({ type: zod_1.default.literal("extend") });
291
+ exports2.doneMessageSchema = zod_1.default.object({
292
+ type: zod_1.default.literal("done"),
293
+ result: zod_1.default.any(),
294
+ success: zod_1.default.boolean()
295
+ });
296
+ exports2.pongMessageSchema = zod_1.default.object({
297
+ type: zod_1.default.literal("pong")
298
+ });
299
+ exports2.outputRunApiSchema = zod_1.default.union([
300
+ exports2.extendMessageSchema,
301
+ exports2.doneMessageSchema,
302
+ exports2.pongMessageSchema
303
+ ]);
304
+ }
305
+ });
306
+
307
+ // ../packages/runtime-interface/dist/interfaceClient.js
308
+ var require_interfaceClient = __commonJS({
309
+ "../packages/runtime-interface/dist/interfaceClient.js"(exports2) {
310
+ "use strict";
311
+ var __createBinding = exports2 && exports2.__createBinding || (Object.create ? (function(o, m, k, k2) {
312
+ if (k2 === void 0) k2 = k;
313
+ var desc = Object.getOwnPropertyDescriptor(m, k);
314
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
315
+ desc = { enumerable: true, get: function() {
316
+ return m[k];
317
+ } };
318
+ }
319
+ Object.defineProperty(o, k2, desc);
320
+ }) : (function(o, m, k, k2) {
321
+ if (k2 === void 0) k2 = k;
322
+ o[k2] = m[k];
323
+ }));
324
+ var __setModuleDefault = exports2 && exports2.__setModuleDefault || (Object.create ? (function(o, v) {
325
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
326
+ }) : function(o, v) {
327
+ o["default"] = v;
328
+ });
329
+ var __importStar = exports2 && exports2.__importStar || function(mod) {
330
+ if (mod && mod.__esModule) return mod;
331
+ var result = {};
332
+ if (mod != null) {
333
+ for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
334
+ }
335
+ __setModuleDefault(result, mod);
336
+ return result;
337
+ };
338
+ Object.defineProperty(exports2, "__esModule", { value: true });
339
+ exports2.JSONLFileClient = exports2.TCPSocketClient = exports2.UnixSocketClient = exports2.SocketClient = void 0;
340
+ var net = __importStar(require("net"));
341
+ var fs6 = __importStar(require("fs-extra"));
342
+ var readline_1 = require("readline");
343
+ var promises_1 = require("timers/promises");
344
+ var SocketClient = class _SocketClient {
345
+ constructor(socket) {
346
+ this.socket = socket;
347
+ }
348
+ sendJSON(data) {
349
+ const dataToSend = JSON.stringify(data);
350
+ const length = Buffer.byteLength(dataToSend);
351
+ const buffer = Buffer.alloc(_SocketClient.LENGTH_HEADER_LENGTH + length);
352
+ buffer.writeUInt32BE(length, 0);
353
+ buffer.write(dataToSend, _SocketClient.LENGTH_HEADER_LENGTH);
354
+ this.socket.write(buffer);
355
+ }
356
+ async *receiveJSON() {
357
+ let buffer = Buffer.alloc(0);
358
+ const endPromise = new Promise((resolve2, reject) => {
359
+ this.socket.once("end", () => {
360
+ resolve2();
361
+ });
362
+ this.socket.once("error", reject);
363
+ });
364
+ while (true) {
365
+ const chunk = await Promise.race([
366
+ new Promise((resolve2) => this.socket.once("data", (data2) => {
367
+ if (typeof data2 === "string") {
368
+ return resolve2(Buffer.from(data2));
369
+ }
370
+ resolve2(data2);
371
+ })),
372
+ endPromise
373
+ ]);
374
+ if (!(chunk instanceof Buffer)) {
375
+ break;
376
+ }
377
+ buffer = Buffer.concat([buffer, chunk]);
378
+ const length = buffer.readUInt32BE(0);
379
+ if (buffer.length < length + _SocketClient.LENGTH_HEADER_LENGTH) {
380
+ continue;
381
+ }
382
+ const data = buffer.subarray(_SocketClient.LENGTH_HEADER_LENGTH, length + _SocketClient.LENGTH_HEADER_LENGTH);
383
+ buffer = buffer.subarray(length + _SocketClient.LENGTH_HEADER_LENGTH);
384
+ yield JSON.parse(data.toString());
385
+ }
386
+ }
387
+ async close() {
388
+ this.socket.end();
389
+ await Promise.race([
390
+ new Promise((resolve2) => this.socket.once("close", resolve2)),
391
+ new Promise((resolve2) => this.socket.once("error", resolve2)),
392
+ (0, promises_1.setTimeout)(3e3)
393
+ ]);
394
+ }
395
+ get closed() {
396
+ return this.socket.closed;
397
+ }
398
+ };
399
+ exports2.SocketClient = SocketClient;
400
+ SocketClient.LENGTH_HEADER_LENGTH = 4;
401
+ var UnixSocketClient = class extends SocketClient {
402
+ constructor(path3) {
403
+ super(net.createConnection(path3));
404
+ }
405
+ };
406
+ exports2.UnixSocketClient = UnixSocketClient;
407
+ var TCPSocketClient = class extends SocketClient {
408
+ constructor(host, port) {
409
+ super(net.createConnection(port, host));
410
+ }
411
+ };
412
+ exports2.TCPSocketClient = TCPSocketClient;
413
+ var JSONLFileClient = class {
414
+ constructor(filePath) {
415
+ this.fileStream = fs6.createReadStream(filePath, { encoding: "utf-8" });
416
+ }
417
+ sendJSON(data) {
418
+ console.log("Sending message", data);
419
+ }
420
+ async *receiveJSON() {
421
+ const rl = (0, readline_1.createInterface)({
422
+ input: this.fileStream,
423
+ crlfDelay: Infinity
424
+ });
425
+ for await (const line of rl) {
426
+ if (line.trim() === "") {
427
+ continue;
428
+ }
429
+ yield JSON.parse(line);
430
+ }
431
+ }
432
+ async close() {
433
+ this.fileStream.close();
434
+ await Promise.race([
435
+ new Promise((resolve2) => this.fileStream.once("close", resolve2)),
436
+ new Promise((resolve2) => this.fileStream.once("error", resolve2)),
437
+ (0, promises_1.setTimeout)(3e3)
438
+ ]);
439
+ }
440
+ get closed() {
441
+ return this.fileStream.closed;
442
+ }
443
+ };
444
+ exports2.JSONLFileClient = JSONLFileClient;
445
+ }
446
+ });
447
+
448
+ // ../packages/runtime-interface/dist/index.js
449
+ var require_dist = __commonJS({
450
+ "../packages/runtime-interface/dist/index.js"(exports2) {
451
+ "use strict";
452
+ var __createBinding = exports2 && exports2.__createBinding || (Object.create ? (function(o, m, k, k2) {
453
+ if (k2 === void 0) k2 = k;
454
+ var desc = Object.getOwnPropertyDescriptor(m, k);
455
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
456
+ desc = { enumerable: true, get: function() {
457
+ return m[k];
458
+ } };
459
+ }
460
+ Object.defineProperty(o, k2, desc);
461
+ }) : (function(o, m, k, k2) {
462
+ if (k2 === void 0) k2 = k;
463
+ o[k2] = m[k];
464
+ }));
465
+ var __exportStar = exports2 && exports2.__exportStar || function(m, exports3) {
466
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports3, p)) __createBinding(exports3, m, p);
467
+ };
468
+ Object.defineProperty(exports2, "__esModule", { value: true });
469
+ __exportStar(require_errors(), exports2);
470
+ __exportStar(require_types(), exports2);
471
+ __exportStar(require_interfaceClient(), exports2);
472
+ }
473
+ });
474
+
475
+ // src/commands/api/run.ts
476
+ var run_exports = {};
477
+ __export(run_exports, {
478
+ executeCLI: () => executeCLI
479
+ });
480
+ module.exports = __toCommonJS(run_exports);
481
+ var import_commander = require("commander");
482
+ var fs5 = __toESM(require("fs-extra"));
483
+
484
+ // src/commands/common/utils/fileUtils.ts
485
+ var path = __toESM(require("path"));
486
+ var fs = __toESM(require("fs-extra"));
487
+ var import_dotenv = __toESM(require("dotenv"));
488
+ import_dotenv.default.config();
489
+ function getFullPathInProject(...paths) {
490
+ return path.resolve(process.cwd(), process.env.ROOT || "./", ...paths);
491
+ }
492
+
493
+ // src/common/settingsSchema.ts
494
+ var z = __toESM(require("zod"));
495
+ var baseCaptchaSchema = z.object({
496
+ enabled: z.boolean()
497
+ });
498
+ var customCaptchaSchema = baseCaptchaSchema.extend({
499
+ imageLocators: z.array(z.string()).min(1, "At least one image locator is required"),
500
+ submitLocators: z.array(z.string()).min(1, "At least one submit locator is required"),
501
+ inputLocators: z.array(z.string()).min(1, "At least one input locator is required")
502
+ });
503
+ var textCaptchaSchema = baseCaptchaSchema.extend({
504
+ labelLocators: z.array(z.string()).min(1, "At least one image locator is required"),
505
+ submitLocators: z.array(z.string()).min(1, "At least one submit locator is required"),
506
+ inputLocators: z.array(z.string()).min(1, "At least one input locator is required")
507
+ });
508
+ var captchaSolverSolveSettingsSchema = z.object({
509
+ autoSolve: z.boolean().default(true),
510
+ solveDelay: z.number().min(0).default(2e3),
511
+ maxRetries: z.number().min(0).default(3),
512
+ timeout: z.number().min(0).default(3e4)
513
+ });
514
+ var captchaSolverSettingsSchema = z.object({
515
+ enabled: z.boolean().default(false),
516
+ port: z.number().int().min(1).max(65535).optional(),
517
+ cloudflare: baseCaptchaSchema.optional(),
518
+ googleRecaptchaV2: baseCaptchaSchema.optional(),
519
+ googleRecaptchaV3: baseCaptchaSchema.optional(),
520
+ awscaptcha: baseCaptchaSchema.optional(),
521
+ hcaptcha: baseCaptchaSchema.optional(),
522
+ funcaptcha: baseCaptchaSchema.optional(),
523
+ geetest: baseCaptchaSchema.optional(),
524
+ lemin: baseCaptchaSchema.optional(),
525
+ customCaptcha: customCaptchaSchema.optional(),
526
+ text: textCaptchaSchema.optional(),
527
+ settings: captchaSolverSolveSettingsSchema.default(
528
+ captchaSolverSolveSettingsSchema.parse({})
529
+ )
530
+ }).default({});
531
+ var authSessionsSchema = z.object({
532
+ enabled: z.boolean()
533
+ }).optional().default({
534
+ enabled: false
535
+ });
536
+ var stealthModeSchema = z.object({
537
+ enabled: z.boolean()
538
+ }).optional().default({
539
+ enabled: false
540
+ });
541
+ var settingsSchema = z.object({
542
+ authSessions: authSessionsSchema,
543
+ stealthMode: stealthModeSchema,
544
+ captchaSolver: captchaSolverSettingsSchema.optional()
545
+ });
546
+
547
+ // src/commands/common/utils/settings.ts
548
+ var import_promises = require("fs/promises");
549
+ async function getSettings() {
550
+ const settingsFilePath = getFullPathInProject("Intuned.json");
551
+ const settings = await (0, import_promises.readFile)(settingsFilePath, { encoding: "utf-8" });
552
+ if (settings) {
553
+ const parsed = settingsSchema.safeParse(JSON.parse(settings));
554
+ if (parsed.success) {
555
+ return parsed.data;
556
+ } else {
557
+ throw new Error(parsed.error.message);
558
+ }
559
+ }
560
+ return {
561
+ authSessions: { enabled: false },
562
+ stealthMode: { enabled: false }
563
+ };
564
+ }
565
+
566
+ // src/commands/api/run.ts
567
+ var import_dotenv2 = __toESM(require("dotenv"));
568
+
569
+ // src/common/asyncLocalStorage/index.ts
570
+ var import_node_async_hooks = require("async_hooks");
571
+ var asyncLocalStorage = new import_node_async_hooks.AsyncLocalStorage();
572
+ function runWithContext(contextData, callback, ...args) {
573
+ return asyncLocalStorage.run(contextData, callback, ...args);
574
+ }
575
+ function getExecutionContext() {
576
+ const contextData = asyncLocalStorage.getStore();
577
+ return contextData;
578
+ }
579
+
580
+ // src/common/cleanEnvironmentVariables.ts
581
+ function cleanEnvironmentVariables() {
582
+ Object.keys(process.env).filter((i) => {
583
+ if (i.toLocaleLowerCase().startsWith("npm")) {
584
+ return true;
585
+ }
586
+ if (i.toLocaleLowerCase().startsWith("fly") && i !== "FLY_ALLOC_ID") {
587
+ return true;
588
+ }
589
+ }).forEach((i) => delete process.env[i]);
590
+ }
591
+
592
+ // src/common/Logger/index.ts
593
+ var import_node_util = require("util");
594
+ var import_chalk = __toESM(require("chalk"));
595
+ var format = import_node_util.formatWithOptions.bind(void 0, { colors: true });
596
+ var LOG_LEVEL_COLORS = {
597
+ TRACE: import_chalk.default.gray,
598
+ DEBUG: import_chalk.default.blue,
599
+ INFO: import_chalk.default.green,
600
+ WARN: import_chalk.default.yellow,
601
+ ERROR: import_chalk.default.red
602
+ };
603
+ var Logger = class {
604
+ logFunction(entry) {
605
+ const { level, timestamp, message, meta } = entry;
606
+ const date = new Date(timestamp);
607
+ const levelColor = LOG_LEVEL_COLORS[level];
608
+ if (meta === void 0) {
609
+ process.stderr.write(
610
+ `${format(date)} [@intuned/sdk][${levelColor(level)}] ${message}
611
+ `
612
+ );
613
+ } else {
614
+ process.stderr.write(
615
+ `${format(date)} [@intuned/sdk][${levelColor(
616
+ level
617
+ )}] ${message} ${format(meta)}
618
+ `
619
+ );
620
+ }
621
+ }
622
+ log(level, message, meta) {
623
+ this.logFunction({
624
+ level,
625
+ message,
626
+ meta: meta && Object.keys(meta).length === 0 ? void 0 : meta,
627
+ timestamp: Date.now()
628
+ });
629
+ }
630
+ trace(message, meta) {
631
+ this.log("TRACE", message, meta);
632
+ }
633
+ debug(message, meta) {
634
+ this.log("DEBUG", message, meta);
635
+ }
636
+ info(message, meta) {
637
+ this.log("INFO", message, meta);
638
+ }
639
+ warn(message, meta) {
640
+ this.log("WARN", message, meta);
641
+ }
642
+ error(message, meta) {
643
+ this.log("ERROR", message, meta);
644
+ }
645
+ };
646
+ var logger = new Logger();
647
+
648
+ // src/commands/api/run.ts
649
+ var import_nanoid = require("nanoid");
650
+ var import_chalk2 = __toESM(require("chalk"));
651
+
652
+ // src/runtime/attemptStore.ts
653
+ var attemptStore = Object.freeze({
654
+ get: (key) => {
655
+ const context = getExecutionContext();
656
+ return context?.store ? context.store[key] : void 0;
657
+ },
658
+ set: (key, value) => {
659
+ const context = getExecutionContext();
660
+ if (!context) {
661
+ throw new Error("store.set failed due to an internal error.");
662
+ }
663
+ if (!context.store) {
664
+ context.store = {};
665
+ }
666
+ context.store[key] = value;
667
+ }
668
+ });
669
+
670
+ // src/runtime/persistentStore.ts
671
+ var import_zod = require("zod");
672
+
673
+ // src/common/jwtTokenManager.ts
674
+ var import_cross_fetch = __toESM(require("cross-fetch"));
675
+ var jwt = __toESM(require("jsonwebtoken"));
676
+ var import_neverthrow = require("neverthrow");
677
+
678
+ // src/common/constants.ts
679
+ var API_FOLDER_NAME = "api";
680
+ var WORKSPACE_ID_ENV_VAR_KEY = "INTUNED_WORKSPACE_ID";
681
+ var PROJECT_ID_ENV_VAR_KEY = "INTUNED_PROJECT_ID";
682
+ var API_KEY_ENV_VAR_KEY = "INTUNED_API_KEY";
683
+ var AUTH_TOKEN_ENV_VAR_KEY = "INTUNED_AUTH_TOKEN";
684
+ var API_KEY_HEADER_NAME = "x-api-key";
685
+ var CLI_ENV_VAR_KEY = "INTUNED_CLI";
686
+
687
+ // src/common/jwtTokenManager.ts
688
+ var JwtTokenManager = class {
689
+ constructor(refreshTokenPath) {
690
+ this.refreshTokenPath = refreshTokenPath;
691
+ this._token = void 0;
692
+ }
693
+ get token() {
694
+ return this._token;
695
+ }
696
+ // When the token is set, the schedule for renewal is issued automatically
697
+ // This is currently being set it two places:
698
+ // 1. Whenever the runner starts, initializes it from the environment variable (set whenever the api is run from the authoring IDE)
699
+ // 2. Whenever a published api is called to run (/api/run/*), it is set to the token received in the run request.
700
+ set token(newToken) {
701
+ if (this._token != newToken) {
702
+ this._token = newToken;
703
+ void this.scheduleTokenRefresh();
704
+ }
705
+ }
706
+ get timeToRefresh() {
707
+ if (!this._token) return;
708
+ const payload = jwt.decode(this._token);
709
+ if (!payload || typeof payload == "string") return;
710
+ const expiry = payload.expiry;
711
+ if (!expiry || typeof expiry !== "number") return;
712
+ const timeWindow = 60 * 1e3;
713
+ const timeToRefresh = expiry - Date.now() - timeWindow;
714
+ return Math.max(timeToRefresh, timeWindow);
715
+ }
716
+ async scheduleTokenRefresh() {
717
+ if (process.env.RUN_ENVIRONMENT?.toLowerCase() !== "authoring") return;
718
+ const timeToRefresh = this.timeToRefresh;
719
+ if (timeToRefresh === void 0) return;
720
+ if (this.tokenRefreshTimeout) clearTimeout(this.tokenRefreshTimeout);
721
+ this.tokenRefreshTimeout = setTimeout(async () => {
722
+ const result = await this.refreshToken();
723
+ if (result && result.isErr()) {
724
+ console.error(`[Internal Error] ${result.error}`);
725
+ return;
726
+ }
727
+ await this.scheduleTokenRefresh();
728
+ }, timeToRefresh);
729
+ }
730
+ async refreshToken() {
731
+ if (process.env.RUN_ENVIRONMENT?.toLowerCase() !== "authoring") return;
732
+ const res = await this.fetchWithToken(
733
+ `${this.backendFunctionsBaseUrl}/${this.refreshTokenPath}`,
734
+ {
735
+ method: "GET"
736
+ }
737
+ );
738
+ if (res.status === 401) {
739
+ return (0, import_neverthrow.err)("Unauthorized");
740
+ }
741
+ const jsonResult = await import_neverthrow.ResultAsync.fromPromise(
742
+ res.json(),
743
+ () => "not json"
744
+ );
745
+ if (jsonResult.isErr()) return;
746
+ const newToken = jsonResult.value.token;
747
+ if (newToken) this._token = newToken;
748
+ }
749
+ async fetchWithToken(...[input, init]) {
750
+ const headers = new Headers(init?.headers);
751
+ const apiKey = process.env[API_KEY_ENV_VAR_KEY];
752
+ if (apiKey) {
753
+ headers.set(API_KEY_HEADER_NAME, apiKey);
754
+ }
755
+ const token = process.env[AUTH_TOKEN_ENV_VAR_KEY];
756
+ if (token) {
757
+ headers.set("Authorization", `Bearer ${token}`);
758
+ }
759
+ if (this.token !== void 0) {
760
+ headers.set("Authorization", `Bearer ${this.token}`);
761
+ }
762
+ const result = await (0, import_cross_fetch.default)(input, {
763
+ ...init,
764
+ headers
765
+ });
766
+ if (result.status === 401 && process.env[CLI_ENV_VAR_KEY] === "true") {
767
+ console.warn(
768
+ "Unauthorized backend function call - make sure to provision your project to Intuned to set up the correct API credentials.\nRun 'intuned provision' or see https://docs.intunedhq.com/docs/05-references/cli#provision-project for more information."
769
+ );
770
+ }
771
+ return result;
772
+ }
773
+ get backendFunctionsBaseUrl() {
774
+ try {
775
+ if (!process.env.FUNCTIONS_DOMAIN) {
776
+ throw new Error(
777
+ `Cannot call backend function - FUNCTIONS_DOMAIN not set`
778
+ );
779
+ }
780
+ const domain = process.env.FUNCTIONS_DOMAIN;
781
+ if (!process.env[WORKSPACE_ID_ENV_VAR_KEY]) {
782
+ throw new Error(
783
+ `Cannot call backend function - ${WORKSPACE_ID_ENV_VAR_KEY} not set`
784
+ );
785
+ }
786
+ const workspaceId = process.env[WORKSPACE_ID_ENV_VAR_KEY];
787
+ if (!process.env[PROJECT_ID_ENV_VAR_KEY] && !process.env.INTUNED_INTEGRATION_ID) {
788
+ throw new Error(
789
+ `Cannot call backend function - ${PROJECT_ID_ENV_VAR_KEY} or INTUNED_INTEGRATION_ID not set`
790
+ );
791
+ }
792
+ const projectId = process.env.INTUNED_INTEGRATION_ID ?? process.env[PROJECT_ID_ENV_VAR_KEY];
793
+ return `${domain}/api/${workspaceId}/functions/${projectId}`;
794
+ } catch (e) {
795
+ if (process.env[CLI_ENV_VAR_KEY] === "true") {
796
+ throw new Error(
797
+ `API credentials not set - make sure to provision your project to Intuned to set up the correct API credentials.
798
+ Run 'intuned provision' or see https://docs.intunedhq.com/docs/05-references/cli#provision-project for more information.
799
+ Original error: ${e.message}`
800
+ );
801
+ }
802
+ }
803
+ }
804
+ };
805
+ var backendFunctionsTokenManager = new JwtTokenManager(
806
+ `refreshBackendFunctionsToken`
807
+ );
808
+ backendFunctionsTokenManager.token = process.env.INTUNED_AUTHORING_SESSION_BACKEND_FUNCTIONS_TOKEN;
809
+ function callBackendFunctionWithToken(path3, init) {
810
+ return backendFunctionsTokenManager.fetchWithToken(
811
+ `${backendFunctionsTokenManager.backendFunctionsBaseUrl}/${path3}`,
812
+ init
813
+ );
814
+ }
815
+
816
+ // src/runtime/persistentStore.ts
817
+ var forbiddenCharacters = /[:#]/g;
818
+ var keySchema = import_zod.z.string().min(1, "Key must be at least 1 character long").refine(
819
+ (key) => (key.match(forbiddenCharacters)?.length ?? 0) === 0,
820
+ 'Key cannot contain the following characters: ":" or "#"'
821
+ );
822
+ var persistentStore = Object.freeze({
823
+ get: async (key) => {
824
+ const parsedKey = keySchema.parse(key);
825
+ const response = await callBackendFunctionWithToken(
826
+ `kv-store/${parsedKey}`,
827
+ {
828
+ method: "GET"
829
+ }
830
+ );
831
+ const json = await response.json();
832
+ if (!response.ok) {
833
+ throw new Error(json.message);
834
+ }
835
+ return json.value;
836
+ },
837
+ set: async (key, value) => {
838
+ const keyResult = keySchema.parse(key);
839
+ const response = await callBackendFunctionWithToken(
840
+ `kv-store/${keyResult}`,
841
+ {
842
+ method: "PUT",
843
+ body: JSON.stringify(value),
844
+ headers: {
845
+ "Content-Type": "application/json"
846
+ }
847
+ }
848
+ );
849
+ const json = await response.json();
850
+ if (!response.ok) {
851
+ throw new Error(json.message);
852
+ }
853
+ }
854
+ });
855
+
856
+ // src/runtime/RunError.ts
857
+ var RunError = class _RunError extends Error {
858
+ constructor(message, options) {
859
+ super(message);
860
+ this.message = message;
861
+ this.name = "USER_GENERATED_ERROR";
862
+ this.options = options ?? {
863
+ retryable: false
864
+ };
865
+ Object.setPrototypeOf(this, _RunError.prototype);
866
+ }
867
+ };
868
+ new RunError("", {});
869
+
870
+ // src/common/extension/intunedExtensionServer.ts
871
+ var import_fastify = __toESM(require("fastify"));
872
+
873
+ // src/common/extension/types.ts
874
+ var import_zod2 = require("zod");
875
+ var captchaTypeSchema = import_zod2.z.enum([
876
+ "aws",
877
+ "cloudflare",
878
+ "customcaptcha",
879
+ "funcaptcha",
880
+ "geetest",
881
+ "hcaptcha",
882
+ "lemincaptcha",
883
+ "recaptcha",
884
+ "textcaptcha"
885
+ ]);
886
+ var captchaStatusSchema = import_zod2.z.enum([
887
+ "attached",
888
+ "solving",
889
+ "solved",
890
+ "error",
891
+ "detached"
892
+ ]);
893
+ var captchaErrorCodeSchema = import_zod2.z.enum([
894
+ "HIT_LIMIT",
895
+ "MAX_RETRIES",
896
+ "UNEXPECTED_SERVER_RESPONSE",
897
+ "UNEXPECTED_ERROR"
898
+ ]);
899
+ var captchaErrorSchema = import_zod2.z.object({
900
+ code: captchaErrorCodeSchema,
901
+ error: import_zod2.z.unknown().optional()
902
+ });
903
+ var captchaBaseSchema = import_zod2.z.object({
904
+ id: import_zod2.z.string().min(1),
905
+ tabId: import_zod2.z.number().int().positive(),
906
+ type: captchaTypeSchema,
907
+ retryCount: import_zod2.z.number().int().nonnegative().optional()
908
+ });
909
+ var captchaNonErrorSchema = captchaBaseSchema.extend({
910
+ status: captchaStatusSchema.exclude(["error"])
911
+ });
912
+ var captchaErrorStatusSchema = captchaBaseSchema.extend({
913
+ status: import_zod2.z.literal("error"),
914
+ error: captchaErrorSchema
915
+ });
916
+ var captchaSchema = import_zod2.z.discriminatedUnion("status", [
917
+ captchaBaseSchema.extend({
918
+ status: import_zod2.z.literal("attached")
919
+ }),
920
+ captchaBaseSchema.extend({
921
+ status: import_zod2.z.literal("solving")
922
+ }),
923
+ captchaBaseSchema.extend({
924
+ status: import_zod2.z.literal("solved")
925
+ }),
926
+ captchaBaseSchema.extend({
927
+ status: import_zod2.z.literal("detached")
928
+ }),
929
+ captchaErrorStatusSchema
930
+ ]);
931
+
932
+ // src/common/extension/extensionsHelpers.ts
933
+ var import_path = __toESM(require("path"));
934
+ var import_promises2 = require("fs/promises");
935
+ var import_portfinder = require("portfinder");
936
+ var INTUNED_WORKER = "intunedWorker.js";
937
+ var INTUNED_EXTENSION_SETTINGS_FILE_NAME = "intunedSettings.json";
938
+ var captchaSolverSettings = null;
939
+ async function getIntunedCaptchaExtensionPort() {
940
+ if (process.env.INTUNED_CAPTCHA_EXTENSION_PORT) {
941
+ return parseInt(process.env.INTUNED_CAPTCHA_EXTENSION_PORT, 10);
942
+ }
943
+ return await (0, import_portfinder.getPort)({});
944
+ }
945
+ async function resolveCaptchaSolverSettings(input) {
946
+ const parsed = captchaSolverSettingsSchema.parse(input || {});
947
+ if (!parsed.port) {
948
+ parsed.port = await getIntunedCaptchaExtensionPort();
949
+ }
950
+ return parsed;
951
+ }
952
+ function isIntunedExtensionLoaded() {
953
+ return !!getIntunedExtensionPath();
954
+ }
955
+ function buildExtensionsList() {
956
+ const extensionsList = [];
957
+ if (isIntunedExtensionLoaded()) {
958
+ const intunedExtensionPath = getIntunedExtensionPath();
959
+ extensionsList.push(intunedExtensionPath);
960
+ }
961
+ return extensionsList;
962
+ }
963
+ function getIntunedExtensionPath() {
964
+ return process.env.INTUNED_EXTENSION_PATH;
965
+ }
966
+ async function isIntunedExtensionEnabled() {
967
+ const path3 = getIntunedExtensionPath();
968
+ if (!path3) {
969
+ return false;
970
+ }
971
+ const captchaSolverSettings2 = await getIntunedCaptchaSolverSettings();
972
+ return captchaSolverSettings2.enabled;
973
+ }
974
+ async function getIntunedExtensionWorker(context) {
975
+ if (!await isIntunedExtensionEnabled()) {
976
+ return null;
977
+ }
978
+ let attemptCount = 0;
979
+ while (attemptCount < 4) {
980
+ const intunedServiceWorker = context.serviceWorkers().find((serviceWorker) => serviceWorker.url().includes(INTUNED_WORKER));
981
+ if (intunedServiceWorker) {
982
+ return intunedServiceWorker;
983
+ }
984
+ try {
985
+ await context.waitForEvent("serviceworker", { timeout: 3e3 });
986
+ } catch (err7) {
987
+ console.log(`Error accessing service workers (attempt ${attemptCount})`);
988
+ }
989
+ attemptCount++;
990
+ }
991
+ console.error("Failed to get intuned worker after 5 attmepts");
992
+ return null;
993
+ }
994
+ async function getIntunedExtensionSettings(captchaSolverSettings2) {
995
+ const [domain, workspaceId, projectId] = [
996
+ process.env.FUNCTIONS_DOMAIN,
997
+ process.env[WORKSPACE_ID_ENV_VAR_KEY],
998
+ process.env.INTUNED_INTEGRATION_ID ?? process.env[PROJECT_ID_ENV_VAR_KEY]
999
+ ];
1000
+ if (!domain || !workspaceId || !projectId) {
1001
+ const missingEnvVars = [
1002
+ domain && "FUNCTIONS_DOMAIN",
1003
+ workspaceId && WORKSPACE_ID_ENV_VAR_KEY,
1004
+ projectId && `INTUNED_INTEGRATION_ID OR ${PROJECT_ID_ENV_VAR_KEY}`
1005
+ ];
1006
+ throw new Error(
1007
+ `Missing required environment variables: ${missingEnvVars}`
1008
+ );
1009
+ }
1010
+ const authentication = (() => {
1011
+ if (process.env.INTUNED_API_KEY) {
1012
+ return { type: "apiKey", apiKey: process.env.INTUNED_API_KEY };
1013
+ }
1014
+ if (process.env.INTUNED_BASIC_AUTH_USERNAME && process.env.INTUNED_BASIC_AUTH_PASSWORD) {
1015
+ const credentials = `${process.env.INTUNED_BASIC_AUTH_USERNAME}:${process.env.INTUNED_BASIC_AUTH_PASSWORD}`;
1016
+ const token = Buffer.from(credentials, "utf-8").toString("base64");
1017
+ return { type: "basic", token };
1018
+ }
1019
+ return {
1020
+ type: "bearer",
1021
+ token: backendFunctionsTokenManager.token
1022
+ };
1023
+ })();
1024
+ const baseUrl = process.env.INTUNED_API_BASE_URL ?? domain;
1025
+ return {
1026
+ ...captchaSolverSettings2,
1027
+ workspaceId,
1028
+ projectId,
1029
+ baseUrl,
1030
+ authentication
1031
+ };
1032
+ }
1033
+ async function getIntunedCaptchaSolverSettings() {
1034
+ if (captchaSolverSettings) {
1035
+ return captchaSolverSettings;
1036
+ }
1037
+ const settings = await getSettings();
1038
+ captchaSolverSettings = await resolveCaptchaSolverSettings(
1039
+ settings.captchaSolver
1040
+ );
1041
+ return captchaSolverSettings;
1042
+ }
1043
+ async function setupIntunedExtension() {
1044
+ if (!await isIntunedExtensionEnabled()) {
1045
+ return;
1046
+ }
1047
+ const intunedExtensionPath = getIntunedExtensionPath();
1048
+ const intunedExtensionSettingsPath = import_path.default.join(
1049
+ intunedExtensionPath,
1050
+ INTUNED_EXTENSION_SETTINGS_FILE_NAME
1051
+ );
1052
+ const captchaSolverSettings2 = await getIntunedCaptchaSolverSettings();
1053
+ await setupIntunedExtensionServer(captchaSolverSettings2);
1054
+ const captchaSolverSettingsWithRunContext = await getIntunedExtensionSettings(
1055
+ captchaSolverSettings2
1056
+ );
1057
+ await (0, import_promises2.writeFile)(
1058
+ intunedExtensionSettingsPath,
1059
+ JSON.stringify(captchaSolverSettingsWithRunContext)
1060
+ );
1061
+ }
1062
+
1063
+ // src/common/extension/intunedExtensionServer.ts
1064
+ var TabCaptchaState = class {
1065
+ constructor(tabId) {
1066
+ this.tabId = tabId;
1067
+ this.captchasById = /* @__PURE__ */ new Map();
1068
+ this.subscribers = new Array();
1069
+ }
1070
+ subscribe(handler) {
1071
+ this.subscribers.push(handler);
1072
+ }
1073
+ unsubscribe(handler, status) {
1074
+ const index = this.subscribers.findIndex(
1075
+ (subscriber) => subscriber.handler === handler && (subscriber.status === status || !status)
1076
+ );
1077
+ if (index !== -1) {
1078
+ this.subscribers.splice(index, 1);
1079
+ }
1080
+ }
1081
+ getCaptchas() {
1082
+ return [...this.captchasById.values()];
1083
+ }
1084
+ async upsertCaptcha(captcha) {
1085
+ this.captchasById.set(captcha.id, captcha);
1086
+ for (const subscriber of this.subscribers) {
1087
+ if (!subscriber.status || subscriber.status === captcha.status) {
1088
+ await subscriber.handler(captcha);
1089
+ }
1090
+ }
1091
+ }
1092
+ };
1093
+ var ExtensionServer = class {
1094
+ constructor() {
1095
+ this.app = null;
1096
+ this.tabs = /* @__PURE__ */ new Map();
1097
+ }
1098
+ getOrCreateTab(tabId) {
1099
+ const existing = this.tabs.get(tabId);
1100
+ if (existing) return existing;
1101
+ const created = new TabCaptchaState(tabId);
1102
+ this.tabs.set(tabId, created);
1103
+ return created;
1104
+ }
1105
+ async handleUpsertCaptcha(captcha) {
1106
+ const tab = this.getOrCreateTab(captcha.tabId);
1107
+ await tab.upsertCaptcha(captcha);
1108
+ }
1109
+ async start({
1110
+ port,
1111
+ host = "0.0.0.0"
1112
+ }) {
1113
+ if (this.app) {
1114
+ return;
1115
+ }
1116
+ this.app = (0, import_fastify.default)({
1117
+ logger: false,
1118
+ bodyLimit: 1e6
1119
+ });
1120
+ this.app.post("/state", async (request, reply) => {
1121
+ try {
1122
+ const result = captchaSchema.safeParse(request.body);
1123
+ if (!result.success) {
1124
+ return reply.code(400).send({
1125
+ error: "Invalid captcha payload",
1126
+ details: result.error.flatten().fieldErrors
1127
+ });
1128
+ }
1129
+ await this.handleUpsertCaptcha(result.data);
1130
+ return reply.code(200).send({});
1131
+ } catch (error) {
1132
+ console.error("Error processing captcha state update:", error);
1133
+ return reply.code(500).send({
1134
+ error: "Internal server error",
1135
+ message: error?.message ?? String(error)
1136
+ });
1137
+ }
1138
+ });
1139
+ this.app.setNotFoundHandler((_request, reply) => {
1140
+ return reply.code(404).send({ error: "Not found" });
1141
+ });
1142
+ await this.app.listen({
1143
+ port,
1144
+ host
1145
+ });
1146
+ }
1147
+ async stop() {
1148
+ if (!this.app) return;
1149
+ const toClose = this.app;
1150
+ this.app = null;
1151
+ await toClose.close();
1152
+ }
1153
+ async getCaptchas(page, status) {
1154
+ const tabId = await getTabId(page);
1155
+ const tab = this.tabs.get(tabId);
1156
+ if (!tab) return [];
1157
+ const captchas = tab.getCaptchas();
1158
+ if (!status) return captchas;
1159
+ return captchas.filter((c) => c.status === status && c.tabId === tabId);
1160
+ }
1161
+ async subscribe(page, handler, status) {
1162
+ const tabId = await getTabId(page);
1163
+ const tab = this.getOrCreateTab(tabId);
1164
+ tab.subscribe({ handler, status });
1165
+ }
1166
+ async unsubscribe(page, handler, status) {
1167
+ const tabId = await getTabId(page);
1168
+ const tab = this.tabs.get(tabId);
1169
+ if (!tab) return;
1170
+ tab.unsubscribe(handler, status);
1171
+ }
1172
+ removeTab(tabId) {
1173
+ this.tabs.delete(tabId);
1174
+ }
1175
+ async getTabId(page) {
1176
+ return await getTabId(page);
1177
+ }
1178
+ };
1179
+ var extensionServerSingleton = null;
1180
+ async function setupIntunedExtensionServer(captchaSolverSettings2) {
1181
+ if (!captchaSolverSettings2) {
1182
+ captchaSolverSettings2 = await resolveCaptchaSolverSettings();
1183
+ }
1184
+ if (!extensionServerSingleton) {
1185
+ extensionServerSingleton = new ExtensionServer();
1186
+ }
1187
+ await extensionServerSingleton.start({
1188
+ port: captchaSolverSettings2.port ?? 9e3
1189
+ });
1190
+ }
1191
+ async function cleanIntunedExtensionServer() {
1192
+ if (extensionServerSingleton) {
1193
+ await extensionServerSingleton.stop();
1194
+ extensionServerSingleton = null;
1195
+ }
1196
+ }
1197
+ var TAB_ID_CACHE_KEY = "__INTUNED_CACHED_TAB_ID__";
1198
+ async function getTabId(page) {
1199
+ const cached = page[TAB_ID_CACHE_KEY];
1200
+ if (cached !== void 0 && typeof cached === "number") {
1201
+ return cached;
1202
+ }
1203
+ let tabId;
1204
+ try {
1205
+ tabId = await page.evaluate("window.__INTUNED_TAB_ID__", { timeout: 100 });
1206
+ } catch {
1207
+ await page.waitForFunction("window.__INTUNED_TAB_ID__ !== undefined", {
1208
+ timeout: 15e3
1209
+ });
1210
+ tabId = await page.evaluate("window.__INTUNED_TAB_ID__");
1211
+ }
1212
+ const numericTabId = Number(tabId);
1213
+ page[TAB_ID_CACHE_KEY] = numericTabId;
1214
+ return numericTabId;
1215
+ }
1216
+
1217
+ // src/runtime/downloadDirectory.ts
1218
+ var import_fs_extra = require("fs-extra");
1219
+ function getDownloadDirectoryPath() {
1220
+ const context = getExecutionContext();
1221
+ if (!context) {
1222
+ throw new Error("ExecutionContext not found");
1223
+ }
1224
+ const path3 = `/tmp/downloads/${context.runId}`;
1225
+ (0, import_fs_extra.ensureDirSync)(path3, {
1226
+ mode: 1533
1227
+ });
1228
+ return path3;
1229
+ }
1230
+
1231
+ // src/common/runApi/index.ts
1232
+ var fs4 = __toESM(require("fs-extra"));
1233
+ var import_fs_extra3 = require("fs-extra");
1234
+ var import_neverthrow6 = require("neverthrow");
1235
+
1236
+ // src/common/contextStorageStateHelpers.ts
1237
+ async function setStorageState(context, state) {
1238
+ if ("cookies" in state && state.cookies) {
1239
+ await context.addCookies(state.cookies);
1240
+ }
1241
+ const page = await context.newPage();
1242
+ if ("origins" in state && state.origins) {
1243
+ for (const originData of state.origins || []) {
1244
+ const origin = originData.origin;
1245
+ await page.route(
1246
+ `${origin}/*`,
1247
+ (route) => route.fulfill({
1248
+ body: "<html><head><title>Set Storage</title></head><body><h1>Set Storage</h1></body></html>",
1249
+ contentType: "text/html",
1250
+ status: 200
1251
+ })
1252
+ );
1253
+ try {
1254
+ await page.goto(origin);
1255
+ for (const item of originData.localStorage) {
1256
+ await page.evaluate(
1257
+ ([key, value]) => {
1258
+ window.localStorage.setItem(key, value);
1259
+ },
1260
+ [item.name, item.value]
1261
+ );
1262
+ }
1263
+ } finally {
1264
+ await page.unroute(`${origin}/*`);
1265
+ }
1266
+ }
1267
+ }
1268
+ if ("sessionStorage" in state && state.sessionStorage) {
1269
+ await context.addInitScript((storage) => {
1270
+ for (const { origin, sessionStorage } of storage) {
1271
+ if (window.location.origin === origin) {
1272
+ for (const item of sessionStorage) {
1273
+ const value = window.sessionStorage.getItem(item.name);
1274
+ if (!value) {
1275
+ window.sessionStorage.setItem(item.name, item.value);
1276
+ }
1277
+ }
1278
+ }
1279
+ }
1280
+ }, state.sessionStorage);
1281
+ }
1282
+ await page.close();
1283
+ }
1284
+ async function getStorageState(context) {
1285
+ const result = {
1286
+ cookies: [],
1287
+ origins: []
1288
+ };
1289
+ const storageState = await context.storageState();
1290
+ result.cookies = storageState.cookies;
1291
+ result.origins = storageState.origins;
1292
+ const sessionDataList = [];
1293
+ const pages = await context.pages();
1294
+ for (const page of pages) {
1295
+ if (page.isClosed()) continue;
1296
+ try {
1297
+ const sessionData = await page.evaluate(() => {
1298
+ const items = { ...window.sessionStorage };
1299
+ return {
1300
+ origin: window.location.origin,
1301
+ sessionStorage: Object.entries(items).map(([name, value]) => ({
1302
+ name,
1303
+ value
1304
+ }))
1305
+ };
1306
+ });
1307
+ sessionDataList.push(sessionData);
1308
+ } catch (error) {
1309
+ console.error("Error getting sessionStorage:", error);
1310
+ }
1311
+ }
1312
+ result["sessionStorage"] = sessionDataList;
1313
+ return result;
1314
+ }
1315
+
1316
+ // src/common/playwrightContext.ts
1317
+ var fs3 = __toESM(require("fs-extra"));
1318
+ var import_neverthrow5 = require("neverthrow");
1319
+
1320
+ // src/common/setupContextHook.ts
1321
+ var import_neverthrow3 = require("neverthrow");
1322
+
1323
+ // src/common/runApi/importUsingImportFunction.ts
1324
+ var import_neverthrow2 = require("neverthrow");
1325
+ var import_runtime_interface = __toESM(require_dist());
1326
+ async function importUsingImportFunction({
1327
+ path: path3,
1328
+ allowGenerators = true,
1329
+ importFunction
1330
+ }) {
1331
+ try {
1332
+ const importedResult = await importFunction(path3);
1333
+ if (importedResult.isErr()) {
1334
+ if (importedResult.error.type === "not_found") {
1335
+ return (0, import_neverthrow2.err)(new import_runtime_interface.ApiNotFoundError(path3));
1336
+ }
1337
+ return (0, import_neverthrow2.err)(new import_runtime_interface.AutomationError(importedResult.error.error));
1338
+ }
1339
+ const imported = importedResult.value;
1340
+ if (!imported || !imported.default || !imported.default.constructor) {
1341
+ return (0, import_neverthrow2.err)(new import_runtime_interface.InvalidApiError(`${path3} does not have a default export`));
1342
+ }
1343
+ if (imported.default.constructor.name === "AsyncGeneratorFunction") {
1344
+ if (!allowGenerators) {
1345
+ return (0, import_neverthrow2.err)(
1346
+ new import_runtime_interface.InvalidApiError(
1347
+ `${path3} default export must be an async function`
1348
+ )
1349
+ );
1350
+ }
1351
+ return (0, import_neverthrow2.ok)(async (...args) => {
1352
+ const generator = imported.default(
1353
+ ...args
1354
+ );
1355
+ const result = await generator.next();
1356
+ if (!result.done) {
1357
+ throw new Error("Yield is not supported");
1358
+ }
1359
+ return result.value;
1360
+ });
1361
+ }
1362
+ if (imported.default.constructor.name === "AsyncFunction") {
1363
+ return (0, import_neverthrow2.ok)(imported.default);
1364
+ }
1365
+ return (0, import_neverthrow2.err)(
1366
+ new import_runtime_interface.InvalidApiError(`${path3} default export must be an async function`)
1367
+ );
1368
+ } catch (error) {
1369
+ return (0, import_neverthrow2.err)(new import_runtime_interface.AutomationError(error));
1370
+ }
1371
+ }
1372
+
1373
+ // src/common/setupContextHook.ts
1374
+ var import_runtime_interface2 = __toESM(require_dist());
1375
+ var setupContextHookPath = "hooks/setupContext";
1376
+ async function loadSetupContextHook({
1377
+ importFunction
1378
+ }) {
1379
+ const importedFunctionResult = await importUsingImportFunction({
1380
+ path: setupContextHookPath,
1381
+ importFunction
1382
+ });
1383
+ if (importedFunctionResult.isErr() && importedFunctionResult.error instanceof import_runtime_interface2.ApiNotFoundError) {
1384
+ return (0, import_neverthrow3.ok)(null);
1385
+ }
1386
+ return importedFunctionResult;
1387
+ }
1388
+
1389
+ // src/common/playwrightContext.ts
1390
+ var import_portfinder2 = require("portfinder");
1391
+
1392
+ // src/common/launchBrowser.ts
1393
+ var playwright = __toESM(require("playwright"));
1394
+ var import_fs_extra2 = require("fs-extra");
1395
+ var import_path2 = require("path");
1396
+ var fs2 = __toESM(require("fs-extra"));
1397
+ var import_wait_on = __toESM(require("wait-on"));
1398
+ var import_child_process = require("child_process");
1399
+ var import_util = require("util");
1400
+ var import_neverthrow4 = require("neverthrow");
1401
+ var import_zod3 = require("zod");
1402
+ var execAsync = (0, import_util.promisify)(import_child_process.exec);
1403
+ async function createUserDirWithPreferences() {
1404
+ const playwrightTempDir = await (0, import_fs_extra2.mkdtemp)("/tmp/pw-");
1405
+ const userDir = (0, import_path2.join)(playwrightTempDir, "userdir");
1406
+ const defaultDir = (0, import_path2.join)(userDir, "Default");
1407
+ await (0, import_fs_extra2.mkdir)(defaultDir, {
1408
+ recursive: true
1409
+ });
1410
+ const preferences = {
1411
+ plugins: {
1412
+ always_open_pdf_externally: true
1413
+ }
1414
+ };
1415
+ await (0, import_fs_extra2.writeFile)((0, import_path2.join)(defaultDir, "Preferences"), JSON.stringify(preferences));
1416
+ return userDir;
1417
+ }
1418
+ async function launchChromium(options) {
1419
+ if ("cdpAddress" in options) {
1420
+ if (await isIntunedExtensionEnabled()) {
1421
+ await setupIntunedExtensionServer();
1422
+ }
1423
+ const browser = await playwright.chromium.connectOverCDP(
1424
+ options.cdpAddress
1425
+ );
1426
+ if (browser.contexts().length === 0) {
1427
+ throw new Error("No browser contexts found in the connected browser");
1428
+ }
1429
+ const context2 = browser.contexts()[0];
1430
+ let page2 = context2.pages().at(0) ?? await context2.newPage();
1431
+ const targetId = options.cdpTargetId;
1432
+ if (targetId) {
1433
+ for (const p of context2.pages()) {
1434
+ let cdp = null;
1435
+ try {
1436
+ cdp = await context2.newCDPSession(p);
1437
+ const result = await cdp.send("Target.getTargetInfo");
1438
+ if (result.targetInfo.targetId === targetId) {
1439
+ page2 = p;
1440
+ break;
1441
+ }
1442
+ } catch (error) {
1443
+ } finally {
1444
+ await cdp?.detach();
1445
+ }
1446
+ }
1447
+ }
1448
+ return { page: page2, context: context2 };
1449
+ }
1450
+ const { headless, appModeInitialUrl, cdpPort, proxy, downloadsPath } = options;
1451
+ let { executablePath } = options;
1452
+ const defaultArgsToIgnore = [
1453
+ "--disable-extensions",
1454
+ "--disable-component-extensions-with-background-pages",
1455
+ "--disable-background-networking",
1456
+ "--disable-backgrounding-occluded-windows",
1457
+ "--disable-background-timer-throttling"
1458
+ ];
1459
+ const extraArgs = [];
1460
+ const userDataDir = await createUserDirWithPreferences();
1461
+ if (isIntunedExtensionLoaded()) {
1462
+ const extensionsList = buildExtensionsList();
1463
+ const extensions = extensionsList.join(",");
1464
+ extraArgs.push(`--disable-extensions-except=${extensions}`);
1465
+ extraArgs.push(`--load-extension=${extensions}`);
1466
+ }
1467
+ if (await isIntunedExtensionEnabled()) {
1468
+ await setupIntunedExtension();
1469
+ if (proxy) {
1470
+ extraArgs.push('--proxy-bypass-list="<-loopback>"');
1471
+ }
1472
+ }
1473
+ if (cdpPort) {
1474
+ extraArgs.push(`--remote-debugging-port=${cdpPort}`);
1475
+ }
1476
+ if (headless) {
1477
+ defaultArgsToIgnore.push("--headless=old");
1478
+ extraArgs.push("--headless=new");
1479
+ }
1480
+ if (appModeInitialUrl) {
1481
+ extraArgs.push(`--app=${appModeInitialUrl}`);
1482
+ }
1483
+ if (executablePath) {
1484
+ executablePath = await fs2.realpath(executablePath);
1485
+ if (!await fs2.exists(executablePath)) {
1486
+ console.log(
1487
+ `Warning: Executable path ${executablePath} does not exist. Falling back to default.`
1488
+ );
1489
+ executablePath = void 0;
1490
+ }
1491
+ }
1492
+ const viewport = null;
1493
+ const userAgent = process.env.__PLAYWRIGHT_USER_AGENT_OVERRIDE ?? await getHeadlessUserAgent({
1494
+ executablePath
1495
+ });
1496
+ const context = await playwright.chromium.launchPersistentContext(
1497
+ userDataDir,
1498
+ {
1499
+ userAgent,
1500
+ executablePath,
1501
+ headless,
1502
+ viewport,
1503
+ proxy,
1504
+ downloadsPath,
1505
+ args: extraArgs,
1506
+ ignoreDefaultArgs: defaultArgsToIgnore
1507
+ }
1508
+ );
1509
+ context.once("close", async () => {
1510
+ try {
1511
+ await (0, import_fs_extra2.rm)(userDataDir, {
1512
+ recursive: true,
1513
+ force: true,
1514
+ retryDelay: 1e3,
1515
+ maxRetries: 5
1516
+ });
1517
+ if (await isIntunedExtensionEnabled()) {
1518
+ await cleanIntunedExtensionServer();
1519
+ }
1520
+ } catch (error) {
1521
+ console.error("Failed to remove user data dir", error);
1522
+ }
1523
+ });
1524
+ if (cdpPort) {
1525
+ const createdCdpAddress = getLocalCdpAddress(cdpPort);
1526
+ await waitOnCdpAddress(createdCdpAddress);
1527
+ }
1528
+ const page = context.pages().at(0) ?? await context.newPage();
1529
+ if (await isIntunedExtensionEnabled()) {
1530
+ await getIntunedExtensionWorker(context);
1531
+ }
1532
+ return {
1533
+ page,
1534
+ context
1535
+ };
1536
+ }
1537
+ async function getBrowserExecutablePath() {
1538
+ const browserType = getBrowserType();
1539
+ if (browserType === "brave") {
1540
+ return await getBraveExecutablePath();
1541
+ }
1542
+ }
1543
+ async function launchBrowser(options) {
1544
+ if ("cdpAddress" in options) {
1545
+ return launchChromium(options);
1546
+ }
1547
+ return launchChromium({
1548
+ ...options,
1549
+ executablePath: await getBrowserExecutablePath()
1550
+ });
1551
+ }
1552
+ function getBrowserType() {
1553
+ if (process.env.BROWSER_TYPE === "brave") {
1554
+ return "brave";
1555
+ }
1556
+ return "chromium";
1557
+ }
1558
+ async function getBraveExecutablePath() {
1559
+ const { stdout } = await execAsync("which brave-browser-stable");
1560
+ const bravePath = stdout.trim();
1561
+ if (bravePath.length === 0) {
1562
+ throw new Error("Brave browser not found");
1563
+ }
1564
+ return bravePath;
1565
+ }
1566
+ function getLocalCdpAddress(port) {
1567
+ return `http://localhost:${port}`;
1568
+ }
1569
+ async function getCdpWebSocketUrl(cdpAddress) {
1570
+ let response;
1571
+ try {
1572
+ response = await fetch(`${cdpAddress}/json/version`);
1573
+ } catch (error) {
1574
+ return (0, import_neverthrow4.err)(
1575
+ `Failed to fetch CDP version from ${cdpAddress}: ${error instanceof Error ? error.message : String(error)}`
1576
+ );
1577
+ }
1578
+ if (!response.ok) {
1579
+ return (0, import_neverthrow4.err)(
1580
+ `Failed to get CDP WebSocket URL from ${cdpAddress}: ${response.status} ${response.statusText}`
1581
+ );
1582
+ }
1583
+ let data;
1584
+ try {
1585
+ data = await response.json();
1586
+ } catch (error) {
1587
+ return (0, import_neverthrow4.err)(
1588
+ `Invalid CDP version response from ${cdpAddress}: ${error instanceof Error ? error.message : String(error)}`
1589
+ );
1590
+ }
1591
+ const cdpJsonVersionResponseSchema = import_zod3.z.object({
1592
+ webSocketDebuggerUrl: import_zod3.z.string().url()
1593
+ });
1594
+ const parseResult = cdpJsonVersionResponseSchema.safeParse(data);
1595
+ if (!parseResult.success) {
1596
+ return (0, import_neverthrow4.err)(
1597
+ `Invalid CDP version response from ${cdpAddress}: ${JSON.stringify(
1598
+ parseResult.error.format()
1599
+ )}`
1600
+ );
1601
+ }
1602
+ return (0, import_neverthrow4.ok)(parseResult.data.webSocketDebuggerUrl);
1603
+ }
1604
+ async function waitOnCdpAddress(cdpAddress) {
1605
+ const cdpAddressWithoutProtocol = cdpAddress.replace("http://", "").replace("https://", "").replace("localhost", "127.0.0.1");
1606
+ await (0, import_wait_on.default)({
1607
+ resources: [`http-get://${cdpAddressWithoutProtocol}/json/version`],
1608
+ delay: 100,
1609
+ interval: 100,
1610
+ timeout: 5e3,
1611
+ tcpTimeout: 1e3,
1612
+ window: 1e3
1613
+ });
1614
+ }
1615
+ async function getHeadlessUserAgent({
1616
+ executablePath,
1617
+ args,
1618
+ ignoreDefaultArgs
1619
+ }) {
1620
+ const browser = await playwright.chromium.launch({
1621
+ headless: true,
1622
+ executablePath,
1623
+ args,
1624
+ ignoreDefaultArgs
1625
+ });
1626
+ const context = await browser.newContext();
1627
+ const page = await context.newPage();
1628
+ let userAgent = await page.evaluate(() => navigator.userAgent);
1629
+ await browser.close();
1630
+ if (!userAgent || typeof userAgent !== "string") {
1631
+ return void 0;
1632
+ }
1633
+ userAgent = userAgent.replace("HeadlessChrome", "Chrome");
1634
+ return userAgent;
1635
+ }
1636
+
1637
+ // src/common/playwrightContext.ts
1638
+ var import_runtime_interface3 = __toESM(require_dist());
1639
+ async function withPlaywrightContext({
1640
+ cdpAddress,
1641
+ cdpTargetId,
1642
+ proxy,
1643
+ headless = true,
1644
+ downloadsPath,
1645
+ importFunction,
1646
+ apiName,
1647
+ apiParameters
1648
+ }, fn) {
1649
+ let context;
1650
+ let page;
1651
+ try {
1652
+ const setupContextHookResult = importFunction ? await loadSetupContextHook({
1653
+ importFunction
1654
+ }) : (0, import_neverthrow5.ok)(null);
1655
+ if (setupContextHookResult.isErr()) {
1656
+ return setupContextHookResult;
1657
+ }
1658
+ const setupContextHook = setupContextHookResult.value;
1659
+ if (setupContextHook === null) {
1660
+ if (cdpAddress !== void 0) {
1661
+ ({ page, context } = await launchBrowser({ cdpAddress, cdpTargetId }));
1662
+ } else {
1663
+ ({ page, context } = await launchBrowser({
1664
+ proxy,
1665
+ headless,
1666
+ downloadsPath
1667
+ }));
1668
+ }
1669
+ return await fn(context, page);
1670
+ }
1671
+ let hookCdpUrl = null;
1672
+ if (cdpAddress) {
1673
+ hookCdpUrl = cdpAddress;
1674
+ ({ context, page } = await launchBrowser({ cdpAddress, cdpTargetId }));
1675
+ } else {
1676
+ const port = await (0, import_portfinder2.getPort)({
1677
+ port: 9222
1678
+ });
1679
+ ({ context, page } = await launchBrowser({
1680
+ proxy,
1681
+ headless,
1682
+ downloadsPath,
1683
+ cdpPort: port
1684
+ }));
1685
+ hookCdpUrl = getLocalCdpAddress(port);
1686
+ }
1687
+ let hookResult;
1688
+ try {
1689
+ const wsUrlResult = await getCdpWebSocketUrl(hookCdpUrl);
1690
+ if (wsUrlResult.isOk()) {
1691
+ hookCdpUrl = wsUrlResult.value;
1692
+ } else {
1693
+ throw new Error(wsUrlResult.error);
1694
+ }
1695
+ hookResult = await setupContextHook({
1696
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
1697
+ apiName,
1698
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
1699
+ apiParameters,
1700
+ cdpUrl: hookCdpUrl
1701
+ });
1702
+ } catch (error) {
1703
+ return (0, import_neverthrow5.err)(new import_runtime_interface3.AutomationError(error));
1704
+ }
1705
+ if (!hookResult) {
1706
+ return await fn(context, page);
1707
+ }
1708
+ const { page: newPage, context: newContext, cleanup } = hookResult;
1709
+ let result;
1710
+ try {
1711
+ result = { return: await fn(newContext, newPage ?? page) };
1712
+ } catch (e) {
1713
+ result = { throw: e };
1714
+ }
1715
+ try {
1716
+ await cleanup?.();
1717
+ } catch (e) {
1718
+ result = { return: (0, import_neverthrow5.err)(new import_runtime_interface3.AutomationError(e)) };
1719
+ }
1720
+ if ("throw" in result) {
1721
+ throw result.throw;
1722
+ }
1723
+ return result.return;
1724
+ } finally {
1725
+ await context?.close();
1726
+ }
1727
+ }
1728
+ async function loadSessionToContext({
1729
+ context,
1730
+ session
1731
+ }) {
1732
+ let sessionToLoad;
1733
+ if (session.type === "state") {
1734
+ const state = session.state;
1735
+ if (state === void 0 || state === null) {
1736
+ return;
1737
+ }
1738
+ sessionToLoad = state;
1739
+ } else {
1740
+ const fullPath = getFullPathInProject(session.path);
1741
+ sessionToLoad = await fs3.readJson(fullPath);
1742
+ }
1743
+ await setStorageState(context, sessionToLoad);
1744
+ }
1745
+
1746
+ // src/common/formatZodError.ts
1747
+ function formatZodError(zodError) {
1748
+ const formattedErrors = zodError.errors.map((error) => {
1749
+ const path3 = error.path.map((segment) => {
1750
+ return typeof segment === "number" ? `[${segment}]` : segment;
1751
+ }).join(".");
1752
+ if (path3) {
1753
+ return `${path3} is invalid - ${error.message}`;
1754
+ }
1755
+ return error.message;
1756
+ });
1757
+ return formattedErrors;
1758
+ }
1759
+
1760
+ // src/common/runApi/index.ts
1761
+ var import_runtime_interface4 = __toESM(require_dist());
1762
+ function getObjectSizeInBytes(obj) {
1763
+ try {
1764
+ return new TextEncoder().encode(JSON.stringify(obj)).length;
1765
+ } catch (error) {
1766
+ return new TextEncoder().encode(String(obj)).length;
1767
+ }
1768
+ }
1769
+ async function runApi({
1770
+ abortSignal,
1771
+ importFunction,
1772
+ retrieveSession = false,
1773
+ ...input
1774
+ }) {
1775
+ let traceStarted = false;
1776
+ const inputParseResult = import_runtime_interface4.runApiParametersSchema.safeParse(input);
1777
+ if (!inputParseResult.success) {
1778
+ return (0, import_neverthrow6.err)(
1779
+ new import_runtime_interface4.InternalInvalidInputError(
1780
+ "Input validation failed",
1781
+ formatZodError(inputParseResult.error)
1782
+ )
1783
+ );
1784
+ }
1785
+ const {
1786
+ automationFunction: { name, params },
1787
+ runOptions,
1788
+ tracing,
1789
+ auth
1790
+ } = inputParseResult.data;
1791
+ const abortSymbol = /* @__PURE__ */ Symbol("abort");
1792
+ const abortPromise = new Promise((resolve2) => {
1793
+ if (!abortSignal) return;
1794
+ abortSignal.addEventListener("abort", () => {
1795
+ resolve2(abortSymbol);
1796
+ });
1797
+ });
1798
+ async function runAutomation() {
1799
+ const validatedModuleResult = await importUsingImportFunction({
1800
+ path: name,
1801
+ importFunction
1802
+ });
1803
+ if (validatedModuleResult.isErr()) {
1804
+ return (0, import_neverthrow6.err)(validatedModuleResult.error);
1805
+ }
1806
+ const automationFunction = validatedModuleResult.value;
1807
+ if (auth && auth.session.type === "state") {
1808
+ const state = auth.session.state;
1809
+ if (state === void 0 || state === null) {
1810
+ return (0, import_neverthrow6.err)(new import_runtime_interface4.AuthRequiredError());
1811
+ }
1812
+ }
1813
+ const playwrightContextParameters = {
1814
+ apiName: name,
1815
+ apiParameters: params,
1816
+ importFunction
1817
+ };
1818
+ const intunedContext = getExecutionContext();
1819
+ const runAutomationWithContext = async (context, page) => {
1820
+ async function saveTraceIfNeeded({
1821
+ errorMessage
1822
+ }) {
1823
+ if (!tracing.enabled || !traceStarted) {
1824
+ return;
1825
+ }
1826
+ try {
1827
+ await context?.tracing.stop({ path: tracing.filePath });
1828
+ } catch (error) {
1829
+ console.log(errorMessage, error?.message);
1830
+ await (0, import_fs_extra3.remove)(tracing.filePath);
1831
+ }
1832
+ }
1833
+ if (auth) {
1834
+ await loadSessionToContext({
1835
+ context,
1836
+ session: auth.session
1837
+ });
1838
+ if (intunedContext && auth.parameters) {
1839
+ intunedContext.getAuthSessionParameters = async () => auth.parameters;
1840
+ }
1841
+ }
1842
+ if (tracing.enabled) {
1843
+ await context.tracing.start({
1844
+ screenshots: true,
1845
+ snapshots: true,
1846
+ sources: true
1847
+ });
1848
+ traceStarted = true;
1849
+ }
1850
+ cleanEnvironmentVariables();
1851
+ const automationFunctionParameters = [
1852
+ ...params !== void 0 ? [params] : [],
1853
+ page,
1854
+ context
1855
+ ];
1856
+ try {
1857
+ const [automationFunctionResult] = await Promise.all([
1858
+ automationFunction(...automationFunctionParameters)
1859
+ ]);
1860
+ const MAX_RESULT_SIZE_BYTES = 2 * 1024 * 1024;
1861
+ const resultSizeInBytes = getObjectSizeInBytes(
1862
+ automationFunctionResult
1863
+ );
1864
+ if (resultSizeInBytes > MAX_RESULT_SIZE_BYTES) {
1865
+ return (0, import_neverthrow6.err)(
1866
+ new import_runtime_interface4.ResultTooBigError(resultSizeInBytes, MAX_RESULT_SIZE_BYTES)
1867
+ );
1868
+ }
1869
+ if (retrieveSession) {
1870
+ return (0, import_neverthrow6.ok)({
1871
+ result: automationFunctionResult,
1872
+ extendedPayloads: intunedContext?.extendedPayloads,
1873
+ session: await getStorageState(context)
1874
+ });
1875
+ }
1876
+ return (0, import_neverthrow6.ok)({
1877
+ result: automationFunctionResult,
1878
+ extendedPayloads: intunedContext?.extendedPayloads
1879
+ });
1880
+ } catch (error) {
1881
+ return (0, import_neverthrow6.err)(new import_runtime_interface4.AutomationError(error));
1882
+ } finally {
1883
+ await saveTraceIfNeeded({ errorMessage: "failed to save trace" });
1884
+ }
1885
+ };
1886
+ if (intunedContext?.store) {
1887
+ intunedContext.store = {};
1888
+ }
1889
+ if (runOptions.environment === "standalone") {
1890
+ const downloadsPath = getDownloadDirectoryPath();
1891
+ try {
1892
+ return await withPlaywrightContext(
1893
+ {
1894
+ headless: runOptions.headless,
1895
+ proxy: runOptions.proxy,
1896
+ downloadsPath,
1897
+ ...playwrightContextParameters
1898
+ },
1899
+ runAutomationWithContext
1900
+ );
1901
+ } finally {
1902
+ await fs4.remove(downloadsPath);
1903
+ }
1904
+ } else {
1905
+ return await withPlaywrightContext(
1906
+ {
1907
+ cdpAddress: runOptions.cdpAddress,
1908
+ cdpTargetId: runOptions.cdpTargetId,
1909
+ ...playwrightContextParameters
1910
+ },
1911
+ runAutomationWithContext
1912
+ );
1913
+ }
1914
+ }
1915
+ const result = await Promise.race([await runAutomation(), abortPromise]);
1916
+ if (result === abortSymbol) {
1917
+ return (0, import_neverthrow6.err)(new import_runtime_interface4.AbortedError());
1918
+ }
1919
+ return result;
1920
+ }
1921
+
1922
+ // src/commands/api/run.ts
1923
+ var import_runtime_interface5 = __toESM(require_dist());
1924
+
1925
+ // src/commands/common/tsNodeImport.ts
1926
+ var import_neverthrow7 = require("neverthrow");
1927
+ var tsNodeImport = async (apiName) => {
1928
+ require("ts-node").register({
1929
+ transpileOnly: true,
1930
+ compilerOptions: {
1931
+ lib: ["dom", "es2020"]
1932
+ }
1933
+ });
1934
+ const path3 = getFullPathInProject(...apiName.split("/"));
1935
+ try {
1936
+ const imported = await Promise.resolve().then(() => require(path3));
1937
+ return (0, import_neverthrow7.ok)(imported);
1938
+ } catch (e) {
1939
+ if ("code" in e && e.code.includes("MODULE_NOT_FOUND")) {
1940
+ if ("requireStack" in e && Array.isArray(e.requireStack)) {
1941
+ const isStackTopThisFile = e.requireStack[0] === __filename;
1942
+ if (!isStackTopThisFile) {
1943
+ return (0, import_neverthrow7.err)({
1944
+ type: "other",
1945
+ error: e
1946
+ });
1947
+ }
1948
+ }
1949
+ return (0, import_neverthrow7.err)({
1950
+ type: "not_found"
1951
+ });
1952
+ }
1953
+ return (0, import_neverthrow7.err)({
1954
+ type: "other",
1955
+ error: e
1956
+ });
1957
+ }
1958
+ };
1959
+
1960
+ // src/commands/api/run.ts
1961
+ var import_isNil = __toESM(require("lodash/isNil"));
1962
+ import_dotenv2.default.config({
1963
+ path: `.env`
1964
+ });
1965
+ async function executeCLI(apiName, mode, inputData, options) {
1966
+ cleanEnvironmentVariables();
1967
+ let authSessionPathToUse = null;
1968
+ const settings = await getSettings();
1969
+ if (settings.authSessions.enabled) {
1970
+ if (!options.authSessionPath) {
1971
+ throw new Error(
1972
+ "AuthSessions are enabled but no AuthSession provided.\nPlease provide --auth-session-path option.\nSee https://docs.intunedhq.com/docs/cli/auth-sessions for more information."
1973
+ );
1974
+ }
1975
+ authSessionPathToUse = options.authSessionPath;
1976
+ } else {
1977
+ if (options.authSessionPath) {
1978
+ throw new Error(
1979
+ "Authentication is not enabled but AuthSession was provided.\nEnable AuthSessions in Intuned.json to use this feature.\nSee https://docs.intunedhq.com/docs/cli/auth-sessions for more information."
1980
+ );
1981
+ }
1982
+ }
1983
+ const runApiResult = await runApi({
1984
+ automationFunction: {
1985
+ name: `${API_FOLDER_NAME}/${apiName}`,
1986
+ params: inputData
1987
+ },
1988
+ auth: authSessionPathToUse ? {
1989
+ session: {
1990
+ type: "file",
1991
+ path: authSessionPathToUse
1992
+ }
1993
+ } : void 0,
1994
+ runOptions: {
1995
+ environment: "cdp",
1996
+ cdpAddress: options.cdpAddress
1997
+ },
1998
+ importFunction: tsNodeImport
1999
+ });
2000
+ if (runApiResult.isErr()) {
2001
+ if (runApiResult.error instanceof import_runtime_interface5.AutomationError) {
2002
+ throw runApiResult.error.error;
2003
+ }
2004
+ console.error(runApiResult.error);
2005
+ throw new Error("An error occurred while running the API");
2006
+ }
2007
+ const { result, extendedPayloads: payloadToAppend } = runApiResult.value;
2008
+ const resultsDir = "/tmp/run-results";
2009
+ if (options.outputFileId) {
2010
+ logger.info(
2011
+ import_chalk2.default.underline.bgBlue.white(
2012
+ `Click to Open: Results saved (Run: ${options.outputFileId})`
2013
+ )
2014
+ );
2015
+ fs5.ensureDirSync(resultsDir);
2016
+ const path3 = `${resultsDir}/${options.outputFileId}.json`;
2017
+ await fs5.writeJson(
2018
+ path3,
2019
+ {
2020
+ input: inputData,
2021
+ output: result
2022
+ },
2023
+ {
2024
+ spaces: 2
2025
+ }
2026
+ );
2027
+ } else {
2028
+ console.log("result:", result);
2029
+ }
2030
+ const hasPayloadToAppend = payloadToAppend && payloadToAppend.length > 0;
2031
+ if (hasPayloadToAppend && options.outputFileId) {
2032
+ logger.info(
2033
+ import_chalk2.default.underline.bgBlue.white(
2034
+ `Click to Open: payloads to append (Run: ${options.outputFileId})`
2035
+ )
2036
+ );
2037
+ fs5.ensureDirSync(resultsDir);
2038
+ const path3 = `${resultsDir}/${options.outputFileId}-payloads-to-append.json`;
2039
+ await fs5.writeJson(path3, payloadToAppend, {
2040
+ spaces: 2
2041
+ });
2042
+ } else if (hasPayloadToAppend) {
2043
+ logger.info("payload to append:", payloadToAppend);
2044
+ logger.info(
2045
+ "This will only take an effect if this API run was part of a job."
2046
+ );
2047
+ }
2048
+ }
2049
+ import_commander.program.description("run the user function in the cli for testing purposes").option("-i, --input [file]", "input json file").option("-j, --json [json]", "input json string").option("--cdpAddress <cdpAddress>", "CDP address", "http://localhost:9222").option(
2050
+ "--authSessionPath <authSession>",
2051
+ "AuthSession to use when executing the api"
2052
+ ).option(
2053
+ "--outputFileId <outputFileId>",
2054
+ "the output file id to save the result in"
2055
+ ).option("--proxy <proxy>", "proxy to use").option(
2056
+ "--authSessionParameters <authSessionParameters>",
2057
+ "parameters used to create the used AuthSession"
2058
+ ).argument("[apiName]", "name of the api", "default").allowUnknownOption().addArgument(
2059
+ new import_commander.Argument("<mode>", "mode of execution").choices([
2060
+ "vanilla",
2061
+ "playwright",
2062
+ "playwright-standalone",
2063
+ "playwright-headless"
2064
+ ]).default("playwright-standalone").argOptional()
2065
+ ).action(async (apiName, mode, options) => {
2066
+ let inputData = null;
2067
+ if (options.input) {
2068
+ inputData = await fs5.readJSON(options.input);
2069
+ } else if (options.json) {
2070
+ inputData = JSON.parse(options.json);
2071
+ } else {
2072
+ inputData = {};
2073
+ }
2074
+ let authSessionParametersJson = void 0;
2075
+ if (!(0, import_isNil.default)(options?.authSessionParameters)) {
2076
+ authSessionParametersJson = JSON.parse(options.authSessionParameters);
2077
+ }
2078
+ await runWithContext(
2079
+ {
2080
+ runEnvironment: "IDE" /* IDE */,
2081
+ extendedPayloads: [],
2082
+ runId: (0, import_nanoid.nanoid)(),
2083
+ proxy: options.proxy,
2084
+ getAuthSessionParameters: authSessionParametersJson !== void 0 ? async () => authSessionParametersJson : void 0
2085
+ },
2086
+ () => executeCLI(apiName, mode, inputData, options)
2087
+ );
2088
+ process.exit(0);
2089
+ });
2090
+ import_commander.program.parse(process.argv);
2091
+ // Annotate the CommonJS export names for ESM import in node:
2092
+ 0 && (module.exports = {
2093
+ executeCLI
2094
+ });