@testdroid-ai/cli 0.1.2

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 (229) hide show
  1. package/dist/adb/adb-path-resolver.d.ts +6 -0
  2. package/dist/adb/adb-path-resolver.d.ts.map +1 -0
  3. package/dist/adb/adb-path-resolver.js +93 -0
  4. package/dist/adb/adb-path-resolver.js.map +1 -0
  5. package/dist/adb/adb-shell.d.ts +29 -0
  6. package/dist/adb/adb-shell.d.ts.map +1 -0
  7. package/dist/adb/adb-shell.js +128 -0
  8. package/dist/adb/adb-shell.js.map +1 -0
  9. package/dist/adb/detector.d.ts +12 -0
  10. package/dist/adb/detector.d.ts.map +1 -0
  11. package/dist/adb/detector.js +44 -0
  12. package/dist/adb/detector.js.map +1 -0
  13. package/dist/adb/device-tracker.d.ts +23 -0
  14. package/dist/adb/device-tracker.d.ts.map +1 -0
  15. package/dist/adb/device-tracker.js +134 -0
  16. package/dist/adb/device-tracker.js.map +1 -0
  17. package/dist/adb/emulator-detector.d.ts +11 -0
  18. package/dist/adb/emulator-detector.d.ts.map +1 -0
  19. package/dist/adb/emulator-detector.js +54 -0
  20. package/dist/adb/emulator-detector.js.map +1 -0
  21. package/dist/appium/health-check.d.ts +6 -0
  22. package/dist/appium/health-check.d.ts.map +1 -0
  23. package/dist/appium/health-check.js +61 -0
  24. package/dist/appium/health-check.js.map +1 -0
  25. package/dist/appium/session-manager.d.ts +15 -0
  26. package/dist/appium/session-manager.d.ts.map +1 -0
  27. package/dist/appium/session-manager.js +82 -0
  28. package/dist/appium/session-manager.js.map +1 -0
  29. package/dist/auth/credential-store.d.ts +31 -0
  30. package/dist/auth/credential-store.d.ts.map +1 -0
  31. package/dist/auth/credential-store.js +154 -0
  32. package/dist/auth/credential-store.js.map +1 -0
  33. package/dist/auth/pairing-auth.d.ts +26 -0
  34. package/dist/auth/pairing-auth.d.ts.map +1 -0
  35. package/dist/auth/pairing-auth.js +177 -0
  36. package/dist/auth/pairing-auth.js.map +1 -0
  37. package/dist/auth/token-manager.d.ts +20 -0
  38. package/dist/auth/token-manager.d.ts.map +1 -0
  39. package/dist/auth/token-manager.js +101 -0
  40. package/dist/auth/token-manager.js.map +1 -0
  41. package/dist/commands/connect.d.ts +9 -0
  42. package/dist/commands/connect.d.ts.map +1 -0
  43. package/dist/commands/connect.js +270 -0
  44. package/dist/commands/connect.js.map +1 -0
  45. package/dist/commands/devices.d.ts +2 -0
  46. package/dist/commands/devices.d.ts.map +1 -0
  47. package/dist/commands/devices.js +52 -0
  48. package/dist/commands/devices.js.map +1 -0
  49. package/dist/commands/doctor.d.ts +2 -0
  50. package/dist/commands/doctor.d.ts.map +1 -0
  51. package/dist/commands/doctor.js +214 -0
  52. package/dist/commands/doctor.js.map +1 -0
  53. package/dist/commands/mirror.d.ts +2 -0
  54. package/dist/commands/mirror.d.ts.map +1 -0
  55. package/dist/commands/mirror.js +77 -0
  56. package/dist/commands/mirror.js.map +1 -0
  57. package/dist/commands/prompt.d.ts +12 -0
  58. package/dist/commands/prompt.d.ts.map +1 -0
  59. package/dist/commands/prompt.js +174 -0
  60. package/dist/commands/prompt.js.map +1 -0
  61. package/dist/commands/record.d.ts +7 -0
  62. package/dist/commands/record.d.ts.map +1 -0
  63. package/dist/commands/record.js +129 -0
  64. package/dist/commands/record.js.map +1 -0
  65. package/dist/commands/run.d.ts +16 -0
  66. package/dist/commands/run.d.ts.map +1 -0
  67. package/dist/commands/run.js +271 -0
  68. package/dist/commands/run.js.map +1 -0
  69. package/dist/commands/setup.d.ts +6 -0
  70. package/dist/commands/setup.d.ts.map +1 -0
  71. package/dist/commands/setup.js +50 -0
  72. package/dist/commands/setup.js.map +1 -0
  73. package/dist/commands/status.d.ts +2 -0
  74. package/dist/commands/status.d.ts.map +1 -0
  75. package/dist/commands/status.js +32 -0
  76. package/dist/commands/status.js.map +1 -0
  77. package/dist/healing/ai-adapter.d.ts +44 -0
  78. package/dist/healing/ai-adapter.d.ts.map +1 -0
  79. package/dist/healing/ai-adapter.js +87 -0
  80. package/dist/healing/ai-adapter.js.map +1 -0
  81. package/dist/healing/healer.d.ts +109 -0
  82. package/dist/healing/healer.d.ts.map +1 -0
  83. package/dist/healing/healer.js +475 -0
  84. package/dist/healing/healer.js.map +1 -0
  85. package/dist/healing/similarity.d.ts +63 -0
  86. package/dist/healing/similarity.d.ts.map +1 -0
  87. package/dist/healing/similarity.js +172 -0
  88. package/dist/healing/similarity.js.map +1 -0
  89. package/dist/index.d.ts +3 -0
  90. package/dist/index.d.ts.map +1 -0
  91. package/dist/index.js +107 -0
  92. package/dist/index.js.map +1 -0
  93. package/dist/recording/capture-strategies/capture-strategy.interface.d.ts +6 -0
  94. package/dist/recording/capture-strategies/capture-strategy.interface.d.ts.map +1 -0
  95. package/dist/recording/capture-strategies/capture-strategy.interface.js +3 -0
  96. package/dist/recording/capture-strategies/capture-strategy.interface.js.map +1 -0
  97. package/dist/recording/capture-strategies/getevent.strategy.d.ts +27 -0
  98. package/dist/recording/capture-strategies/getevent.strategy.d.ts.map +1 -0
  99. package/dist/recording/capture-strategies/getevent.strategy.js +163 -0
  100. package/dist/recording/capture-strategies/getevent.strategy.js.map +1 -0
  101. package/dist/recording/capture-strategies/key-capture.strategy.d.ts +16 -0
  102. package/dist/recording/capture-strategies/key-capture.strategy.d.ts.map +1 -0
  103. package/dist/recording/capture-strategies/key-capture.strategy.js +67 -0
  104. package/dist/recording/capture-strategies/key-capture.strategy.js.map +1 -0
  105. package/dist/recording/capture-strategies/logcat-capture.strategy.d.ts +32 -0
  106. package/dist/recording/capture-strategies/logcat-capture.strategy.d.ts.map +1 -0
  107. package/dist/recording/capture-strategies/logcat-capture.strategy.js +176 -0
  108. package/dist/recording/capture-strategies/logcat-capture.strategy.js.map +1 -0
  109. package/dist/recording/confidence-scorer.d.ts +13 -0
  110. package/dist/recording/confidence-scorer.d.ts.map +1 -0
  111. package/dist/recording/confidence-scorer.js +34 -0
  112. package/dist/recording/confidence-scorer.js.map +1 -0
  113. package/dist/recording/event-capture.d.ts +40 -0
  114. package/dist/recording/event-capture.d.ts.map +1 -0
  115. package/dist/recording/event-capture.js +168 -0
  116. package/dist/recording/event-capture.js.map +1 -0
  117. package/dist/recording/recorder.d.ts +32 -0
  118. package/dist/recording/recorder.d.ts.map +1 -0
  119. package/dist/recording/recorder.js +107 -0
  120. package/dist/recording/recorder.js.map +1 -0
  121. package/dist/runner/adb-executor.d.ts +116 -0
  122. package/dist/runner/adb-executor.d.ts.map +1 -0
  123. package/dist/runner/adb-executor.js +323 -0
  124. package/dist/runner/adb-executor.js.map +1 -0
  125. package/dist/runner/appium-executor.d.ts +35 -0
  126. package/dist/runner/appium-executor.d.ts.map +1 -0
  127. package/dist/runner/appium-executor.js +191 -0
  128. package/dist/runner/appium-executor.js.map +1 -0
  129. package/dist/runner/parse-strategies/appium-default.strategy.d.ts +12 -0
  130. package/dist/runner/parse-strategies/appium-default.strategy.d.ts.map +1 -0
  131. package/dist/runner/parse-strategies/appium-default.strategy.js +342 -0
  132. package/dist/runner/parse-strategies/appium-default.strategy.js.map +1 -0
  133. package/dist/runner/parse-strategies/index.d.ts +7 -0
  134. package/dist/runner/parse-strategies/index.d.ts.map +1 -0
  135. package/dist/runner/parse-strategies/index.js +14 -0
  136. package/dist/runner/parse-strategies/index.js.map +1 -0
  137. package/dist/runner/parse-strategies/java.strategy.d.ts +8 -0
  138. package/dist/runner/parse-strategies/java.strategy.d.ts.map +1 -0
  139. package/dist/runner/parse-strategies/java.strategy.js +170 -0
  140. package/dist/runner/parse-strategies/java.strategy.js.map +1 -0
  141. package/dist/runner/parse-strategies/maestro.strategy.d.ts +8 -0
  142. package/dist/runner/parse-strategies/maestro.strategy.d.ts.map +1 -0
  143. package/dist/runner/parse-strategies/maestro.strategy.js +105 -0
  144. package/dist/runner/parse-strategies/maestro.strategy.js.map +1 -0
  145. package/dist/runner/parse-strategies/python.strategy.d.ts +8 -0
  146. package/dist/runner/parse-strategies/python.strategy.d.ts.map +1 -0
  147. package/dist/runner/parse-strategies/python.strategy.js +140 -0
  148. package/dist/runner/parse-strategies/python.strategy.js.map +1 -0
  149. package/dist/runner/parse-strategies/robot.strategy.d.ts +8 -0
  150. package/dist/runner/parse-strategies/robot.strategy.d.ts.map +1 -0
  151. package/dist/runner/parse-strategies/robot.strategy.js +81 -0
  152. package/dist/runner/parse-strategies/robot.strategy.js.map +1 -0
  153. package/dist/runner/parse-strategies/types.d.ts +17 -0
  154. package/dist/runner/parse-strategies/types.d.ts.map +1 -0
  155. package/dist/runner/parse-strategies/types.js +3 -0
  156. package/dist/runner/parse-strategies/types.js.map +1 -0
  157. package/dist/runner/parse-strategies/utils.d.ts +39 -0
  158. package/dist/runner/parse-strategies/utils.d.ts.map +1 -0
  159. package/dist/runner/parse-strategies/utils.js +217 -0
  160. package/dist/runner/parse-strategies/utils.js.map +1 -0
  161. package/dist/runner/script-parser.d.ts +43 -0
  162. package/dist/runner/script-parser.d.ts.map +1 -0
  163. package/dist/runner/script-parser.js +68 -0
  164. package/dist/runner/script-parser.js.map +1 -0
  165. package/dist/runner/selector-resolver.d.ts +87 -0
  166. package/dist/runner/selector-resolver.d.ts.map +1 -0
  167. package/dist/runner/selector-resolver.js +206 -0
  168. package/dist/runner/selector-resolver.js.map +1 -0
  169. package/dist/runner/step-executor.d.ts +112 -0
  170. package/dist/runner/step-executor.d.ts.map +1 -0
  171. package/dist/runner/step-executor.js +518 -0
  172. package/dist/runner/step-executor.js.map +1 -0
  173. package/dist/runner/test-runner.d.ts +89 -0
  174. package/dist/runner/test-runner.d.ts.map +1 -0
  175. package/dist/runner/test-runner.js +320 -0
  176. package/dist/runner/test-runner.js.map +1 -0
  177. package/dist/scrcpy/launcher.d.ts +16 -0
  178. package/dist/scrcpy/launcher.d.ts.map +1 -0
  179. package/dist/scrcpy/launcher.js +87 -0
  180. package/dist/scrcpy/launcher.js.map +1 -0
  181. package/dist/services/device-poller.d.ts +10 -0
  182. package/dist/services/device-poller.d.ts.map +1 -0
  183. package/dist/services/device-poller.js +44 -0
  184. package/dist/services/device-poller.js.map +1 -0
  185. package/dist/services/execution-manager.d.ts +18 -0
  186. package/dist/services/execution-manager.d.ts.map +1 -0
  187. package/dist/services/execution-manager.js +185 -0
  188. package/dist/services/execution-manager.js.map +1 -0
  189. package/dist/services/recording-manager.d.ts +13 -0
  190. package/dist/services/recording-manager.d.ts.map +1 -0
  191. package/dist/services/recording-manager.js +234 -0
  192. package/dist/services/recording-manager.js.map +1 -0
  193. package/dist/setup/adb-installer.d.ts +8 -0
  194. package/dist/setup/adb-installer.d.ts.map +1 -0
  195. package/dist/setup/adb-installer.js +115 -0
  196. package/dist/setup/adb-installer.js.map +1 -0
  197. package/dist/setup/appium-installer.d.ts +8 -0
  198. package/dist/setup/appium-installer.d.ts.map +1 -0
  199. package/dist/setup/appium-installer.js +113 -0
  200. package/dist/setup/appium-installer.js.map +1 -0
  201. package/dist/setup/base-installer.d.ts +33 -0
  202. package/dist/setup/base-installer.d.ts.map +1 -0
  203. package/dist/setup/base-installer.js +170 -0
  204. package/dist/setup/base-installer.js.map +1 -0
  205. package/dist/setup/build-tools-installer.d.ts +8 -0
  206. package/dist/setup/build-tools-installer.d.ts.map +1 -0
  207. package/dist/setup/build-tools-installer.js +109 -0
  208. package/dist/setup/build-tools-installer.js.map +1 -0
  209. package/dist/setup/java-installer.d.ts +8 -0
  210. package/dist/setup/java-installer.d.ts.map +1 -0
  211. package/dist/setup/java-installer.js +132 -0
  212. package/dist/setup/java-installer.js.map +1 -0
  213. package/dist/setup/paths.d.ts +12 -0
  214. package/dist/setup/paths.d.ts.map +1 -0
  215. package/dist/setup/paths.js +90 -0
  216. package/dist/setup/paths.js.map +1 -0
  217. package/dist/setup/scrcpy-installer.d.ts +8 -0
  218. package/dist/setup/scrcpy-installer.d.ts.map +1 -0
  219. package/dist/setup/scrcpy-installer.js +143 -0
  220. package/dist/setup/scrcpy-installer.js.map +1 -0
  221. package/dist/setup/setup-manager.d.ts +18 -0
  222. package/dist/setup/setup-manager.d.ts.map +1 -0
  223. package/dist/setup/setup-manager.js +113 -0
  224. package/dist/setup/setup-manager.js.map +1 -0
  225. package/dist/ws/socket.d.ts +38 -0
  226. package/dist/ws/socket.d.ts.map +1 -0
  227. package/dist/ws/socket.js +91 -0
  228. package/dist/ws/socket.js.map +1 -0
  229. package/package.json +53 -0
@@ -0,0 +1,61 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.isAppiumRunning = isAppiumRunning;
37
+ /**
38
+ * Checks if Appium server is reachable AND webdriverio is importable.
39
+ * Both conditions must be true for Appium mode to work.
40
+ */
41
+ async function isAppiumRunning(url = 'http://localhost:4723') {
42
+ // 1. Check webdriverio is importable (optional dep)
43
+ try {
44
+ await Promise.resolve().then(() => __importStar(require('webdriverio')));
45
+ }
46
+ catch {
47
+ return false;
48
+ }
49
+ // 2. Check Appium server is reachable
50
+ try {
51
+ const parsedUrl = new URL(`${url}/status`);
52
+ const response = await fetch(parsedUrl.toString(), {
53
+ signal: AbortSignal.timeout(2000),
54
+ });
55
+ return response.ok;
56
+ }
57
+ catch {
58
+ return false;
59
+ }
60
+ }
61
+ //# sourceMappingURL=health-check.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"health-check.js","sourceRoot":"","sources":["../../src/appium/health-check.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAIA,0CAkBC;AAtBD;;;GAGG;AACI,KAAK,UAAU,eAAe,CAAC,GAAG,GAAG,uBAAuB;IACjE,oDAAoD;IACpD,IAAI,CAAC;QACH,wDAAa,aAAa,GAAC,CAAC;IAC9B,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;IAED,sCAAsC;IACtC,IAAI,CAAC;QACH,MAAM,SAAS,GAAG,IAAI,GAAG,CAAC,GAAG,GAAG,SAAS,CAAC,CAAC;QAC3C,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,SAAS,CAAC,QAAQ,EAAE,EAAE;YACjD,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC;SAClC,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC,EAAE,CAAC;IACrB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC"}
@@ -0,0 +1,15 @@
1
+ /**
2
+ * Manages Appium/WebDriverIO session lifecycle.
3
+ * Uses dynamic import to avoid crashing when webdriverio is not installed.
4
+ */
5
+ export declare class AppiumSessionManager {
6
+ private driver;
7
+ /**
8
+ * Creates an Appium/WebDriverIO session for the given device.
9
+ * Precondition: webdriverio must be installed (`npm install webdriverio`).
10
+ */
11
+ createSession(serial: string, appiumUrl?: string): Promise<any>;
12
+ destroySession(): Promise<void>;
13
+ getDriver(): any | null;
14
+ }
15
+ //# sourceMappingURL=session-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-manager.d.ts","sourceRoot":"","sources":["../../src/appium/session-manager.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,qBAAa,oBAAoB;IAC/B,OAAO,CAAC,MAAM,CAAa;IAE3B;;;OAGG;IACG,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,SAA0B,GAAG,OAAO,CAAC,GAAG,CAAC;IAyBhF,cAAc,IAAI,OAAO,CAAC,IAAI,CAAC;IAOrC,SAAS,IAAI,GAAG,GAAG,IAAI;CAGxB"}
@@ -0,0 +1,82 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.AppiumSessionManager = void 0;
37
+ /**
38
+ * Manages Appium/WebDriverIO session lifecycle.
39
+ * Uses dynamic import to avoid crashing when webdriverio is not installed.
40
+ */
41
+ class AppiumSessionManager {
42
+ driver = null;
43
+ /**
44
+ * Creates an Appium/WebDriverIO session for the given device.
45
+ * Precondition: webdriverio must be installed (`npm install webdriverio`).
46
+ */
47
+ async createSession(serial, appiumUrl = 'http://localhost:4723') {
48
+ let remote;
49
+ try {
50
+ ({ remote } = await Promise.resolve().then(() => __importStar(require('webdriverio'))));
51
+ }
52
+ catch {
53
+ throw new Error('webdriverio nao instalado. Execute: npm install webdriverio');
54
+ }
55
+ const parsed = new URL(appiumUrl);
56
+ this.driver = await remote({
57
+ hostname: parsed.hostname,
58
+ port: Number(parsed.port) || 4723,
59
+ path: '/',
60
+ capabilities: {
61
+ platformName: 'Android',
62
+ 'appium:automationName': 'UiAutomator2',
63
+ 'appium:udid': serial,
64
+ 'appium:noReset': true,
65
+ 'appium:autoLaunch': false,
66
+ 'appium:newCommandTimeout': 600,
67
+ },
68
+ });
69
+ return this.driver;
70
+ }
71
+ async destroySession() {
72
+ if (this.driver) {
73
+ await this.driver.deleteSession().catch(() => { });
74
+ this.driver = null;
75
+ }
76
+ }
77
+ getDriver() {
78
+ return this.driver;
79
+ }
80
+ }
81
+ exports.AppiumSessionManager = AppiumSessionManager;
82
+ //# sourceMappingURL=session-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-manager.js","sourceRoot":"","sources":["../../src/appium/session-manager.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;;;GAGG;AACH,MAAa,oBAAoB;IACvB,MAAM,GAAQ,IAAI,CAAC;IAE3B;;;OAGG;IACH,KAAK,CAAC,aAAa,CAAC,MAAc,EAAE,SAAS,GAAG,uBAAuB;QACrE,IAAI,MAAM,CAAC;QACX,IAAI,CAAC;YACH,CAAC,EAAE,MAAM,EAAE,GAAG,wDAAa,aAAa,GAAC,CAAC,CAAC;QAC7C,CAAC;QAAC,MAAM,CAAC;YACP,MAAM,IAAI,KAAK,CAAC,6DAA6D,CAAC,CAAC;QACjF,CAAC;QACD,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,SAAS,CAAC,CAAC;QAElC,IAAI,CAAC,MAAM,GAAG,MAAM,MAAM,CAAC;YACzB,QAAQ,EAAE,MAAM,CAAC,QAAQ;YACzB,IAAI,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,IAAI;YACjC,IAAI,EAAE,GAAG;YACT,YAAY,EAAE;gBACZ,YAAY,EAAE,SAAS;gBACvB,uBAAuB,EAAE,cAAc;gBACvC,aAAa,EAAE,MAAM;gBACrB,gBAAgB,EAAE,IAAI;gBACtB,mBAAmB,EAAE,KAAK;gBAC1B,0BAA0B,EAAE,GAAG;aAChC;SACF,CAAC,CAAC;QACH,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED,KAAK,CAAC,cAAc;QAClB,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,MAAM,IAAI,CAAC,MAAM,CAAC,aAAa,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;YAClD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QACrB,CAAC;IACH,CAAC;IAED,SAAS;QACP,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;CACF;AA1CD,oDA0CC"}
@@ -0,0 +1,31 @@
1
+ interface StoredCredentials {
2
+ serverUrl: string;
3
+ accessToken: string;
4
+ refreshToken: string;
5
+ expiresAt: number;
6
+ }
7
+ interface Config {
8
+ serverUrl?: string;
9
+ }
10
+ /**
11
+ * Encrypted credential storage at ~/.testdroid/credentials.json
12
+ * Uses AES-256-GCM with PBKDF2-derived key.
13
+ */
14
+ export declare class CredentialStore {
15
+ private ensureDir;
16
+ save(creds: StoredCredentials): void;
17
+ load(): StoredCredentials | null;
18
+ clear(): void;
19
+ saveConfig(config: Config): void;
20
+ loadConfig(): Config;
21
+ /**
22
+ * Resolves server URL from: flag → env → config → default
23
+ */
24
+ resolveServerUrl(flagValue?: string): string;
25
+ private getKey;
26
+ private getMachineId;
27
+ private encrypt;
28
+ private decrypt;
29
+ }
30
+ export {};
31
+ //# sourceMappingURL=credential-store.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"credential-store.d.ts","sourceRoot":"","sources":["../../src/auth/credential-store.ts"],"names":[],"mappings":"AAKA,UAAU,iBAAiB;IACzB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,UAAU,MAAM;IACd,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAOD;;;GAGG;AACH,qBAAa,eAAe;IAC1B,OAAO,CAAC,SAAS;IAMjB,IAAI,CAAC,KAAK,EAAE,iBAAiB,GAAG,IAAI;IAMpC,IAAI,IAAI,iBAAiB,GAAG,IAAI;IAWhC,KAAK,IAAI,IAAI;IAMb,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAOhC,UAAU,IAAI,MAAM;IASpB;;OAEG;IACH,gBAAgB,CAAC,SAAS,CAAC,EAAE,MAAM,GAAG,MAAM;IAU5C,OAAO,CAAC,MAAM;IAed,OAAO,CAAC,YAAY;IASpB,OAAO,CAAC,OAAO;IAaf,OAAO,CAAC,OAAO;CAUhB"}
@@ -0,0 +1,154 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.CredentialStore = void 0;
37
+ const fs = __importStar(require("fs"));
38
+ const path = __importStar(require("path"));
39
+ const os = __importStar(require("os"));
40
+ const crypto = __importStar(require("crypto"));
41
+ const TESTDROID_DIR = path.join(os.homedir(), '.testdroid');
42
+ const CREDENTIALS_FILE = path.join(TESTDROID_DIR, 'credentials.json');
43
+ const CONFIG_FILE = path.join(TESTDROID_DIR, 'config.json');
44
+ const SALT_FILE = path.join(TESTDROID_DIR, '.salt');
45
+ /**
46
+ * Encrypted credential storage at ~/.testdroid/credentials.json
47
+ * Uses AES-256-GCM with PBKDF2-derived key.
48
+ */
49
+ class CredentialStore {
50
+ ensureDir() {
51
+ if (!fs.existsSync(TESTDROID_DIR)) {
52
+ fs.mkdirSync(TESTDROID_DIR, { recursive: true, mode: 0o700 });
53
+ }
54
+ }
55
+ save(creds) {
56
+ this.ensureDir();
57
+ const encrypted = this.encrypt(JSON.stringify(creds));
58
+ fs.writeFileSync(CREDENTIALS_FILE, encrypted, { mode: 0o600 });
59
+ }
60
+ load() {
61
+ if (!fs.existsSync(CREDENTIALS_FILE))
62
+ return null;
63
+ try {
64
+ const encrypted = fs.readFileSync(CREDENTIALS_FILE, 'utf-8');
65
+ const json = this.decrypt(encrypted);
66
+ return JSON.parse(json);
67
+ }
68
+ catch {
69
+ return null;
70
+ }
71
+ }
72
+ clear() {
73
+ if (fs.existsSync(CREDENTIALS_FILE)) {
74
+ fs.unlinkSync(CREDENTIALS_FILE);
75
+ }
76
+ }
77
+ saveConfig(config) {
78
+ this.ensureDir();
79
+ fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2), {
80
+ mode: 0o600,
81
+ });
82
+ }
83
+ loadConfig() {
84
+ if (!fs.existsSync(CONFIG_FILE))
85
+ return {};
86
+ try {
87
+ return JSON.parse(fs.readFileSync(CONFIG_FILE, 'utf-8'));
88
+ }
89
+ catch {
90
+ return {};
91
+ }
92
+ }
93
+ /**
94
+ * Resolves server URL from: flag → env → config → default
95
+ */
96
+ resolveServerUrl(flagValue) {
97
+ if (flagValue)
98
+ return flagValue;
99
+ if (process.env.TESTDROID_SERVER)
100
+ return process.env.TESTDROID_SERVER;
101
+ const config = this.loadConfig();
102
+ if (config.serverUrl)
103
+ return config.serverUrl;
104
+ return 'http://localhost:3001';
105
+ }
106
+ // ─── Encryption ──────────────────────────────────────────────────
107
+ getKey() {
108
+ this.ensureDir();
109
+ let salt;
110
+ if (fs.existsSync(SALT_FILE)) {
111
+ salt = fs.readFileSync(SALT_FILE);
112
+ }
113
+ else {
114
+ salt = crypto.randomBytes(16);
115
+ fs.writeFileSync(SALT_FILE, salt, { mode: 0o600 });
116
+ }
117
+ // Machine-stable identifier + random salt
118
+ const machineId = this.getMachineId();
119
+ return crypto.pbkdf2Sync(machineId, salt, 600_000, 32, 'sha256');
120
+ }
121
+ getMachineId() {
122
+ try {
123
+ if (process.platform === 'linux' && fs.existsSync('/etc/machine-id')) {
124
+ return fs.readFileSync('/etc/machine-id', 'utf-8').trim();
125
+ }
126
+ }
127
+ catch { }
128
+ return `${os.hostname()}-${os.userInfo().username}-${os.arch()}`;
129
+ }
130
+ encrypt(plaintext) {
131
+ const key = this.getKey();
132
+ const iv = crypto.randomBytes(12);
133
+ const cipher = crypto.createCipheriv('aes-256-gcm', key, iv);
134
+ const encrypted = Buffer.concat([
135
+ cipher.update(plaintext, 'utf-8'),
136
+ cipher.final(),
137
+ ]);
138
+ const tag = cipher.getAuthTag();
139
+ // Format: base64(iv + tag + encrypted)
140
+ return Buffer.concat([iv, tag, encrypted]).toString('base64');
141
+ }
142
+ decrypt(data) {
143
+ const key = this.getKey();
144
+ const buf = Buffer.from(data, 'base64');
145
+ const iv = buf.subarray(0, 12);
146
+ const tag = buf.subarray(12, 28);
147
+ const encrypted = buf.subarray(28);
148
+ const decipher = crypto.createDecipheriv('aes-256-gcm', key, iv);
149
+ decipher.setAuthTag(tag);
150
+ return decipher.update(encrypted) + decipher.final('utf-8');
151
+ }
152
+ }
153
+ exports.CredentialStore = CredentialStore;
154
+ //# sourceMappingURL=credential-store.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"credential-store.js","sourceRoot":"","sources":["../../src/auth/credential-store.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,26 @@
1
+ import { TokenManager } from './token-manager';
2
+ /**
3
+ * RFC 8628 Device Authorization Grant — client side.
4
+ * Handles: request code, display, auto-open browser, poll for approval.
5
+ */
6
+ export declare class PairingAuth {
7
+ private serverUrl;
8
+ private tokenManager;
9
+ private devices;
10
+ constructor(serverUrl: string, tokenManager: TokenManager);
11
+ pair(options?: {
12
+ noBrowser?: boolean;
13
+ devices?: Array<{
14
+ serial: string;
15
+ model: string;
16
+ osVersion: string;
17
+ type: string;
18
+ connectionType: string;
19
+ }>;
20
+ }): Promise<boolean>;
21
+ private deviceAuthorize;
22
+ private pollForToken;
23
+ private sleep;
24
+ private formatTime;
25
+ }
26
+ //# sourceMappingURL=pairing-auth.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pairing-auth.d.ts","sourceRoot":"","sources":["../../src/auth/pairing-auth.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAW/C;;;GAGG;AACH,qBAAa,WAAW;IAIpB,OAAO,CAAC,SAAS;IACjB,OAAO,CAAC,YAAY;IAJtB,OAAO,CAAC,OAAO,CAAyG;gBAG9G,SAAS,EAAE,MAAM,EACjB,YAAY,EAAE,YAAY;IAG9B,IAAI,CAAC,OAAO,GAAE;QAAE,SAAS,CAAC,EAAE,OAAO,CAAC;QAAC,OAAO,CAAC,EAAE,KAAK,CAAC;YAAE,MAAM,EAAE,MAAM,CAAC;YAAC,KAAK,EAAE,MAAM,CAAC;YAAC,SAAS,EAAE,MAAM,CAAC;YAAC,IAAI,EAAE,MAAM,CAAC;YAAC,cAAc,EAAE,MAAM,CAAA;SAAE,CAAC,CAAA;KAAO,GAAG,OAAO,CAAC,OAAO,CAAC;YA4ClK,eAAe;YAyCf,YAAY;IA0E1B,OAAO,CAAC,KAAK;IAIb,OAAO,CAAC,UAAU;CAKnB"}
@@ -0,0 +1,177 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
37
+ };
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ exports.PairingAuth = void 0;
40
+ const chalk_1 = __importDefault(require("chalk"));
41
+ const ora_1 = __importDefault(require("ora"));
42
+ /**
43
+ * RFC 8628 Device Authorization Grant — client side.
44
+ * Handles: request code, display, auto-open browser, poll for approval.
45
+ */
46
+ class PairingAuth {
47
+ serverUrl;
48
+ tokenManager;
49
+ devices = [];
50
+ constructor(serverUrl, tokenManager) {
51
+ this.serverUrl = serverUrl;
52
+ this.tokenManager = tokenManager;
53
+ }
54
+ async pair(options = {}) {
55
+ this.devices = options.devices || [];
56
+ // Step 1: Request device authorization
57
+ const authResponse = await this.deviceAuthorize();
58
+ if (!authResponse)
59
+ return false;
60
+ // Step 2: Display code and open browser
61
+ console.log('');
62
+ console.log(chalk_1.default.bold(` Pairing code: ${chalk_1.default.cyan(authResponse.user_code)}`));
63
+ console.log('');
64
+ if (!options.noBrowser) {
65
+ try {
66
+ const open = (await Promise.resolve().then(() => __importStar(require('open')))).default;
67
+ await open(authResponse.verification_uri_complete);
68
+ console.log(chalk_1.default.dim(' Browser opened.'));
69
+ }
70
+ catch {
71
+ console.log(chalk_1.default.dim(` Visit: ${authResponse.verification_uri_complete}`));
72
+ }
73
+ }
74
+ else {
75
+ console.log(` Visit: ${authResponse.verification_uri_complete}`);
76
+ console.log('');
77
+ console.log(` TESTDROID_PAIR_URL=${authResponse.verification_uri_complete}`);
78
+ console.log(` TESTDROID_PAIR_CODE=${authResponse.user_code}`);
79
+ }
80
+ console.log('');
81
+ // Step 3: Poll for approval
82
+ return this.pollForToken(authResponse);
83
+ }
84
+ async deviceAuthorize() {
85
+ try {
86
+ const res = await fetch(`${this.serverUrl}/api/v1/oauth/device/authorize`, {
87
+ method: 'POST',
88
+ headers: { 'Content-Type': 'application/json' },
89
+ body: JSON.stringify((() => {
90
+ const body = { client_id: 'testdroid-cli' };
91
+ if (this.devices.length > 0) {
92
+ const d = this.devices[0];
93
+ body.deviceInfo = {
94
+ serial: d.serial,
95
+ model: d.model || 'Unknown',
96
+ osVersion: d.osVersion || 'Unknown',
97
+ type: d.type || 'physical',
98
+ connectionType: d.connectionType || 'usb',
99
+ };
100
+ }
101
+ return body;
102
+ })()),
103
+ });
104
+ if (!res.ok) {
105
+ console.error(chalk_1.default.red('Failed to request pairing code'));
106
+ return null;
107
+ }
108
+ return (await res.json());
109
+ }
110
+ catch (err) {
111
+ console.error(chalk_1.default.red(`Cannot reach server at ${this.serverUrl}`));
112
+ console.error(chalk_1.default.dim('Check if the server is running and accessible.'));
113
+ return null;
114
+ }
115
+ }
116
+ async pollForToken(auth) {
117
+ const spinner = (0, ora_1.default)('Waiting for approval...').start();
118
+ let interval = auth.interval * 1000;
119
+ const deadline = Date.now() + auth.expires_in * 1000;
120
+ while (Date.now() < deadline) {
121
+ await this.sleep(interval);
122
+ const remaining = Math.ceil((deadline - Date.now()) / 1000);
123
+ spinner.text = `Waiting for approval... (${this.formatTime(remaining)})`;
124
+ try {
125
+ const res = await fetch(`${this.serverUrl}/api/v1/oauth/token`, {
126
+ method: 'POST',
127
+ headers: { 'Content-Type': 'application/json' },
128
+ body: JSON.stringify({
129
+ grant_type: 'urn:ietf:params:oauth:grant-type:device_code',
130
+ device_code: auth.device_code,
131
+ client_id: 'testdroid-cli',
132
+ }),
133
+ });
134
+ if (res.ok) {
135
+ const data = (await res.json());
136
+ spinner.succeed(chalk_1.default.green('Approved!'));
137
+ this.tokenManager.saveTokens(data.access_token, data.refresh_token, data.expires_in);
138
+ return true;
139
+ }
140
+ const error = (await res.json());
141
+ switch (error.error) {
142
+ case 'authorization_pending':
143
+ continue;
144
+ case 'slow_down':
145
+ interval += 5000;
146
+ continue;
147
+ case 'expired_token':
148
+ spinner.fail(chalk_1.default.red('Pairing code expired.'));
149
+ console.log(chalk_1.default.dim('Run `testdroid connect` to generate a new code.'));
150
+ return false;
151
+ case 'access_denied':
152
+ spinner.fail(chalk_1.default.red('Pairing denied by user.'));
153
+ return false;
154
+ default:
155
+ spinner.fail(chalk_1.default.red(`Pairing error: ${error.error}`));
156
+ return false;
157
+ }
158
+ }
159
+ catch {
160
+ // Network error — keep polling
161
+ continue;
162
+ }
163
+ }
164
+ spinner.fail(chalk_1.default.red('Pairing code expired.'));
165
+ return false;
166
+ }
167
+ sleep(ms) {
168
+ return new Promise((resolve) => setTimeout(resolve, ms));
169
+ }
170
+ formatTime(seconds) {
171
+ const m = Math.floor(seconds / 60);
172
+ const s = seconds % 60;
173
+ return `${m}:${s.toString().padStart(2, '0')}`;
174
+ }
175
+ }
176
+ exports.PairingAuth = PairingAuth;
177
+ //# sourceMappingURL=pairing-auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pairing-auth.js","sourceRoot":"","sources":["../../src/auth/pairing-auth.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,20 @@
1
+ /**
2
+ * Manages token lifecycle: storage, retrieval, auto-refresh.
3
+ */
4
+ export declare class TokenManager {
5
+ private serverUrl;
6
+ private store;
7
+ private refreshTimer;
8
+ private onRefreshFailed?;
9
+ constructor(serverUrl: string);
10
+ setOnRefreshFailed(cb: () => void): void;
11
+ saveTokens(accessToken: string, refreshToken: string, expiresIn: number): void;
12
+ getAccessToken(): string | null;
13
+ getRefreshToken(): string | null;
14
+ isAuthenticated(): boolean;
15
+ refresh(): Promise<boolean>;
16
+ clear(): void;
17
+ startAutoRefresh(): void;
18
+ private scheduleRefresh;
19
+ }
20
+ //# sourceMappingURL=token-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token-manager.d.ts","sourceRoot":"","sources":["../../src/auth/token-manager.ts"],"names":[],"mappings":"AAEA;;GAEG;AACH,qBAAa,YAAY;IAKX,OAAO,CAAC,SAAS;IAJ7B,OAAO,CAAC,KAAK,CAAyB;IACtC,OAAO,CAAC,YAAY,CAA+B;IACnD,OAAO,CAAC,eAAe,CAAC,CAAa;gBAEjB,SAAS,EAAE,MAAM;IAErC,kBAAkB,CAAC,EAAE,EAAE,MAAM,IAAI,GAAG,IAAI;IAIxC,UAAU,CACR,WAAW,EAAE,MAAM,EACnB,YAAY,EAAE,MAAM,EACpB,SAAS,EAAE,MAAM,GAChB,IAAI;IAWP,cAAc,IAAI,MAAM,GAAG,IAAI;IAO/B,eAAe,IAAI,MAAM,GAAG,IAAI;IAMhC,eAAe,IAAI,OAAO;IAIpB,OAAO,IAAI,OAAO,CAAC,OAAO,CAAC;IAgCjC,KAAK,IAAI,IAAI;IAQb,gBAAgB,IAAI,IAAI;IAUxB,OAAO,CAAC,eAAe;CAaxB"}