@flrande/browserctl 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (90) hide show
  1. package/LICENSE +21 -0
  2. package/README-CN.md +1155 -0
  3. package/README.md +1155 -0
  4. package/apps/browserctl/src/commands/a11y-snapshot.ts +20 -0
  5. package/apps/browserctl/src/commands/act.ts +20 -0
  6. package/apps/browserctl/src/commands/common.test.ts +87 -0
  7. package/apps/browserctl/src/commands/common.ts +191 -0
  8. package/apps/browserctl/src/commands/console-list.ts +20 -0
  9. package/apps/browserctl/src/commands/cookie-clear.ts +18 -0
  10. package/apps/browserctl/src/commands/cookie-get.ts +18 -0
  11. package/apps/browserctl/src/commands/cookie-set.ts +22 -0
  12. package/apps/browserctl/src/commands/dialog-arm.ts +20 -0
  13. package/apps/browserctl/src/commands/dom-query-all.ts +18 -0
  14. package/apps/browserctl/src/commands/dom-query.ts +18 -0
  15. package/apps/browserctl/src/commands/download-trigger.ts +22 -0
  16. package/apps/browserctl/src/commands/download-wait.test.ts +67 -0
  17. package/apps/browserctl/src/commands/download-wait.ts +27 -0
  18. package/apps/browserctl/src/commands/element-screenshot.ts +20 -0
  19. package/apps/browserctl/src/commands/frame-list.ts +16 -0
  20. package/apps/browserctl/src/commands/frame-snapshot.ts +18 -0
  21. package/apps/browserctl/src/commands/network-wait-for.ts +100 -0
  22. package/apps/browserctl/src/commands/profile-list.ts +16 -0
  23. package/apps/browserctl/src/commands/profile-use.ts +18 -0
  24. package/apps/browserctl/src/commands/response-body.ts +24 -0
  25. package/apps/browserctl/src/commands/screenshot.ts +16 -0
  26. package/apps/browserctl/src/commands/snapshot.ts +16 -0
  27. package/apps/browserctl/src/commands/status.ts +10 -0
  28. package/apps/browserctl/src/commands/storage-get.ts +20 -0
  29. package/apps/browserctl/src/commands/storage-set.ts +22 -0
  30. package/apps/browserctl/src/commands/tab-close.ts +20 -0
  31. package/apps/browserctl/src/commands/tab-focus.ts +20 -0
  32. package/apps/browserctl/src/commands/tab-open.ts +19 -0
  33. package/apps/browserctl/src/commands/tabs.ts +13 -0
  34. package/apps/browserctl/src/commands/upload-arm.ts +26 -0
  35. package/apps/browserctl/src/daemon-client.test.ts +253 -0
  36. package/apps/browserctl/src/daemon-client.ts +632 -0
  37. package/apps/browserctl/src/e2e.test.ts +99 -0
  38. package/apps/browserctl/src/main.test.ts +215 -0
  39. package/apps/browserctl/src/main.ts +372 -0
  40. package/apps/browserctl/src/smoke.test.ts +16 -0
  41. package/apps/browserctl/src/smoke.ts +5 -0
  42. package/apps/browserd/src/bootstrap.ts +432 -0
  43. package/apps/browserd/src/chrome-relay-extension-bridge.test.ts +275 -0
  44. package/apps/browserd/src/chrome-relay-extension-bridge.ts +506 -0
  45. package/apps/browserd/src/container.ts +1531 -0
  46. package/apps/browserd/src/main.test.ts +864 -0
  47. package/apps/browserd/src/main.ts +7 -0
  48. package/bin/browserctl.cjs +21 -0
  49. package/bin/browserd.cjs +21 -0
  50. package/extensions/chrome-relay/README.md +36 -0
  51. package/extensions/chrome-relay/background.js +1687 -0
  52. package/extensions/chrome-relay/manifest.json +15 -0
  53. package/extensions/chrome-relay/popup.html +369 -0
  54. package/extensions/chrome-relay/popup.js +972 -0
  55. package/package.json +51 -0
  56. package/packages/core/src/bootstrap.test.ts +10 -0
  57. package/packages/core/src/driver-registry.test.ts +45 -0
  58. package/packages/core/src/driver-registry.ts +22 -0
  59. package/packages/core/src/driver.ts +47 -0
  60. package/packages/core/src/index.ts +5 -0
  61. package/packages/core/src/ref-cache.test.ts +61 -0
  62. package/packages/core/src/ref-cache.ts +28 -0
  63. package/packages/core/src/session-store.test.ts +49 -0
  64. package/packages/core/src/session-store.ts +33 -0
  65. package/packages/core/src/types.ts +9 -0
  66. package/packages/driver-chrome-relay/src/chrome-relay-driver.test.ts +634 -0
  67. package/packages/driver-chrome-relay/src/chrome-relay-driver.ts +2206 -0
  68. package/packages/driver-chrome-relay/src/chrome-relay-extension-runtime.test.ts +264 -0
  69. package/packages/driver-chrome-relay/src/chrome-relay-extension-runtime.ts +521 -0
  70. package/packages/driver-chrome-relay/src/index.ts +26 -0
  71. package/packages/driver-managed/src/index.ts +22 -0
  72. package/packages/driver-managed/src/managed-driver.test.ts +59 -0
  73. package/packages/driver-managed/src/managed-driver.ts +125 -0
  74. package/packages/driver-managed/src/managed-local-driver.test.ts +506 -0
  75. package/packages/driver-managed/src/managed-local-driver.ts +2021 -0
  76. package/packages/driver-remote-cdp/src/index.ts +19 -0
  77. package/packages/driver-remote-cdp/src/remote-cdp-driver.test.ts +617 -0
  78. package/packages/driver-remote-cdp/src/remote-cdp-driver.ts +2042 -0
  79. package/packages/protocol/src/envelope.test.ts +25 -0
  80. package/packages/protocol/src/envelope.ts +31 -0
  81. package/packages/protocol/src/errors.test.ts +17 -0
  82. package/packages/protocol/src/errors.ts +11 -0
  83. package/packages/protocol/src/index.ts +3 -0
  84. package/packages/protocol/src/tools.ts +3 -0
  85. package/packages/transport-mcp-stdio/src/index.ts +3 -0
  86. package/packages/transport-mcp-stdio/src/sdk-server.ts +139 -0
  87. package/packages/transport-mcp-stdio/src/server.test.ts +281 -0
  88. package/packages/transport-mcp-stdio/src/server.ts +183 -0
  89. package/packages/transport-mcp-stdio/src/tool-map.ts +67 -0
  90. package/scripts/smoke.ps1 +127 -0
@@ -0,0 +1,15 @@
1
+ {
2
+ "manifest_version": 3,
3
+ "name": "BrowserCtl Relay",
4
+ "version": "0.1.0",
5
+ "description": "BrowserCtl local relay extension for chrome-relay extension mode.",
6
+ "permissions": ["tabs", "scripting", "storage", "debugger", "cookies"],
7
+ "host_permissions": ["<all_urls>"],
8
+ "background": {
9
+ "service_worker": "background.js"
10
+ },
11
+ "action": {
12
+ "default_title": "BrowserCtl Relay",
13
+ "default_popup": "popup.html"
14
+ }
15
+ }
@@ -0,0 +1,369 @@
1
+ <!doctype html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="utf-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1" />
6
+ <title>BrowserCtl Relay</title>
7
+ <style>
8
+ :root {
9
+ --bg: #f3f6fa;
10
+ --surface: #ffffff;
11
+ --surface-muted: #f8fafc;
12
+ --text: #1f2937;
13
+ --text-muted: #6b7280;
14
+ --border: #d1d9e6;
15
+ --primary: #0f6fda;
16
+ --primary-hover: #095ec1;
17
+ --secondary: #4b5563;
18
+ --secondary-hover: #374151;
19
+ --ok: #1f8f3a;
20
+ --ok-soft: #e8f6ec;
21
+ --error: #b02a37;
22
+ --error-soft: #fdecec;
23
+ }
24
+
25
+ body {
26
+ margin: 0;
27
+ padding: 12px 12px 14px;
28
+ font-family: "Segoe UI", sans-serif;
29
+ width: 360px;
30
+ color: var(--text);
31
+ background: radial-gradient(
32
+ circle at top right,
33
+ rgba(15, 111, 218, 0.15),
34
+ transparent 50%
35
+ ),
36
+ var(--bg);
37
+ }
38
+
39
+ .layout {
40
+ display: grid;
41
+ gap: 12px;
42
+ }
43
+
44
+ .header {
45
+ display: flex;
46
+ justify-content: space-between;
47
+ align-items: flex-start;
48
+ gap: 10px;
49
+ padding: 10px;
50
+ border: 1px solid var(--border);
51
+ border-radius: 10px;
52
+ background: linear-gradient(180deg, #ffffff, #f8fbff);
53
+ box-shadow: 0 6px 14px rgba(15, 23, 42, 0.06);
54
+ }
55
+
56
+ .header-title {
57
+ display: grid;
58
+ gap: 2px;
59
+ }
60
+
61
+ .title {
62
+ font-size: 14px;
63
+ font-weight: 700;
64
+ }
65
+
66
+ .subtitle {
67
+ font-size: 11px;
68
+ color: var(--text-muted);
69
+ }
70
+
71
+ .pill {
72
+ align-self: center;
73
+ padding: 4px 8px;
74
+ border-radius: 999px;
75
+ font-size: 11px;
76
+ font-weight: 700;
77
+ border: 1px solid var(--border);
78
+ background: var(--surface-muted);
79
+ color: var(--text-muted);
80
+ }
81
+
82
+ .pill.ok {
83
+ border-color: #9ad5a9;
84
+ color: var(--ok);
85
+ background: var(--ok-soft);
86
+ }
87
+
88
+ .pill.error {
89
+ border-color: #f0b7bf;
90
+ color: var(--error);
91
+ background: var(--error-soft);
92
+ }
93
+
94
+ .toolbar {
95
+ display: flex;
96
+ justify-content: flex-start;
97
+ }
98
+
99
+ .language-control {
100
+ display: grid;
101
+ gap: 4px;
102
+ font-size: 11px;
103
+ font-weight: 600;
104
+ color: #555;
105
+ }
106
+
107
+ select {
108
+ height: 28px;
109
+ border: 1px solid var(--border);
110
+ border-radius: 6px;
111
+ padding: 0 8px;
112
+ font-size: 12px;
113
+ background: var(--surface);
114
+ color: var(--text);
115
+ }
116
+
117
+ .section-title {
118
+ font-size: 11px;
119
+ font-weight: 700;
120
+ color: var(--text-muted);
121
+ letter-spacing: 0.02em;
122
+ text-transform: uppercase;
123
+ }
124
+
125
+ .panel {
126
+ display: grid;
127
+ gap: 8px;
128
+ border: 1px solid var(--border);
129
+ border-radius: 10px;
130
+ background: var(--surface);
131
+ padding: 10px;
132
+ }
133
+
134
+ label {
135
+ display: grid;
136
+ gap: 4px;
137
+ font-size: 12px;
138
+ font-weight: 600;
139
+ }
140
+
141
+ input {
142
+ height: 30px;
143
+ border: 1px solid var(--border);
144
+ border-radius: 6px;
145
+ padding: 0 8px;
146
+ font-size: 12px;
147
+ background: var(--surface);
148
+ color: var(--text);
149
+ }
150
+
151
+ .buttons {
152
+ display: flex;
153
+ gap: 8px;
154
+ }
155
+
156
+ button {
157
+ flex: 1;
158
+ height: 30px;
159
+ border: 0;
160
+ border-radius: 6px;
161
+ background: var(--primary);
162
+ color: #fff;
163
+ font-size: 12px;
164
+ font-weight: 600;
165
+ cursor: pointer;
166
+ transition: background-color 120ms ease;
167
+ }
168
+
169
+ button:hover:not(:disabled) {
170
+ background: var(--primary-hover);
171
+ }
172
+
173
+ button.secondary {
174
+ background: var(--secondary);
175
+ }
176
+
177
+ button.secondary:hover:not(:disabled) {
178
+ background: var(--secondary-hover);
179
+ }
180
+
181
+ button:disabled {
182
+ opacity: 0.6;
183
+ cursor: not-allowed;
184
+ }
185
+
186
+ .metrics {
187
+ display: grid;
188
+ grid-template-columns: repeat(2, minmax(0, 1fr));
189
+ gap: 8px;
190
+ }
191
+
192
+ .metric {
193
+ border: 1px solid var(--border);
194
+ border-radius: 8px;
195
+ background: var(--surface-muted);
196
+ padding: 8px;
197
+ display: grid;
198
+ gap: 4px;
199
+ }
200
+
201
+ .metric-label {
202
+ font-size: 10px;
203
+ letter-spacing: 0.02em;
204
+ text-transform: uppercase;
205
+ color: var(--text-muted);
206
+ font-weight: 700;
207
+ }
208
+
209
+ .metric-value {
210
+ font-size: 12px;
211
+ font-weight: 700;
212
+ color: var(--text);
213
+ word-break: break-word;
214
+ }
215
+
216
+ .metric-value.ok {
217
+ color: var(--ok);
218
+ }
219
+
220
+ .metric-value.error {
221
+ color: var(--error);
222
+ }
223
+
224
+ .details-block {
225
+ border: 1px solid var(--border);
226
+ border-radius: 8px;
227
+ background: var(--surface-muted);
228
+ overflow: hidden;
229
+ }
230
+
231
+ .details-block summary {
232
+ cursor: pointer;
233
+ padding: 8px 10px;
234
+ font-size: 12px;
235
+ font-weight: 700;
236
+ color: var(--text-muted);
237
+ }
238
+
239
+ .status {
240
+ font-size: 12px;
241
+ line-height: 1.4;
242
+ background: #fff;
243
+ border-top: 1px solid var(--border);
244
+ padding: 8px 10px;
245
+ white-space: pre-wrap;
246
+ margin: 0;
247
+ }
248
+
249
+ .status.ok {
250
+ background: var(--ok-soft);
251
+ }
252
+
253
+ .status.error {
254
+ background: var(--error-soft);
255
+ }
256
+
257
+ .hint {
258
+ font-size: 11px;
259
+ color: var(--text-muted);
260
+ }
261
+
262
+ .guidance {
263
+ margin: 0 0 0 2px;
264
+ padding-left: 18px;
265
+ display: grid;
266
+ gap: 4px;
267
+ font-size: 12px;
268
+ }
269
+
270
+ .guidance li {
271
+ line-height: 1.35;
272
+ }
273
+
274
+ .feedback {
275
+ font-size: 12px;
276
+ border-radius: 7px;
277
+ padding: 7px 9px;
278
+ border: 1px solid var(--border);
279
+ background: var(--surface-muted);
280
+ }
281
+
282
+ .feedback.ok {
283
+ border-color: #9ad5a9;
284
+ color: var(--ok);
285
+ background: var(--ok-soft);
286
+ }
287
+
288
+ .feedback.error {
289
+ border-color: #f0b7bf;
290
+ color: var(--error);
291
+ background: var(--error-soft);
292
+ }
293
+ </style>
294
+ </head>
295
+ <body>
296
+ <div class="layout">
297
+ <div class="header">
298
+ <div class="header-title">
299
+ <div id="headerTitle" class="title">BrowserCtl Relay</div>
300
+ <div id="headerSubtitle" class="subtitle">Local browser bridge console</div>
301
+ </div>
302
+ <div id="headerConnectionPill" class="pill">Unknown</div>
303
+ </div>
304
+
305
+ <div class="panel">
306
+ <div class="toolbar">
307
+ <label class="language-control">
308
+ <span id="languageLabel">Language</span>
309
+ <select id="languageSelect">
310
+ <option value="en">English</option>
311
+ <option value="zh-CN">中文</option>
312
+ </select>
313
+ </label>
314
+ </div>
315
+ </div>
316
+
317
+ <div class="panel">
318
+ <div id="connectionSectionTitle" class="section-title">Connection</div>
319
+ <label>
320
+ <span id="bridgeUrlLabel">Bridge URL</span>
321
+ <input id="bridgeUrl" type="text" placeholder="ws://127.0.0.1:9223/bridge" />
322
+ </label>
323
+
324
+ <label>
325
+ <span id="tokenLabel">Token (required)</span>
326
+ <input id="token" type="password" placeholder="BROWSERD_CHROME_RELAY_EXTENSION_TOKEN" />
327
+ </label>
328
+
329
+ <div class="buttons">
330
+ <button id="saveButton" type="button">Save</button>
331
+ <button id="reconnectButton" class="secondary" type="button">Reconnect</button>
332
+ </div>
333
+ </div>
334
+
335
+ <div class="panel">
336
+ <div id="statusSectionTitle" class="section-title">Live Status</div>
337
+ <div id="statusSummary" class="metrics"></div>
338
+ <details class="details-block">
339
+ <summary id="statusDetailsLabel">Status details</summary>
340
+ <div id="statusBox" class="status">Loading relay status...</div>
341
+ </details>
342
+ </div>
343
+
344
+ <div class="panel">
345
+ <div id="troubleshootingSectionTitle" class="section-title">Troubleshooting</div>
346
+ <div id="diagnosticsHighlights" class="metrics"></div>
347
+ <div class="buttons">
348
+ <button id="diagnoseButton" class="secondary" type="button">Run Diagnostics</button>
349
+ <button id="copyReportButton" class="secondary" type="button">Copy Report</button>
350
+ </div>
351
+ <details class="details-block">
352
+ <summary id="diagnosticsDetailsLabel">Diagnostics details</summary>
353
+ <div id="diagnosticsBox" class="status">No diagnostics yet.</div>
354
+ </details>
355
+ <ul id="guidanceList" class="guidance">
356
+ <li>Run diagnostics when status and CLI result look inconsistent.</li>
357
+ </ul>
358
+ </div>
359
+
360
+ <div id="actionFeedback" class="feedback" hidden></div>
361
+
362
+ <div id="hintText" class="hint">
363
+ Keep this extension enabled while using browserctl chrome-relay extension mode.
364
+ </div>
365
+ </div>
366
+
367
+ <script src="popup.js"></script>
368
+ </body>
369
+ </html>