@myclaw163/clawclaw-cli 0.6.55 → 0.6.57
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 +440 -440
- package/bin/clawclaw-cli.mjs +3 -3
- package/package.json +48 -48
- package/personas//347/220/206/346/231/272/346/270/251/345/222/214.md +23 -23
- package/personas//350/200/201/350/260/213/346/267/261/347/256/227.md +22 -22
- package/personas//350/257/232/346/201/263/347/233/264/347/216/207.md +22 -22
- package/personas//350/275/273/346/235/276/346/264/273/346/263/274.md +22 -22
- package/personas//351/207/216/346/200/247/345/217/233/351/200/206.md +23 -23
- package/scripts/postinstall.mjs +20 -20
- package/scripts/sync-bundled-skill.mjs +245 -245
- package/scripts/sync-bundled-skill.test.mjs +152 -152
- package/skills/clawclaw/SKILL.md +244 -240
- package/skills/clawclaw/references/CHATTERBOX.md +142 -142
- package/skills/clawclaw/references/COMMANDS.md +132 -132
- package/skills/clawclaw/references/GAME-MECHANICS.md +186 -186
- package/skills/clawclaw/references/HUB.md +48 -48
- package/skills/clawclaw/references/KNOWLEDGE.md +43 -43
- package/skills/clawclaw/references/STRATEGIES.md +57 -57
- package/skills/clawclaw/references/STREAM.md +59 -58
- package/skills/clawclaw/references/TACTICS.md +65 -65
- package/src/assets/clawclaw-ascii-map.txt +40 -40
- package/src/cli.ts +110 -153
- package/src/commands/_schema.ts +109 -109
- package/src/commands/account.ts +209 -209
- package/src/commands/do.test.ts +37 -37
- package/src/commands/do.ts +95 -95
- package/src/commands/events.ts +22 -22
- package/src/commands/game-map.test.ts +28 -28
- package/src/commands/game-start-plan.test.ts +84 -142
- package/src/commands/game.ts +1027 -882
- package/src/commands/history-player.test.ts +102 -102
- package/src/commands/history.ts +573 -573
- package/src/commands/hub.test.ts +96 -96
- package/src/commands/hub.ts +234 -234
- package/src/commands/knowledge.test.ts +19 -19
- package/src/commands/knowledge.ts +168 -168
- package/src/commands/load.test.ts +51 -51
- package/src/commands/load.ts +13 -13
- package/src/commands/meeting-history.test.ts +106 -106
- package/src/commands/memory.ts +40 -40
- package/src/commands/peek.ts +45 -38
- package/src/commands/persona.ts +57 -57
- package/src/commands/skill.ts +128 -128
- package/src/commands/state.ts +46 -46
- package/src/commands/strategy.test.ts +135 -135
- package/src/commands/strategy.ts +180 -189
- package/src/commands/tts.ts +128 -128
- package/src/commands/upgrade.test.ts +82 -82
- package/src/commands/upgrade.ts +148 -148
- package/src/commands/watch.test.ts +969 -973
- package/src/commands/watch.ts +720 -709
- package/src/lib/auth.test.ts +59 -59
- package/src/lib/auth.ts +186 -186
- package/src/lib/command-meta.ts +37 -37
- package/src/lib/game-client.ts +391 -391
- package/src/lib/http-keepalive.ts +15 -15
- package/src/lib/http-transport.test.ts +42 -42
- package/src/lib/http-transport.ts +113 -113
- package/src/lib/hub-client.test.ts +56 -56
- package/src/lib/hub-client.ts +88 -88
- package/src/lib/hub-install.test.ts +98 -98
- package/src/lib/hub-install.ts +121 -121
- package/src/lib/hub-reminder.ts +75 -75
- package/src/lib/hub-unzip.test.ts +69 -69
- package/src/lib/hub-unzip.ts +62 -62
- package/src/lib/init-command.test.ts +75 -75
- package/src/lib/init-command.ts +120 -120
- package/src/lib/knowledge-store.test.ts +180 -180
- package/src/lib/knowledge-store.ts +374 -374
- package/src/lib/load-context.test.ts +52 -52
- package/src/lib/load-context.ts +52 -52
- package/src/lib/match-state.test.ts +134 -134
- package/src/lib/match-state.ts +94 -94
- package/src/lib/netease-tts.ts +83 -83
- package/src/lib/normalize.ts +42 -42
- package/src/lib/persona.test.ts +41 -41
- package/src/lib/persona.ts +72 -72
- package/src/lib/server-registry.ts +152 -152
- package/src/lib/skill-version.test.ts +48 -48
- package/src/lib/skill-version.ts +19 -19
- package/src/lib/strategy-export.test.ts +232 -232
- package/src/lib/strategy-export.ts +242 -242
- package/src/lib/tts-keys.ts +7 -7
- package/src/lib/tts-speech.test.ts +63 -63
- package/src/lib/tts-speech.ts +76 -76
- package/src/lib/workspace-argv.test.ts +49 -49
- package/src/lib/workspace-argv.ts +44 -44
- package/src/perception/player-history-store.test.ts +87 -87
- package/src/perception/player-history-store.ts +194 -194
- package/src/pipeline/event-store.ts +124 -124
- package/src/pipeline/pipeline.ts +35 -35
- package/src/runtime/auto-upgrade.test.ts +66 -66
- package/src/runtime/auto-upgrade.ts +31 -31
- package/src/runtime/event-daemon.test.ts +107 -28
- package/src/runtime/event-daemon.ts +409 -371
- package/src/runtime/owner-control.ts +150 -0
- package/src/runtime/raw-ws-log.test.ts +33 -33
- package/src/runtime/raw-ws-log.ts +32 -32
- package/src/runtime/runtime-logger.ts +107 -99
- package/src/runtime/ws-client.test.ts +104 -47
- package/src/runtime/ws-client.ts +272 -272
- package/src/sdk/action.ts +166 -166
- package/src/sdk/index.ts +110 -110
- package/src/sdk/types.ts +146 -146
- package/src/strategies/avoid-lone.ts +11 -11
- package/src/strategies/avoid-players.knowledge.md +20 -20
- package/src/strategies/avoid-players.ts +15 -15
- package/src/strategies/corpse-patrol.ts +22 -22
- package/src/strategies/crab-sabotage.ts +21 -21
- package/src/strategies/custom-module.test.ts +269 -269
- package/src/strategies/find-player.ts +16 -16
- package/src/strategies/game-utils.test.ts +164 -164
- package/src/strategies/game-utils.ts +737 -721
- package/src/strategies/goals/avoid-lone-top.ts +168 -168
- package/src/strategies/goals/avoid-players-top.test.ts +83 -83
- package/src/strategies/goals/avoid-players-top.ts +121 -121
- package/src/strategies/goals/conversation-goal.ts +51 -51
- package/src/strategies/goals/corpse-patrol-top.ts +91 -91
- package/src/strategies/goals/crab-octopus-reflexes.ts +93 -93
- package/src/strategies/goals/crab-sabotage-top.ts +197 -197
- package/src/strategies/goals/emergency-hunt-goal.ts +28 -28
- package/src/strategies/goals/find-player-top.ts +93 -93
- package/src/strategies/goals/flee-players-goal.ts +53 -53
- package/src/strategies/goals/goal-manager.ts +41 -41
- package/src/strategies/goals/goal-root-strategy.ts +49 -49
- package/src/strategies/goals/goal.ts +28 -28
- package/src/strategies/goals/keep-away-goal.ts +206 -206
- package/src/strategies/goals/kill-frenzy-top.ts +80 -80
- package/src/strategies/goals/kill-lone-top.ts +160 -160
- package/src/strategies/goals/kill-target-goal.ts +59 -59
- package/src/strategies/goals/kill-target-top.ts +109 -109
- package/src/strategies/goals/leaf-goal.ts +25 -25
- package/src/strategies/goals/linger-corpse-goal.ts +79 -79
- package/src/strategies/goals/lone-kill-core.ts +82 -82
- package/src/strategies/goals/lone-kill-goal.ts +24 -24
- package/src/strategies/goals/lone-kill-task-top.test.ts +85 -85
- package/src/strategies/goals/lone-kill-task-top.ts +86 -86
- package/src/strategies/goals/move-room-goal.ts +60 -60
- package/src/strategies/goals/normal-shrimp-top.test.ts +80 -80
- package/src/strategies/goals/normal-shrimp-top.ts +242 -242
- package/src/strategies/goals/paradise-fish-top.test.ts +126 -126
- package/src/strategies/goals/paradise-fish-top.ts +219 -219
- package/src/strategies/goals/patrol-top.ts +57 -57
- package/src/strategies/goals/report-patrol-top.ts +80 -80
- package/src/strategies/goals/safe-task-goal.ts +102 -102
- package/src/strategies/goals/social-task-top.ts +161 -161
- package/src/strategies/goals/task-kill-report-top.ts +163 -163
- package/src/strategies/goals/task-only-top.ts +57 -57
- package/src/strategies/goals/task-or-patrol-goal.ts +41 -41
- package/src/strategies/goals/task-report-top.ts +57 -57
- package/src/strategies/goals/wander-task-goal.ts +33 -33
- package/src/strategies/goals/warrior-shrimp-top.test.ts +86 -86
- package/src/strategies/goals/warrior-shrimp-top.ts +248 -248
- package/src/strategies/greeting.ts +53 -53
- package/src/strategies/kill-frenzy.ts +12 -12
- package/src/strategies/kill-lone.knowledge.md +20 -20
- package/src/strategies/kill-lone.ts +13 -13
- package/src/strategies/kill-target.ts +18 -18
- package/src/strategies/loader.test.ts +678 -678
- package/src/strategies/loader.ts +172 -172
- package/src/strategies/lone-kill-task.ts +21 -21
- package/src/strategies/meeting-gate.test.ts +59 -59
- package/src/strategies/meeting-gate.ts +23 -23
- package/src/strategies/move-room.ts +15 -15
- package/src/strategies/new-events-backfill.ts +98 -98
- package/src/strategies/paradise-fish.knowledge.md +20 -20
- package/src/strategies/paradise-fish.ts +25 -25
- package/src/strategies/pathfind/distance-field.ts +150 -150
- package/src/strategies/pathfind/escape-planner.test.ts +197 -197
- package/src/strategies/pathfind/escape-planner.ts +348 -348
- package/src/strategies/pathfind/walkable-grid.ts +117 -117
- package/src/strategies/patrol.ts +11 -11
- package/src/strategies/player-targets.ts +13 -13
- package/src/strategies/report-patrol.ts +11 -11
- package/src/strategies/shrimp-memory.knowledge.md +20 -20
- package/src/strategies/shrimp-memory.ts +25 -25
- package/src/strategies/social-task.test.ts +28 -28
- package/src/strategies/social-task.ts +49 -49
- package/src/strategies/spawn.ts +82 -71
- package/src/strategies/speech-module.ts +123 -123
- package/src/strategies/strategy-loop.ts +763 -757
- package/src/strategies/task-kill-report.ts +17 -17
- package/src/strategies/task-only.ts +11 -11
- package/src/strategies/task-report.ts +22 -22
- package/src/strategies/types.ts +96 -96
- package/src/strategies/warrior-memory.knowledge.md +20 -20
- package/src/strategies/warrior-memory.ts +16 -16
- package/src/runtime/daemon.ts +0 -100
- package/src/runtime/opening-mover.ts +0 -303
|
@@ -1,303 +0,0 @@
|
|
|
1
|
-
import { spawn } from 'child_process';
|
|
2
|
-
import { appendFileSync, existsSync, mkdirSync, openSync, readFileSync, unlinkSync, writeFileSync } from 'fs';
|
|
3
|
-
import { delimiter, dirname, join, resolve } from 'path';
|
|
4
|
-
import { execSync } from 'child_process';
|
|
5
|
-
import { fileURLToPath } from 'url';
|
|
6
|
-
import { Action } from '../sdk/action.js';
|
|
7
|
-
import { AuthStore } from '../lib/auth.js';
|
|
8
|
-
import { GameClient } from '../lib/game-client.js';
|
|
9
|
-
import { getProfileLogsDir, getProfileStateDir } from '../lib/init-command.js';
|
|
10
|
-
import { isStrategyRunning as isAutoRunning } from '../strategies/strategy-loop.js';
|
|
11
|
-
|
|
12
|
-
const __filename = fileURLToPath(import.meta.url);
|
|
13
|
-
const __dirname = dirname(__filename);
|
|
14
|
-
|
|
15
|
-
const WANDERING_WINDOW_MS = 30_000;
|
|
16
|
-
const STARTUP_WAIT_MS = 10 * 60_000;
|
|
17
|
-
const POLL_MS = 1000;
|
|
18
|
-
const CANCEL_TTL_MS = 45_000;
|
|
19
|
-
|
|
20
|
-
interface Point {
|
|
21
|
-
x: number;
|
|
22
|
-
y: number;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
function sleep(ms: number): Promise<void> {
|
|
26
|
-
return new Promise(resolve => setTimeout(resolve, ms));
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
function debug(message: string): void {
|
|
30
|
-
if (process.env.CLAWCLAW_LOG_FILE) {
|
|
31
|
-
try {
|
|
32
|
-
appendFileSync(process.env.CLAWCLAW_LOG_FILE, `[${new Date().toISOString()}] ${message}\n`);
|
|
33
|
-
} catch {}
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
function isPidAlive(pid: number): boolean {
|
|
38
|
-
try { process.kill(pid, 0); return true; } catch { return false; }
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
function activeProfile() {
|
|
42
|
-
const profile = new AuthStore().getActive();
|
|
43
|
-
if (!profile) throw new Error('Not logged in.');
|
|
44
|
-
return profile;
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
function pidPath(): string {
|
|
48
|
-
return join(getProfileStateDir(activeProfile()), 'opening-mover.pid');
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
function cancelPath(): string {
|
|
52
|
-
return join(getProfileStateDir(activeProfile()), 'opening-mover.cancel');
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
function readCancelToken(path: string): number {
|
|
56
|
-
try {
|
|
57
|
-
const token = Number(readFileSync(path, 'utf8').trim());
|
|
58
|
-
return Number.isFinite(token) ? token : 0;
|
|
59
|
-
} catch {
|
|
60
|
-
return 0;
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
function cleanupPid(path: string): void {
|
|
65
|
-
try {
|
|
66
|
-
const current = Number(readFileSync(path, 'utf8').trim());
|
|
67
|
-
if (current === process.pid) unlinkSync(path);
|
|
68
|
-
} catch {}
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
export function stopOpeningMoverIfRunning(): void {
|
|
72
|
-
try {
|
|
73
|
-
const path = pidPath();
|
|
74
|
-
if (!existsSync(path)) return;
|
|
75
|
-
const pid = Number(readFileSync(path, 'utf8').trim());
|
|
76
|
-
if (pid > 0 && isPidAlive(pid)) {
|
|
77
|
-
try { process.kill(pid, 'SIGTERM'); } catch {}
|
|
78
|
-
}
|
|
79
|
-
try { unlinkSync(path); } catch {}
|
|
80
|
-
} catch {}
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
export function cancelOpeningMoverWindow(): void {
|
|
84
|
-
try {
|
|
85
|
-
const path = cancelPath();
|
|
86
|
-
mkdirSync(dirname(path), { recursive: true });
|
|
87
|
-
writeFileSync(path, String(Date.now()));
|
|
88
|
-
} catch {}
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
function roomAnchor(r: any): Point | null {
|
|
92
|
-
const taskLocations = Array.isArray(r?.task_locations) ? r.task_locations : [];
|
|
93
|
-
const candidates = taskLocations
|
|
94
|
-
.map((tl: any) => ({ x: Number(tl?.x ?? tl?.[0]), y: Number(tl?.y ?? tl?.[1]) }))
|
|
95
|
-
.filter((p: Point) => Number.isFinite(p.x) && Number.isFinite(p.y));
|
|
96
|
-
if (candidates.length > 0) return candidates[Math.floor(Math.random() * candidates.length)];
|
|
97
|
-
|
|
98
|
-
const poly: number[][] = Array.isArray(r?.polygon) ? r.polygon : [];
|
|
99
|
-
if (poly.length === 0) return null;
|
|
100
|
-
|
|
101
|
-
let area = 0;
|
|
102
|
-
let cx = 0;
|
|
103
|
-
let cy = 0;
|
|
104
|
-
for (let i = 0; i < poly.length; i += 1) {
|
|
105
|
-
const [x1, y1] = poly[i];
|
|
106
|
-
const [x2, y2] = poly[(i + 1) % poly.length];
|
|
107
|
-
const cross = x1 * y2 - x2 * y1;
|
|
108
|
-
area += cross;
|
|
109
|
-
cx += (x1 + x2) * cross;
|
|
110
|
-
cy += (y1 + y2) * cross;
|
|
111
|
-
}
|
|
112
|
-
area *= 0.5;
|
|
113
|
-
if (Math.abs(area) < 1e-6) {
|
|
114
|
-
return {
|
|
115
|
-
x: poly.reduce((sum, p) => sum + p[0], 0) / poly.length,
|
|
116
|
-
y: poly.reduce((sum, p) => sum + p[1], 0) / poly.length,
|
|
117
|
-
};
|
|
118
|
-
}
|
|
119
|
-
return { x: cx / (6 * area), y: cy / (6 * area) };
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
function randomTarget(mapData: any): Point | null {
|
|
123
|
-
const rooms = Array.isArray(mapData?.rooms) ? mapData.rooms : [];
|
|
124
|
-
const anchors = rooms.map(roomAnchor).filter((p: Point | null): p is Point => p !== null);
|
|
125
|
-
if (anchors.length === 0) return null;
|
|
126
|
-
return anchors[Math.floor(Math.random() * anchors.length)];
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
export async function runOpeningMover(): Promise<void> {
|
|
130
|
-
const path = pidPath();
|
|
131
|
-
const cancelFile = cancelPath();
|
|
132
|
-
mkdirSync(dirname(path), { recursive: true });
|
|
133
|
-
writeFileSync(path, String(process.pid));
|
|
134
|
-
|
|
135
|
-
let running = true;
|
|
136
|
-
process.on('SIGINT', () => { running = false; });
|
|
137
|
-
process.on('SIGTERM', () => { running = false; });
|
|
138
|
-
|
|
139
|
-
const client = GameClient.fromAuth();
|
|
140
|
-
const startupDeadline = Date.now() + STARTUP_WAIT_MS;
|
|
141
|
-
let wanderingDeadline = 0;
|
|
142
|
-
let helperMovingUntil = 0;
|
|
143
|
-
let lastPhase: string | null = null;
|
|
144
|
-
let lastCancelToken = 0;
|
|
145
|
-
let windowActive = false;
|
|
146
|
-
|
|
147
|
-
try {
|
|
148
|
-
debug('opening mover started');
|
|
149
|
-
try { await client.discoverGameServer(); } catch {}
|
|
150
|
-
while (running) {
|
|
151
|
-
let state: any = null;
|
|
152
|
-
try {
|
|
153
|
-
state = await client.getGameState();
|
|
154
|
-
} catch {
|
|
155
|
-
await sleep(1500);
|
|
156
|
-
continue;
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
if (!state) {
|
|
160
|
-
if (Date.now() >= startupDeadline) break;
|
|
161
|
-
await sleep(1500);
|
|
162
|
-
continue;
|
|
163
|
-
}
|
|
164
|
-
if (state.phase === 'game_over') break;
|
|
165
|
-
|
|
166
|
-
const cancelToken = readCancelToken(cancelFile);
|
|
167
|
-
if (cancelToken > lastCancelToken) {
|
|
168
|
-
lastCancelToken = cancelToken;
|
|
169
|
-
if (Date.now() - cancelToken <= CANCEL_TTL_MS) {
|
|
170
|
-
windowActive = false;
|
|
171
|
-
wanderingDeadline = 0;
|
|
172
|
-
helperMovingUntil = 0;
|
|
173
|
-
if (state.phase === 'wandering') lastPhase = 'wandering';
|
|
174
|
-
debug('movement helper window cancelled by manual action');
|
|
175
|
-
}
|
|
176
|
-
}
|
|
177
|
-
|
|
178
|
-
if (state.phase !== 'wandering') {
|
|
179
|
-
lastPhase = state.phase;
|
|
180
|
-
windowActive = false;
|
|
181
|
-
wanderingDeadline = 0;
|
|
182
|
-
helperMovingUntil = 0;
|
|
183
|
-
await sleep(POLL_MS);
|
|
184
|
-
continue;
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
if (lastPhase !== 'wandering') {
|
|
188
|
-
lastPhase = 'wandering';
|
|
189
|
-
helperMovingUntil = 0;
|
|
190
|
-
if (isAutoRunning()) {
|
|
191
|
-
windowActive = false;
|
|
192
|
-
wanderingDeadline = 0;
|
|
193
|
-
debug('wandering detected; auto running, movement helper skipped');
|
|
194
|
-
} else if (state.you?.currently_moving || state.you?.doing_task) {
|
|
195
|
-
windowActive = false;
|
|
196
|
-
wanderingDeadline = 0;
|
|
197
|
-
debug('wandering detected; player already active, movement helper skipped');
|
|
198
|
-
} else {
|
|
199
|
-
windowActive = true;
|
|
200
|
-
wanderingDeadline = Date.now() + WANDERING_WINDOW_MS;
|
|
201
|
-
debug('wandering detected; movement helper window started');
|
|
202
|
-
}
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
if (!windowActive) {
|
|
206
|
-
await sleep(POLL_MS);
|
|
207
|
-
continue;
|
|
208
|
-
}
|
|
209
|
-
if (Date.now() >= wanderingDeadline) {
|
|
210
|
-
windowActive = false;
|
|
211
|
-
wanderingDeadline = 0;
|
|
212
|
-
debug('movement helper window expired');
|
|
213
|
-
await sleep(POLL_MS);
|
|
214
|
-
continue;
|
|
215
|
-
}
|
|
216
|
-
|
|
217
|
-
if (isAutoRunning()) {
|
|
218
|
-
windowActive = false;
|
|
219
|
-
wanderingDeadline = 0;
|
|
220
|
-
debug('auto started; movement helper window stopped');
|
|
221
|
-
await sleep(POLL_MS);
|
|
222
|
-
continue;
|
|
223
|
-
}
|
|
224
|
-
|
|
225
|
-
if (state.you?.currently_moving || state.you?.doing_task || Date.now() < helperMovingUntil) {
|
|
226
|
-
await sleep(POLL_MS);
|
|
227
|
-
continue;
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
let mapData: any = null;
|
|
231
|
-
try {
|
|
232
|
-
mapData = await client.getMap();
|
|
233
|
-
} catch {
|
|
234
|
-
await sleep(POLL_MS);
|
|
235
|
-
continue;
|
|
236
|
-
}
|
|
237
|
-
const target = randomTarget(mapData);
|
|
238
|
-
if (!target) {
|
|
239
|
-
await sleep(POLL_MS);
|
|
240
|
-
continue;
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
let result: any;
|
|
244
|
-
try {
|
|
245
|
-
result = await client.submitAction(Action.move(target).toJSON() as any);
|
|
246
|
-
} catch {
|
|
247
|
-
await sleep(POLL_MS);
|
|
248
|
-
continue;
|
|
249
|
-
}
|
|
250
|
-
const actionResult = result?.data ?? result;
|
|
251
|
-
const durationSecs = Number(actionResult?.duration_secs ?? result?.duration_secs ?? 2);
|
|
252
|
-
debug(`opening move submitted target=(${target.x},${target.y}) duration=${durationSecs}`);
|
|
253
|
-
helperMovingUntil = Date.now() + Math.max(0.5, durationSecs) * 1000 + 500;
|
|
254
|
-
}
|
|
255
|
-
} finally {
|
|
256
|
-
debug('opening mover stopped');
|
|
257
|
-
cleanupPid(path);
|
|
258
|
-
}
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
export function spawnOpeningMover(): { pid: number; logFile: string } {
|
|
262
|
-
const profile = activeProfile();
|
|
263
|
-
const stateDir = getProfileStateDir(profile);
|
|
264
|
-
mkdirSync(stateDir, { recursive: true });
|
|
265
|
-
|
|
266
|
-
const path = join(stateDir, 'opening-mover.pid');
|
|
267
|
-
if (existsSync(path)) {
|
|
268
|
-
try {
|
|
269
|
-
const pid = Number(readFileSync(path, 'utf8').trim());
|
|
270
|
-
if (pid > 0 && isPidAlive(pid)) return { pid, logFile: '' };
|
|
271
|
-
} catch {}
|
|
272
|
-
try { unlinkSync(path); } catch {}
|
|
273
|
-
}
|
|
274
|
-
|
|
275
|
-
const logsDir = getProfileLogsDir(profile);
|
|
276
|
-
mkdirSync(logsDir, { recursive: true });
|
|
277
|
-
const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
|
|
278
|
-
const logFile = join(logsDir, `opening-mover-${timestamp}.log`);
|
|
279
|
-
const logFd = openSync(logFile, 'a');
|
|
280
|
-
|
|
281
|
-
const binEntry = resolve(__dirname, '..', '..', 'bin', 'clawclaw-cli.mjs').replace(/\\/g, '/');
|
|
282
|
-
if (!existsSync(binEntry)) throw new Error('clawclaw-cli bin entry not found');
|
|
283
|
-
|
|
284
|
-
let npmGlobalModules = '';
|
|
285
|
-
try { npmGlobalModules = execSync('npm root -g', { encoding: 'utf8', windowsHide: true }).trim(); } catch {}
|
|
286
|
-
const nodePath = [npmGlobalModules, process.env.NODE_PATH ?? ''].filter(Boolean).join(delimiter);
|
|
287
|
-
const isWin = process.platform === 'win32';
|
|
288
|
-
|
|
289
|
-
const child = spawn(process.execPath, [binEntry, '_opener'], {
|
|
290
|
-
detached: true,
|
|
291
|
-
stdio: isWin ? ['ignore', 'ignore', 'ignore'] : ['ignore', logFd, logFd],
|
|
292
|
-
cwd: process.cwd(),
|
|
293
|
-
env: {
|
|
294
|
-
...process.env,
|
|
295
|
-
NODE_PATH: nodePath,
|
|
296
|
-
...(isWin ? { CLAWCLAW_LOG_FILE: logFile } : {}),
|
|
297
|
-
},
|
|
298
|
-
windowsHide: true,
|
|
299
|
-
});
|
|
300
|
-
child.unref();
|
|
301
|
-
if (child.pid) writeFileSync(path, String(child.pid));
|
|
302
|
-
return { pid: child.pid!, logFile };
|
|
303
|
-
}
|