@web-auto/camo 0.1.18 → 0.1.19
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.
- package/README.md +18 -19
- package/bin/browser-service.mjs +11 -0
- package/package.json +7 -2
- package/scripts/install.mjs +3 -3
- package/src/cli.mjs +8 -5
- package/src/commands/attach.mjs +141 -0
- package/src/commands/browser.mjs +5 -16
- package/src/commands/mouse.mjs +2 -12
- package/src/container/runtime-core/operations/index.mjs +6 -15
- package/src/container/subscription-registry.mjs +6 -6
- package/src/core/actions.mjs +0 -12
- package/src/core/index.mjs +0 -1
- package/src/lifecycle/lock.mjs +7 -3
- package/src/services/browser-service/index.js +651 -0
- package/src/services/browser-service/index.js.map +1 -0
- package/src/services/browser-service/internal/BrowserSession.input.test.js +322 -0
- package/src/services/browser-service/internal/BrowserSession.input.test.js.map +1 -0
- package/src/services/browser-service/internal/BrowserSession.js +304 -0
- package/src/services/browser-service/internal/BrowserSession.js.map +1 -0
- package/src/services/browser-service/internal/ElementRegistry.js +61 -0
- package/src/services/browser-service/internal/ElementRegistry.js.map +1 -0
- package/src/services/browser-service/internal/ProfileLock.js +85 -0
- package/src/services/browser-service/internal/ProfileLock.js.map +1 -0
- package/src/services/browser-service/internal/SessionManager.js +184 -0
- package/src/services/browser-service/internal/SessionManager.js.map +1 -0
- package/src/services/browser-service/internal/SessionManager.test.js +40 -0
- package/src/services/browser-service/internal/SessionManager.test.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/cookies.js +145 -0
- package/src/services/browser-service/internal/browser-session/cookies.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/input-ops.js +127 -0
- package/src/services/browser-service/internal/browser-session/input-ops.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/input-pipeline.js +133 -0
- package/src/services/browser-service/internal/browser-session/input-pipeline.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/logging.js +46 -0
- package/src/services/browser-service/internal/browser-session/navigation.js +39 -0
- package/src/services/browser-service/internal/browser-session/navigation.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/page-hooks.js +443 -0
- package/src/services/browser-service/internal/browser-session/page-hooks.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/page-management.js +212 -0
- package/src/services/browser-service/internal/browser-session/page-management.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/recording.js +199 -0
- package/src/services/browser-service/internal/browser-session/recording.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/runtime-events.js +62 -0
- package/src/services/browser-service/internal/browser-session/runtime-events.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/session-core.js +85 -0
- package/src/services/browser-service/internal/browser-session/session-core.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/session-state.js +39 -0
- package/src/services/browser-service/internal/browser-session/session-state.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/types.js +15 -0
- package/src/services/browser-service/internal/browser-session/types.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/utils.js +69 -0
- package/src/services/browser-service/internal/browser-session/utils.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/viewport-manager.js +47 -0
- package/src/services/browser-service/internal/browser-session/viewport-manager.js.map +1 -0
- package/src/services/browser-service/internal/browser-session/viewport.js +216 -0
- package/src/services/browser-service/internal/browser-session/viewport.js.map +1 -0
- package/src/services/browser-service/internal/container-matcher.js +852 -0
- package/src/services/browser-service/internal/container-matcher.js.map +1 -0
- package/src/services/browser-service/internal/container-registry.js +182 -0
- package/src/services/browser-service/internal/engine-manager.js +259 -0
- package/src/services/browser-service/internal/engine-manager.js.map +1 -0
- package/src/services/browser-service/internal/fingerprint.js +203 -0
- package/src/services/browser-service/internal/fingerprint.js.map +1 -0
- package/src/services/browser-service/internal/heartbeat.js +137 -0
- package/src/services/browser-service/internal/logging.js +46 -0
- package/src/services/browser-service/internal/pageRuntime.js +29 -0
- package/src/services/browser-service/internal/pageRuntime.js.map +1 -0
- package/src/services/browser-service/internal/runtimeInjector.js +30 -0
- package/src/services/browser-service/internal/runtimeInjector.js.map +1 -0
- package/src/services/browser-service/internal/service-process-logger.js +140 -0
- package/src/services/browser-service/internal/state-bus.js +46 -0
- package/src/services/browser-service/internal/state-bus.js.map +1 -0
- package/src/services/browser-service/internal/storage-paths.js +42 -0
- package/src/services/browser-service/internal/storage-paths.js.map +1 -0
- package/src/services/browser-service/internal/ws-server.js +1194 -0
- package/src/services/browser-service/internal/ws-server.js.map +1 -0
- package/src/services/browser-service/internal/ws-server.test.js +59 -0
- package/src/services/browser-service/internal/ws-server.test.js.map +1 -0
- package/src/services/browser-service/server.mjs +6 -0
- package/src/services/controller/cli-bridge.js +93 -0
- package/src/services/controller/container-index.js +50 -0
- package/src/services/controller/container-storage.js +36 -0
- package/src/services/controller/controller-actions.js +207 -0
- package/src/services/controller/controller.js +1138 -0
- package/src/services/controller/selectors.js +54 -0
- package/src/services/controller/transport.js +118 -0
- package/src/utils/browser-service.mjs +100 -125
- package/src/utils/config.mjs +22 -21
- package/src/utils/help.mjs +11 -9
- package/src/utils/ws-client.mjs +30 -0
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
import fs from 'fs';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
import { resolveRecordsRoot } from '../storage-paths.js';
|
|
4
|
+
export class BrowserSessionRecording {
|
|
5
|
+
profileId;
|
|
6
|
+
getCurrentUrl;
|
|
7
|
+
getContext;
|
|
8
|
+
recordingStream = null;
|
|
9
|
+
recording = {
|
|
10
|
+
active: false,
|
|
11
|
+
enabled: false,
|
|
12
|
+
name: null,
|
|
13
|
+
outputPath: null,
|
|
14
|
+
overlay: false,
|
|
15
|
+
startedAt: null,
|
|
16
|
+
endedAt: null,
|
|
17
|
+
eventCount: 0,
|
|
18
|
+
lastEventAt: null,
|
|
19
|
+
lastError: null,
|
|
20
|
+
};
|
|
21
|
+
bindRecorderBridge = () => { };
|
|
22
|
+
installRecorderRuntime = async () => { };
|
|
23
|
+
constructor(profileId, getCurrentUrl, getContext) {
|
|
24
|
+
this.profileId = profileId;
|
|
25
|
+
this.getCurrentUrl = getCurrentUrl;
|
|
26
|
+
this.getContext = getContext;
|
|
27
|
+
}
|
|
28
|
+
setBindRecorderBridge(fn) {
|
|
29
|
+
this.bindRecorderBridge = fn;
|
|
30
|
+
}
|
|
31
|
+
setInstallRecorderRuntime(fn) {
|
|
32
|
+
this.installRecorderRuntime = fn;
|
|
33
|
+
}
|
|
34
|
+
getRecordingStatus() {
|
|
35
|
+
return { ...this.recording };
|
|
36
|
+
}
|
|
37
|
+
normalizeRecordingName(raw) {
|
|
38
|
+
const text = String(raw || '').trim();
|
|
39
|
+
const fallback = `record-${this.profileId}`;
|
|
40
|
+
if (!text)
|
|
41
|
+
return fallback;
|
|
42
|
+
return text.replace(/[^a-zA-Z0-9._-]+/g, '-').replace(/^-+|-+$/g, '').slice(0, 120) || fallback;
|
|
43
|
+
}
|
|
44
|
+
buildRecordingFilename(name) {
|
|
45
|
+
const stamp = new Date().toISOString().replace(/[:.]/g, '-');
|
|
46
|
+
return `${stamp}-${name}.jsonl`;
|
|
47
|
+
}
|
|
48
|
+
resolveRecordingOutputPath(options) {
|
|
49
|
+
const name = this.normalizeRecordingName(options?.name);
|
|
50
|
+
const rawOutput = String(options?.outputPath || '').trim();
|
|
51
|
+
if (!rawOutput) {
|
|
52
|
+
const root = path.join(resolveRecordsRoot(), this.profileId);
|
|
53
|
+
return path.join(root, this.buildRecordingFilename(name));
|
|
54
|
+
}
|
|
55
|
+
const absolute = path.isAbsolute(rawOutput) ? rawOutput : path.resolve(rawOutput);
|
|
56
|
+
if (absolute.endsWith(path.sep)) {
|
|
57
|
+
return path.join(absolute, this.buildRecordingFilename(name));
|
|
58
|
+
}
|
|
59
|
+
if (fs.existsSync(absolute) && fs.statSync(absolute).isDirectory()) {
|
|
60
|
+
return path.join(absolute, this.buildRecordingFilename(name));
|
|
61
|
+
}
|
|
62
|
+
return absolute;
|
|
63
|
+
}
|
|
64
|
+
async startRecording(options = {}) {
|
|
65
|
+
const outputPath = this.resolveRecordingOutputPath(options);
|
|
66
|
+
const name = this.normalizeRecordingName(options?.name);
|
|
67
|
+
const overlay = options?.overlay !== false;
|
|
68
|
+
if (this.recordingStream) {
|
|
69
|
+
await this.stopRecording({ reason: 'restart' });
|
|
70
|
+
}
|
|
71
|
+
await fs.promises.mkdir(path.dirname(outputPath), { recursive: true });
|
|
72
|
+
const stream = fs.createWriteStream(outputPath, { flags: 'a', encoding: 'utf-8' });
|
|
73
|
+
stream.on('error', (err) => {
|
|
74
|
+
this.recording.lastError = err?.message || String(err);
|
|
75
|
+
this.recording.enabled = false;
|
|
76
|
+
});
|
|
77
|
+
this.recordingStream = stream;
|
|
78
|
+
this.recording = {
|
|
79
|
+
active: true,
|
|
80
|
+
enabled: true,
|
|
81
|
+
name,
|
|
82
|
+
outputPath,
|
|
83
|
+
overlay,
|
|
84
|
+
startedAt: Date.now(),
|
|
85
|
+
endedAt: null,
|
|
86
|
+
eventCount: 0,
|
|
87
|
+
lastEventAt: null,
|
|
88
|
+
lastError: null,
|
|
89
|
+
};
|
|
90
|
+
const context = this.getContext();
|
|
91
|
+
if (context) {
|
|
92
|
+
const pages = context.pages().filter((p) => !p.isClosed());
|
|
93
|
+
for (const page of pages) {
|
|
94
|
+
this.bindRecorderBridge(page);
|
|
95
|
+
// eslint-disable-next-line no-await-in-loop
|
|
96
|
+
await this.installRecorderRuntime(page, 'recording_start').catch(() => { });
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
this.writeRecordingEvent('recording.start', { name, outputPath, overlay }, { allowWhenDisabled: true });
|
|
100
|
+
return this.getRecordingStatus();
|
|
101
|
+
}
|
|
102
|
+
async stopRecording(options = {}) {
|
|
103
|
+
if (!this.recordingStream) {
|
|
104
|
+
this.recording.active = false;
|
|
105
|
+
this.recording.enabled = false;
|
|
106
|
+
this.recording.overlay = false;
|
|
107
|
+
this.recording.endedAt = Date.now();
|
|
108
|
+
const context = this.getContext();
|
|
109
|
+
if (context) {
|
|
110
|
+
const pages = context.pages().filter((p) => !p.isClosed());
|
|
111
|
+
for (const page of pages) {
|
|
112
|
+
// eslint-disable-next-line no-await-in-loop
|
|
113
|
+
await this.destroyRecorderRuntimeOnPage(page).catch(() => { });
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
return this.getRecordingStatus();
|
|
117
|
+
}
|
|
118
|
+
this.writeRecordingEvent('recording.stop', { reason: options.reason || 'manual' }, { allowWhenDisabled: true });
|
|
119
|
+
this.recording.enabled = false;
|
|
120
|
+
this.recording.active = false;
|
|
121
|
+
this.recording.overlay = false;
|
|
122
|
+
this.recording.endedAt = Date.now();
|
|
123
|
+
const context = this.getContext();
|
|
124
|
+
if (context) {
|
|
125
|
+
const pages = context.pages().filter((p) => !p.isClosed());
|
|
126
|
+
for (const page of pages) {
|
|
127
|
+
// eslint-disable-next-line no-await-in-loop
|
|
128
|
+
await this.destroyRecorderRuntimeOnPage(page).catch(() => { });
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
const stream = this.recordingStream;
|
|
132
|
+
this.recordingStream = null;
|
|
133
|
+
await new Promise((resolve) => {
|
|
134
|
+
stream.end(() => resolve());
|
|
135
|
+
});
|
|
136
|
+
return this.getRecordingStatus();
|
|
137
|
+
}
|
|
138
|
+
writeRecordingEvent(type, payload = {}, options = {}) {
|
|
139
|
+
if (!this.recordingStream || !this.recording.active)
|
|
140
|
+
return;
|
|
141
|
+
if (!this.recording.enabled && !options.allowWhenDisabled)
|
|
142
|
+
return;
|
|
143
|
+
const eventTs = Date.now();
|
|
144
|
+
const entry = {
|
|
145
|
+
ts: eventTs,
|
|
146
|
+
profileId: this.profileId,
|
|
147
|
+
sessionId: this.profileId,
|
|
148
|
+
type,
|
|
149
|
+
url: options.pageUrl || this.getCurrentUrl() || null,
|
|
150
|
+
payload,
|
|
151
|
+
};
|
|
152
|
+
try {
|
|
153
|
+
this.recordingStream.write(`${JSON.stringify(entry)}\n`);
|
|
154
|
+
this.recording.eventCount += 1;
|
|
155
|
+
this.recording.lastEventAt = eventTs;
|
|
156
|
+
}
|
|
157
|
+
catch (err) {
|
|
158
|
+
this.recording.lastError = err?.message || String(err);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
handleRecorderEvent(page, evt) {
|
|
162
|
+
const type = String(evt?.type || '').trim();
|
|
163
|
+
if (!type)
|
|
164
|
+
return;
|
|
165
|
+
const pageUrl = String(evt?.href || page?.url?.() || this.getCurrentUrl() || '');
|
|
166
|
+
const payload = evt?.payload && typeof evt.payload === 'object' ? evt.payload : {};
|
|
167
|
+
if (type === 'recording.toggled') {
|
|
168
|
+
if (!this.recording.active) {
|
|
169
|
+
this.recording.enabled = false;
|
|
170
|
+
return;
|
|
171
|
+
}
|
|
172
|
+
this.recording.enabled = payload.enabled !== false;
|
|
173
|
+
this.writeRecordingEvent(type, payload, { pageUrl, allowWhenDisabled: true });
|
|
174
|
+
return;
|
|
175
|
+
}
|
|
176
|
+
if (type === 'recording.runtime_ready') {
|
|
177
|
+
this.writeRecordingEvent(type, payload, { pageUrl, allowWhenDisabled: true });
|
|
178
|
+
return;
|
|
179
|
+
}
|
|
180
|
+
this.writeRecordingEvent(type, payload, { pageUrl });
|
|
181
|
+
}
|
|
182
|
+
recordPageVisit(page, reason) {
|
|
183
|
+
const pageUrl = page?.url?.() || this.getCurrentUrl() || null;
|
|
184
|
+
if (!pageUrl)
|
|
185
|
+
return;
|
|
186
|
+
this.writeRecordingEvent('page.visit', { reason, title: null }, { pageUrl });
|
|
187
|
+
}
|
|
188
|
+
async destroyRecorderRuntimeOnPage(page) {
|
|
189
|
+
if (!page || page.isClosed())
|
|
190
|
+
return;
|
|
191
|
+
await page.evaluate(() => {
|
|
192
|
+
const runtime = window.__camoRecorderV1__;
|
|
193
|
+
if (!runtime || typeof runtime.destroy !== 'function')
|
|
194
|
+
return null;
|
|
195
|
+
return runtime.destroy();
|
|
196
|
+
});
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
//# sourceMappingURL=recording.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"recording.js","sourceRoot":"","sources":["../../../../../../modules/camo-backend/src/internal/browser-session/recording.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { getStateBus } from '../state-bus.js';
|
|
2
|
+
import { logDebug } from '../logging.js';
|
|
3
|
+
const stateBus = getStateBus();
|
|
4
|
+
export function createRuntimeEventManager(sessionId) {
|
|
5
|
+
const observers = new Set();
|
|
6
|
+
function addObserver(observer) {
|
|
7
|
+
observers.add(observer);
|
|
8
|
+
logDebug('browser-service', 'runtimeObserver:add', { sessionId, total: observers.size });
|
|
9
|
+
return () => {
|
|
10
|
+
observers.delete(observer);
|
|
11
|
+
logDebug('browser-service', 'runtimeObserver:remove', { sessionId, total: observers.size });
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
function emit(event) {
|
|
15
|
+
const payload = {
|
|
16
|
+
ts: Date.now(),
|
|
17
|
+
sessionId,
|
|
18
|
+
...event,
|
|
19
|
+
};
|
|
20
|
+
logDebug('browser-service', 'runtimeEvent', {
|
|
21
|
+
sessionId,
|
|
22
|
+
type: event?.type || 'unknown',
|
|
23
|
+
observers: observers.size,
|
|
24
|
+
});
|
|
25
|
+
observers.forEach((observer) => {
|
|
26
|
+
try {
|
|
27
|
+
observer(payload);
|
|
28
|
+
}
|
|
29
|
+
catch (err) {
|
|
30
|
+
console.warn('[BrowserSession] runtime observer error', err);
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
publishState(payload);
|
|
34
|
+
}
|
|
35
|
+
function publishState(payload) {
|
|
36
|
+
try {
|
|
37
|
+
stateBus.setState(`browser-session:${sessionId}`, {
|
|
38
|
+
status: 'running',
|
|
39
|
+
lastRuntimeEvent: payload?.type || 'unknown',
|
|
40
|
+
lastUrl: payload?.pageUrl || '',
|
|
41
|
+
lastUpdate: payload?.ts || Date.now(),
|
|
42
|
+
});
|
|
43
|
+
stateBus.publish('browser.runtime.event', payload);
|
|
44
|
+
}
|
|
45
|
+
catch (err) {
|
|
46
|
+
logDebug('browser-service', 'runtimeEvent:stateBus:error', {
|
|
47
|
+
sessionId,
|
|
48
|
+
error: err?.message || err,
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
function clearObservers() {
|
|
53
|
+
observers.clear();
|
|
54
|
+
}
|
|
55
|
+
return {
|
|
56
|
+
addObserver,
|
|
57
|
+
emit,
|
|
58
|
+
publishState,
|
|
59
|
+
clearObservers,
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
//# sourceMappingURL=runtime-events.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runtime-events.js","sourceRoot":"","sources":["../../../../../../modules/camo-backend/src/internal/browser-session/runtime-events.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAE,QAAQ,EAAE,MAAM,kCAAkC,CAAC;AAE5D,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;AAU/B,MAAM,UAAU,yBAAyB,CAAC,SAAiB;IACzD,MAAM,SAAS,GAAG,IAAI,GAAG,EAAwC,CAAC;IAElE,SAAS,WAAW,CAAC,QAA8C;QACjE,SAAS,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACxB,QAAQ,CAAC,iBAAiB,EAAE,qBAAqB,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC;QACzF,OAAO,GAAG,EAAE;YACV,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;YAC3B,QAAQ,CAAC,iBAAiB,EAAE,wBAAwB,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC;QAC9F,CAAC,CAAC;IACJ,CAAC;IAED,SAAS,IAAI,CAAC,KAA0B;QACtC,MAAM,OAAO,GAAwB;YACnC,EAAE,EAAE,IAAI,CAAC,GAAG,EAAE;YACd,SAAS;YACT,GAAG,KAAK;SACT,CAAC;QACF,QAAQ,CAAC,iBAAiB,EAAE,cAAc,EAAE;YAC1C,SAAS;YACT,IAAI,EAAE,KAAK,EAAE,IAAI,IAAI,SAAS;YAC9B,SAAS,EAAE,SAAS,CAAC,IAAI;SAC1B,CAAC,CAAC;QACH,SAAS,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE;YAC7B,IAAI,CAAC;gBACH,QAAQ,CAAC,OAAO,CAAC,CAAC;YACpB,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,OAAO,CAAC,IAAI,CAAC,yCAAyC,EAAE,GAAG,CAAC,CAAC;YAC/D,CAAC;QACH,CAAC,CAAC,CAAC;QACH,YAAY,CAAC,OAAO,CAAC,CAAC;IACxB,CAAC;IAED,SAAS,YAAY,CAAC,OAA4B;QAChD,IAAI,CAAC;YACH,QAAQ,CAAC,QAAQ,CAAC,mBAAmB,SAAS,EAAE,EAAE;gBAChD,MAAM,EAAE,SAAS;gBACjB,gBAAgB,EAAE,OAAO,EAAE,IAAI,IAAI,SAAS;gBAC5C,OAAO,EAAE,OAAO,EAAE,OAAO,IAAI,EAAE;gBAC/B,UAAU,EAAE,OAAO,EAAE,EAAE,IAAI,IAAI,CAAC,GAAG,EAAE;aACtC,CAAC,CAAC;YACH,QAAQ,CAAC,OAAO,CAAC,uBAAuB,EAAE,OAAO,CAAC,CAAC;QACrD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,QAAQ,CAAC,iBAAiB,EAAE,6BAA6B,EAAE;gBACzD,SAAS;gBACT,KAAK,EAAG,GAAa,EAAE,OAAO,IAAI,GAAG;aACtC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,SAAS,cAAc;QACrB,SAAS,CAAC,KAAK,EAAE,CAAC;IACpB,CAAC;IAED,OAAO;QACL,WAAW;QACX,IAAI;QACJ,YAAY;QACZ,cAAc;KACf,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { ProfileLock } from '../profile-lock';
|
|
2
|
+
import { resolveProfilesRoot } from '../paths';
|
|
3
|
+
export class BrowserSessionCore {
|
|
4
|
+
options;
|
|
5
|
+
browser;
|
|
6
|
+
context;
|
|
7
|
+
page;
|
|
8
|
+
lock;
|
|
9
|
+
profileDir;
|
|
10
|
+
lastKnownUrl = null;
|
|
11
|
+
mode = 'dev';
|
|
12
|
+
recording = {
|
|
13
|
+
active: false,
|
|
14
|
+
enabled: false,
|
|
15
|
+
name: null,
|
|
16
|
+
outputPath: null,
|
|
17
|
+
overlay: false,
|
|
18
|
+
startedAt: null,
|
|
19
|
+
endedAt: null,
|
|
20
|
+
eventCount: 0,
|
|
21
|
+
lastEventAt: null,
|
|
22
|
+
lastError: null,
|
|
23
|
+
};
|
|
24
|
+
exitNotified = false;
|
|
25
|
+
runtimeObservers = new Set();
|
|
26
|
+
onExit;
|
|
27
|
+
constructor(options) {
|
|
28
|
+
this.options = options;
|
|
29
|
+
const profileId = options.profileId || 'default';
|
|
30
|
+
const root = resolveProfilesRoot();
|
|
31
|
+
this.profileDir = `${root}/${profileId}`;
|
|
32
|
+
fs.mkdirSync(this.profileDir, { recursive: true });
|
|
33
|
+
this.lock = new ProfileLock(profileId);
|
|
34
|
+
}
|
|
35
|
+
get id() {
|
|
36
|
+
return this.options.profileId;
|
|
37
|
+
}
|
|
38
|
+
get currentPage() {
|
|
39
|
+
return this.page;
|
|
40
|
+
}
|
|
41
|
+
get modeName() {
|
|
42
|
+
return this.mode;
|
|
43
|
+
}
|
|
44
|
+
setMode(next = 'dev') {
|
|
45
|
+
this.mode = next === 'run' ? 'run' : 'dev';
|
|
46
|
+
}
|
|
47
|
+
getInfo() {
|
|
48
|
+
return {
|
|
49
|
+
session_id: this.options.profileId,
|
|
50
|
+
profileId: this.options.profileId,
|
|
51
|
+
current_url: this.getCurrentUrl(),
|
|
52
|
+
mode: this.mode,
|
|
53
|
+
headless: !!this.options.headless,
|
|
54
|
+
recording: this.getRecordingStatus(),
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
getRecordingStatus() {
|
|
58
|
+
return { ...this.recording };
|
|
59
|
+
}
|
|
60
|
+
async close() {
|
|
61
|
+
try {
|
|
62
|
+
await this.stopRecording({ reason: 'session_close' }).catch(() => { });
|
|
63
|
+
await this.context?.close();
|
|
64
|
+
}
|
|
65
|
+
finally {
|
|
66
|
+
await this.browser?.close();
|
|
67
|
+
this.lock.release();
|
|
68
|
+
this.runtimeObservers.clear();
|
|
69
|
+
this.notifyExit();
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
notifyExit() {
|
|
73
|
+
if (this.exitNotified)
|
|
74
|
+
return;
|
|
75
|
+
this.exitNotified = true;
|
|
76
|
+
this.onExit?.(this.options.profileId);
|
|
77
|
+
}
|
|
78
|
+
addRuntimeEventObserver(observer) {
|
|
79
|
+
this.runtimeObservers.add(observer);
|
|
80
|
+
return () => {
|
|
81
|
+
this.runtimeObservers.delete(observer);
|
|
82
|
+
};
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
//# sourceMappingURL=session-core.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-core.js","sourceRoot":"","sources":["../../../../../../modules/camo-backend/src/internal/browser-session/session-core.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import { createEmptyRecordingState } from './types.js';
|
|
2
|
+
export function createInitialSessionState(options) {
|
|
3
|
+
return {
|
|
4
|
+
options,
|
|
5
|
+
browser: undefined,
|
|
6
|
+
context: undefined,
|
|
7
|
+
page: undefined,
|
|
8
|
+
mode: 'dev',
|
|
9
|
+
lastKnownUrl: null,
|
|
10
|
+
lastCookieSignature: null,
|
|
11
|
+
lastCookieSaveTs: 0,
|
|
12
|
+
lastViewport: {
|
|
13
|
+
lastViewport: null,
|
|
14
|
+
followWindowViewport: !options.headless,
|
|
15
|
+
},
|
|
16
|
+
fingerprint: null,
|
|
17
|
+
recording: createEmptyRecordingState(),
|
|
18
|
+
recordingStream: null,
|
|
19
|
+
exitNotified: false,
|
|
20
|
+
wheelMode: String(process.env.CAMO_SCROLL_INPUT_MODE || '').trim().toLowerCase() === 'keyboard'
|
|
21
|
+
? 'keyboard'
|
|
22
|
+
: 'wheel',
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
export function getActivePage(state) {
|
|
26
|
+
if (state.page && !state.page.isClosed()) {
|
|
27
|
+
return state.page;
|
|
28
|
+
}
|
|
29
|
+
if (!state.context)
|
|
30
|
+
return null;
|
|
31
|
+
const alive = state.context.pages().find((p) => !p.isClosed());
|
|
32
|
+
if (alive) {
|
|
33
|
+
state.page = alive;
|
|
34
|
+
return alive;
|
|
35
|
+
}
|
|
36
|
+
state.page = undefined;
|
|
37
|
+
return null;
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=session-state.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-state.js","sourceRoot":"","sources":["../../../../../../modules/camo-backend/src/internal/browser-session/session-state.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,yBAAyB,EAAE,MAAM,YAAY,CAAC;AAoBvD,MAAM,UAAU,yBAAyB,CAAC,OAA8B;IACtE,OAAO;QACL,OAAO;QACP,OAAO,EAAE,SAAS;QAClB,OAAO,EAAE,SAAS;QAClB,IAAI,EAAE,SAAS;QACf,IAAI,EAAE,KAAK;QACX,YAAY,EAAE,IAAI;QAClB,mBAAmB,EAAE,IAAI;QACzB,gBAAgB,EAAE,CAAC;QACnB,YAAY,EAAE;YACZ,YAAY,EAAE,IAAI;YAClB,oBAAoB,EAAE,CAAC,OAAO,CAAC,QAAQ;SACxC;QACD,WAAW,EAAE,IAAI;QACjB,SAAS,EAAE,yBAAyB,EAAE;QACtC,eAAe,EAAE,IAAI;QACrB,YAAY,EAAE,KAAK;QACnB,SAAS,EAAE,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,KAAK,UAAU;YAC7F,CAAC,CAAC,UAAU;YACZ,CAAC,CAAC,OAAO;KACZ,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAmB;IAC/C,IAAI,KAAK,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;QACzC,OAAO,KAAK,CAAC,IAAI,CAAC;IACpB,CAAC;IACD,IAAI,CAAC,KAAK,CAAC,OAAO;QAAE,OAAO,IAAI,CAAC;IAChC,MAAM,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;IAC/D,IAAI,KAAK,EAAE,CAAC;QACV,KAAK,CAAC,IAAI,GAAG,KAAK,CAAC;QACnB,OAAO,KAAK,CAAC;IACf,CAAC;IACD,KAAK,CAAC,IAAI,GAAG,SAAS,CAAC;IACvB,OAAO,IAAI,CAAC;AACd,CAAC"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
export function createEmptyRecordingState() {
|
|
2
|
+
return {
|
|
3
|
+
active: false,
|
|
4
|
+
enabled: false,
|
|
5
|
+
name: null,
|
|
6
|
+
outputPath: null,
|
|
7
|
+
overlay: false,
|
|
8
|
+
startedAt: null,
|
|
9
|
+
endedAt: null,
|
|
10
|
+
eventCount: 0,
|
|
11
|
+
lastEventAt: null,
|
|
12
|
+
lastError: null,
|
|
13
|
+
};
|
|
14
|
+
}
|
|
15
|
+
//# sourceMappingURL=types.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../modules/camo-backend/src/internal/browser-session/types.ts"],"names":[],"mappings":"AA+BA,MAAM,UAAU,yBAAyB;IACvC,OAAO;QACL,MAAM,EAAE,KAAK;QACb,OAAO,EAAE,KAAK;QACd,IAAI,EAAE,IAAI;QACV,UAAU,EAAE,IAAI;QAChB,OAAO,EAAE,KAAK;QACd,SAAS,EAAE,IAAI;QACf,OAAO,EAAE,IAAI;QACb,UAAU,EAAE,CAAC;QACb,WAAW,EAAE,IAAI;QACjB,SAAS,EAAE,IAAI;KAChB,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
export function resolveInputActionTimeoutMs() {
|
|
2
|
+
const raw = Number(process.env.CAMO_INPUT_ACTION_TIMEOUT_MS ?? process.env.CAMO_API_TIMEOUT_MS ?? 30000);
|
|
3
|
+
return Math.max(1000, Number.isFinite(raw) ? raw : 30000);
|
|
4
|
+
}
|
|
5
|
+
export function resolveNavigationWaitUntil() {
|
|
6
|
+
const raw = String(process.env.CAMO_NAV_WAIT_UNTIL ?? 'commit').trim().toLowerCase();
|
|
7
|
+
if (raw === 'load')
|
|
8
|
+
return 'load';
|
|
9
|
+
if (raw === 'domcontentloaded' || raw === 'dom')
|
|
10
|
+
return 'domcontentloaded';
|
|
11
|
+
if (raw === 'networkidle')
|
|
12
|
+
return 'networkidle';
|
|
13
|
+
return 'commit';
|
|
14
|
+
}
|
|
15
|
+
export function resolveInputActionMaxAttempts() {
|
|
16
|
+
const raw = Number(process.env.CAMO_INPUT_ACTION_MAX_ATTEMPTS ?? 2);
|
|
17
|
+
return Math.max(1, Math.min(3, Number.isFinite(raw) ? Math.floor(raw) : 2));
|
|
18
|
+
}
|
|
19
|
+
export function resolveInputRecoveryDelayMs() {
|
|
20
|
+
const raw = Number(process.env.CAMO_INPUT_RECOVERY_DELAY_MS ?? 120);
|
|
21
|
+
return Math.max(0, Number.isFinite(raw) ? Math.floor(raw) : 120);
|
|
22
|
+
}
|
|
23
|
+
export function resolveInputRecoveryBringToFrontTimeoutMs() {
|
|
24
|
+
const raw = Number(process.env.CAMO_INPUT_RECOVERY_BRING_TO_FRONT_TIMEOUT_MS ?? 800);
|
|
25
|
+
return Math.max(100, Number.isFinite(raw) ? Math.floor(raw) : 800);
|
|
26
|
+
}
|
|
27
|
+
export function resolveInputReadySettleMs() {
|
|
28
|
+
const raw = Number(process.env.CAMO_INPUT_READY_SETTLE_MS ?? 80);
|
|
29
|
+
return Math.max(0, Number.isFinite(raw) ? Math.floor(raw) : 80);
|
|
30
|
+
}
|
|
31
|
+
export function isTimeoutLikeError(error) {
|
|
32
|
+
const message = String(error?.message || error || '').toLowerCase();
|
|
33
|
+
return message.includes('timed out') || message.includes('timeout');
|
|
34
|
+
}
|
|
35
|
+
export function normalizeUrl(raw) {
|
|
36
|
+
try {
|
|
37
|
+
const url = new URL(raw);
|
|
38
|
+
return `${url.origin}${url.pathname}`;
|
|
39
|
+
}
|
|
40
|
+
catch {
|
|
41
|
+
return raw;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
export async function ensureInputReadyOnPage(page, headless, bringToFrontTimeoutMs, settleMs) {
|
|
45
|
+
if (headless)
|
|
46
|
+
return;
|
|
47
|
+
let bringToFrontTimer = null;
|
|
48
|
+
try {
|
|
49
|
+
await Promise.race([
|
|
50
|
+
page.bringToFront(),
|
|
51
|
+
new Promise((_resolve, reject) => {
|
|
52
|
+
bringToFrontTimer = setTimeout(() => {
|
|
53
|
+
reject(new Error(`input ready bringToFront timed out after ${bringToFrontTimeoutMs}ms`));
|
|
54
|
+
}, bringToFrontTimeoutMs);
|
|
55
|
+
}),
|
|
56
|
+
]);
|
|
57
|
+
}
|
|
58
|
+
catch {
|
|
59
|
+
// Best-effort only
|
|
60
|
+
}
|
|
61
|
+
finally {
|
|
62
|
+
if (bringToFrontTimer)
|
|
63
|
+
clearTimeout(bringToFrontTimer);
|
|
64
|
+
}
|
|
65
|
+
if (settleMs > 0) {
|
|
66
|
+
await page.waitForTimeout(settleMs).catch(() => { });
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=utils.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../modules/camo-backend/src/internal/browser-session/utils.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { ensurePageViewport, refreshViewportFromWindow, setViewportSizeOnPage, maybeCenterWindow } from './viewport.js';
|
|
2
|
+
export class BrowserSessionViewportManager {
|
|
3
|
+
profileId;
|
|
4
|
+
getContext;
|
|
5
|
+
getEngine;
|
|
6
|
+
isHeadless;
|
|
7
|
+
state = { lastViewport: null, followWindowViewport: false };
|
|
8
|
+
constructor(profileId, getContext, getEngine, isHeadless) {
|
|
9
|
+
this.profileId = profileId;
|
|
10
|
+
this.getContext = getContext;
|
|
11
|
+
this.getEngine = getEngine;
|
|
12
|
+
this.isHeadless = isHeadless;
|
|
13
|
+
}
|
|
14
|
+
setInitialViewport(viewport, followWindowViewport) {
|
|
15
|
+
this.state.followWindowViewport = followWindowViewport;
|
|
16
|
+
this.state.lastViewport = followWindowViewport
|
|
17
|
+
? null
|
|
18
|
+
: { width: viewport.width, height: viewport.height };
|
|
19
|
+
}
|
|
20
|
+
isFollowingWindow() {
|
|
21
|
+
return this.state.followWindowViewport;
|
|
22
|
+
}
|
|
23
|
+
getLastViewport() {
|
|
24
|
+
return this.state.lastViewport;
|
|
25
|
+
}
|
|
26
|
+
async refreshFromWindow(page) {
|
|
27
|
+
const refreshed = await refreshViewportFromWindow(page).catch(() => null);
|
|
28
|
+
if (refreshed) {
|
|
29
|
+
this.state.lastViewport = refreshed;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
async ensurePageViewport(page) {
|
|
33
|
+
this.state = await ensurePageViewport(page, this.state, this.getContext(), this.getEngine(), this.isHeadless());
|
|
34
|
+
}
|
|
35
|
+
async setViewportSize(page, opts) {
|
|
36
|
+
const next = await setViewportSizeOnPage(page, opts, this.state, this.getContext(), this.getEngine(), this.isHeadless());
|
|
37
|
+
this.state.lastViewport = next;
|
|
38
|
+
return next;
|
|
39
|
+
}
|
|
40
|
+
async maybeCenter(page, fallback) {
|
|
41
|
+
const target = this.state.lastViewport || fallback;
|
|
42
|
+
if (!target)
|
|
43
|
+
return;
|
|
44
|
+
await maybeCenterWindow(page, target, this.isHeadless());
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
//# sourceMappingURL=viewport-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"viewport-manager.js","sourceRoot":"","sources":["../../../../../../modules/camo-backend/src/internal/browser-session/viewport-manager.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,kBAAkB,EAAE,yBAAyB,EAAE,qBAAqB,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAExH,MAAM,OAAO,6BAA6B;IAI9B;IACA;IACA;IACA;IANF,KAAK,GAAkB,EAAE,YAAY,EAAE,IAAI,EAAE,oBAAoB,EAAE,KAAK,EAAE,CAAC;IAEnF,YACU,SAAiB,EACjB,UAA4C,EAC5C,SAAuB,EACvB,UAAyB;QAHzB,cAAS,GAAT,SAAS,CAAQ;QACjB,eAAU,GAAV,UAAU,CAAkC;QAC5C,cAAS,GAAT,SAAS,CAAc;QACvB,eAAU,GAAV,UAAU,CAAe;IAChC,CAAC;IAEJ,kBAAkB,CAAC,QAA2C,EAAE,oBAA6B;QAC3F,IAAI,CAAC,KAAK,CAAC,oBAAoB,GAAG,oBAAoB,CAAC;QACvD,IAAI,CAAC,KAAK,CAAC,YAAY,GAAG,oBAAoB;YAC5C,CAAC,CAAC,IAAI;YACN,CAAC,CAAC,EAAE,KAAK,EAAE,QAAQ,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,CAAC,MAAM,EAAE,CAAC;IACzD,CAAC;IAED,iBAAiB;QACf,OAAO,IAAI,CAAC,KAAK,CAAC,oBAAoB,CAAC;IACzC,CAAC;IAED,eAAe;QACb,OAAO,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC;IACjC,CAAC;IAED,KAAK,CAAC,iBAAiB,CAAC,IAAU;QAChC,MAAM,SAAS,GAAG,MAAM,yBAAyB,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,GAAS,EAAE,CAAC,IAAI,CAAC,CAAC;QAChF,IAAI,SAAS,EAAE,CAAC;YACd,IAAI,CAAC,KAAK,CAAC,YAAY,GAAG,SAAS,CAAC;QACtC,CAAC;IACH,CAAC;IAED,KAAK,CAAC,kBAAkB,CAAC,IAAU;QACjC,IAAI,CAAC,KAAK,GAAG,MAAM,kBAAkB,CACnC,IAAI,EACJ,IAAI,CAAC,KAAK,EACV,IAAI,CAAC,UAAU,EAAE,EACjB,IAAI,CAAC,SAAS,EAAE,EAChB,IAAI,CAAC,UAAU,EAAE,CAClB,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,IAAU,EAAE,IAAuC;QACvE,MAAM,IAAI,GAAG,MAAM,qBAAqB,CACtC,IAAI,EACJ,IAAI,EACJ,IAAI,CAAC,KAAK,EACV,IAAI,CAAC,UAAU,EAAE,EACjB,IAAI,CAAC,SAAS,EAAE,EAChB,IAAI,CAAC,UAAU,EAAE,CAClB,CAAC;QACF,IAAI,CAAC,KAAK,CAAC,YAAY,GAAG,IAAI,CAAC;QAC/B,OAAO,IAAI,CAAC;IACd,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,IAAU,EAAE,QAA4C;QACxE,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,IAAI,QAAQ,CAAC;QACnD,IAAI,CAAC,MAAM;YAAE,OAAO;QACpB,MAAM,iBAAiB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC;IAC3D,CAAC;CACF"}
|