@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.
- package/dist/cli.js +42 -20
- package/dist/cli.js.map +1 -1
- package/dist/commands/doctor.d.ts +1 -0
- package/dist/commands/doctor.js +38 -0
- package/dist/commands/doctor.js.map +1 -1
- package/dist/commands/join.js +18 -6
- package/dist/commands/join.js.map +1 -1
- package/dist/commands/override.js +4 -0
- package/dist/commands/override.js.map +1 -1
- package/dist/commands/pull.d.ts +2 -0
- package/dist/commands/pull.js +29 -5
- package/dist/commands/pull.js.map +1 -1
- package/dist/commands/push.d.ts +3 -0
- package/dist/commands/push.js +231 -197
- package/dist/commands/push.js.map +1 -1
- package/dist/lockfile.d.ts +22 -0
- package/dist/lockfile.js +119 -0
- package/dist/lockfile.js.map +1 -0
- package/dist/logger.d.ts +23 -0
- package/dist/logger.js +83 -0
- package/dist/logger.js.map +1 -0
- package/package.json +1 -1
package/dist/lockfile.js
ADDED
|
@@ -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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"}
|
package/dist/logger.d.ts
ADDED
|
@@ -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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"}
|