btcp-browser-agent 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 (117) hide show
  1. package/CLAUDE.md +230 -0
  2. package/LICENSE +21 -0
  3. package/README.md +309 -0
  4. package/SKILL.md +143 -0
  5. package/SNAPSHOT_IMPROVEMENTS.md +302 -0
  6. package/USAGE.md +146 -0
  7. package/dist/index.d.ts +34 -0
  8. package/dist/index.d.ts.map +1 -0
  9. package/dist/index.js +35 -0
  10. package/dist/index.js.map +1 -0
  11. package/docs/browser-cli-design.md +500 -0
  12. package/examples/chrome-extension/CHANGELOG.md +210 -0
  13. package/examples/chrome-extension/DEBUG.md +231 -0
  14. package/examples/chrome-extension/ERROR_FIXED.md +147 -0
  15. package/examples/chrome-extension/QUICK_TEST.md +189 -0
  16. package/examples/chrome-extension/README.md +149 -0
  17. package/examples/chrome-extension/SESSION_ONLY_MODE.md +305 -0
  18. package/examples/chrome-extension/TEST_WITH_YOUR_TABS.md +97 -0
  19. package/examples/chrome-extension/build.js +43 -0
  20. package/examples/chrome-extension/manifest.json +37 -0
  21. package/examples/chrome-extension/package-lock.json +1063 -0
  22. package/examples/chrome-extension/package.json +21 -0
  23. package/examples/chrome-extension/popup.html +195 -0
  24. package/examples/chrome-extension/src/background.ts +12 -0
  25. package/examples/chrome-extension/src/content.ts +7 -0
  26. package/examples/chrome-extension/src/popup.ts +303 -0
  27. package/examples/chrome-extension/src/scenario-google-github.ts +389 -0
  28. package/examples/chrome-extension/test-page.html +127 -0
  29. package/examples/chrome-extension/tests/README.md +206 -0
  30. package/examples/chrome-extension/tests/scenario-google-to-github-star.ts +380 -0
  31. package/examples/chrome-extension/tsconfig.json +14 -0
  32. package/examples/snapshots/README.md +207 -0
  33. package/examples/snapshots/amazon-com-detail.html +9528 -0
  34. package/examples/snapshots/amazon-com-detail.snapshot.txt +997 -0
  35. package/examples/snapshots/convert-snapshots.ts +97 -0
  36. package/examples/snapshots/edition-cnn-com.html +13292 -0
  37. package/examples/snapshots/edition-cnn-com.snapshot.txt +562 -0
  38. package/examples/snapshots/github-com-microsoft-vscode.html +2916 -0
  39. package/examples/snapshots/github-com-microsoft-vscode.snapshot.txt +455 -0
  40. package/examples/snapshots/google-search.html +20012 -0
  41. package/examples/snapshots/google-search.snapshot.txt +195 -0
  42. package/examples/snapshots/metadata.json +86 -0
  43. package/examples/snapshots/npr-org-templates.html +2031 -0
  44. package/examples/snapshots/npr-org-templates.snapshot.txt +224 -0
  45. package/examples/snapshots/stackoverflow-com.html +5216 -0
  46. package/examples/snapshots/stackoverflow-com.snapshot.txt +2404 -0
  47. package/examples/snapshots/test-all-mode.html +46 -0
  48. package/examples/snapshots/test-all-mode.snapshot.txt +5 -0
  49. package/examples/snapshots/validate.test.ts +296 -0
  50. package/package.json +65 -0
  51. package/packages/cli/package.json +42 -0
  52. package/packages/cli/src/__tests__/cli.test.ts +434 -0
  53. package/packages/cli/src/__tests__/errors.test.ts +226 -0
  54. package/packages/cli/src/__tests__/executor.test.ts +275 -0
  55. package/packages/cli/src/__tests__/formatter.test.ts +260 -0
  56. package/packages/cli/src/__tests__/parser.test.ts +288 -0
  57. package/packages/cli/src/__tests__/suggestions.test.ts +255 -0
  58. package/packages/cli/src/commands/back.ts +22 -0
  59. package/packages/cli/src/commands/check.ts +33 -0
  60. package/packages/cli/src/commands/clear.ts +33 -0
  61. package/packages/cli/src/commands/click.ts +32 -0
  62. package/packages/cli/src/commands/closetab.ts +31 -0
  63. package/packages/cli/src/commands/eval.ts +41 -0
  64. package/packages/cli/src/commands/fill.ts +30 -0
  65. package/packages/cli/src/commands/focus.ts +33 -0
  66. package/packages/cli/src/commands/forward.ts +22 -0
  67. package/packages/cli/src/commands/goto.ts +34 -0
  68. package/packages/cli/src/commands/help.ts +162 -0
  69. package/packages/cli/src/commands/hover.ts +34 -0
  70. package/packages/cli/src/commands/index.ts +129 -0
  71. package/packages/cli/src/commands/newtab.ts +35 -0
  72. package/packages/cli/src/commands/press.ts +40 -0
  73. package/packages/cli/src/commands/reload.ts +25 -0
  74. package/packages/cli/src/commands/screenshot.ts +27 -0
  75. package/packages/cli/src/commands/scroll.ts +64 -0
  76. package/packages/cli/src/commands/select.ts +35 -0
  77. package/packages/cli/src/commands/snapshot.ts +21 -0
  78. package/packages/cli/src/commands/tab.ts +32 -0
  79. package/packages/cli/src/commands/tabs.ts +26 -0
  80. package/packages/cli/src/commands/text.ts +27 -0
  81. package/packages/cli/src/commands/title.ts +17 -0
  82. package/packages/cli/src/commands/type.ts +38 -0
  83. package/packages/cli/src/commands/uncheck.ts +33 -0
  84. package/packages/cli/src/commands/url.ts +17 -0
  85. package/packages/cli/src/commands/wait.ts +54 -0
  86. package/packages/cli/src/errors.ts +164 -0
  87. package/packages/cli/src/executor.ts +68 -0
  88. package/packages/cli/src/formatter.ts +215 -0
  89. package/packages/cli/src/index.ts +257 -0
  90. package/packages/cli/src/parser.ts +195 -0
  91. package/packages/cli/src/suggestions.ts +207 -0
  92. package/packages/cli/src/terminal/Terminal.ts +365 -0
  93. package/packages/cli/src/terminal/index.ts +5 -0
  94. package/packages/cli/src/types.ts +155 -0
  95. package/packages/cli/tsconfig.json +20 -0
  96. package/packages/core/package.json +35 -0
  97. package/packages/core/src/actions.ts +1210 -0
  98. package/packages/core/src/errors.ts +296 -0
  99. package/packages/core/src/index.test.ts +638 -0
  100. package/packages/core/src/index.ts +220 -0
  101. package/packages/core/src/ref-map.ts +107 -0
  102. package/packages/core/src/snapshot.ts +873 -0
  103. package/packages/core/src/types.ts +536 -0
  104. package/packages/core/tsconfig.json +23 -0
  105. package/packages/extension/README.md +129 -0
  106. package/packages/extension/package.json +43 -0
  107. package/packages/extension/src/background.ts +888 -0
  108. package/packages/extension/src/content.ts +172 -0
  109. package/packages/extension/src/index.ts +579 -0
  110. package/packages/extension/src/session-manager.ts +385 -0
  111. package/packages/extension/src/session-types.ts +144 -0
  112. package/packages/extension/src/types.ts +162 -0
  113. package/packages/extension/tsconfig.json +28 -0
  114. package/src/index.ts +64 -0
  115. package/tsconfig.build.json +12 -0
  116. package/tsconfig.json +26 -0
  117. package/vitest.config.ts +13 -0
@@ -0,0 +1,172 @@
1
+ /**
2
+ * @btcp/extension - Content Script
3
+ *
4
+ * Runs in web pages, handles DOM commands from the background script.
5
+ * Uses ContentAgent from @btcp/core for DOM operations.
6
+ */
7
+
8
+ import { createContentAgent, type ContentAgent, type Command as CoreCommand, type Response } from '@btcp/core';
9
+ import type { ExtensionMessage, ExtensionResponse, Command } from './types.js';
10
+
11
+ let agent: ContentAgent | null = null;
12
+ let isContentScriptReady = false;
13
+
14
+ /**
15
+ * Get or create the ContentAgent instance for this page
16
+ */
17
+ function getContentAgent(): ContentAgent {
18
+ if (!agent) {
19
+ agent = createContentAgent(document, window);
20
+ isContentScriptReady = true;
21
+ console.log('[ContentScript] Agent initialized');
22
+ }
23
+ return agent;
24
+ }
25
+
26
+ // Initialize agent immediately
27
+ getContentAgent();
28
+
29
+ /**
30
+ * Check if a command is a core DOM command
31
+ */
32
+ function isCoreCommand(command: Command): command is CoreCommand {
33
+ const extensionActions = [
34
+ 'navigate', 'back', 'forward', 'reload',
35
+ 'getUrl', 'getTitle', 'screenshot',
36
+ 'tabNew', 'tabClose', 'tabSwitch', 'tabList',
37
+ ];
38
+ return !extensionActions.includes(command.action);
39
+ }
40
+
41
+ /**
42
+ * Handle a command from the background script
43
+ */
44
+ async function handleCommand(command: Command): Promise<Response> {
45
+ // Core DOM commands are handled by ContentAgent
46
+ if (isCoreCommand(command)) {
47
+ return getContentAgent().execute(command);
48
+ }
49
+
50
+ // Extension commands that need content script execution
51
+ switch (command.action) {
52
+ case 'getUrl':
53
+ return {
54
+ id: command.id,
55
+ success: true,
56
+ data: { url: window.location.href },
57
+ };
58
+
59
+ case 'getTitle':
60
+ return {
61
+ id: command.id,
62
+ success: true,
63
+ data: { title: document.title },
64
+ };
65
+
66
+ default:
67
+ // Forward to background script
68
+ return {
69
+ id: command.id,
70
+ success: false,
71
+ error: `Command ${command.action} must be handled by background script`,
72
+ };
73
+ }
74
+ }
75
+
76
+ /**
77
+ * Listen for messages from background script
78
+ */
79
+ chrome.runtime.onMessage.addListener((message, _sender, sendResponse) => {
80
+ const msg = message as ExtensionMessage;
81
+
82
+ // Handle ping messages for heartbeat
83
+ if (msg.type === 'btcp:ping') {
84
+ sendResponse({ type: 'btcp:pong', ready: isContentScriptReady } satisfies ExtensionResponse);
85
+ return true;
86
+ }
87
+
88
+ if (msg.type !== 'btcp:command') {
89
+ return false;
90
+ }
91
+
92
+ handleCommand(msg.command)
93
+ .then((response) => {
94
+ sendResponse({ type: 'btcp:response', response } satisfies ExtensionResponse);
95
+ })
96
+ .catch((error) => {
97
+ sendResponse({
98
+ type: 'btcp:response',
99
+ response: {
100
+ id: msg.command.id,
101
+ success: false,
102
+ error: error instanceof Error ? error.message : String(error),
103
+ },
104
+ } satisfies ExtensionResponse);
105
+ });
106
+
107
+ // Return true to indicate async response
108
+ return true;
109
+ });
110
+
111
+ /**
112
+ * Also listen for postMessage from page scripts
113
+ * This allows scripts injected into the page to use the agent
114
+ */
115
+ window.addEventListener('message', async (event) => {
116
+ if (event.source !== window) return;
117
+
118
+ const msg = event.data as ExtensionMessage;
119
+ if (msg?.type !== 'btcp:command') return;
120
+
121
+ const response = await handleCommand(msg.command);
122
+
123
+ window.postMessage({
124
+ type: 'btcp:response',
125
+ response,
126
+ } satisfies ExtensionResponse, '*');
127
+ });
128
+
129
+ /**
130
+ * Lifecycle event listeners for session keep-alive
131
+ */
132
+
133
+ // Detect page visibility changes
134
+ document.addEventListener('visibilitychange', () => {
135
+ if (!document.hidden) {
136
+ console.log('[ContentScript] Page became visible, checking connection...');
137
+ }
138
+ });
139
+
140
+ // Detect freeze/resume events (Chrome 68+)
141
+ document.addEventListener('freeze', () => {
142
+ console.log('[ContentScript] Page frozen');
143
+ });
144
+
145
+ document.addEventListener('resume', () => {
146
+ console.log('[ContentScript] Page resumed, re-initializing agent...');
147
+ // Re-initialize agent to ensure fresh state
148
+ agent = null;
149
+ getContentAgent();
150
+ });
151
+
152
+ // Clear refs on navigation (full page navigation)
153
+ window.addEventListener('beforeunload', () => {
154
+ console.log('[ContentScript] Page navigating, clearing refs...');
155
+ const currentAgent = getContentAgent();
156
+ currentAgent.clearRefs();
157
+ });
158
+
159
+ // Detect SPA navigation (URL changes without full page reload)
160
+ let lastUrl = window.location.href;
161
+ setInterval(() => {
162
+ const currentUrl = window.location.href;
163
+ if (currentUrl !== lastUrl) {
164
+ console.log('[ContentScript] URL changed, clearing refs...', { from: lastUrl, to: currentUrl });
165
+ lastUrl = currentUrl;
166
+ const currentAgent = getContentAgent();
167
+ currentAgent.clearRefs();
168
+ }
169
+ }, 1000);
170
+
171
+ // Export for programmatic use
172
+ export { getContentAgent, handleCommand };