openchrome-mcp 1.2.3 → 1.2.5

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 (107) hide show
  1. package/dist/cdp/client.d.ts.map +1 -1
  2. package/dist/cdp/client.js +10 -0
  3. package/dist/cdp/client.js.map +1 -1
  4. package/dist/chrome/launcher.d.ts +32 -15
  5. package/dist/chrome/launcher.d.ts.map +1 -1
  6. package/dist/chrome/launcher.js +136 -116
  7. package/dist/chrome/launcher.js.map +1 -1
  8. package/dist/chrome/profile-manager.d.ts +108 -0
  9. package/dist/chrome/profile-manager.d.ts.map +1 -0
  10. package/dist/chrome/profile-manager.js +372 -0
  11. package/dist/chrome/profile-manager.js.map +1 -0
  12. package/dist/chrome/sqlite-cookie-copy.d.ts +46 -0
  13. package/dist/chrome/sqlite-cookie-copy.d.ts.map +1 -0
  14. package/dist/chrome/sqlite-cookie-copy.js +145 -0
  15. package/dist/chrome/sqlite-cookie-copy.js.map +1 -0
  16. package/dist/cli/claude-session.js +0 -0
  17. package/dist/cli/index.js +0 -0
  18. package/dist/config/defaults.d.ts +8 -0
  19. package/dist/config/defaults.d.ts.map +1 -1
  20. package/dist/config/defaults.js +9 -1
  21. package/dist/config/defaults.js.map +1 -1
  22. package/dist/config/global.d.ts +2 -0
  23. package/dist/config/global.d.ts.map +1 -1
  24. package/dist/config/global.js.map +1 -1
  25. package/dist/hints/hint-engine.d.ts +36 -2
  26. package/dist/hints/hint-engine.d.ts.map +1 -1
  27. package/dist/hints/hint-engine.js +96 -27
  28. package/dist/hints/hint-engine.js.map +1 -1
  29. package/dist/hints/index.d.ts +1 -1
  30. package/dist/hints/index.d.ts.map +1 -1
  31. package/dist/hints/rules/error-recovery.d.ts.map +1 -1
  32. package/dist/hints/rules/error-recovery.js +5 -1
  33. package/dist/hints/rules/error-recovery.js.map +1 -1
  34. package/dist/hints/rules/repetition-detection.d.ts.map +1 -1
  35. package/dist/hints/rules/repetition-detection.js +42 -0
  36. package/dist/hints/rules/repetition-detection.js.map +1 -1
  37. package/dist/index.js +17 -1
  38. package/dist/index.js.map +1 -1
  39. package/dist/mcp-server.d.ts +12 -1
  40. package/dist/mcp-server.d.ts.map +1 -1
  41. package/dist/mcp-server.js +162 -20
  42. package/dist/mcp-server.js.map +1 -1
  43. package/dist/resources/usage-guide.d.ts +1 -1
  44. package/dist/resources/usage-guide.d.ts.map +1 -1
  45. package/dist/resources/usage-guide.js +22 -0
  46. package/dist/resources/usage-guide.js.map +1 -1
  47. package/dist/session-manager.d.ts +4 -0
  48. package/dist/session-manager.d.ts.map +1 -1
  49. package/dist/session-manager.js +56 -0
  50. package/dist/session-manager.js.map +1 -1
  51. package/dist/tools/click-element.d.ts.map +1 -1
  52. package/dist/tools/click-element.js +6 -2
  53. package/dist/tools/click-element.js.map +1 -1
  54. package/dist/tools/computer.d.ts.map +1 -1
  55. package/dist/tools/computer.js +101 -32
  56. package/dist/tools/computer.js.map +1 -1
  57. package/dist/tools/find.d.ts.map +1 -1
  58. package/dist/tools/find.js +16 -6
  59. package/dist/tools/find.js.map +1 -1
  60. package/dist/tools/form-input.d.ts.map +1 -1
  61. package/dist/tools/form-input.js +20 -0
  62. package/dist/tools/form-input.js.map +1 -1
  63. package/dist/tools/index.d.ts.map +1 -1
  64. package/dist/tools/index.js +2 -0
  65. package/dist/tools/index.js.map +1 -1
  66. package/dist/tools/javascript.d.ts.map +1 -1
  67. package/dist/tools/javascript.js +38 -2
  68. package/dist/tools/javascript.js.map +1 -1
  69. package/dist/tools/navigate.d.ts.map +1 -1
  70. package/dist/tools/navigate.js +27 -0
  71. package/dist/tools/navigate.js.map +1 -1
  72. package/dist/tools/profile-status.d.ts +10 -0
  73. package/dist/tools/profile-status.d.ts.map +1 -0
  74. package/dist/tools/profile-status.js +101 -0
  75. package/dist/tools/profile-status.js.map +1 -0
  76. package/dist/tools/wait-and-click.d.ts.map +1 -1
  77. package/dist/tools/wait-and-click.js +3 -1
  78. package/dist/tools/wait-and-click.js.map +1 -1
  79. package/dist/utils/format-age.d.ts +5 -0
  80. package/dist/utils/format-age.d.ts.map +1 -0
  81. package/dist/utils/format-age.js +15 -0
  82. package/dist/utils/format-age.js.map +1 -0
  83. package/dist/utils/index.d.ts +1 -0
  84. package/dist/utils/index.d.ts.map +1 -1
  85. package/dist/utils/index.js +1 -0
  86. package/dist/utils/index.js.map +1 -1
  87. package/dist/utils/pid-manager.d.ts +6 -0
  88. package/dist/utils/pid-manager.d.ts.map +1 -0
  89. package/dist/utils/pid-manager.js +132 -0
  90. package/dist/utils/pid-manager.js.map +1 -0
  91. package/dist/utils/ref-id-manager.d.ts +19 -1
  92. package/dist/utils/ref-id-manager.d.ts.map +1 -1
  93. package/dist/utils/ref-id-manager.js +49 -2
  94. package/dist/utils/ref-id-manager.js.map +1 -1
  95. package/package.json +4 -1
  96. package/dist/chrome/profile-detector.d.ts +0 -52
  97. package/dist/chrome/profile-detector.d.ts.map +0 -1
  98. package/dist/chrome/profile-detector.js +0 -246
  99. package/dist/chrome/profile-detector.js.map +0 -1
  100. package/dist/hints/rules/dom-mode-suggestion.d.ts +0 -6
  101. package/dist/hints/rules/dom-mode-suggestion.d.ts.map +0 -1
  102. package/dist/hints/rules/dom-mode-suggestion.js +0 -24
  103. package/dist/hints/rules/dom-mode-suggestion.js.map +0 -1
  104. package/dist/types/profile.d.ts +0 -76
  105. package/dist/types/profile.d.ts.map +0 -1
  106. package/dist/types/profile.js +0 -35
  107. package/dist/types/profile.js.map +0 -1
@@ -0,0 +1,372 @@
1
+ "use strict";
2
+ /**
3
+ * ProfileManager - Persistent OpenChrome Profile Architecture
4
+ *
5
+ * Manages a persistent Chrome profile directory at ~/.openchrome/profile/
6
+ * instead of creating disposable temp profiles on every launch.
7
+ * Provides atomic cookie sync using the SQLite backup API.
8
+ */
9
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ var desc = Object.getOwnPropertyDescriptor(m, k);
12
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
13
+ desc = { enumerable: true, get: function() { return m[k]; } };
14
+ }
15
+ Object.defineProperty(o, k2, desc);
16
+ }) : (function(o, m, k, k2) {
17
+ if (k2 === undefined) k2 = k;
18
+ o[k2] = m[k];
19
+ }));
20
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
21
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
22
+ }) : function(o, v) {
23
+ o["default"] = v;
24
+ });
25
+ var __importStar = (this && this.__importStar) || (function () {
26
+ var ownKeys = function(o) {
27
+ ownKeys = Object.getOwnPropertyNames || function (o) {
28
+ var ar = [];
29
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
30
+ return ar;
31
+ };
32
+ return ownKeys(o);
33
+ };
34
+ return function (mod) {
35
+ if (mod && mod.__esModule) return mod;
36
+ var result = {};
37
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
38
+ __setModuleDefault(result, mod);
39
+ return result;
40
+ };
41
+ })();
42
+ Object.defineProperty(exports, "__esModule", { value: true });
43
+ exports.ProfileManager = void 0;
44
+ const fs = __importStar(require("fs"));
45
+ const path = __importStar(require("path"));
46
+ const os = __importStar(require("os"));
47
+ const child_process_1 = require("child_process");
48
+ // ---------------------------------------------------------------------------
49
+ // ProfileManager
50
+ // ---------------------------------------------------------------------------
51
+ /**
52
+ * Manages persistent OpenChrome Chrome profiles and cookie synchronisation.
53
+ *
54
+ * Key responsibilities:
55
+ * - Maintain a reusable profile at `~/.openchrome/profile/` so cookies and
56
+ * session data survive across OpenChrome restarts.
57
+ * - Track whether the persistent profile is stale relative to the real
58
+ * Chrome profile using lightweight file-stat hashing.
59
+ * - Perform atomic cookie sync via the `sqlite3` CLI `.backup` command when
60
+ * available, falling back to a plain file copy otherwise.
61
+ * - Decide which profile directory Chrome should use (`resolveProfile`).
62
+ */
63
+ class ProfileManager {
64
+ // -------------------------------------------------------------------------
65
+ // Constants (configurable via Object.defineProperty for testing)
66
+ // -------------------------------------------------------------------------
67
+ /** Root directory for the persistent OpenChrome profile. */
68
+ static PERSISTENT_PROFILE_DIR = path.join(os.homedir(), '.openchrome', 'profile');
69
+ /** Path to the JSON file that tracks the last sync state. */
70
+ static SYNC_METADATA_PATH = path.join(os.homedir(), '.openchrome', 'sync-metadata.json');
71
+ /** Cookie data is considered fresh if synced within this window (30 min). */
72
+ static COOKIE_FRESHNESS_MS = 30 * 60 * 1000;
73
+ // -------------------------------------------------------------------------
74
+ // Public methods
75
+ // -------------------------------------------------------------------------
76
+ /**
77
+ * Return the persistent profile directory, creating it (including the
78
+ * `Default/` subdirectory) if it does not already exist.
79
+ */
80
+ getOrCreatePersistentProfile() {
81
+ const profileDir = ProfileManager.PERSISTENT_PROFILE_DIR;
82
+ const defaultDir = path.join(profileDir, 'Default');
83
+ if (!fs.existsSync(profileDir)) {
84
+ fs.mkdirSync(profileDir, { recursive: true });
85
+ console.error(`[ProfileManager] Created persistent profile directory: ${profileDir}`);
86
+ }
87
+ if (!fs.existsSync(defaultDir)) {
88
+ fs.mkdirSync(defaultDir, { recursive: true });
89
+ }
90
+ return profileDir;
91
+ }
92
+ /**
93
+ * Determine whether the persistent profile needs a fresh cookie sync from
94
+ * `sourceDir`.
95
+ *
96
+ * Returns `true` when:
97
+ * - No sync metadata exists (never synced before), OR
98
+ * - The source Cookies file has changed since the last sync (different
99
+ * mtime or size), OR
100
+ * - The last sync is older than `COOKIE_FRESHNESS_MS`.
101
+ */
102
+ needsSync(sourceDir) {
103
+ const metadata = this.getSyncMetadata();
104
+ if (!metadata) {
105
+ return true; // Never synced
106
+ }
107
+ // Compute current hash of the source Cookies file
108
+ const sourceCookiesPath = path.join(sourceDir, 'Default', 'Cookies');
109
+ let currentHash;
110
+ try {
111
+ const stat = fs.statSync(sourceCookiesPath);
112
+ currentHash = `${stat.mtimeMs}:${stat.size}`;
113
+ }
114
+ catch {
115
+ // Cookies file doesn't exist in source — no sync possible
116
+ if (!metadata) {
117
+ console.error('[ProfileManager] Source Cookies not found and no prior sync — persistent profile will have no cookies');
118
+ }
119
+ return false;
120
+ }
121
+ if (currentHash !== metadata.sourceProfileHash) {
122
+ return true; // Source has changed
123
+ }
124
+ if (Date.now() - metadata.lastSyncTimestamp > ProfileManager.COOKIE_FRESHNESS_MS) {
125
+ return true; // Stale
126
+ }
127
+ return false;
128
+ }
129
+ /**
130
+ * Synchronise cookies (and Preferences) from `sourceDir` into `destDir`.
131
+ *
132
+ * Uses the `sqlite3` CLI `.backup` command for an atomic, consistent
133
+ * snapshot of the Cookies database. Falls back to a plain file copy when
134
+ * `sqlite3` is not available.
135
+ *
136
+ * After a successful sync the metadata file is updated via
137
+ * `updateSyncMetadata`.
138
+ *
139
+ * @returns `{ atomic: true, success: true }` when sqlite3 backup was used,
140
+ * `{ atomic: false, success: true }` when the plain-copy fallback was used,
141
+ * `{ atomic: false, success: false }` when all methods failed.
142
+ */
143
+ syncCookies(sourceDir, destDir) {
144
+ try {
145
+ const destDefault = path.join(destDir, 'Default');
146
+ fs.mkdirSync(destDefault, { recursive: true });
147
+ // --- 1. Copy Local State -----------------------------------------------
148
+ const localStateSrc = path.join(sourceDir, 'Local State');
149
+ if (fs.existsSync(localStateSrc)) {
150
+ fs.copyFileSync(localStateSrc, path.join(destDir, 'Local State'));
151
+ }
152
+ // --- 2. Sync Cookies (atomic via sqlite3, or plain copy fallback) ------
153
+ const sourceCookiesPath = path.join(sourceDir, 'Default', 'Cookies');
154
+ const destCookiesPath = path.join(destDefault, 'Cookies');
155
+ let atomic = false;
156
+ if (fs.existsSync(sourceCookiesPath)) {
157
+ const sqlite3Available = this._isSqlite3Available();
158
+ if (sqlite3Available) {
159
+ // Atomic backup using the SQLite .backup command.
160
+ // This works even when Chrome is actively writing to the DB.
161
+ // Uses execFileSync (no shell) to prevent injection via path characters.
162
+ (0, child_process_1.execFileSync)('sqlite3', [
163
+ sourceCookiesPath,
164
+ `.backup '${destCookiesPath.replace(/'/g, "''")}'`,
165
+ ], { stdio: 'ignore', timeout: 10000 });
166
+ // .backup produces a clean WAL-checkpoint DB — remove stale WAL/SHM/journal
167
+ // at the destination so Chrome doesn't get confused.
168
+ for (const suffix of ['Cookies-wal', 'Cookies-shm', 'Cookies-journal']) {
169
+ const stale = path.join(destDefault, suffix);
170
+ if (fs.existsSync(stale)) {
171
+ try {
172
+ fs.unlinkSync(stale);
173
+ }
174
+ catch {
175
+ // Non-fatal
176
+ }
177
+ }
178
+ }
179
+ atomic = true;
180
+ }
181
+ else {
182
+ // sqlite3 not available — fall back to plain file copy (same as
183
+ // the legacy copyEssentialProfileData behaviour).
184
+ console.error('[ProfileManager] sqlite3 not found, falling back to non-atomic cookie copy');
185
+ const cookieFiles = [
186
+ 'Cookies',
187
+ 'Cookies-wal',
188
+ 'Cookies-shm',
189
+ 'Cookies-journal',
190
+ ];
191
+ for (const file of cookieFiles) {
192
+ const src = path.join(sourceDir, 'Default', file);
193
+ if (fs.existsSync(src)) {
194
+ try {
195
+ fs.copyFileSync(src, path.join(destDefault, file));
196
+ }
197
+ catch {
198
+ // Individual file copy failure is non-fatal
199
+ }
200
+ }
201
+ }
202
+ atomic = false;
203
+ }
204
+ }
205
+ // --- 3. Copy and patch Preferences ------------------------------------
206
+ const prefsSrc = path.join(sourceDir, 'Default', 'Preferences');
207
+ if (fs.existsSync(prefsSrc)) {
208
+ try {
209
+ const prefsContent = fs.readFileSync(prefsSrc, 'utf8');
210
+ const prefs = JSON.parse(prefsContent);
211
+ // Prevent "Chrome didn't shut down correctly" prompt
212
+ if (prefs.profile) {
213
+ prefs.profile.exit_type = 'Normal';
214
+ prefs.profile.exited_cleanly = true;
215
+ }
216
+ // Suppress session restore so copied profile doesn't reopen old tabs
217
+ if (!prefs.session)
218
+ prefs.session = {};
219
+ prefs.session.restore_on_startup = 5; // 5 = open new tab page
220
+ delete prefs.session.startup_urls;
221
+ fs.writeFileSync(path.join(destDefault, 'Preferences'), JSON.stringify(prefs));
222
+ }
223
+ catch {
224
+ // JSON parse failed — skip Preferences entirely.
225
+ // Chrome will create fresh defaults.
226
+ }
227
+ }
228
+ // --- 4. Update metadata -----------------------------------------------
229
+ this.updateSyncMetadata(sourceDir);
230
+ console.error(`[ProfileManager] Cookie sync complete (atomic=${atomic}) from ${sourceDir} → ${destDir}`);
231
+ return { atomic, success: true };
232
+ }
233
+ catch (err) {
234
+ console.error('[ProfileManager] syncCookies failed (non-fatal):', err);
235
+ return { atomic: false, success: false };
236
+ }
237
+ }
238
+ /**
239
+ * Read the current sync metadata from disk.
240
+ *
241
+ * @returns Parsed `SyncMetadata` or `null` if the file does not exist or
242
+ * cannot be parsed.
243
+ */
244
+ getSyncMetadata() {
245
+ try {
246
+ const raw = fs.readFileSync(ProfileManager.SYNC_METADATA_PATH, 'utf8');
247
+ return JSON.parse(raw);
248
+ }
249
+ catch {
250
+ return null;
251
+ }
252
+ }
253
+ /**
254
+ * Persist sync metadata for `sourceDir` using a temp-file + rename pattern
255
+ * to prevent corruption on concurrent writes.
256
+ */
257
+ updateSyncMetadata(sourceDir) {
258
+ try {
259
+ // Compute source hash
260
+ const sourceCookiesPath = path.join(sourceDir, 'Default', 'Cookies');
261
+ let sourceProfileHash = '';
262
+ try {
263
+ const stat = fs.statSync(sourceCookiesPath);
264
+ sourceProfileHash = `${stat.mtimeMs}:${stat.size}`;
265
+ }
266
+ catch {
267
+ // Cookies file missing — leave hash empty
268
+ }
269
+ const existing = this.getSyncMetadata();
270
+ const updated = {
271
+ lastSyncTimestamp: Date.now(),
272
+ sourceProfileHash,
273
+ syncCount: existing ? existing.syncCount + 1 : 1,
274
+ sourceProfileDir: sourceDir,
275
+ };
276
+ const metaPath = ProfileManager.SYNC_METADATA_PATH;
277
+ const metaDir = path.dirname(metaPath);
278
+ // Ensure parent directory exists
279
+ if (!fs.existsSync(metaDir)) {
280
+ fs.mkdirSync(metaDir, { recursive: true });
281
+ }
282
+ // Atomic write: write to temp file, then rename
283
+ const tmpPath = `${metaPath}.tmp-${Date.now()}`;
284
+ fs.writeFileSync(tmpPath, JSON.stringify(updated, null, 2), 'utf8');
285
+ fs.renameSync(tmpPath, metaPath);
286
+ }
287
+ catch (err) {
288
+ console.error('[ProfileManager] updateSyncMetadata failed (non-fatal):', err);
289
+ }
290
+ }
291
+ /**
292
+ * Determine which Chrome `userDataDir` to use and whether a cookie sync
293
+ * was performed.
294
+ *
295
+ * Priority order:
296
+ * 1. `explicitUserDataDir` — caller has specified an exact directory.
297
+ * 2. `useTempProfile` or `usingHeadlessShell` — create a fresh temp dir.
298
+ * 3. `realProfileDir` exists and **not** locked — use real profile directly.
299
+ * 4. `realProfileDir` exists and **is** locked — use persistent profile,
300
+ * syncing cookies from the real profile when stale.
301
+ * 5. No `realProfileDir` — use persistent profile without a sync.
302
+ */
303
+ resolveProfile(options) {
304
+ const { realProfileDir, isProfileLocked, explicitUserDataDir, useTempProfile, usingHeadlessShell, } = options;
305
+ // 1. Explicit user-data-dir
306
+ if (explicitUserDataDir) {
307
+ return {
308
+ userDataDir: explicitUserDataDir,
309
+ profileType: 'explicit',
310
+ syncPerformed: false,
311
+ };
312
+ }
313
+ // 2. Temp profile requested or headless-shell (no profile support)
314
+ if (useTempProfile || usingHeadlessShell) {
315
+ const tempDir = path.join(os.tmpdir(), `openchrome-${Date.now()}`);
316
+ return {
317
+ userDataDir: tempDir,
318
+ profileType: 'temp',
319
+ syncPerformed: false,
320
+ };
321
+ }
322
+ // 3. Real profile available and NOT locked
323
+ if (realProfileDir && !isProfileLocked) {
324
+ return {
325
+ userDataDir: realProfileDir,
326
+ profileType: 'real',
327
+ syncPerformed: false,
328
+ };
329
+ }
330
+ // 4. Real profile exists but IS locked — use persistent profile
331
+ if (realProfileDir && isProfileLocked) {
332
+ const persistentDir = this.getOrCreatePersistentProfile();
333
+ if (!this.needsSync(realProfileDir)) {
334
+ // Persistent profile is fresh — reuse without re-sync
335
+ return {
336
+ userDataDir: persistentDir,
337
+ profileType: 'persistent',
338
+ syncPerformed: false,
339
+ };
340
+ }
341
+ // Stale — sync cookies from real profile into persistent profile
342
+ const syncResult = this.syncCookies(realProfileDir, persistentDir);
343
+ return {
344
+ userDataDir: persistentDir,
345
+ profileType: 'persistent',
346
+ syncPerformed: syncResult.atomic || syncResult.success,
347
+ };
348
+ }
349
+ // 5. No real profile at all — use persistent profile (no sync needed)
350
+ const persistentDir = this.getOrCreatePersistentProfile();
351
+ return {
352
+ userDataDir: persistentDir,
353
+ profileType: 'persistent',
354
+ syncPerformed: false,
355
+ };
356
+ }
357
+ // -------------------------------------------------------------------------
358
+ // Private helpers
359
+ // -------------------------------------------------------------------------
360
+ /** Check whether the `sqlite3` CLI is available on PATH. */
361
+ _isSqlite3Available() {
362
+ try {
363
+ (0, child_process_1.execFileSync)(os.platform() === 'win32' ? 'where' : 'which', ['sqlite3'], { stdio: 'ignore', timeout: 3000 });
364
+ return true;
365
+ }
366
+ catch {
367
+ return false;
368
+ }
369
+ }
370
+ }
371
+ exports.ProfileManager = ProfileManager;
372
+ //# sourceMappingURL=profile-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"profile-manager.js","sourceRoot":"","sources":["../../src/chrome/profile-manager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,46 @@
1
+ /**
2
+ * Atomic SQLite Cookie Copy
3
+ *
4
+ * Chrome stores cookies in a SQLite database using WAL (Write-Ahead Log) mode.
5
+ * The database state spans up to three files: `Cookies`, `Cookies-wal`, and
6
+ * `Cookies-shm`. A naive sequential `fs.copyFileSync` of these files while
7
+ * Chrome is writing produces an inconsistent copy — the main DB and WAL may
8
+ * represent different transaction states, causing Chrome to silently discard
9
+ * the corrupted database and start with an empty cookie jar.
10
+ *
11
+ * This module implements a 3-tier fallback chain for atomic cookie copying:
12
+ * 1. `better-sqlite3` — VACUUM INTO for a clean, synchronous atomic copy
13
+ * 2. `sqlite3` CLI — `.backup` command via child_process
14
+ * 3. Raw `fs.copyFileSync` — last resort, copies main DB only (no WAL/SHM)
15
+ */
16
+ export interface CookieCopyResult {
17
+ method: 'better-sqlite3' | 'sqlite3-cli' | 'file-copy' | 'none';
18
+ success: boolean;
19
+ warning?: string;
20
+ }
21
+ /**
22
+ * @internal Attempt atomic copy via better-sqlite3's VACUUM INTO.
23
+ */
24
+ declare function attemptBetterSqlite3Copy(sourcePath: string, destPath: string): void;
25
+ /**
26
+ * @internal Mutable dispatch table for testability.
27
+ * Tests can spy on `_deps.attemptBetterSqlite3Copy` to control the fallback chain.
28
+ */
29
+ export declare const _deps: {
30
+ attemptBetterSqlite3Copy: typeof attemptBetterSqlite3Copy;
31
+ };
32
+ /**
33
+ * Copy the Chrome Cookies SQLite database from one Default profile directory
34
+ * to another using a 3-tier fallback chain for atomic consistency:
35
+ *
36
+ * Tier 1: better-sqlite3 `VACUUM INTO` — synchronous, atomic, WAL-aware
37
+ * Tier 2: sqlite3 CLI `.backup` — atomic, WAL-aware, no npm dep required
38
+ * Tier 3: fs.copyFileSync — fast but non-atomic; recent WAL transactions may be absent
39
+ *
40
+ * @param sourceDefaultDir Path to the source Chrome `Default` profile directory
41
+ * @param destDefaultDir Path to the destination `Default` directory (must already exist)
42
+ * @returns Result indicating which method succeeded, or `none` if all failed
43
+ */
44
+ export declare function copyCookiesAtomic(sourceDefaultDir: string, destDefaultDir: string): CookieCopyResult;
45
+ export {};
46
+ //# sourceMappingURL=sqlite-cookie-copy.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sqlite-cookie-copy.d.ts","sourceRoot":"","sources":["../../src/chrome/sqlite-cookie-copy.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAMH,MAAM,WAAW,gBAAgB;IAC/B,MAAM,EAAE,gBAAgB,GAAG,aAAa,GAAG,WAAW,GAAG,MAAM,CAAC;IAChE,OAAO,EAAE,OAAO,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED;;GAEG;AACH,iBAAS,wBAAwB,CAAC,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,IAAI,CAY5E;AAED;;;GAGG;AACH,eAAO,MAAM,KAAK;;CAEjB,CAAC;AAEF;;;;;;;;;;;GAWG;AACH,wBAAgB,iBAAiB,CAC/B,gBAAgB,EAAE,MAAM,EACxB,cAAc,EAAE,MAAM,GACrB,gBAAgB,CAsDlB"}
@@ -0,0 +1,145 @@
1
+ "use strict";
2
+ /**
3
+ * Atomic SQLite Cookie Copy
4
+ *
5
+ * Chrome stores cookies in a SQLite database using WAL (Write-Ahead Log) mode.
6
+ * The database state spans up to three files: `Cookies`, `Cookies-wal`, and
7
+ * `Cookies-shm`. A naive sequential `fs.copyFileSync` of these files while
8
+ * Chrome is writing produces an inconsistent copy — the main DB and WAL may
9
+ * represent different transaction states, causing Chrome to silently discard
10
+ * the corrupted database and start with an empty cookie jar.
11
+ *
12
+ * This module implements a 3-tier fallback chain for atomic cookie copying:
13
+ * 1. `better-sqlite3` — VACUUM INTO for a clean, synchronous atomic copy
14
+ * 2. `sqlite3` CLI — `.backup` command via child_process
15
+ * 3. Raw `fs.copyFileSync` — last resort, copies main DB only (no WAL/SHM)
16
+ */
17
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
18
+ if (k2 === undefined) k2 = k;
19
+ var desc = Object.getOwnPropertyDescriptor(m, k);
20
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
21
+ desc = { enumerable: true, get: function() { return m[k]; } };
22
+ }
23
+ Object.defineProperty(o, k2, desc);
24
+ }) : (function(o, m, k, k2) {
25
+ if (k2 === undefined) k2 = k;
26
+ o[k2] = m[k];
27
+ }));
28
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
29
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
30
+ }) : function(o, v) {
31
+ o["default"] = v;
32
+ });
33
+ var __importStar = (this && this.__importStar) || (function () {
34
+ var ownKeys = function(o) {
35
+ ownKeys = Object.getOwnPropertyNames || function (o) {
36
+ var ar = [];
37
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
38
+ return ar;
39
+ };
40
+ return ownKeys(o);
41
+ };
42
+ return function (mod) {
43
+ if (mod && mod.__esModule) return mod;
44
+ var result = {};
45
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
46
+ __setModuleDefault(result, mod);
47
+ return result;
48
+ };
49
+ })();
50
+ Object.defineProperty(exports, "__esModule", { value: true });
51
+ exports._deps = void 0;
52
+ exports.copyCookiesAtomic = copyCookiesAtomic;
53
+ const fs = __importStar(require("fs"));
54
+ const path = __importStar(require("path"));
55
+ const child_process_1 = require("child_process");
56
+ /**
57
+ * @internal Attempt atomic copy via better-sqlite3's VACUUM INTO.
58
+ */
59
+ function attemptBetterSqlite3Copy(sourcePath, destPath) {
60
+ // eslint-disable-next-line @typescript-eslint/no-var-requires
61
+ const Database = require('better-sqlite3');
62
+ const db = new Database(sourcePath, { readonly: true, fileMustExist: true });
63
+ try {
64
+ // VACUUM INTO creates a fully checkpointed, single-file copy —
65
+ // no WAL file is produced and no WAL file needs to be present.
66
+ const escapedDest = destPath.replace(/'/g, "''");
67
+ db.exec(`VACUUM INTO '${escapedDest}'`);
68
+ }
69
+ finally {
70
+ db.close();
71
+ }
72
+ }
73
+ /**
74
+ * @internal Mutable dispatch table for testability.
75
+ * Tests can spy on `_deps.attemptBetterSqlite3Copy` to control the fallback chain.
76
+ */
77
+ exports._deps = {
78
+ attemptBetterSqlite3Copy,
79
+ };
80
+ /**
81
+ * Copy the Chrome Cookies SQLite database from one Default profile directory
82
+ * to another using a 3-tier fallback chain for atomic consistency:
83
+ *
84
+ * Tier 1: better-sqlite3 `VACUUM INTO` — synchronous, atomic, WAL-aware
85
+ * Tier 2: sqlite3 CLI `.backup` — atomic, WAL-aware, no npm dep required
86
+ * Tier 3: fs.copyFileSync — fast but non-atomic; recent WAL transactions may be absent
87
+ *
88
+ * @param sourceDefaultDir Path to the source Chrome `Default` profile directory
89
+ * @param destDefaultDir Path to the destination `Default` directory (must already exist)
90
+ * @returns Result indicating which method succeeded, or `none` if all failed
91
+ */
92
+ function copyCookiesAtomic(sourceDefaultDir, destDefaultDir) {
93
+ const sourcePath = path.join(sourceDefaultDir, 'Cookies');
94
+ const destPath = path.join(destDefaultDir, 'Cookies');
95
+ // Guard: source Cookies file must exist
96
+ if (!fs.existsSync(sourcePath)) {
97
+ return {
98
+ method: 'none',
99
+ success: false,
100
+ warning: 'Source Cookies file not found',
101
+ };
102
+ }
103
+ // -------------------------------------------------------------------------
104
+ // Tier 1: better-sqlite3 — VACUUM INTO (synchronous, atomic, WAL-aware)
105
+ // -------------------------------------------------------------------------
106
+ try {
107
+ // VACUUM INTO requires the destination to not exist; remove stale file first.
108
+ if (fs.existsSync(destPath)) {
109
+ fs.unlinkSync(destPath);
110
+ }
111
+ exports._deps.attemptBetterSqlite3Copy(sourcePath, destPath);
112
+ return { method: 'better-sqlite3', success: true };
113
+ }
114
+ catch {
115
+ // better-sqlite3 not installed, or VACUUM INTO failed — fall through
116
+ }
117
+ // -------------------------------------------------------------------------
118
+ // Tier 2: sqlite3 CLI — .backup command
119
+ // -------------------------------------------------------------------------
120
+ try {
121
+ (0, child_process_1.execFileSync)('sqlite3', [sourcePath, `.backup '${destPath.replace(/'/g, "''")}'`], {
122
+ timeout: 5000,
123
+ stdio: 'ignore',
124
+ });
125
+ return { method: 'sqlite3-cli', success: true };
126
+ }
127
+ catch {
128
+ // sqlite3 CLI not available or backup failed — fall through
129
+ }
130
+ // -------------------------------------------------------------------------
131
+ // Tier 3: Raw file copy (last resort — non-atomic, WAL not included)
132
+ // -------------------------------------------------------------------------
133
+ try {
134
+ fs.copyFileSync(sourcePath, destPath);
135
+ // Explicitly do NOT copy WAL/SHM files — an inconsistent WAL applied to
136
+ // a partial main-db snapshot is worse than no WAL at all.
137
+ const warning = 'Used non-atomic file copy (sqlite3 unavailable). Some recent cookies may be missing.';
138
+ return { method: 'file-copy', success: true, warning };
139
+ }
140
+ catch {
141
+ const warning = 'All cookie copy methods failed.';
142
+ return { method: 'none', success: false, warning };
143
+ }
144
+ }
145
+ //# sourceMappingURL=sqlite-cookie-copy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sqlite-cookie-copy.js","sourceRoot":"","sources":["../../src/chrome/sqlite-cookie-copy.ts"],"names":[],"mappings":"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"}
File without changes
package/dist/cli/index.js CHANGED
File without changes
@@ -17,4 +17,12 @@ export declare const DEFAULT_VIEWPORT: {
17
17
  export declare const DEFAULT_NAVIGATION_TIMEOUT_MS = 30000;
18
18
  /** Maximum number of candidate elements returned by element-finding queries. */
19
19
  export declare const MAX_SEARCH_CANDIDATES = 30;
20
+ /** CDP protocol timeout in milliseconds. Prevents 180s default hangs. */
21
+ export declare const DEFAULT_PROTOCOL_TIMEOUT_MS = 30000;
22
+ /** Screenshot-specific timeout. Shorter than protocol timeout for fast fallback. */
23
+ export declare const DEFAULT_SCREENSHOT_TIMEOUT_MS = 15000;
24
+ /** Maximum number of tabs (targets) per worker. Oldest tab is closed when limit is reached. */
25
+ export declare const DEFAULT_MAX_TARGETS_PER_WORKER = 5;
26
+ /** Memory pressure threshold in bytes (500MB). Below this free memory, aggressive cleanup triggers. */
27
+ export declare const DEFAULT_MEMORY_PRESSURE_THRESHOLD: number;
20
28
  //# sourceMappingURL=defaults.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"defaults.d.ts","sourceRoot":"","sources":["../../src/config/defaults.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,+FAA+F;AAC/F,eAAO,MAAM,0BAA0B,KAAK,CAAC;AAE7C,sGAAsG;AACtG,eAAO,MAAM,gBAAgB,QAAQ,CAAC;AAEtC,2CAA2C;AAC3C,eAAO,MAAM,gBAAgB;;;CAAyC,CAAC;AAEvE,kDAAkD;AAClD,eAAO,MAAM,6BAA6B,QAAQ,CAAC;AAEnD,gFAAgF;AAChF,eAAO,MAAM,qBAAqB,KAAK,CAAC"}
1
+ {"version":3,"file":"defaults.d.ts","sourceRoot":"","sources":["../../src/config/defaults.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,+FAA+F;AAC/F,eAAO,MAAM,0BAA0B,KAAK,CAAC;AAE7C,sGAAsG;AACtG,eAAO,MAAM,gBAAgB,QAAQ,CAAC;AAEtC,2CAA2C;AAC3C,eAAO,MAAM,gBAAgB;;;CAAyC,CAAC;AAEvE,kDAAkD;AAClD,eAAO,MAAM,6BAA6B,QAAQ,CAAC;AAEnD,gFAAgF;AAChF,eAAO,MAAM,qBAAqB,KAAK,CAAC;AAExC,yEAAyE;AACzE,eAAO,MAAM,2BAA2B,QAAQ,CAAC;AAEjD,oFAAoF;AACpF,eAAO,MAAM,6BAA6B,QAAQ,CAAC;AAEnD,+FAA+F;AAC/F,eAAO,MAAM,8BAA8B,IAAI,CAAC;AAEhD,uGAAuG;AACvG,eAAO,MAAM,iCAAiC,QAAoB,CAAC"}
@@ -6,7 +6,7 @@
6
6
  * Update this single file instead of hunting across tools/cdp/chrome.
7
7
  */
8
8
  Object.defineProperty(exports, "__esModule", { value: true });
9
- exports.MAX_SEARCH_CANDIDATES = exports.DEFAULT_NAVIGATION_TIMEOUT_MS = exports.DEFAULT_VIEWPORT = exports.MAX_OUTPUT_CHARS = exports.DEFAULT_SCREENSHOT_QUALITY = void 0;
9
+ exports.DEFAULT_MEMORY_PRESSURE_THRESHOLD = exports.DEFAULT_MAX_TARGETS_PER_WORKER = exports.DEFAULT_SCREENSHOT_TIMEOUT_MS = exports.DEFAULT_PROTOCOL_TIMEOUT_MS = exports.MAX_SEARCH_CANDIDATES = exports.DEFAULT_NAVIGATION_TIMEOUT_MS = exports.DEFAULT_VIEWPORT = exports.MAX_OUTPUT_CHARS = exports.DEFAULT_SCREENSHOT_QUALITY = void 0;
10
10
  /** WebP screenshot quality (0-100). Directly affects LLM token consumption via base64 size. */
11
11
  exports.DEFAULT_SCREENSHOT_QUALITY = 60;
12
12
  /** Maximum characters returned in page content output (read_page, DOM serializer, batch_paginate). */
@@ -17,4 +17,12 @@ exports.DEFAULT_VIEWPORT = { width: 1920, height: 1080 };
17
17
  exports.DEFAULT_NAVIGATION_TIMEOUT_MS = 30000;
18
18
  /** Maximum number of candidate elements returned by element-finding queries. */
19
19
  exports.MAX_SEARCH_CANDIDATES = 30;
20
+ /** CDP protocol timeout in milliseconds. Prevents 180s default hangs. */
21
+ exports.DEFAULT_PROTOCOL_TIMEOUT_MS = 30000;
22
+ /** Screenshot-specific timeout. Shorter than protocol timeout for fast fallback. */
23
+ exports.DEFAULT_SCREENSHOT_TIMEOUT_MS = 15000;
24
+ /** Maximum number of tabs (targets) per worker. Oldest tab is closed when limit is reached. */
25
+ exports.DEFAULT_MAX_TARGETS_PER_WORKER = 5;
26
+ /** Memory pressure threshold in bytes (500MB). Below this free memory, aggressive cleanup triggers. */
27
+ exports.DEFAULT_MEMORY_PRESSURE_THRESHOLD = 500 * 1024 * 1024;
20
28
  //# sourceMappingURL=defaults.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"defaults.js","sourceRoot":"","sources":["../../src/config/defaults.ts"],"names":[],"mappings":";AAAA;;;;;GAKG;;;AAEH,+FAA+F;AAClF,QAAA,0BAA0B,GAAG,EAAE,CAAC;AAE7C,sGAAsG;AACzF,QAAA,gBAAgB,GAAG,KAAK,CAAC;AAEtC,2CAA2C;AAC9B,QAAA,gBAAgB,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAW,CAAC;AAEvE,kDAAkD;AACrC,QAAA,6BAA6B,GAAG,KAAK,CAAC;AAEnD,gFAAgF;AACnE,QAAA,qBAAqB,GAAG,EAAE,CAAC"}
1
+ {"version":3,"file":"defaults.js","sourceRoot":"","sources":["../../src/config/defaults.ts"],"names":[],"mappings":";AAAA;;;;;GAKG;;;AAEH,+FAA+F;AAClF,QAAA,0BAA0B,GAAG,EAAE,CAAC;AAE7C,sGAAsG;AACzF,QAAA,gBAAgB,GAAG,KAAK,CAAC;AAEtC,2CAA2C;AAC9B,QAAA,gBAAgB,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAW,CAAC;AAEvE,kDAAkD;AACrC,QAAA,6BAA6B,GAAG,KAAK,CAAC;AAEnD,gFAAgF;AACnE,QAAA,qBAAqB,GAAG,EAAE,CAAC;AAExC,yEAAyE;AAC5D,QAAA,2BAA2B,GAAG,KAAK,CAAC;AAEjD,oFAAoF;AACvE,QAAA,6BAA6B,GAAG,KAAK,CAAC;AAEnD,+FAA+F;AAClF,QAAA,8BAA8B,GAAG,CAAC,CAAC;AAEhD,uGAAuG;AAC1F,QAAA,iCAAiC,GAAG,GAAG,GAAG,IAAI,GAAG,IAAI,CAAC"}
@@ -14,6 +14,8 @@ export interface GlobalConfig {
14
14
  useHeadlessShell?: boolean;
15
15
  /** Run Chrome in headless mode (default: true when auto-launch is enabled) */
16
16
  headless?: boolean;
17
+ /** If true, quit running Chrome to reuse the real profile instead of using temp profile (default: false) */
18
+ restartChrome?: boolean;
17
19
  /** Chrome Pool settings for managing multiple Chrome instances */
18
20
  pool?: {
19
21
  /** Enable the Chrome pool (default: true) */