@cryptiklemur/lattice 1.41.1 → 1.41.2

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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@cryptiklemur/lattice",
3
- "version": "1.41.1",
3
+ "version": "1.41.2",
4
4
  "description": "Multi-machine agentic dashboard for Claude Code. Monitor sessions, manage MCP servers and skills, orchestrate across mesh-networked nodes.",
5
5
  "license": "MIT",
6
6
  "author": "Aaron Scherer <me@aaronscherer.me>",
@@ -4,7 +4,7 @@ import type { SDKMessage, SDKPartialAssistantMessage, SDKResultMessage, SDKUserM
4
4
  import type { CanUseTool, PermissionMode, PermissionResult, PermissionUpdate } from "@anthropic-ai/claude-agent-sdk";
5
5
  type MessageParam = SDKUserMessage["message"];
6
6
  import type { Attachment } from "@lattice/shared";
7
- import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
7
+ import { existsSync, readFileSync, writeFileSync, mkdirSync, statSync } from "node:fs";
8
8
  import { join, resolve } from "node:path";
9
9
  import { homedir } from "node:os";
10
10
  import { sendTo, broadcast } from "../ws/broadcast";
@@ -227,77 +227,48 @@ export function isSessionBusy(sessionId: string): boolean {
227
227
  * The SDK spawns child processes (e.g. claude-agent-sdk/cli.js) that hold
228
228
  * lock files — those are NOT external.
229
229
  */
230
- function isOwnProcess(pid: number): boolean {
231
- var myPid = process.pid;
232
- if (pid === myPid) return true;
233
- // Walk up the process tree to see if pid is a descendant of us
234
- var current = pid;
235
- for (var i = 0; i < 10; i++) {
236
- try {
237
- var stat = readFileSync("/proc/" + current + "/stat", "utf-8");
238
- // Format: pid (comm) state ppid ...
239
- var match = stat.match(/^\d+\s+\([^)]*\)\s+\S+\s+(\d+)/);
240
- if (!match) return false;
241
- var ppid = parseInt(match[1], 10);
242
- if (ppid === myPid) return true;
243
- if (ppid <= 1) return false;
244
- current = ppid;
245
- } catch {
246
- return false;
230
+ function isSessionLockedByExternal(sessionId: string): boolean {
231
+ if (activeStreams.has(sessionId)) return false;
232
+
233
+ var config = loadConfig();
234
+ for (var i = 0; i < config.projects.length; i++) {
235
+ var projectPath = config.projects[i].path;
236
+ var hash = projectPath.replace(/\//g, "-");
237
+ var jsonlPath = join(homedir(), ".claude", "projects", hash, sessionId + ".jsonl");
238
+ if (existsSync(jsonlPath)) {
239
+ try {
240
+ var stat = statSync(jsonlPath);
241
+ var mtime = stat.mtimeMs;
242
+ if (Date.now() - mtime < 10000) {
243
+ return true;
244
+ }
245
+ } catch {}
247
246
  }
248
247
  }
249
- return false;
250
- }
251
248
 
252
- /**
253
- * Get PIDs holding the session lock file, excluding Lattice's own process tree.
254
- * Returns the list of truly external PIDs.
255
- */
256
- function getExternalLockPids(sessionId: string): number[] {
257
- var lockPath = join(homedir(), ".claude", "tasks", sessionId, ".lock");
258
- if (!existsSync(lockPath)) return [];
259
- try {
260
- var result = Bun.spawnSync(["fuser", lockPath], {
261
- stderr: "ignore",
262
- });
263
- if (result.exitCode !== 0) return [];
264
- var output = result.stdout.toString().trim();
265
- var pids = output.split(/\s+/)
266
- .map(function (s) { return parseInt(s, 10); })
267
- .filter(function (p) { return !isNaN(p) && !isOwnProcess(p); });
268
- return pids;
269
- } catch {
270
- return [];
271
- }
272
- }
273
-
274
- function isSessionLockedByExternal(sessionId: string): boolean {
275
- return getExternalLockPids(sessionId).length > 0;
276
- }
277
-
278
- /**
279
- * Get the first external PID holding the session lock file.
280
- * Used to send SIGINT to stop the external process.
281
- */
282
- function getExternalLockPid(sessionId: string): number | null {
283
- var pids = getExternalLockPids(sessionId);
284
- return pids.length > 0 ? pids[0] : null;
249
+ return false;
285
250
  }
286
251
 
287
- /**
288
- * Gracefully stop an external Claude Code CLI process controlling a session.
289
- * Sends SIGINT which triggers Claude Code's graceful shutdown.
290
- * Returns true if a signal was sent.
291
- */
292
252
  export function stopExternalSession(sessionId: string): boolean {
293
- var pid = getExternalLockPid(sessionId);
294
- if (pid === null) return false;
295
253
  try {
296
- process.kill(pid, "SIGINT");
297
- return true;
298
- } catch {
299
- return false;
300
- }
254
+ var config = loadConfig();
255
+ for (var i = 0; i < config.projects.length; i++) {
256
+ var hash = config.projects[i].path.replace(/\//g, "-");
257
+ var jsonlPath = join(homedir(), ".claude", "projects", hash, sessionId + ".jsonl");
258
+ if (existsSync(jsonlPath)) {
259
+ var result = Bun.spawnSync(["fuser", jsonlPath], { stderr: "ignore" });
260
+ if (result.exitCode === 0) {
261
+ var output = result.stdout.toString().trim();
262
+ var pids = output.split(/\s+/).map(Number).filter(function (p) { return !isNaN(p) && p !== process.pid; });
263
+ if (pids.length > 0) {
264
+ process.kill(pids[0], "SIGINT");
265
+ return true;
266
+ }
267
+ }
268
+ }
269
+ }
270
+ } catch {}
271
+ return false;
301
272
  }
302
273
 
303
274
  export function getSessionStreamClientId(sessionId: string): string | undefined {