@bradygaster/squad-cli 0.9.3-insider.1 → 0.9.4-insider.1

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.
Files changed (93) hide show
  1. package/README.md +33 -0
  2. package/dist/cli/commands/externalize.d.ts +27 -0
  3. package/dist/cli/commands/externalize.d.ts.map +1 -0
  4. package/dist/cli/commands/externalize.js +207 -0
  5. package/dist/cli/commands/externalize.js.map +1 -0
  6. package/dist/cli/commands/loop.d.ts +6 -1
  7. package/dist/cli/commands/loop.d.ts.map +1 -1
  8. package/dist/cli/commands/loop.js +4 -4
  9. package/dist/cli/commands/loop.js.map +1 -1
  10. package/dist/cli/commands/migrate.js +2 -2
  11. package/dist/cli/commands/migrate.js.map +1 -1
  12. package/dist/cli/commands/rc.js +2 -2
  13. package/dist/cli/commands/rc.js.map +1 -1
  14. package/dist/cli/commands/watch/config.d.ts +0 -8
  15. package/dist/cli/commands/watch/config.d.ts.map +1 -1
  16. package/dist/cli/commands/watch/config.js +0 -5
  17. package/dist/cli/commands/watch/config.js.map +1 -1
  18. package/dist/cli/core/email-scrub.js +2 -2
  19. package/dist/cli/core/email-scrub.js.map +1 -1
  20. package/dist/cli/core/init.d.ts.map +1 -1
  21. package/dist/cli/core/init.js +4 -1
  22. package/dist/cli/core/init.js.map +1 -1
  23. package/dist/cli/core/migrations.js +1 -1
  24. package/dist/cli/core/migrations.js.map +1 -1
  25. package/dist/cli/core/nap.js +2 -2
  26. package/dist/cli/core/nap.js.map +1 -1
  27. package/dist/cli/core/project-type.js +1 -1
  28. package/dist/cli/core/project-type.js.map +1 -1
  29. package/dist/cli/core/templates.d.ts.map +1 -1
  30. package/dist/cli/core/templates.js +48 -0
  31. package/dist/cli/core/templates.js.map +1 -1
  32. package/dist/cli/core/upgrade.d.ts +4 -2
  33. package/dist/cli/core/upgrade.d.ts.map +1 -1
  34. package/dist/cli/core/upgrade.js +28 -21
  35. package/dist/cli/core/upgrade.js.map +1 -1
  36. package/dist/cli/shell/session-store.js +2 -2
  37. package/dist/cli/shell/session-store.js.map +1 -1
  38. package/dist/cli-entry.js +52 -83
  39. package/dist/cli-entry.js.map +1 -1
  40. package/package.json +2 -2
  41. package/templates/casting-reference.md +104 -104
  42. package/templates/ceremonies.md +28 -28
  43. package/templates/fact-checker-charter.md +83 -0
  44. package/templates/skills/external-comms/SKILL.md +329 -329
  45. package/templates/skills/gh-auth-isolation/SKILL.md +183 -183
  46. package/templates/skills/humanizer/SKILL.md +105 -105
  47. package/templates/skills/pr-review-response/SKILL.md +268 -0
  48. package/templates/skills/pr-screenshots/SKILL.md +149 -149
  49. package/templates/skills/versioning-policy/SKILL.md +119 -0
  50. package/dist/cli/commands/watch/capabilities/budget-check.d.ts +0 -29
  51. package/dist/cli/commands/watch/capabilities/budget-check.d.ts.map +0 -1
  52. package/dist/cli/commands/watch/capabilities/budget-check.js +0 -38
  53. package/dist/cli/commands/watch/capabilities/budget-check.js.map +0 -1
  54. package/dist/cli/commands/watch/capabilities/circuit-breaker.d.ts +0 -52
  55. package/dist/cli/commands/watch/capabilities/circuit-breaker.d.ts.map +0 -1
  56. package/dist/cli/commands/watch/capabilities/circuit-breaker.js +0 -152
  57. package/dist/cli/commands/watch/capabilities/circuit-breaker.js.map +0 -1
  58. package/dist/cli/commands/watch/capabilities/health-check.d.ts +0 -29
  59. package/dist/cli/commands/watch/capabilities/health-check.d.ts.map +0 -1
  60. package/dist/cli/commands/watch/capabilities/health-check.js +0 -139
  61. package/dist/cli/commands/watch/capabilities/health-check.js.map +0 -1
  62. package/dist/cli/commands/watch/capabilities/heartbeat.d.ts +0 -48
  63. package/dist/cli/commands/watch/capabilities/heartbeat.d.ts.map +0 -1
  64. package/dist/cli/commands/watch/capabilities/heartbeat.js +0 -115
  65. package/dist/cli/commands/watch/capabilities/heartbeat.js.map +0 -1
  66. package/dist/cli/commands/watch/capabilities/lockfile.d.ts +0 -30
  67. package/dist/cli/commands/watch/capabilities/lockfile.d.ts.map +0 -1
  68. package/dist/cli/commands/watch/capabilities/lockfile.js +0 -100
  69. package/dist/cli/commands/watch/capabilities/lockfile.js.map +0 -1
  70. package/dist/cli/commands/watch/capabilities/machine-capabilities.d.ts +0 -30
  71. package/dist/cli/commands/watch/capabilities/machine-capabilities.d.ts.map +0 -1
  72. package/dist/cli/commands/watch/capabilities/machine-capabilities.js +0 -103
  73. package/dist/cli/commands/watch/capabilities/machine-capabilities.js.map +0 -1
  74. package/dist/cli/commands/watch/capabilities/post-failure.d.ts +0 -19
  75. package/dist/cli/commands/watch/capabilities/post-failure.d.ts.map +0 -1
  76. package/dist/cli/commands/watch/capabilities/post-failure.js +0 -58
  77. package/dist/cli/commands/watch/capabilities/post-failure.js.map +0 -1
  78. package/dist/cli/commands/watch/capabilities/priority.d.ts +0 -59
  79. package/dist/cli/commands/watch/capabilities/priority.d.ts.map +0 -1
  80. package/dist/cli/commands/watch/capabilities/priority.js +0 -101
  81. package/dist/cli/commands/watch/capabilities/priority.js.map +0 -1
  82. package/dist/cli/commands/watch/capabilities/rate-pool.d.ts +0 -67
  83. package/dist/cli/commands/watch/capabilities/rate-pool.d.ts.map +0 -1
  84. package/dist/cli/commands/watch/capabilities/rate-pool.js +0 -187
  85. package/dist/cli/commands/watch/capabilities/rate-pool.js.map +0 -1
  86. package/dist/cli/commands/watch/capabilities/stale-reclaim.d.ts +0 -23
  87. package/dist/cli/commands/watch/capabilities/stale-reclaim.d.ts.map +0 -1
  88. package/dist/cli/commands/watch/capabilities/stale-reclaim.js +0 -87
  89. package/dist/cli/commands/watch/capabilities/stale-reclaim.js.map +0 -1
  90. package/dist/cli/commands/watch/capabilities/webhook-alerts.d.ts +0 -29
  91. package/dist/cli/commands/watch/capabilities/webhook-alerts.d.ts.map +0 -1
  92. package/dist/cli/commands/watch/capabilities/webhook-alerts.js +0 -114
  93. package/dist/cli/commands/watch/capabilities/webhook-alerts.js.map +0 -1
@@ -1,115 +0,0 @@
1
- /**
2
- * Heartbeat capability — write status JSON every round.
3
- *
4
- * Ported from ralph-watch.ps1 `Update-Heartbeat`.
5
- * Writes to `.squad/ralph-heartbeat.json` every round with:
6
- * - round number, status, PID, timestamp
7
- * - consecutive failures counter
8
- * - duration of last round
9
- *
10
- * Also maintains a structured log at `.squad/ralph-watch.log` with rotation.
11
- *
12
- * Runs in the `housekeeping` phase.
13
- *
14
- * Config (via squad.config.ts → watch.capabilities["heartbeat"]):
15
- * heartbeatPath – override heartbeat file location (default: .squad/ralph-heartbeat.json)
16
- * logPath – override log file location (default: .squad/ralph-watch.log)
17
- * maxLogEntries – max log lines before rotation (default: 500)
18
- * maxLogBytes – max log file size before rotation (default: 1MB)
19
- */
20
- import path from 'node:path';
21
- import fs from 'node:fs';
22
- // Module-level state tracked across rounds
23
- let consecutiveFailures = 0;
24
- let lastRoundStart = Date.now();
25
- export class HeartbeatCapability {
26
- name = 'heartbeat';
27
- description = 'Write heartbeat JSON and structured log every round';
28
- configShape = 'object';
29
- requires = [];
30
- phase = 'housekeeping';
31
- async preflight(_context) {
32
- return { ok: true };
33
- }
34
- async execute(context) {
35
- const config = context.config;
36
- const squadDir = path.join(context.teamRoot, '.squad');
37
- const heartbeatPath = config['heartbeatPath'] ?? path.join(squadDir, 'ralph-heartbeat.json');
38
- const logPath = config['logPath'] ?? path.join(squadDir, 'ralph-watch.log');
39
- const maxLogEntries = config['maxLogEntries'] ?? 500;
40
- const maxLogBytes = config['maxLogBytes'] ?? 1_048_576; // 1 MB
41
- const now = new Date();
42
- const durationSeconds = Math.round((Date.now() - lastRoundStart) / 1000 * 100) / 100;
43
- lastRoundStart = Date.now();
44
- // Write heartbeat
45
- const heartbeat = {
46
- lastRun: now.toISOString(),
47
- lastHeartbeat: now.toISOString(),
48
- round: context.round,
49
- status: 'idle',
50
- exitCode: 0,
51
- durationSeconds,
52
- consecutiveFailures,
53
- pid: process.pid,
54
- };
55
- try {
56
- ensureDir(path.dirname(heartbeatPath));
57
- fs.writeFileSync(heartbeatPath, JSON.stringify(heartbeat, null, 2), 'utf-8');
58
- }
59
- catch { /* best-effort */ }
60
- // Append structured log entry
61
- const logEntry = `${now.toISOString()} | Round=${context.round} | Duration=${durationSeconds}s | Failures=${consecutiveFailures} | Status=idle`;
62
- try {
63
- ensureDir(path.dirname(logPath));
64
- fs.appendFileSync(logPath, logEntry + '\n', 'utf-8');
65
- rotateLog(logPath, maxLogEntries, maxLogBytes);
66
- }
67
- catch { /* best-effort */ }
68
- return {
69
- success: true,
70
- summary: `heartbeat written (round ${context.round})`,
71
- data: { heartbeatPath, durationSeconds },
72
- };
73
- }
74
- }
75
- /** Increment consecutive failure counter (called by the main loop on error). */
76
- export function recordFailure() {
77
- consecutiveFailures++;
78
- }
79
- /** Reset consecutive failure counter (called by the main loop on success). */
80
- export function recordSuccess() {
81
- consecutiveFailures = 0;
82
- }
83
- /** Get current consecutive failure count. */
84
- export function getConsecutiveFailures() {
85
- return consecutiveFailures;
86
- }
87
- /** Mark round start time for duration tracking. */
88
- export function markRoundStart() {
89
- lastRoundStart = Date.now();
90
- }
91
- function ensureDir(dir) {
92
- if (!fs.existsSync(dir)) {
93
- fs.mkdirSync(dir, { recursive: true });
94
- }
95
- }
96
- function rotateLog(logPath, maxEntries, maxBytes) {
97
- try {
98
- const stat = fs.statSync(logPath);
99
- let needsRotation = stat.size > maxBytes;
100
- if (!needsRotation) {
101
- const content = fs.readFileSync(logPath, 'utf-8');
102
- const lines = content.split('\n').filter(Boolean);
103
- needsRotation = lines.length > maxEntries;
104
- }
105
- if (needsRotation) {
106
- const content = fs.readFileSync(logPath, 'utf-8');
107
- const lines = content.split('\n').filter(Boolean);
108
- const kept = lines.slice(-Math.max(maxEntries - 1, 1));
109
- const header = `# Ralph Watch Log — Rotated ${new Date().toISOString()} (kept last ${kept.length} entries)`;
110
- fs.writeFileSync(logPath, [header, ...kept].join('\n') + '\n', 'utf-8');
111
- }
112
- }
113
- catch { /* best-effort */ }
114
- }
115
- //# sourceMappingURL=heartbeat.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"heartbeat.js","sourceRoot":"","sources":["../../../../../src/cli/commands/watch/capabilities/heartbeat.ts"],"names":[],"mappings":"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"}
@@ -1,30 +0,0 @@
1
- /**
2
- * Lockfile utility — per-repo lock with PID, timestamp, and stale detection.
3
- *
4
- * Ported from ralph-watch.ps1 lockfile logic.
5
- * Written BEFORE a round (status=running) and AFTER (status=idle/error).
6
- * External tools (e.g., squad-monitor) can read this file to know if
7
- * a watch process is active.
8
- *
9
- * This is a utility module — not a WatchCapability.
10
- */
11
- export interface LockfileData {
12
- pid: number;
13
- status: 'running' | 'idle' | 'error';
14
- round: number;
15
- lastRun: string;
16
- exitCode: number;
17
- consecutiveFailures: number;
18
- started: string;
19
- directory: string;
20
- }
21
- /**
22
- * Acquire a lockfile for this watch process.
23
- * Returns false if another non-stale process holds the lock.
24
- */
25
- export declare function acquireLock(teamRoot: string): boolean;
26
- /** Update the lockfile with current round state. */
27
- export declare function updateLock(teamRoot: string, status: 'running' | 'idle' | 'error', round: number, exitCode?: number, consecutiveFailures?: number): void;
28
- /** Release the lockfile on shutdown. */
29
- export declare function releaseLock(teamRoot: string): void;
30
- //# sourceMappingURL=lockfile.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"lockfile.d.ts","sourceRoot":"","sources":["../../../../../src/cli/commands/watch/capabilities/lockfile.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAKH,MAAM,WAAW,YAAY;IAC3B,GAAG,EAAE,MAAM,CAAC;IACZ,MAAM,EAAE,SAAS,GAAG,MAAM,GAAG,OAAO,CAAC;IACrC,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,mBAAmB,EAAE,MAAM,CAAC;IAC5B,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;CACnB;AAID;;;GAGG;AACH,wBAAgB,WAAW,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAiCrD;AAED,oDAAoD;AACpD,wBAAgB,UAAU,CACxB,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,SAAS,GAAG,MAAM,GAAG,OAAO,EACpC,KAAK,EAAE,MAAM,EACb,QAAQ,GAAE,MAAU,EACpB,mBAAmB,GAAE,MAAU,GAC9B,IAAI,CAiBN;AAED,wCAAwC;AACxC,wBAAgB,WAAW,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI,CAUlD"}
@@ -1,100 +0,0 @@
1
- /**
2
- * Lockfile utility — per-repo lock with PID, timestamp, and stale detection.
3
- *
4
- * Ported from ralph-watch.ps1 lockfile logic.
5
- * Written BEFORE a round (status=running) and AFTER (status=idle/error).
6
- * External tools (e.g., squad-monitor) can read this file to know if
7
- * a watch process is active.
8
- *
9
- * This is a utility module — not a WatchCapability.
10
- */
11
- import path from 'node:path';
12
- import fs from 'node:fs';
13
- const STALE_THRESHOLD_MS = 30 * 60 * 1000; // 30 minutes
14
- /**
15
- * Acquire a lockfile for this watch process.
16
- * Returns false if another non-stale process holds the lock.
17
- */
18
- export function acquireLock(teamRoot) {
19
- const lockPath = getLockPath(teamRoot);
20
- if (fs.existsSync(lockPath)) {
21
- try {
22
- const existing = JSON.parse(fs.readFileSync(lockPath, 'utf-8'));
23
- // Check if the PID is still alive
24
- if (isProcessAlive(existing.pid)) {
25
- // Check if it's stale
26
- const age = Date.now() - new Date(existing.lastRun).getTime();
27
- if (age < STALE_THRESHOLD_MS) {
28
- return false; // Another active process holds the lock
29
- }
30
- // Stale — we can take over
31
- }
32
- // Dead or stale PID — safe to overwrite
33
- }
34
- catch {
35
- // Corrupt lockfile — overwrite
36
- }
37
- }
38
- writeLock(teamRoot, {
39
- pid: process.pid,
40
- status: 'running',
41
- round: 0,
42
- lastRun: new Date().toISOString(),
43
- exitCode: 0,
44
- consecutiveFailures: 0,
45
- started: new Date().toISOString(),
46
- directory: teamRoot,
47
- });
48
- return true;
49
- }
50
- /** Update the lockfile with current round state. */
51
- export function updateLock(teamRoot, status, round, exitCode = 0, consecutiveFailures = 0) {
52
- const lockPath = getLockPath(teamRoot);
53
- let existing = {};
54
- try {
55
- existing = JSON.parse(fs.readFileSync(lockPath, 'utf-8'));
56
- }
57
- catch { /* use defaults */ }
58
- writeLock(teamRoot, {
59
- pid: process.pid,
60
- status,
61
- round,
62
- lastRun: new Date().toISOString(),
63
- exitCode,
64
- consecutiveFailures,
65
- started: existing.started ?? new Date().toISOString(),
66
- directory: teamRoot,
67
- });
68
- }
69
- /** Release the lockfile on shutdown. */
70
- export function releaseLock(teamRoot) {
71
- const lockPath = getLockPath(teamRoot);
72
- try {
73
- if (fs.existsSync(lockPath)) {
74
- const data = JSON.parse(fs.readFileSync(lockPath, 'utf-8'));
75
- if (data.pid === process.pid) {
76
- fs.unlinkSync(lockPath);
77
- }
78
- }
79
- }
80
- catch { /* best-effort */ }
81
- }
82
- function getLockPath(teamRoot) {
83
- return path.join(teamRoot, '.ralph-watch.lock');
84
- }
85
- function writeLock(teamRoot, data) {
86
- try {
87
- fs.writeFileSync(getLockPath(teamRoot), JSON.stringify(data, null, 2), 'utf-8');
88
- }
89
- catch { /* best-effort */ }
90
- }
91
- function isProcessAlive(pid) {
92
- try {
93
- process.kill(pid, 0);
94
- return true;
95
- }
96
- catch {
97
- return false;
98
- }
99
- }
100
- //# sourceMappingURL=lockfile.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"lockfile.js","sourceRoot":"","sources":["../../../../../src/cli/commands/watch/capabilities/lockfile.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,MAAM,SAAS,CAAC;AAazB,MAAM,kBAAkB,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,aAAa;AAExD;;;GAGG;AACH,MAAM,UAAU,WAAW,CAAC,QAAgB;IAC1C,MAAM,QAAQ,GAAG,WAAW,CAAC,QAAQ,CAAC,CAAC;IAEvC,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC5B,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAiB,CAAC;YAChF,kCAAkC;YAClC,IAAI,cAAc,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;gBACjC,sBAAsB;gBACtB,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,CAAC;gBAC9D,IAAI,GAAG,GAAG,kBAAkB,EAAE,CAAC;oBAC7B,OAAO,KAAK,CAAC,CAAC,wCAAwC;gBACxD,CAAC;gBACD,2BAA2B;YAC7B,CAAC;YACD,wCAAwC;QAC1C,CAAC;QAAC,MAAM,CAAC;YACP,+BAA+B;QACjC,CAAC;IACH,CAAC;IAED,SAAS,CAAC,QAAQ,EAAE;QAClB,GAAG,EAAE,OAAO,CAAC,GAAG;QAChB,MAAM,EAAE,SAAS;QACjB,KAAK,EAAE,CAAC;QACR,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACjC,QAAQ,EAAE,CAAC;QACX,mBAAmB,EAAE,CAAC;QACtB,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACjC,SAAS,EAAE,QAAQ;KACpB,CAAC,CAAC;IAEH,OAAO,IAAI,CAAC;AACd,CAAC;AAED,oDAAoD;AACpD,MAAM,UAAU,UAAU,CACxB,QAAgB,EAChB,MAAoC,EACpC,KAAa,EACb,WAAmB,CAAC,EACpB,sBAA8B,CAAC;IAE/B,MAAM,QAAQ,GAAG,WAAW,CAAC,QAAQ,CAAC,CAAC;IACvC,IAAI,QAAQ,GAA0B,EAAE,CAAC;IACzC,IAAI,CAAC;QACH,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,CAAC;IAC5D,CAAC;IAAC,MAAM,CAAC,CAAC,kBAAkB,CAAC,CAAC;IAE9B,SAAS,CAAC,QAAQ,EAAE;QAClB,GAAG,EAAE,OAAO,CAAC,GAAG;QAChB,MAAM;QACN,KAAK;QACL,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACjC,QAAQ;QACR,mBAAmB;QACnB,OAAO,EAAG,QAAyB,CAAC,OAAO,IAAI,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACvE,SAAS,EAAE,QAAQ;KACpB,CAAC,CAAC;AACL,CAAC;AAED,wCAAwC;AACxC,MAAM,UAAU,WAAW,CAAC,QAAgB;IAC1C,MAAM,QAAQ,GAAG,WAAW,CAAC,QAAQ,CAAC,CAAC;IACvC,IAAI,CAAC;QACH,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC5B,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAiB,CAAC;YAC5E,IAAI,IAAI,CAAC,GAAG,KAAK,OAAO,CAAC,GAAG,EAAE,CAAC;gBAC7B,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;YAC1B,CAAC;QACH,CAAC;IACH,CAAC;IAAC,MAAM,CAAC,CAAC,iBAAiB,CAAC,CAAC;AAC/B,CAAC;AAED,SAAS,WAAW,CAAC,QAAgB;IACnC,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,mBAAmB,CAAC,CAAC;AAClD,CAAC;AAED,SAAS,SAAS,CAAC,QAAgB,EAAE,IAAkB;IACrD,IAAI,CAAC;QACH,EAAE,CAAC,aAAa,CAAC,WAAW,CAAC,QAAQ,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;IAClF,CAAC;IAAC,MAAM,CAAC,CAAC,iBAAiB,CAAC,CAAC;AAC/B,CAAC;AAED,SAAS,cAAc,CAAC,GAAW;IACjC,IAAI,CAAC;QACH,OAAO,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;QACrB,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC"}
@@ -1,30 +0,0 @@
1
- /**
2
- * Machine capability checking — match issue needs:* labels to local machine.
3
- *
4
- * Ported from ralph-watch.ps1 `Test-MachineCapability`.
5
- * Reads `needs:*` labels from issues and compares against a local
6
- * capabilities list provided via:
7
- * 1. CLI flag: --capabilities gpu,browser,docker
8
- * 2. Config: watch.capabilities["machine-capabilities"].list
9
- * 3. Auto-detect: probe for common tools (nvidia-smi, playwright, docker)
10
- *
11
- * This is a utility module — not a WatchCapability.
12
- */
13
- export interface MachineCapabilityResult {
14
- canHandle: boolean;
15
- reason: string;
16
- missing?: string[];
17
- }
18
- export interface MachineCapabilityConfig {
19
- /** Explicit list of capabilities this machine has. */
20
- list?: string[];
21
- /** Whether to auto-detect capabilities (default: true). */
22
- autoDetect?: boolean;
23
- }
24
- /** Detect capabilities available on this machine. */
25
- export declare function detectCapabilities(config?: MachineCapabilityConfig): Promise<string[]>;
26
- /**
27
- * Check if this machine can handle an issue based on its needs:* labels.
28
- */
29
- export declare function checkMachineCapability(issueLabels: string[], machineCaps: string[]): MachineCapabilityResult;
30
- //# sourceMappingURL=machine-capabilities.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"machine-capabilities.d.ts","sourceRoot":"","sources":["../../../../../src/cli/commands/watch/capabilities/machine-capabilities.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAOH,MAAM,WAAW,uBAAuB;IACtC,SAAS,EAAE,OAAO,CAAC;IACnB,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC;CACpB;AAED,MAAM,WAAW,uBAAuB;IACtC,sDAAsD;IACtD,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;IAChB,2DAA2D;IAC3D,UAAU,CAAC,EAAE,OAAO,CAAC;CACtB;AAyCD,qDAAqD;AACrD,wBAAsB,kBAAkB,CAAC,MAAM,CAAC,EAAE,uBAAuB,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,CAW5F;AAED;;GAEG;AACH,wBAAgB,sBAAsB,CACpC,WAAW,EAAE,MAAM,EAAE,EACrB,WAAW,EAAE,MAAM,EAAE,GACpB,uBAAuB,CAqBzB"}
@@ -1,103 +0,0 @@
1
- /**
2
- * Machine capability checking — match issue needs:* labels to local machine.
3
- *
4
- * Ported from ralph-watch.ps1 `Test-MachineCapability`.
5
- * Reads `needs:*` labels from issues and compares against a local
6
- * capabilities list provided via:
7
- * 1. CLI flag: --capabilities gpu,browser,docker
8
- * 2. Config: watch.capabilities["machine-capabilities"].list
9
- * 3. Auto-detect: probe for common tools (nvidia-smi, playwright, docker)
10
- *
11
- * This is a utility module — not a WatchCapability.
12
- */
13
- import { execFile } from 'node:child_process';
14
- import { promisify } from 'node:util';
15
- const execFileAsync = promisify(execFile);
16
- /** Probes for common capabilities on the local machine. */
17
- const CAPABILITY_PROBES = {
18
- gpu: async () => {
19
- try {
20
- await execFileAsync('nvidia-smi', ['--query-gpu=name', '--format=csv,noheader'], { timeout: 5_000 });
21
- return true;
22
- }
23
- catch {
24
- return false;
25
- }
26
- },
27
- browser: async () => {
28
- try {
29
- await execFileAsync('npx', ['playwright', '--version'], { timeout: 10_000 });
30
- return true;
31
- }
32
- catch {
33
- return false;
34
- }
35
- },
36
- docker: async () => {
37
- try {
38
- await execFileAsync('docker', ['info'], { timeout: 5_000 });
39
- return true;
40
- }
41
- catch {
42
- return false;
43
- }
44
- },
45
- node: async () => {
46
- try {
47
- await execFileAsync('node', ['--version'], { timeout: 5_000 });
48
- return true;
49
- }
50
- catch {
51
- return false;
52
- }
53
- },
54
- python: async () => {
55
- try {
56
- await execFileAsync('python3', ['--version'], { timeout: 5_000 });
57
- return true;
58
- }
59
- catch {
60
- try {
61
- await execFileAsync('python', ['--version'], { timeout: 5_000 });
62
- return true;
63
- }
64
- catch {
65
- return false;
66
- }
67
- }
68
- },
69
- };
70
- /** Detect capabilities available on this machine. */
71
- export async function detectCapabilities(config) {
72
- const caps = new Set(config?.list ?? []);
73
- if (config?.autoDetect !== false) {
74
- const probes = Object.entries(CAPABILITY_PROBES).map(async ([name, probe]) => {
75
- if (await probe())
76
- caps.add(name);
77
- });
78
- await Promise.all(probes);
79
- }
80
- return [...caps];
81
- }
82
- /**
83
- * Check if this machine can handle an issue based on its needs:* labels.
84
- */
85
- export function checkMachineCapability(issueLabels, machineCaps) {
86
- const needsLabels = issueLabels
87
- .filter(l => l.startsWith('needs:'))
88
- .map(l => l.replace(/^needs:/, ''));
89
- if (needsLabels.length === 0) {
90
- return { canHandle: true, reason: 'No needs:* labels on issue' };
91
- }
92
- const capSet = new Set(machineCaps.map(c => c.toLowerCase()));
93
- const missing = needsLabels.filter(need => !capSet.has(need.toLowerCase()));
94
- if (missing.length === 0) {
95
- return { canHandle: true, reason: 'All required capabilities present' };
96
- }
97
- return {
98
- canHandle: false,
99
- reason: `Missing capabilities: ${missing.join(', ')}`,
100
- missing,
101
- };
102
- }
103
- //# sourceMappingURL=machine-capabilities.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"machine-capabilities.js","sourceRoot":"","sources":["../../../../../src/cli/commands/watch/capabilities/machine-capabilities.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAEH,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAC9C,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAEtC,MAAM,aAAa,GAAG,SAAS,CAAC,QAAQ,CAAC,CAAC;AAe1C,2DAA2D;AAC3D,MAAM,iBAAiB,GAA2C;IAChE,GAAG,EAAE,KAAK,IAAI,EAAE;QACd,IAAI,CAAC;YACH,MAAM,aAAa,CAAC,YAAY,EAAE,CAAC,kBAAkB,EAAE,uBAAuB,CAAC,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YACrG,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YAAC,OAAO,KAAK,CAAC;QAAC,CAAC;IAC3B,CAAC;IACD,OAAO,EAAE,KAAK,IAAI,EAAE;QAClB,IAAI,CAAC;YACH,MAAM,aAAa,CAAC,KAAK,EAAE,CAAC,YAAY,EAAE,WAAW,CAAC,EAAE,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC;YAC7E,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YAAC,OAAO,KAAK,CAAC;QAAC,CAAC;IAC3B,CAAC;IACD,MAAM,EAAE,KAAK,IAAI,EAAE;QACjB,IAAI,CAAC;YACH,MAAM,aAAa,CAAC,QAAQ,EAAE,CAAC,MAAM,CAAC,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YAC5D,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YAAC,OAAO,KAAK,CAAC;QAAC,CAAC;IAC3B,CAAC;IACD,IAAI,EAAE,KAAK,IAAI,EAAE;QACf,IAAI,CAAC;YACH,MAAM,aAAa,CAAC,MAAM,EAAE,CAAC,WAAW,CAAC,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YAC/D,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YAAC,OAAO,KAAK,CAAC;QAAC,CAAC;IAC3B,CAAC;IACD,MAAM,EAAE,KAAK,IAAI,EAAE;QACjB,IAAI,CAAC;YACH,MAAM,aAAa,CAAC,SAAS,EAAE,CAAC,WAAW,CAAC,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YAClE,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YACP,IAAI,CAAC;gBACH,MAAM,aAAa,CAAC,QAAQ,EAAE,CAAC,WAAW,CAAC,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;gBACjE,OAAO,IAAI,CAAC;YACd,CAAC;YAAC,MAAM,CAAC;gBAAC,OAAO,KAAK,CAAC;YAAC,CAAC;QAC3B,CAAC;IACH,CAAC;CACF,CAAC;AAEF,qDAAqD;AACrD,MAAM,CAAC,KAAK,UAAU,kBAAkB,CAAC,MAAgC;IACvE,MAAM,IAAI,GAAG,IAAI,GAAG,CAAS,MAAM,EAAE,IAAI,IAAI,EAAE,CAAC,CAAC;IAEjD,IAAI,MAAM,EAAE,UAAU,KAAK,KAAK,EAAE,CAAC;QACjC,MAAM,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC,IAAI,EAAE,KAAK,CAAC,EAAE,EAAE;YAC3E,IAAI,MAAM,KAAK,EAAE;gBAAE,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACpC,CAAC,CAAC,CAAC;QACH,MAAM,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAC5B,CAAC;IAED,OAAO,CAAC,GAAG,IAAI,CAAC,CAAC;AACnB,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,sBAAsB,CACpC,WAAqB,EACrB,WAAqB;IAErB,MAAM,WAAW,GAAG,WAAW;SAC5B,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;SACnC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC,CAAC;IAEtC,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC7B,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,4BAA4B,EAAE,CAAC;IACnE,CAAC;IAED,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;IAC9D,MAAM,OAAO,GAAG,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;IAE5E,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACzB,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,mCAAmC,EAAE,CAAC;IAC1E,CAAC;IAED,OAAO;QACL,SAAS,EAAE,KAAK;QAChB,MAAM,EAAE,yBAAyB,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;QACrD,OAAO;KACR,CAAC;AACJ,CAAC"}
@@ -1,19 +0,0 @@
1
- /**
2
- * Post-failure remediation — tiered self-healing response.
3
- *
4
- * Ported from ralph-watch.ps1 `Invoke-PostFailureRemediation`.
5
- * Tiered response based on consecutive failure count:
6
- * Tier 1 (≥3): Reset circuit breaker state
7
- * Tier 2 (≥6): Re-verify auth
8
- * Tier 3 (≥9): Git pull to get latest fixes
9
- * Tier 4 (≥15): Extended pause + webhook alert
10
- *
11
- * This is a utility module — called from the main watch loop on errors.
12
- */
13
- import type { ModelCircuitBreaker } from './circuit-breaker.js';
14
- export interface RemediationResult {
15
- actions: string[];
16
- pauseSeconds: number;
17
- }
18
- export declare function runPostFailureRemediation(consecutiveFailures: number, round: number, teamRoot: string, circuitBreaker?: ModelCircuitBreaker): Promise<RemediationResult>;
19
- //# sourceMappingURL=post-failure.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"post-failure.d.ts","sourceRoot":"","sources":["../../../../../src/cli/commands/watch/capabilities/post-failure.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAIH,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,sBAAsB,CAAC;AAIhE,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,YAAY,EAAE,MAAM,CAAC;CACtB;AAED,wBAAsB,yBAAyB,CAC7C,mBAAmB,EAAE,MAAM,EAC3B,KAAK,EAAE,MAAM,EACb,QAAQ,EAAE,MAAM,EAChB,cAAc,CAAC,EAAE,mBAAmB,GACnC,OAAO,CAAC,iBAAiB,CAAC,CA4C5B"}
@@ -1,58 +0,0 @@
1
- /**
2
- * Post-failure remediation — tiered self-healing response.
3
- *
4
- * Ported from ralph-watch.ps1 `Invoke-PostFailureRemediation`.
5
- * Tiered response based on consecutive failure count:
6
- * Tier 1 (≥3): Reset circuit breaker state
7
- * Tier 2 (≥6): Re-verify auth
8
- * Tier 3 (≥9): Git pull to get latest fixes
9
- * Tier 4 (≥15): Extended pause + webhook alert
10
- *
11
- * This is a utility module — called from the main watch loop on errors.
12
- */
13
- import { execFile } from 'node:child_process';
14
- import { promisify } from 'node:util';
15
- const execFileAsync = promisify(execFile);
16
- export async function runPostFailureRemediation(consecutiveFailures, round, teamRoot, circuitBreaker) {
17
- const actions = [];
18
- let pauseSeconds = 0;
19
- if (consecutiveFailures >= 3 && consecutiveFailures < 6) {
20
- // Tier 1: Reset circuit breaker
21
- if (circuitBreaker) {
22
- circuitBreaker.reset();
23
- actions.push('Tier1: Reset circuit breaker to defaults');
24
- }
25
- }
26
- if (consecutiveFailures >= 6 && consecutiveFailures < 9) {
27
- // Tier 2: Re-verify gh auth
28
- try {
29
- const { stdout } = await execFileAsync('gh', ['api', 'user', '--jq', '.login'], {
30
- timeout: 10_000,
31
- });
32
- actions.push(`Tier2: GH auth verified (${stdout.trim()})`);
33
- }
34
- catch {
35
- actions.push('Tier2: GH auth still failing');
36
- }
37
- }
38
- if (consecutiveFailures >= 9 && consecutiveFailures < 15) {
39
- // Tier 3: Git pull latest
40
- try {
41
- await execFileAsync('git', ['pull', '--rebase', '--quiet'], {
42
- cwd: teamRoot,
43
- timeout: 30_000,
44
- });
45
- actions.push('Tier3: Git pull --rebase succeeded');
46
- }
47
- catch {
48
- actions.push('Tier3: Git pull failed');
49
- }
50
- }
51
- if (consecutiveFailures >= 15) {
52
- // Tier 4: Extended pause
53
- pauseSeconds = 30 * 60; // 30 minutes
54
- actions.push(`Tier4: ${consecutiveFailures} failures — pausing ${pauseSeconds / 60} minutes`);
55
- }
56
- return { actions, pauseSeconds };
57
- }
58
- //# sourceMappingURL=post-failure.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"post-failure.js","sourceRoot":"","sources":["../../../../../src/cli/commands/watch/capabilities/post-failure.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAEH,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAC9C,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAGtC,MAAM,aAAa,GAAG,SAAS,CAAC,QAAQ,CAAC,CAAC;AAO1C,MAAM,CAAC,KAAK,UAAU,yBAAyB,CAC7C,mBAA2B,EAC3B,KAAa,EACb,QAAgB,EAChB,cAAoC;IAEpC,MAAM,OAAO,GAAa,EAAE,CAAC;IAC7B,IAAI,YAAY,GAAG,CAAC,CAAC;IAErB,IAAI,mBAAmB,IAAI,CAAC,IAAI,mBAAmB,GAAG,CAAC,EAAE,CAAC;QACxD,gCAAgC;QAChC,IAAI,cAAc,EAAE,CAAC;YACnB,cAAc,CAAC,KAAK,EAAE,CAAC;YACvB,OAAO,CAAC,IAAI,CAAC,0CAA0C,CAAC,CAAC;QAC3D,CAAC;IACH,CAAC;IAED,IAAI,mBAAmB,IAAI,CAAC,IAAI,mBAAmB,GAAG,CAAC,EAAE,CAAC;QACxD,4BAA4B;QAC5B,IAAI,CAAC;YACH,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,aAAa,CAAC,IAAI,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE;gBAC9E,OAAO,EAAE,MAAM;aAChB,CAAC,CAAC;YACH,OAAO,CAAC,IAAI,CAAC,4BAA4B,MAAM,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;QAC7D,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;QAC/C,CAAC;IACH,CAAC;IAED,IAAI,mBAAmB,IAAI,CAAC,IAAI,mBAAmB,GAAG,EAAE,EAAE,CAAC;QACzD,0BAA0B;QAC1B,IAAI,CAAC;YACH,MAAM,aAAa,CAAC,KAAK,EAAE,CAAC,MAAM,EAAE,UAAU,EAAE,SAAS,CAAC,EAAE;gBAC1D,GAAG,EAAE,QAAQ;gBACb,OAAO,EAAE,MAAM;aAChB,CAAC,CAAC;YACH,OAAO,CAAC,IAAI,CAAC,oCAAoC,CAAC,CAAC;QACrD,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAED,IAAI,mBAAmB,IAAI,EAAE,EAAE,CAAC;QAC9B,yBAAyB;QACzB,YAAY,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC,aAAa;QACrC,OAAO,CAAC,IAAI,CAAC,UAAU,mBAAmB,uBAAuB,YAAY,GAAG,EAAE,UAAU,CAAC,CAAC;IAChG,CAAC;IAED,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,CAAC;AACnC,CAAC"}
@@ -1,59 +0,0 @@
1
- /**
2
- * Issue priority scoring — weigh issues for execution order.
3
- *
4
- * Ported from ralph-watch.ps1 issue scoring logic.
5
- * Factors:
6
- * - Priority labels: P0 (100), P1 (60), P2 (30), P3 (10)
7
- * - Age bonus: +1 per day open (max +30)
8
- * - Staleness: +20 if no activity in 7+ days
9
- * - Bug label: +15
10
- * - Size labels: size:S +10, size:M 0, size:L -5
11
- *
12
- * This is a utility module — not a WatchCapability.
13
- * Used by the execute capability to sort issues before picking work.
14
- */
15
- export interface ScoredIssue {
16
- number: number;
17
- title: string;
18
- labels: Array<{
19
- name: string;
20
- }>;
21
- score: number;
22
- breakdown: Record<string, number>;
23
- }
24
- export interface PriorityConfig {
25
- /** Weight multipliers — override defaults. */
26
- weights?: Partial<PriorityWeights>;
27
- }
28
- export interface PriorityWeights {
29
- p0: number;
30
- p1: number;
31
- p2: number;
32
- p3: number;
33
- agePerDay: number;
34
- ageMax: number;
35
- staleThresholdDays: number;
36
- staleBonus: number;
37
- bugBonus: number;
38
- sizeSBonus: number;
39
- sizeLPenalty: number;
40
- }
41
- export interface IssueLike {
42
- number: number;
43
- title: string;
44
- labels: Array<{
45
- name: string;
46
- }>;
47
- createdAt?: string | Date;
48
- updatedAt?: string | Date;
49
- }
50
- /**
51
- * Score a single issue for execution priority.
52
- * Higher score = should be picked first.
53
- */
54
- export declare function scoreIssue(issue: IssueLike, config?: PriorityConfig): ScoredIssue;
55
- /**
56
- * Score and sort a batch of issues, highest score first.
57
- */
58
- export declare function rankIssues(issues: IssueLike[], config?: PriorityConfig): ScoredIssue[];
59
- //# sourceMappingURL=priority.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"priority.d.ts","sourceRoot":"","sources":["../../../../../src/cli/commands/watch/capabilities/priority.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAEH,MAAM,WAAW,WAAW;IAC1B,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAChC,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CACnC;AAED,MAAM,WAAW,cAAc;IAC7B,8CAA8C;IAC9C,OAAO,CAAC,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC;CACpC;AAED,MAAM,WAAW,eAAe;IAC9B,EAAE,EAAE,MAAM,CAAC;IACX,EAAE,EAAE,MAAM,CAAC;IACX,EAAE,EAAE,MAAM,CAAC;IACX,EAAE,EAAE,MAAM,CAAC;IACX,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,kBAAkB,EAAE,MAAM,CAAC;IAC3B,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,UAAU,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;CACtB;AAgBD,MAAM,WAAW,SAAS;IACxB,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAChC,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;CAC3B;AAED;;;GAGG;AACH,wBAAgB,UAAU,CAAC,KAAK,EAAE,SAAS,EAAE,MAAM,CAAC,EAAE,cAAc,GAAG,WAAW,CA8DjF;AAED;;GAEG;AACH,wBAAgB,UAAU,CAAC,MAAM,EAAE,SAAS,EAAE,EAAE,MAAM,CAAC,EAAE,cAAc,GAAG,WAAW,EAAE,CAItF"}
@@ -1,101 +0,0 @@
1
- /**
2
- * Issue priority scoring — weigh issues for execution order.
3
- *
4
- * Ported from ralph-watch.ps1 issue scoring logic.
5
- * Factors:
6
- * - Priority labels: P0 (100), P1 (60), P2 (30), P3 (10)
7
- * - Age bonus: +1 per day open (max +30)
8
- * - Staleness: +20 if no activity in 7+ days
9
- * - Bug label: +15
10
- * - Size labels: size:S +10, size:M 0, size:L -5
11
- *
12
- * This is a utility module — not a WatchCapability.
13
- * Used by the execute capability to sort issues before picking work.
14
- */
15
- const DEFAULT_WEIGHTS = {
16
- p0: 100,
17
- p1: 60,
18
- p2: 30,
19
- p3: 10,
20
- agePerDay: 1,
21
- ageMax: 30,
22
- staleThresholdDays: 7,
23
- staleBonus: 20,
24
- bugBonus: 15,
25
- sizeSBonus: 10,
26
- sizeLPenalty: -5,
27
- };
28
- /**
29
- * Score a single issue for execution priority.
30
- * Higher score = should be picked first.
31
- */
32
- export function scoreIssue(issue, config) {
33
- const w = { ...DEFAULT_WEIGHTS, ...(config?.weights ?? {}) };
34
- const breakdown = {};
35
- let score = 0;
36
- const labels = new Set(issue.labels.map(l => l.name.toLowerCase()));
37
- // Priority label scoring
38
- if (labels.has('p0') || labels.has('priority:p0') || labels.has('priority:critical')) {
39
- breakdown['priority'] = w.p0;
40
- }
41
- else if (labels.has('p1') || labels.has('priority:p1') || labels.has('priority:high')) {
42
- breakdown['priority'] = w.p1;
43
- }
44
- else if (labels.has('p2') || labels.has('priority:p2') || labels.has('priority:medium')) {
45
- breakdown['priority'] = w.p2;
46
- }
47
- else if (labels.has('p3') || labels.has('priority:p3') || labels.has('priority:low')) {
48
- breakdown['priority'] = w.p3;
49
- }
50
- else {
51
- breakdown['priority'] = w.p2; // Default to P2 if unlabeled
52
- }
53
- score += breakdown['priority'];
54
- // Age bonus
55
- if (issue.createdAt) {
56
- const created = new Date(issue.createdAt);
57
- const ageDays = Math.floor((Date.now() - created.getTime()) / (1000 * 60 * 60 * 24));
58
- const ageScore = Math.min(ageDays * w.agePerDay, w.ageMax);
59
- breakdown['age'] = ageScore;
60
- score += ageScore;
61
- }
62
- // Staleness bonus
63
- if (issue.updatedAt) {
64
- const updated = new Date(issue.updatedAt);
65
- const staleDays = Math.floor((Date.now() - updated.getTime()) / (1000 * 60 * 60 * 24));
66
- if (staleDays >= w.staleThresholdDays) {
67
- breakdown['stale'] = w.staleBonus;
68
- score += w.staleBonus;
69
- }
70
- }
71
- // Bug label bonus
72
- if (labels.has('bug') || labels.has('type:bug')) {
73
- breakdown['bug'] = w.bugBonus;
74
- score += w.bugBonus;
75
- }
76
- // Size label adjustment
77
- if (labels.has('size:s') || labels.has('size:small')) {
78
- breakdown['size'] = w.sizeSBonus;
79
- score += w.sizeSBonus;
80
- }
81
- else if (labels.has('size:l') || labels.has('size:large') || labels.has('size:xl')) {
82
- breakdown['size'] = w.sizeLPenalty;
83
- score += w.sizeLPenalty;
84
- }
85
- return {
86
- number: issue.number,
87
- title: issue.title,
88
- labels: issue.labels,
89
- score,
90
- breakdown,
91
- };
92
- }
93
- /**
94
- * Score and sort a batch of issues, highest score first.
95
- */
96
- export function rankIssues(issues, config) {
97
- return issues
98
- .map(i => scoreIssue(i, config))
99
- .sort((a, b) => b.score - a.score);
100
- }
101
- //# sourceMappingURL=priority.js.map