@jackwener/opencli 1.5.6 → 1.5.7

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 (334) hide show
  1. package/CHANGELOG.md +26 -0
  2. package/README.md +4 -2
  3. package/README.zh-CN.md +4 -1
  4. package/SKILL.md +879 -0
  5. package/dist/browser/cdp.d.ts +1 -0
  6. package/dist/browser/cdp.js +30 -27
  7. package/dist/browser/daemon-client.d.ts +7 -1
  8. package/dist/browser/daemon-client.js +3 -0
  9. package/dist/browser/dom-helpers.js +1 -0
  10. package/dist/browser/dom-helpers.test.js +14 -1
  11. package/dist/browser/mcp.js +18 -13
  12. package/dist/browser/page.js +22 -2
  13. package/dist/browser/page.test.d.ts +1 -0
  14. package/dist/browser/page.test.js +44 -0
  15. package/dist/browser/stealth.js +198 -0
  16. package/dist/browser/stealth.test.d.ts +1 -0
  17. package/dist/browser/stealth.test.js +134 -0
  18. package/dist/browser.test.js +1 -1
  19. package/dist/build-manifest.d.ts +1 -0
  20. package/dist/build-manifest.js +5 -1
  21. package/dist/build-manifest.test.js +2 -0
  22. package/dist/cli-manifest.json +544 -137
  23. package/dist/cli.js +20 -3
  24. package/dist/clis/antigravity/serve.d.ts +1 -1
  25. package/dist/clis/antigravity/serve.js +5 -8
  26. package/dist/clis/bilibili/subtitle.js +4 -0
  27. package/dist/clis/bilibili/subtitle.test.d.ts +1 -0
  28. package/dist/clis/bilibili/subtitle.test.js +48 -0
  29. package/dist/clis/chatwise/ask.js +0 -2
  30. package/dist/clis/chatwise/export.js +0 -2
  31. package/dist/clis/chatwise/history.js +0 -2
  32. package/dist/clis/chatwise/model.js +0 -2
  33. package/dist/clis/chatwise/new.js +1 -2
  34. package/dist/clis/chatwise/read.js +0 -2
  35. package/dist/clis/chatwise/screenshot.js +1 -2
  36. package/dist/clis/chatwise/send.js +0 -2
  37. package/dist/clis/chatwise/status.js +1 -2
  38. package/dist/clis/ctrip/search.d.ts +13 -0
  39. package/dist/clis/ctrip/search.js +73 -48
  40. package/dist/clis/ctrip/search.test.d.ts +1 -0
  41. package/dist/clis/ctrip/search.test.js +64 -0
  42. package/dist/clis/douyin/_shared/sts2.js +8 -2
  43. package/dist/clis/douyin/_shared/sts2.test.d.ts +1 -0
  44. package/dist/clis/douyin/_shared/sts2.test.js +27 -0
  45. package/dist/clis/douyin/activities.js +4 -2
  46. package/dist/clis/douyin/activities.test.js +34 -1
  47. package/dist/clis/douyin/collections.js +1 -1
  48. package/dist/clis/douyin/collections.test.js +24 -2
  49. package/dist/clis/douyin/draft.d.ts +8 -11
  50. package/dist/clis/douyin/draft.js +302 -185
  51. package/dist/clis/douyin/draft.test.d.ts +1 -1
  52. package/dist/clis/douyin/draft.test.js +357 -2
  53. package/dist/clis/douyin/hashtag.js +9 -2
  54. package/dist/clis/douyin/hashtag.test.js +35 -2
  55. package/dist/clis/douyin/profile.js +1 -1
  56. package/dist/clis/douyin/profile.test.js +36 -1
  57. package/dist/clis/douyin/videos.js +22 -5
  58. package/dist/clis/douyin/videos.test.js +45 -2
  59. package/dist/clis/facebook/search.test.d.ts +5 -0
  60. package/dist/clis/facebook/search.test.js +60 -0
  61. package/dist/clis/facebook/search.yaml +4 -3
  62. package/dist/clis/instagram/download.d.ts +16 -0
  63. package/dist/clis/instagram/download.js +225 -0
  64. package/dist/clis/instagram/download.test.d.ts +1 -0
  65. package/dist/clis/instagram/download.test.js +118 -0
  66. package/dist/clis/notebooklm/bind-current.d.ts +1 -0
  67. package/dist/clis/notebooklm/bind-current.js +29 -0
  68. package/dist/clis/notebooklm/bind-current.test.d.ts +1 -0
  69. package/dist/clis/notebooklm/bind-current.test.js +35 -0
  70. package/dist/clis/notebooklm/binding.test.d.ts +1 -0
  71. package/dist/clis/notebooklm/binding.test.js +44 -0
  72. package/dist/clis/notebooklm/compat.test.d.ts +3 -0
  73. package/dist/clis/notebooklm/compat.test.js +16 -0
  74. package/dist/clis/notebooklm/current.d.ts +1 -0
  75. package/dist/clis/notebooklm/current.js +28 -0
  76. package/dist/clis/notebooklm/get.d.ts +1 -0
  77. package/dist/clis/notebooklm/get.js +37 -0
  78. package/dist/clis/notebooklm/history.d.ts +1 -0
  79. package/dist/clis/notebooklm/history.js +25 -0
  80. package/dist/clis/notebooklm/history.test.d.ts +1 -0
  81. package/dist/clis/notebooklm/history.test.js +58 -0
  82. package/dist/clis/notebooklm/list.d.ts +1 -0
  83. package/dist/clis/notebooklm/list.js +35 -0
  84. package/dist/clis/notebooklm/note-list.d.ts +1 -0
  85. package/dist/clis/notebooklm/note-list.js +28 -0
  86. package/dist/clis/notebooklm/note-list.test.d.ts +1 -0
  87. package/dist/clis/notebooklm/note-list.test.js +56 -0
  88. package/dist/clis/notebooklm/notes-get.d.ts +1 -0
  89. package/dist/clis/notebooklm/notes-get.js +47 -0
  90. package/dist/clis/notebooklm/notes-get.test.d.ts +1 -0
  91. package/dist/clis/notebooklm/notes-get.test.js +72 -0
  92. package/dist/clis/notebooklm/rpc.d.ts +36 -0
  93. package/dist/clis/notebooklm/rpc.js +189 -0
  94. package/dist/clis/notebooklm/rpc.test.d.ts +1 -0
  95. package/dist/clis/notebooklm/rpc.test.js +105 -0
  96. package/dist/clis/notebooklm/shared.d.ts +87 -0
  97. package/dist/clis/notebooklm/shared.js +3 -0
  98. package/dist/clis/notebooklm/source-fulltext.d.ts +1 -0
  99. package/dist/clis/notebooklm/source-fulltext.js +44 -0
  100. package/dist/clis/notebooklm/source-fulltext.test.d.ts +1 -0
  101. package/dist/clis/notebooklm/source-fulltext.test.js +106 -0
  102. package/dist/clis/notebooklm/source-get.d.ts +1 -0
  103. package/dist/clis/notebooklm/source-get.js +40 -0
  104. package/dist/clis/notebooklm/source-get.test.d.ts +1 -0
  105. package/dist/clis/notebooklm/source-get.test.js +84 -0
  106. package/dist/clis/notebooklm/source-guide.d.ts +1 -0
  107. package/dist/clis/notebooklm/source-guide.js +44 -0
  108. package/dist/clis/notebooklm/source-guide.test.d.ts +1 -0
  109. package/dist/clis/notebooklm/source-guide.test.js +104 -0
  110. package/dist/clis/notebooklm/source-list.d.ts +1 -0
  111. package/dist/clis/notebooklm/source-list.js +30 -0
  112. package/dist/clis/notebooklm/status.d.ts +1 -0
  113. package/dist/clis/notebooklm/status.js +31 -0
  114. package/dist/clis/notebooklm/summary.d.ts +1 -0
  115. package/dist/clis/notebooklm/summary.js +30 -0
  116. package/dist/clis/notebooklm/summary.test.d.ts +1 -0
  117. package/dist/clis/notebooklm/summary.test.js +78 -0
  118. package/dist/clis/notebooklm/utils.d.ts +37 -0
  119. package/dist/clis/notebooklm/utils.js +739 -0
  120. package/dist/clis/notebooklm/utils.test.d.ts +1 -0
  121. package/dist/clis/notebooklm/utils.test.js +390 -0
  122. package/dist/clis/substack/utils.d.ts +4 -0
  123. package/dist/clis/substack/utils.js +8 -2
  124. package/dist/clis/substack/utils.test.d.ts +1 -0
  125. package/dist/clis/substack/utils.test.js +46 -0
  126. package/dist/clis/v2ex/hot.yaml +4 -1
  127. package/dist/clis/v2ex/latest.yaml +4 -1
  128. package/dist/clis/v2ex/topic.yaml +6 -1
  129. package/dist/clis/weixin/download.d.ts +9 -0
  130. package/dist/clis/weixin/download.js +76 -6
  131. package/dist/clis/weread/book.js +108 -2
  132. package/dist/clis/weread/commands.test.js +262 -152
  133. package/dist/clis/weread/utils.d.ts +10 -0
  134. package/dist/clis/weread/utils.js +27 -7
  135. package/dist/clis/xiaohongshu/comments.d.ts +3 -0
  136. package/dist/clis/xiaohongshu/comments.js +76 -17
  137. package/dist/clis/xiaohongshu/comments.test.js +70 -9
  138. package/dist/clis/xiaohongshu/download.d.ts +4 -1
  139. package/dist/clis/xiaohongshu/download.js +83 -22
  140. package/dist/clis/xiaohongshu/download.test.d.ts +1 -0
  141. package/dist/clis/xiaohongshu/download.test.js +75 -0
  142. package/dist/clis/xiaohongshu/note-helpers.d.ts +12 -0
  143. package/dist/clis/xiaohongshu/note-helpers.js +23 -0
  144. package/dist/clis/xiaohongshu/note.d.ts +7 -0
  145. package/dist/clis/xiaohongshu/note.js +76 -0
  146. package/dist/clis/xiaohongshu/note.test.d.ts +1 -0
  147. package/dist/clis/xiaohongshu/note.test.js +136 -0
  148. package/dist/clis/xiaohongshu/search.js +9 -0
  149. package/dist/clis/xiaohongshu/search.test.js +10 -4
  150. package/dist/clis/youtube/search.js +57 -17
  151. package/dist/clis/zhihu/question.js +19 -17
  152. package/dist/clis/zhihu/question.test.d.ts +1 -0
  153. package/dist/clis/zhihu/question.test.js +54 -0
  154. package/dist/commanderAdapter.js +9 -0
  155. package/dist/commanderAdapter.test.js +25 -0
  156. package/dist/commands/daemon.d.ts +9 -0
  157. package/dist/commands/daemon.js +124 -0
  158. package/dist/commands/daemon.test.d.ts +1 -0
  159. package/dist/commands/daemon.test.js +185 -0
  160. package/dist/completion.js +3 -1
  161. package/dist/constants.d.ts +2 -0
  162. package/dist/constants.js +2 -0
  163. package/dist/daemon.d.ts +1 -1
  164. package/dist/daemon.js +25 -14
  165. package/dist/daemon.test.d.ts +1 -0
  166. package/dist/daemon.test.js +65 -0
  167. package/dist/discovery.d.ts +9 -0
  168. package/dist/discovery.js +47 -2
  169. package/dist/electron-apps.d.ts +29 -0
  170. package/dist/electron-apps.js +65 -0
  171. package/dist/electron-apps.test.d.ts +1 -0
  172. package/dist/electron-apps.test.js +43 -0
  173. package/dist/engine.test.js +41 -9
  174. package/dist/execution.js +20 -16
  175. package/dist/idle-manager.d.ts +19 -0
  176. package/dist/idle-manager.js +54 -0
  177. package/dist/launcher.d.ts +36 -0
  178. package/dist/launcher.js +152 -0
  179. package/dist/launcher.test.d.ts +1 -0
  180. package/dist/launcher.test.js +57 -0
  181. package/dist/main.js +3 -3
  182. package/dist/registry.d.ts +1 -0
  183. package/dist/registry.js +31 -3
  184. package/dist/registry.test.js +13 -0
  185. package/dist/runtime.d.ts +5 -3
  186. package/dist/runtime.js +12 -5
  187. package/dist/serialization.d.ts +1 -0
  188. package/dist/serialization.js +3 -0
  189. package/dist/serialization.test.js +17 -1
  190. package/dist/tui.d.ts +7 -0
  191. package/dist/tui.js +52 -0
  192. package/dist/tui.test.d.ts +1 -0
  193. package/dist/tui.test.js +19 -0
  194. package/dist/weixin-download.test.js +14 -0
  195. package/docs/.vitepress/config.mts +1 -0
  196. package/docs/adapters/browser/notebooklm.md +69 -0
  197. package/docs/adapters/browser/xiaohongshu.md +19 -10
  198. package/docs/adapters/index.md +67 -66
  199. package/docs/guide/browser-bridge.md +12 -0
  200. package/docs/guide/troubleshooting.md +9 -4
  201. package/docs/superpowers/plans/2026-03-31-daemon-lifecycle-redesign.md +857 -0
  202. package/docs/superpowers/specs/2026-03-31-daemon-lifecycle-redesign.md +208 -0
  203. package/docs/zh/guide/browser-bridge.md +12 -0
  204. package/extension/dist/background.js +794 -513
  205. package/extension/src/background.test.ts +202 -2
  206. package/extension/src/background.ts +174 -10
  207. package/extension/src/cdp.ts +12 -0
  208. package/extension/src/protocol.ts +7 -5
  209. package/package.json +1 -1
  210. package/src/browser/cdp.ts +24 -17
  211. package/src/browser/daemon-client.ts +7 -1
  212. package/src/browser/dom-helpers.test.ts +15 -1
  213. package/src/browser/dom-helpers.ts +1 -0
  214. package/src/browser/mcp.ts +18 -13
  215. package/src/browser/page.test.ts +58 -0
  216. package/src/browser/page.ts +18 -2
  217. package/src/browser/stealth.test.ts +153 -0
  218. package/src/browser/stealth.ts +198 -0
  219. package/src/browser.test.ts +1 -1
  220. package/src/build-manifest.test.ts +2 -0
  221. package/src/build-manifest.ts +6 -1
  222. package/src/cli.ts +21 -3
  223. package/src/clis/antigravity/SKILL.md +3 -12
  224. package/src/clis/antigravity/serve.ts +5 -10
  225. package/src/clis/bilibili/subtitle.test.ts +60 -0
  226. package/src/clis/bilibili/subtitle.ts +4 -0
  227. package/src/clis/chatwise/ask.ts +0 -2
  228. package/src/clis/chatwise/export.ts +0 -2
  229. package/src/clis/chatwise/history.ts +0 -2
  230. package/src/clis/chatwise/model.ts +0 -2
  231. package/src/clis/chatwise/new.ts +1 -2
  232. package/src/clis/chatwise/read.ts +0 -2
  233. package/src/clis/chatwise/screenshot.ts +1 -2
  234. package/src/clis/chatwise/send.ts +0 -2
  235. package/src/clis/chatwise/status.ts +1 -2
  236. package/src/clis/ctrip/search.test.ts +73 -0
  237. package/src/clis/ctrip/search.ts +97 -47
  238. package/src/clis/douyin/_shared/sts2.test.ts +31 -0
  239. package/src/clis/douyin/_shared/sts2.ts +11 -3
  240. package/src/clis/douyin/activities.test.ts +41 -1
  241. package/src/clis/douyin/activities.ts +12 -3
  242. package/src/clis/douyin/collections.test.ts +35 -2
  243. package/src/clis/douyin/collections.ts +1 -1
  244. package/src/clis/douyin/draft.test.ts +444 -2
  245. package/src/clis/douyin/draft.ts +382 -218
  246. package/src/clis/douyin/hashtag.test.ts +42 -2
  247. package/src/clis/douyin/hashtag.ts +11 -3
  248. package/src/clis/douyin/profile.test.ts +43 -1
  249. package/src/clis/douyin/profile.ts +9 -2
  250. package/src/clis/douyin/videos.test.ts +52 -2
  251. package/src/clis/douyin/videos.ts +49 -15
  252. package/src/clis/facebook/search.test.ts +70 -0
  253. package/src/clis/facebook/search.yaml +4 -3
  254. package/src/clis/instagram/download.test.ts +159 -0
  255. package/src/clis/instagram/download.ts +286 -0
  256. package/src/clis/notebooklm/bind-current.test.ts +43 -0
  257. package/src/clis/notebooklm/bind-current.ts +36 -0
  258. package/src/clis/notebooklm/binding.test.ts +53 -0
  259. package/src/clis/notebooklm/compat.test.ts +19 -0
  260. package/src/clis/notebooklm/current.ts +38 -0
  261. package/src/clis/notebooklm/get.ts +53 -0
  262. package/src/clis/notebooklm/history.test.ts +70 -0
  263. package/src/clis/notebooklm/history.ts +36 -0
  264. package/src/clis/notebooklm/list.ts +40 -0
  265. package/src/clis/notebooklm/note-list.test.ts +64 -0
  266. package/src/clis/notebooklm/note-list.ts +42 -0
  267. package/src/clis/notebooklm/notes-get.test.ts +88 -0
  268. package/src/clis/notebooklm/notes-get.ts +67 -0
  269. package/src/clis/notebooklm/rpc.test.ts +126 -0
  270. package/src/clis/notebooklm/rpc.ts +286 -0
  271. package/src/clis/notebooklm/shared.ts +98 -0
  272. package/src/clis/notebooklm/source-fulltext.test.ts +123 -0
  273. package/src/clis/notebooklm/source-fulltext.ts +69 -0
  274. package/src/clis/notebooklm/source-get.test.ts +100 -0
  275. package/src/clis/notebooklm/source-get.ts +60 -0
  276. package/src/clis/notebooklm/source-guide.test.ts +121 -0
  277. package/src/clis/notebooklm/source-guide.ts +69 -0
  278. package/src/clis/notebooklm/source-list.ts +45 -0
  279. package/src/clis/notebooklm/status.ts +34 -0
  280. package/src/clis/notebooklm/summary.test.ts +94 -0
  281. package/src/clis/notebooklm/summary.ts +45 -0
  282. package/src/clis/notebooklm/utils.test.ts +446 -0
  283. package/src/clis/notebooklm/utils.ts +893 -0
  284. package/src/clis/substack/utils.test.ts +54 -0
  285. package/src/clis/substack/utils.ts +10 -2
  286. package/src/clis/v2ex/hot.yaml +4 -1
  287. package/src/clis/v2ex/latest.yaml +4 -1
  288. package/src/clis/v2ex/topic.yaml +6 -1
  289. package/src/clis/weixin/download.ts +95 -6
  290. package/src/clis/weread/book.ts +142 -2
  291. package/src/clis/weread/commands.test.ts +314 -154
  292. package/src/clis/weread/utils.ts +33 -4
  293. package/src/clis/xiaohongshu/comments.test.ts +85 -9
  294. package/src/clis/xiaohongshu/comments.ts +76 -17
  295. package/src/clis/xiaohongshu/download.test.ts +96 -0
  296. package/src/clis/xiaohongshu/download.ts +83 -22
  297. package/src/clis/xiaohongshu/note-helpers.ts +25 -0
  298. package/src/clis/xiaohongshu/note.test.ts +164 -0
  299. package/src/clis/xiaohongshu/note.ts +86 -0
  300. package/src/clis/xiaohongshu/search.test.ts +11 -4
  301. package/src/clis/xiaohongshu/search.ts +13 -0
  302. package/src/clis/youtube/search.ts +57 -17
  303. package/src/clis/zhihu/question.test.ts +71 -0
  304. package/src/clis/zhihu/question.ts +27 -15
  305. package/src/commanderAdapter.test.ts +30 -0
  306. package/src/commanderAdapter.ts +7 -0
  307. package/src/commands/daemon.test.ts +238 -0
  308. package/src/commands/daemon.ts +135 -0
  309. package/src/completion.ts +2 -1
  310. package/src/constants.ts +3 -0
  311. package/src/daemon.test.ts +88 -0
  312. package/src/daemon.ts +26 -14
  313. package/src/discovery.ts +52 -2
  314. package/src/electron-apps.test.ts +50 -0
  315. package/src/electron-apps.ts +89 -0
  316. package/src/engine.test.ts +45 -9
  317. package/src/execution.ts +24 -19
  318. package/src/idle-manager.ts +60 -0
  319. package/src/launcher.test.ts +67 -0
  320. package/src/launcher.ts +185 -0
  321. package/src/main.ts +3 -2
  322. package/src/registry.test.ts +15 -0
  323. package/src/registry.ts +32 -3
  324. package/src/runtime.ts +13 -7
  325. package/src/serialization.test.ts +19 -1
  326. package/src/serialization.ts +2 -0
  327. package/src/tui.test.ts +23 -0
  328. package/src/tui.ts +65 -0
  329. package/src/weixin-download.test.ts +27 -0
  330. package/tests/e2e/browser-public-extended.test.ts +6 -2
  331. package/chatwise-opencli.ps1 +0 -82
  332. package/dist/clis/chatwise/shared.d.ts +0 -2
  333. package/dist/clis/chatwise/shared.js +0 -6
  334. package/src/clis/chatwise/shared.ts +0 -8
@@ -23,6 +23,7 @@ export declare class CDPBridge implements IBrowserFactory {
23
23
  connect(opts?: {
24
24
  timeout?: number;
25
25
  workspace?: string;
26
+ cdpEndpoint?: string;
26
27
  }): Promise<IPage>;
27
28
  close(): Promise<void>;
28
29
  send(method: string, params?: Record<string, unknown>, timeoutMs?: number): Promise<unknown>;
@@ -15,6 +15,7 @@ import { generateSnapshotJs, scrollToRefJs, getFormStateJs } from './dom-snapsho
15
15
  import { generateStealthJs } from './stealth.js';
16
16
  import { clickJs, typeTextJs, pressKeyJs, waitForTextJs, scrollJs, autoScrollJs, networkRequestsJs, waitForDomStableJs, waitForCaptureJs, waitForSelectorJs, } from './dom-helpers.js';
17
17
  import { isRecord, saveBase64ToFile } from '../utils.js';
18
+ import { getAllElectronApps } from '../electron-apps.js';
18
19
  const CDP_SEND_TIMEOUT = 30_000;
19
20
  export class CDPBridge {
20
21
  _ws = null;
@@ -24,9 +25,9 @@ export class CDPBridge {
24
25
  async connect(opts) {
25
26
  if (this._ws)
26
27
  throw new Error('CDPBridge is already connected. Call close() before reconnecting.');
27
- const endpoint = process.env.OPENCLI_CDP_ENDPOINT;
28
+ const endpoint = opts?.cdpEndpoint ?? process.env.OPENCLI_CDP_ENDPOINT;
28
29
  if (!endpoint)
29
- throw new Error('OPENCLI_CDP_ENDPOINT is not set');
30
+ throw new Error('CDP endpoint not provided (pass cdpEndpoint or set OPENCLI_CDP_ENDPOINT)');
30
31
  let wsUrl = endpoint;
31
32
  if (endpoint.startsWith('http')) {
32
33
  const targets = await fetchJsonDirect(`${endpoint.replace(/\/$/, '')}/json`);
@@ -269,7 +270,19 @@ class CDPPage {
269
270
  return [];
270
271
  }
271
272
  async getCurrentUrl() {
272
- return this._lastUrl;
273
+ if (this._lastUrl)
274
+ return this._lastUrl;
275
+ try {
276
+ const current = await this.evaluate('window.location.href');
277
+ if (typeof current === 'string' && current) {
278
+ this._lastUrl = current;
279
+ return current;
280
+ }
281
+ }
282
+ catch {
283
+ // Best-effort: direct CDP sessions may not have a ready page yet.
284
+ }
285
+ return null;
273
286
  }
274
287
  async installInterceptor(pattern) {
275
288
  const { generateInterceptorJs } = await import('../interceptor.js');
@@ -346,30 +359,20 @@ function scoreCDPTarget(target, preferredPattern) {
346
359
  score -= 40;
347
360
  if (title && title !== 'devtools')
348
361
  score += 25;
349
- if (title.includes('antigravity'))
350
- score += 120;
351
- if (title.includes('codex'))
352
- score += 120;
353
- if (title.includes('cursor'))
354
- score += 120;
355
- if (title.includes('chatwise'))
356
- score += 120;
357
- if (title.includes('notion'))
358
- score += 120;
359
- if (title.includes('discord'))
360
- score += 120;
361
- if (url.includes('antigravity'))
362
- score += 100;
363
- if (url.includes('codex'))
364
- score += 100;
365
- if (url.includes('cursor'))
366
- score += 100;
367
- if (url.includes('chatwise'))
368
- score += 100;
369
- if (url.includes('notion'))
370
- score += 100;
371
- if (url.includes('discord'))
372
- score += 100;
362
+ // Boost score for known Electron app names from the registry (builtin + user-defined)
363
+ const appNames = Object.values(getAllElectronApps()).map(a => (a.displayName ?? a.processName).toLowerCase());
364
+ for (const name of appNames) {
365
+ if (title.includes(name)) {
366
+ score += 120;
367
+ break;
368
+ }
369
+ }
370
+ for (const name of appNames) {
371
+ if (url.includes(name)) {
372
+ score += 100;
373
+ break;
374
+ }
375
+ }
373
376
  return score;
374
377
  }
375
378
  function compilePreferredPattern(raw) {
@@ -6,7 +6,7 @@
6
6
  import type { BrowserSessionInfo } from '../types.js';
7
7
  export interface DaemonCommand {
8
8
  id: string;
9
- action: 'exec' | 'navigate' | 'tabs' | 'cookies' | 'screenshot' | 'close-window' | 'sessions' | 'set-file-input';
9
+ action: 'exec' | 'navigate' | 'tabs' | 'cookies' | 'screenshot' | 'close-window' | 'sessions' | 'set-file-input' | 'bind-current';
10
10
  tabId?: number;
11
11
  code?: string;
12
12
  workspace?: string;
@@ -14,6 +14,8 @@ export interface DaemonCommand {
14
14
  op?: string;
15
15
  index?: number;
16
16
  domain?: string;
17
+ matchDomain?: string;
18
+ matchPathPrefix?: string;
17
19
  format?: 'png' | 'jpeg';
18
20
  quality?: number;
19
21
  fullPage?: boolean;
@@ -43,3 +45,7 @@ export declare function isExtensionConnected(): Promise<boolean>;
43
45
  */
44
46
  export declare function sendCommand(action: DaemonCommand['action'], params?: Omit<DaemonCommand, 'id' | 'action'>): Promise<unknown>;
45
47
  export declare function listSessions(): Promise<BrowserSessionInfo[]>;
48
+ export declare function bindCurrentTab(workspace: string, opts?: {
49
+ matchDomain?: string;
50
+ matchPathPrefix?: string;
51
+ }): Promise<unknown>;
@@ -105,3 +105,6 @@ export async function listSessions() {
105
105
  const result = await sendCommand('sessions');
106
106
  return Array.isArray(result) ? result : [];
107
107
  }
108
+ export async function bindCurrentTab(workspace, opts = {}) {
109
+ return sendCommand('bind-current', { workspace, ...opts });
110
+ }
@@ -103,6 +103,7 @@ export function scrollJs(direction, amount) {
103
103
  export function autoScrollJs(times, delayMs) {
104
104
  return `
105
105
  (async () => {
106
+ if (!document.body) return;
106
107
  for (let i = 0; i < ${times}; i++) {
107
108
  const lastHeight = document.body.scrollHeight;
108
109
  window.scrollTo(0, lastHeight);
@@ -1,5 +1,18 @@
1
1
  import { describe, it, expect } from 'vitest';
2
- import { waitForCaptureJs, waitForSelectorJs } from './dom-helpers.js';
2
+ import { autoScrollJs, waitForCaptureJs, waitForSelectorJs } from './dom-helpers.js';
3
+ describe('autoScrollJs', () => {
4
+ it('returns early without error when document.body is null', async () => {
5
+ const g = globalThis;
6
+ const origDoc = g.document;
7
+ g.document = { body: null, documentElement: {} };
8
+ g.window = g;
9
+ const code = autoScrollJs(3, 500);
10
+ // Should resolve without throwing
11
+ await expect(eval(code)).resolves.not.toThrow();
12
+ g.document = origDoc;
13
+ delete g.window;
14
+ });
15
+ });
3
16
  describe('waitForCaptureJs', () => {
4
17
  it('returns a non-empty string', () => {
5
18
  const code = waitForCaptureJs(1000);
@@ -50,29 +50,36 @@ export class BrowserBridge {
50
50
  this._state = 'closed';
51
51
  }
52
52
  async _ensureDaemon(timeoutSeconds) {
53
- // Use default if not provided, zero, or negative
54
53
  const effectiveSeconds = (timeoutSeconds && timeoutSeconds > 0) ? timeoutSeconds : Math.ceil(DAEMON_SPAWN_TIMEOUT / 1000);
55
54
  const timeoutMs = effectiveSeconds * 1000;
55
+ // Fast path: extension already connected
56
56
  if (await isExtensionConnected())
57
57
  return;
58
+ // Daemon running but no extension — wait for extension with progress
58
59
  if (await isDaemonRunning()) {
60
+ if (process.env.OPENCLI_VERBOSE || process.stderr.isTTY) {
61
+ process.stderr.write('⏳ Waiting for Chrome extension to connect...\n');
62
+ process.stderr.write(' Make sure Chrome is open and the OpenCLI extension is enabled.\n');
63
+ }
64
+ const deadline = Date.now() + timeoutMs;
65
+ while (Date.now() < deadline) {
66
+ await new Promise(resolve => setTimeout(resolve, 200));
67
+ if (await isExtensionConnected())
68
+ return;
69
+ }
59
70
  throw new Error('Daemon is running but the Browser Extension is not connected.\n' +
60
71
  'Please install and enable the opencli Browser Bridge extension in Chrome.');
61
72
  }
62
- // Find daemon relative to this file works for both:
63
- // npx tsx src/main.ts → src/browser/mcp.ts → src/daemon.ts
64
- // node dist/main.js → dist/browser/mcp.js → dist/daemon.js
73
+ // No daemon — spawn one
65
74
  const __dirname = path.dirname(fileURLToPath(import.meta.url));
66
75
  const parentDir = path.resolve(__dirname, '..');
67
76
  const daemonTs = path.join(parentDir, 'daemon.ts');
68
77
  const daemonJs = path.join(parentDir, 'daemon.js');
69
78
  const isTs = fs.existsSync(daemonTs);
70
79
  const daemonPath = isTs ? daemonTs : daemonJs;
71
- if (process.env.OPENCLI_VERBOSE) {
72
- console.error(`[opencli] Starting daemon (${isTs ? 'ts' : 'js'})...`);
80
+ if (process.env.OPENCLI_VERBOSE || process.stderr.isTTY) {
81
+ process.stderr.write('⏳ Starting daemon...\n');
73
82
  }
74
- // For compiled .js, use the current node binary directly (fast).
75
- // For .ts dev mode, node can't run .ts files — use tsx via --import.
76
83
  const spawnArgs = isTs
77
84
  ? [process.execPath, '--import', 'tsx/esm', daemonPath]
78
85
  : [process.execPath, daemonPath];
@@ -82,15 +89,13 @@ export class BrowserBridge {
82
89
  env: { ...process.env },
83
90
  });
84
91
  this._daemonProc.unref();
85
- // Wait for daemon to be ready AND extension to connect (exponential backoff)
86
- const backoffs = [50, 100, 200, 400, 800, 1500, 3000];
92
+ // Wait for daemon + extension with faster polling
87
93
  const deadline = Date.now() + timeoutMs;
88
- for (let i = 0; Date.now() < deadline; i++) {
89
- await new Promise(resolve => setTimeout(resolve, backoffs[Math.min(i, backoffs.length - 1)]));
94
+ while (Date.now() < deadline) {
95
+ await new Promise(resolve => setTimeout(resolve, 200));
90
96
  if (await isExtensionConnected())
91
97
  return;
92
98
  }
93
- // Daemon might be up but extension not connected — give a useful error
94
99
  if (await isDaemonRunning()) {
95
100
  throw new Error('Daemon is running but the Browser Extension is not connected.\n' +
96
101
  'Please install and enable the opencli Browser Bridge extension in Chrome.');
@@ -96,7 +96,19 @@ export class Page {
96
96
  }
97
97
  }
98
98
  async getCurrentUrl() {
99
- return this._lastUrl;
99
+ if (this._lastUrl)
100
+ return this._lastUrl;
101
+ try {
102
+ const current = await this.evaluate('window.location.href');
103
+ if (typeof current === 'string' && current) {
104
+ this._lastUrl = current;
105
+ return current;
106
+ }
107
+ }
108
+ catch {
109
+ // Best-effort: some commands may run before a debuggable tab is ready.
110
+ }
111
+ return null;
100
112
  }
101
113
  /** Close the automation window in the extension */
102
114
  async closeWindow() {
@@ -109,7 +121,15 @@ export class Page {
109
121
  }
110
122
  async evaluate(js) {
111
123
  const code = wrapForEval(js);
112
- return sendCommand('exec', { code, ...this._cmdOpts() });
124
+ try {
125
+ return await sendCommand('exec', { code, ...this._cmdOpts() });
126
+ }
127
+ catch (err) {
128
+ if (!isRetryableSettleError(err))
129
+ throw err;
130
+ await new Promise((resolve) => setTimeout(resolve, 200));
131
+ return sendCommand('exec', { code, ...this._cmdOpts() });
132
+ }
113
133
  }
114
134
  async getCookies(opts = {}) {
115
135
  const result = await sendCommand('cookies', { ...this._wsOpt(), ...opts });
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,44 @@
1
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
2
+ const { sendCommandMock } = vi.hoisted(() => ({
3
+ sendCommandMock: vi.fn(),
4
+ }));
5
+ vi.mock('./daemon-client.js', () => ({
6
+ sendCommand: sendCommandMock,
7
+ }));
8
+ import { Page } from './page.js';
9
+ describe('Page.getCurrentUrl', () => {
10
+ beforeEach(() => {
11
+ sendCommandMock.mockReset();
12
+ });
13
+ it('reads the real browser URL when no local navigation cache exists', async () => {
14
+ sendCommandMock.mockResolvedValueOnce('https://notebooklm.google.com/notebook/nb-live');
15
+ const page = new Page('site:notebooklm');
16
+ const url = await page.getCurrentUrl();
17
+ expect(url).toBe('https://notebooklm.google.com/notebook/nb-live');
18
+ expect(sendCommandMock).toHaveBeenCalledTimes(1);
19
+ expect(sendCommandMock).toHaveBeenCalledWith('exec', expect.objectContaining({
20
+ workspace: 'site:notebooklm',
21
+ }));
22
+ });
23
+ it('caches the discovered browser URL for later reads', async () => {
24
+ sendCommandMock.mockResolvedValueOnce('https://notebooklm.google.com/notebook/nb-live');
25
+ const page = new Page('site:notebooklm');
26
+ expect(await page.getCurrentUrl()).toBe('https://notebooklm.google.com/notebook/nb-live');
27
+ expect(await page.getCurrentUrl()).toBe('https://notebooklm.google.com/notebook/nb-live');
28
+ expect(sendCommandMock).toHaveBeenCalledTimes(1);
29
+ });
30
+ });
31
+ describe('Page.evaluate', () => {
32
+ beforeEach(() => {
33
+ sendCommandMock.mockReset();
34
+ });
35
+ it('retries once when the inspected target navigated during exec', async () => {
36
+ sendCommandMock
37
+ .mockRejectedValueOnce(new Error('{"code":-32000,"message":"Inspected target navigated or closed"}'))
38
+ .mockResolvedValueOnce(42);
39
+ const page = new Page('site:notebooklm');
40
+ const value = await page.evaluate('21 + 21');
41
+ expect(value).toBe(42);
42
+ expect(sendCommandMock).toHaveBeenCalledTimes(2);
43
+ });
44
+ });
@@ -149,6 +149,204 @@ export function generateStealthJs() {
149
149
  }
150
150
  } catch {}
151
151
 
152
+ // ── Shared toString disguise infrastructure ──
153
+ // Save the pristine Function.prototype.toString BEFORE any patches,
154
+ // so all subsequent disguises use the real native reference.
155
+ // Anti-bot scripts detect per-instance toString overrides via:
156
+ // Function.hasOwnProperty('toString') → true if patched
157
+ // Function.prototype.toString.call(fn) !== fn.toString()
158
+ // Instead we patch Function.prototype.toString once with a WeakMap
159
+ // lookup, making disguised functions indistinguishable from native.
160
+ const _origToString = Function.prototype.toString;
161
+ const _disguised = new WeakMap();
162
+ try {
163
+ Object.defineProperty(Function.prototype, 'toString', {
164
+ value: function() {
165
+ const override = _disguised.get(this);
166
+ return override !== undefined ? override : _origToString.call(this);
167
+ },
168
+ writable: true, configurable: true,
169
+ });
170
+ } catch {}
171
+ const _disguise = (fn, name) => {
172
+ _disguised.set(fn, 'function ' + name + '() { [native code] }');
173
+ try { Object.defineProperty(fn, 'name', { value: name, configurable: true }); } catch {}
174
+ return fn;
175
+ };
176
+
177
+ // 8. Anti-debugger statement trap
178
+ // Sites inject debugger statements to detect DevTools/CDP.
179
+ // When a CDP debugger is attached, the statement pauses execution
180
+ // and the site measures the time gap to confirm automation.
181
+ // We neutralize this by overriding the Function constructor and
182
+ // eval to strip debugger statements from dynamically created code.
183
+ // Note: this does NOT affect static debugger statements in parsed
184
+ // scripts — those require CDP Debugger.setBreakpointsActive(false)
185
+ // which we handle at the extension level.
186
+ // Caveat: the regex targets standalone debugger statements (preceded
187
+ // by a statement boundary) to minimise false positives inside string
188
+ // literals, but cannot perfectly distinguish all cases without a
189
+ // full parser. This is an acceptable trade-off for stealth code.
190
+ try {
191
+ const _OrigFunction = Function;
192
+ // Match standalone debugger statements preceded by a statement
193
+ // boundary (start of string, semicolon, brace, or newline).
194
+ // This avoids most false positives inside string literals like
195
+ // "use debugger mode" while still catching the anti-bot patterns.
196
+ const _debuggerRe = /(?:^|(?<=[;{}\\n\\r]))\\s*debugger\\s*;?/g;
197
+ const _cleanDebugger = (src) => typeof src === 'string' ? src.replace(_debuggerRe, '') : src;
198
+ // Patch Function constructor to strip debugger from dynamic code.
199
+ // Support both Function('code') and new Function('code') via
200
+ // new.target / Reflect.construct.
201
+ const _PatchedFunction = function(...args) {
202
+ if (args.length > 0) {
203
+ args[args.length - 1] = _cleanDebugger(args[args.length - 1]);
204
+ }
205
+ if (new.target) {
206
+ return Reflect.construct(_OrigFunction, args, new.target);
207
+ }
208
+ return _OrigFunction.apply(this, args);
209
+ };
210
+ _PatchedFunction.prototype = _OrigFunction.prototype;
211
+ Object.setPrototypeOf(_PatchedFunction, _OrigFunction);
212
+ _disguise(_PatchedFunction, 'Function');
213
+ try { window.Function = _PatchedFunction; } catch {}
214
+
215
+ // Patch eval to strip debugger
216
+ const _origEval = window.eval;
217
+ const _patchedEval = function(code) {
218
+ return _origEval.call(this, _cleanDebugger(code));
219
+ };
220
+ _disguise(_patchedEval, 'eval');
221
+ try { window.eval = _patchedEval; } catch {}
222
+ } catch {}
223
+
224
+ // 9. Console method fingerprinting defense
225
+ // When CDP Runtime.enable is called, Chrome replaces console.log etc.
226
+ // with CDP-bound versions. These bound functions have a different
227
+ // toString() output: "function log() { [native code] }" becomes
228
+ // something like "function () { [native code] }" (no name) or the
229
+ // bound function signature leaks. Anti-bot scripts check:
230
+ // console.log.toString().includes('[native code]')
231
+ // console.log.name === 'log'
232
+ // We re-wrap console methods and register them via the shared
233
+ // _disguise infrastructure so Function.prototype.toString.call()
234
+ // also returns the correct native string.
235
+ try {
236
+ const _consoleMethods = ['log', 'warn', 'error', 'info', 'debug', 'table', 'trace', 'dir', 'group', 'groupEnd', 'groupCollapsed', 'clear', 'count', 'assert', 'profile', 'profileEnd', 'time', 'timeEnd', 'timeStamp'];
237
+ for (const _m of _consoleMethods) {
238
+ if (typeof console[_m] !== 'function') continue;
239
+ const _origMethod = console[_m];
240
+ const _nativeStr = 'function ' + _m + '() { [native code] }';
241
+ // Only patch if toString is wrong (i.e. CDP has replaced it)
242
+ try {
243
+ const _currentStr = _origToString.call(_origMethod);
244
+ if (_currentStr === _nativeStr) continue; // already looks native
245
+ } catch {}
246
+ const _wrapper = function() { return _origMethod.apply(console, arguments); };
247
+ Object.defineProperty(_wrapper, 'length', { value: _origMethod.length || 0, configurable: true });
248
+ _disguise(_wrapper, _m);
249
+ try { console[_m] = _wrapper; } catch {}
250
+ }
251
+ } catch {}
252
+
253
+ // 10. window.outerWidth/outerHeight defense
254
+ // When DevTools or CDP debugger is attached, Chrome may alter the
255
+ // window dimensions. Anti-bot scripts compare outerWidth/innerWidth
256
+ // and outerHeight/innerHeight — a significant difference indicates
257
+ // DevTools is open. We freeze the relationship so the delta stays
258
+ // consistent with a normal browser window.
259
+ // Thresholds: width delta > 100px or height delta > 200px indicates
260
+ // a docked DevTools panel. When triggered, we report outerWidth
261
+ // equal to innerWidth (normal for maximised windows) and
262
+ // outerHeight as innerHeight + the captured "normal" delta (capped
263
+ // to a reasonable range), so the result is plausible across OSes.
264
+ try {
265
+ const _normalWidthDelta = window.outerWidth - window.innerWidth;
266
+ const _normalHeightDelta = window.outerHeight - window.innerHeight;
267
+ // Only patch if the delta looks suspicious (e.g. DevTools docked)
268
+ if (_normalWidthDelta > 100 || _normalHeightDelta > 200) {
269
+ Object.defineProperty(window, 'outerWidth', {
270
+ get: () => window.innerWidth,
271
+ configurable: true,
272
+ });
273
+ // Use a clamped height offset (40-120px covers macOS ~78px,
274
+ // Windows ~40px, and Linux ~37-50px title bar heights).
275
+ const _heightOffset = Math.max(40, Math.min(120, _normalHeightDelta));
276
+ Object.defineProperty(window, 'outerHeight', {
277
+ get: () => window.innerHeight + _heightOffset,
278
+ configurable: true,
279
+ });
280
+ }
281
+ } catch {}
282
+
283
+ // 11. Performance API cleanup
284
+ // CDP injects internal resources and timing entries that don't exist
285
+ // in normal browsing. Filter entries with debugger/devtools URLs.
286
+ try {
287
+ const _origGetEntries = Performance.prototype.getEntries;
288
+ const _origGetByType = Performance.prototype.getEntriesByType;
289
+ const _origGetByName = Performance.prototype.getEntriesByName;
290
+ const _suspiciousPatterns = ['debugger', 'devtools', '__puppeteer', '__playwright', 'pptr:'];
291
+ const _filterEntries = (entries) => {
292
+ if (!Array.isArray(entries)) return entries;
293
+ return entries.filter(e => {
294
+ const name = e.name || '';
295
+ return !_suspiciousPatterns.some(p => name.includes(p));
296
+ });
297
+ };
298
+ Performance.prototype.getEntries = function() {
299
+ return _filterEntries(_origGetEntries.call(this));
300
+ };
301
+ Performance.prototype.getEntriesByType = function(type) {
302
+ return _filterEntries(_origGetByType.call(this, type));
303
+ };
304
+ Performance.prototype.getEntriesByName = function(name, type) {
305
+ return _filterEntries(_origGetByName.call(this, name, type));
306
+ };
307
+ } catch {}
308
+
309
+ // 12. WebDriver-related property defense
310
+ // Some anti-bot systems check additional navigator properties
311
+ // and document properties that may indicate automation.
312
+ try {
313
+ // document.$cdc_ properties (ChromeDriver specific, backup for #6)
314
+ for (const _prop of Object.getOwnPropertyNames(document)) {
315
+ if (_prop.startsWith('$cdc_') || _prop.startsWith('$chrome_')) {
316
+ try { delete document[_prop]; } catch {}
317
+ }
318
+ }
319
+ } catch {}
320
+
321
+ // 13. Iframe contentWindow.chrome consistency
322
+ // Anti-bot scripts create iframes and check if
323
+ // iframe.contentWindow.chrome exists and matches the parent.
324
+ // CDP-controlled pages may have inconsistent iframe contexts.
325
+ try {
326
+ const _origHTMLIFrame = HTMLIFrameElement.prototype;
327
+ const _origContentWindow = Object.getOwnPropertyDescriptor(_origHTMLIFrame, 'contentWindow');
328
+ if (_origContentWindow && _origContentWindow.get) {
329
+ Object.defineProperty(_origHTMLIFrame, 'contentWindow', {
330
+ get: function() {
331
+ const _w = _origContentWindow.get.call(this);
332
+ if (_w) {
333
+ try {
334
+ if (!_w.chrome) {
335
+ Object.defineProperty(_w, 'chrome', {
336
+ value: window.chrome,
337
+ writable: true,
338
+ configurable: true,
339
+ });
340
+ }
341
+ } catch {}
342
+ }
343
+ return _w;
344
+ },
345
+ configurable: true,
346
+ });
347
+ }
348
+ } catch {}
349
+
152
350
  return 'applied';
153
351
  })()
154
352
  `;
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,134 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { generateStealthJs } from './stealth.js';
3
+ /**
4
+ * Tests for the stealth anti-detection module.
5
+ *
6
+ * We test the generated JS string for expected content and structure.
7
+ * Evaluating in Node is fragile because stealth patches target browser
8
+ * globals (navigator, Performance, HTMLIFrameElement) that don't exist
9
+ * or behave differently in Node. Instead we verify the code string
10
+ * contains the right patches and is syntactically valid.
11
+ */
12
+ describe('generateStealthJs', () => {
13
+ it('returns a non-empty string', () => {
14
+ const code = generateStealthJs();
15
+ expect(typeof code).toBe('string');
16
+ expect(code.length).toBeGreaterThan(0);
17
+ });
18
+ it('is a valid self-contained IIFE', () => {
19
+ const code = generateStealthJs();
20
+ // Should start/end as an IIFE
21
+ expect(code.trim()).toMatch(/^\(\(\) => \{/);
22
+ expect(code.trim()).toMatch(/\}\)\(\)$/);
23
+ });
24
+ it('patches navigator.webdriver', () => {
25
+ const code = generateStealthJs();
26
+ expect(code).toContain("navigator, 'webdriver'");
27
+ expect(code).toContain('() => false');
28
+ });
29
+ it('stubs window.chrome', () => {
30
+ const code = generateStealthJs();
31
+ expect(code).toContain('window.chrome');
32
+ expect(code).toContain('runtime');
33
+ expect(code).toContain('loadTimes');
34
+ expect(code).toContain('csi');
35
+ });
36
+ it('fakes navigator.plugins if empty', () => {
37
+ const code = generateStealthJs();
38
+ expect(code).toContain('navigator.plugins');
39
+ expect(code).toContain('PDF Viewer');
40
+ expect(code).toContain('Chrome PDF Viewer');
41
+ });
42
+ it('ensures navigator.languages is non-empty', () => {
43
+ const code = generateStealthJs();
44
+ expect(code).toContain('navigator.languages');
45
+ expect(code).toContain("'en-US'");
46
+ });
47
+ it('normalizes Permissions.query for notifications', () => {
48
+ const code = generateStealthJs();
49
+ expect(code).toContain('Permissions');
50
+ expect(code).toContain('notifications');
51
+ });
52
+ it('cleans automation artifacts', () => {
53
+ const code = generateStealthJs();
54
+ expect(code).toContain('__playwright');
55
+ expect(code).toContain('__puppeteer');
56
+ expect(code).toContain("'cdc_'");
57
+ expect(code).toContain("'__cdc_'");
58
+ });
59
+ it('filters CDP patterns from Error.stack', () => {
60
+ const code = generateStealthJs();
61
+ expect(code).toContain('puppeteer_evaluation_script');
62
+ expect(code).toContain("'pptr:'");
63
+ expect(code).toContain("'debugger://'");
64
+ });
65
+ it('neutralizes debugger statement traps', () => {
66
+ const code = generateStealthJs();
67
+ // Should patch Function constructor with new.target / Reflect.construct
68
+ expect(code).toContain('_OrigFunction');
69
+ expect(code).toContain('_PatchedFunction');
70
+ expect(code).toContain('new.target');
71
+ expect(code).toContain('Reflect.construct');
72
+ // Should patch eval
73
+ expect(code).toContain('_origEval');
74
+ expect(code).toContain('_patchedEval');
75
+ // Regex to strip debugger (lookbehind for statement boundaries)
76
+ expect(code).toContain('_debuggerRe');
77
+ });
78
+ it('uses shared toString disguise via WeakMap', () => {
79
+ const code = generateStealthJs();
80
+ // Shared infrastructure at the top of the IIFE
81
+ expect(code).toContain('_origToString');
82
+ expect(code).toContain('WeakMap');
83
+ expect(code).toContain('_disguised');
84
+ expect(code).toContain('_disguise');
85
+ // Should NOT have per-instance toString overrides on Function/eval
86
+ // (they go through _disguise instead)
87
+ });
88
+ it('defends console method fingerprinting', () => {
89
+ const code = generateStealthJs();
90
+ expect(code).toContain('_consoleMethods');
91
+ expect(code).toContain("'log'");
92
+ expect(code).toContain("'warn'");
93
+ expect(code).toContain("'error'");
94
+ expect(code).toContain('[native code]');
95
+ // Uses saved _origToString reference
96
+ expect(code).toContain('_origToString.call');
97
+ });
98
+ it('defends window dimension detection', () => {
99
+ const code = generateStealthJs();
100
+ expect(code).toContain('outerWidth');
101
+ expect(code).toContain('outerHeight');
102
+ expect(code).toContain('innerWidth');
103
+ expect(code).toContain('innerHeight');
104
+ });
105
+ it('filters Performance API entries', () => {
106
+ const code = generateStealthJs();
107
+ expect(code).toContain('getEntries');
108
+ expect(code).toContain('getEntriesByType');
109
+ expect(code).toContain('getEntriesByName');
110
+ expect(code).toContain('_suspiciousPatterns');
111
+ });
112
+ it('cleans document $cdc_ properties', () => {
113
+ const code = generateStealthJs();
114
+ expect(code).toContain("'$cdc_'");
115
+ expect(code).toContain("'$chrome_'");
116
+ });
117
+ it('patches iframe contentWindow.chrome consistency', () => {
118
+ const code = generateStealthJs();
119
+ expect(code).toContain('contentWindow');
120
+ expect(code).toContain('HTMLIFrameElement');
121
+ });
122
+ it('uses non-enumerable guard flag on EventTarget.prototype', () => {
123
+ const code = generateStealthJs();
124
+ expect(code).toContain('EventTarget.prototype');
125
+ expect(code).toContain("'__lsn'");
126
+ expect(code).toContain('enumerable: false');
127
+ });
128
+ it('generates syntactically valid JavaScript', () => {
129
+ const code = generateStealthJs();
130
+ // new Function() parses the code without executing it in a real
131
+ // browser context, catching syntax errors from template literal issues.
132
+ expect(() => new Function(code)).not.toThrow();
133
+ });
134
+ });
@@ -108,7 +108,7 @@ describe('BrowserBridge state', () => {
108
108
  vi.spyOn(daemonClient, 'isExtensionConnected').mockResolvedValue(false);
109
109
  vi.spyOn(daemonClient, 'isDaemonRunning').mockResolvedValue(true);
110
110
  const mcp = new BrowserBridge();
111
- await expect(mcp.connect()).rejects.toThrow('Browser Extension is not connected');
111
+ await expect(mcp.connect({ timeout: 0.1 })).rejects.toThrow('Browser Extension is not connected');
112
112
  });
113
113
  });
114
114
  describe('stealth anti-detection', () => {