@kodrunhq/claudefy 1.4.3 → 1.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,119 @@
1
+ import { mkdir, readFile, rm, writeFile } from "node:fs/promises";
2
+ import { dirname } from "node:path";
3
+ import { existsSync } from "node:fs";
4
+ export class Lockfile {
5
+ path;
6
+ retries;
7
+ retryDelayMs;
8
+ maxAgeMs;
9
+ operation;
10
+ constructor(path, options = {}) {
11
+ this.path = path;
12
+ this.retries = options.retries ?? 3;
13
+ this.retryDelayMs = options.retryDelayMs ?? 1000;
14
+ this.maxAgeMs = options.maxAgeMs ?? 120_000; // 2 minutes
15
+ this.operation = options.operation ?? "unknown";
16
+ }
17
+ async acquire() {
18
+ for (let attempt = 0; attempt <= this.retries; attempt++) {
19
+ if (attempt > 0) {
20
+ await this.delay(this.retryDelayMs);
21
+ }
22
+ if (await this.tryAcquire()) {
23
+ return true;
24
+ }
25
+ }
26
+ return false;
27
+ }
28
+ async release() {
29
+ try {
30
+ await rm(this.path, { force: true });
31
+ }
32
+ catch {
33
+ // Best effort — stale lock will expire via maxAgeMs
34
+ }
35
+ }
36
+ async tryAcquire() {
37
+ if (!existsSync(this.path)) {
38
+ return this.writeLock();
39
+ }
40
+ // Lock exists — check if stale
41
+ const existing = await this.readLock();
42
+ if (!existing) {
43
+ // Corrupted lockfile (unparseable JSON), overwrite
44
+ await this.release();
45
+ return this.writeLock();
46
+ }
47
+ // Check age
48
+ const age = Date.now() - new Date(existing.timestamp).getTime();
49
+ if (age > this.maxAgeMs) {
50
+ await this.release();
51
+ return this.writeLock();
52
+ }
53
+ // Check if PID is alive
54
+ if (!this.isProcessAlive(existing.pid)) {
55
+ await this.release();
56
+ return this.writeLock();
57
+ }
58
+ // Lock is held by a live, recent process
59
+ return false;
60
+ }
61
+ async writeLock() {
62
+ await mkdir(dirname(this.path), { recursive: true, mode: 0o700 });
63
+ const data = {
64
+ pid: process.pid,
65
+ timestamp: new Date().toISOString(),
66
+ operation: this.operation,
67
+ };
68
+ try {
69
+ await writeFile(this.path, JSON.stringify(data), { flag: "wx", mode: 0o600 });
70
+ return true;
71
+ }
72
+ catch (err) {
73
+ if (err.code === "EEXIST") {
74
+ return false;
75
+ }
76
+ throw err; // Disk full, permissions, etc. must surface
77
+ }
78
+ }
79
+ async readLock() {
80
+ try {
81
+ const content = await readFile(this.path, "utf-8");
82
+ return JSON.parse(content);
83
+ }
84
+ catch (err) {
85
+ if (err instanceof SyntaxError) {
86
+ return null; // Corrupted JSON
87
+ }
88
+ if (err.code === "ENOENT") {
89
+ return null; // Lockfile deleted between existsSync() and readFile()
90
+ }
91
+ // I/O errors (EACCES, EMFILE, etc.) — treat as "lock exists, can't read"
92
+ // so we don't steal it. Return a synthetic entry with current PID to prevent overwrite.
93
+ return { pid: process.pid, timestamp: new Date().toISOString(), operation: "unknown" };
94
+ }
95
+ }
96
+ isProcessAlive(pid) {
97
+ // pid=0 signals the process group; pid<0 signals a group or all processes.
98
+ // Neither represents a real process that owns a lock.
99
+ if (!Number.isInteger(pid) || pid <= 0 || pid > 4194304) {
100
+ return false;
101
+ }
102
+ try {
103
+ process.kill(pid, 0);
104
+ return true;
105
+ }
106
+ catch (err) {
107
+ // EPERM means the process exists but belongs to another user
108
+ if (err.code === "EPERM") {
109
+ return true;
110
+ }
111
+ // ESRCH means process doesn't exist
112
+ return false;
113
+ }
114
+ }
115
+ delay(ms) {
116
+ return new Promise((resolve) => setTimeout(resolve, ms));
117
+ }
118
+ }
119
+ //# sourceMappingURL=lockfile.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lockfile.js","sourceRoot":"","sources":["../src/lockfile.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,23 @@
1
+ export interface LogEntry {
2
+ timestamp: string;
3
+ level: "info" | "warn" | "error";
4
+ operation: string;
5
+ message: string;
6
+ }
7
+ export interface LoggerOptions {
8
+ maxBytes?: number;
9
+ maxFiles?: number;
10
+ }
11
+ export interface ReadRecentFilter {
12
+ level?: "info" | "warn" | "error";
13
+ operation?: string;
14
+ }
15
+ export declare class Logger {
16
+ private readonly filePath;
17
+ private readonly maxBytes;
18
+ private readonly maxFiles;
19
+ constructor(filePath: string, options?: LoggerOptions);
20
+ log(level: LogEntry["level"], operation: string, message: string): Promise<void>;
21
+ readRecent(count: number, filter?: ReadRecentFilter): Promise<LogEntry[]>;
22
+ private rotateIfNeeded;
23
+ }
package/dist/logger.js ADDED
@@ -0,0 +1,83 @@
1
+ import { appendFile, mkdir, readFile, rename, rm, stat } from "node:fs/promises";
2
+ import { dirname } from "node:path";
3
+ import { existsSync } from "node:fs";
4
+ export class Logger {
5
+ filePath;
6
+ maxBytes;
7
+ maxFiles;
8
+ constructor(filePath, options = {}) {
9
+ this.filePath = filePath;
10
+ this.maxBytes = options.maxBytes ?? 512 * 1024; // 512 KB default
11
+ this.maxFiles = options.maxFiles ?? 3;
12
+ }
13
+ async log(level, operation, message) {
14
+ try {
15
+ const entry = {
16
+ timestamp: new Date().toISOString(),
17
+ level,
18
+ operation,
19
+ message,
20
+ };
21
+ const line = JSON.stringify(entry) + "\n";
22
+ await mkdir(dirname(this.filePath), { recursive: true, mode: 0o700 });
23
+ await appendFile(this.filePath, line, { mode: 0o600 });
24
+ await this.rotateIfNeeded();
25
+ }
26
+ catch {
27
+ // Logging is best-effort; never crash the caller
28
+ }
29
+ }
30
+ async readRecent(count, filter) {
31
+ if (!existsSync(this.filePath))
32
+ return [];
33
+ let content;
34
+ try {
35
+ content = await readFile(this.filePath, "utf-8");
36
+ }
37
+ catch {
38
+ return [];
39
+ }
40
+ let entries = content
41
+ .trim()
42
+ .split("\n")
43
+ .filter(Boolean)
44
+ .map((line) => {
45
+ try {
46
+ return JSON.parse(line);
47
+ }
48
+ catch {
49
+ return null;
50
+ }
51
+ })
52
+ .filter((e) => e !== null);
53
+ if (filter?.level) {
54
+ entries = entries.filter((e) => e.level === filter.level);
55
+ }
56
+ if (filter?.operation) {
57
+ entries = entries.filter((e) => e.operation === filter.operation);
58
+ }
59
+ return entries.slice(-count);
60
+ }
61
+ async rotateIfNeeded() {
62
+ try {
63
+ const stats = await stat(this.filePath);
64
+ if (stats.size <= this.maxBytes)
65
+ return;
66
+ }
67
+ catch {
68
+ return;
69
+ }
70
+ // Shift existing rotated files: .2 -> .3, .1 -> .2, current -> .1
71
+ for (let i = this.maxFiles; i >= 1; i--) {
72
+ const src = i === 1 ? this.filePath : `${this.filePath}.${i - 1}`;
73
+ const dest = `${this.filePath}.${i}`;
74
+ if (existsSync(src)) {
75
+ if (i === this.maxFiles) {
76
+ await rm(dest, { force: true });
77
+ }
78
+ await rename(src, dest);
79
+ }
80
+ }
81
+ }
82
+ }
83
+ //# sourceMappingURL=logger.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.js","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@kodrunhq/claudefy",
3
- "version": "1.4.3",
3
+ "version": "1.5.0",
4
4
  "description": "Sync your Claude Code environment across machines",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",