islandflow/.agents/skills/impeccable/scripts/context-signals.mjs
dirtydishes f237916291
Some checks are pending
CI / Validate (push) Waiting to run
Install Impeccable skill for Codex
2026-05-29 03:59:27 -04:00

225 lines
7.9 KiB
JavaScript

#!/usr/bin/env node
/**
* Context-signals gatherer for the bare `{{command_prefix}}impeccable`
* (no-argument) path. Collects cheap, deterministic signals about the current
* project and emits them as JSON.
*
* It does NOT score or rank. The agent reasons over the raw signals using its
* knowledge of the command catalog (see SKILL.md routing rule 1). Deliberately
* light: no LLM calls, no detector run (`npx impeccable detect` is heavier and
* opt-in), no file writes. Every probe is best-effort and never throws; the
* output is always valid JSON.
*
* Signals:
* - setup: PRODUCT.md / DESIGN.md presence, register, whether code exists
* - critique: the latest cached critique score (.impeccable/critique)
* - git: branch + files changed vs the default branch (a scope hint)
* - devServer: whether a local dev server answers on a common port (gates live)
*/
import fs from 'node:fs';
import net from 'node:net';
import path from 'node:path';
import { fileURLToPath } from 'node:url';
import { execFileSync } from 'node:child_process';
import { loadContext, extractRegister } from './context.mjs';
import { getCritiqueDir } from './impeccable-paths.mjs';
/** Is there code here at all, or just context files / an empty repo? */
function hasCode(cwd) {
if (fs.existsSync(path.join(cwd, 'package.json'))) return true;
for (const d of ['src', 'app', 'pages', 'site', 'public', 'components', 'lib']) {
if (fs.existsSync(path.join(cwd, d))) return true;
}
return false;
}
/**
* The most recent critique snapshot across all targets. Filenames are
* timestamp-prefixed (`<iso>__<slug>.md`), so a lexical sort is chronological.
* Parses the small frontmatter for score + P0/P1 counts.
*/
function latestCritique(cwd) {
try {
const dir = getCritiqueDir(cwd);
if (!fs.existsSync(dir)) return null;
const files = fs.readdirSync(dir).filter((f) => f.endsWith('.md')).sort();
if (!files.length) return null;
const newest = files[files.length - 1];
const text = fs.readFileSync(path.join(dir, newest), 'utf-8');
const front = text.split('---')[1] || '';
const get = (k) => {
const m = front.match(new RegExp(`^${k}:\\s*(.+)$`, 'm'));
return m ? m[1].trim() : null;
};
const num = (v) => {
const n = Number(v);
return Number.isFinite(n) ? n : null;
};
return {
slug: get('slug'),
score: num(get('score')),
p0: num(get('p0')),
p1: num(get('p1')),
timestamp: get('timestamp'),
file: path.relative(cwd, path.join(dir, newest)),
};
} catch {
return null;
}
}
/** Branch + a scope hint: files changed vs the default branch, else working tree. */
function gitSignals(cwd) {
const run = (args, { trim = true } = {}) => {
try {
const out = execFileSync('git', args, {
cwd,
encoding: 'utf-8',
stdio: ['ignore', 'pipe', 'ignore'],
});
return trim ? out.trim() : out;
} catch {
return null;
}
};
if (run(['rev-parse', '--is-inside-work-tree']) !== 'true') {
return { isRepo: false, branch: null, base: null, changedFiles: [], changedCount: 0 };
}
const branch = run(['rev-parse', '--abbrev-ref', 'HEAD']);
let base = null;
for (const b of ['main', 'master']) {
if (run(['rev-parse', '--verify', '--quiet', b]) !== null) {
base = b;
break;
}
}
const diffBase = base && branch && branch !== base ? base : null;
const fromDiff = diffBase ? run(['diff', '--name-only', `${diffBase}...HEAD`]) : null;
// porcelain lines are `XY PATH`: a 2-char status + a space, then the path.
// Don't trim the combined output — an unstaged-modified line starts with a
// leading space (` M path`), and a global trim would eat the first line's
// status column and shift the slice. Renames render as `old -> new`.
const fromStatus = run(['-c', 'core.quotepath=false', 'status', '--porcelain'], { trim: false });
let changed = [];
if (fromDiff) {
changed = fromDiff.split('\n').filter(Boolean);
} else if (fromStatus) {
changed = fromStatus.split(/\r?\n/).filter(Boolean).map((l) => {
const p = l.slice(3);
const arrow = p.indexOf(' -> ');
return arrow === -1 ? p : p.slice(arrow + 4);
});
}
return {
isRepo: true,
branch,
base: diffBase,
changedFiles: changed.slice(0, 50),
changedCount: changed.length,
};
}
const COMMON_DEV_PORTS = [4321, 3000, 5173, 5174, 8080, 8000, 4200];
function probePort(port, timeout = 250) {
return new Promise((resolve) => {
const sock = new net.Socket();
let settled = false;
const finish = (ok) => {
if (settled) return;
settled = true;
try { sock.destroy(); } catch { /* ignore */ }
resolve(ok);
};
sock.setTimeout(timeout);
sock.once('connect', () => finish(true));
sock.once('timeout', () => finish(false));
sock.once('error', () => finish(false));
sock.connect(port, '127.0.0.1');
});
}
async function devServerSignals() {
const open = [];
await Promise.all(
COMMON_DEV_PORTS.map(async (p) => {
if (await probePort(p)) open.push(p);
}),
);
open.sort((a, b) => a - b);
return { running: open.length > 0, ports: open };
}
// Extensions the detector scans (mirrors the engine's walkDir set + HTML).
const SCANNABLE_EXT = new Set([
'.html', '.htm', '.css', '.scss',
'.jsx', '.tsx', '.js', '.ts', '.vue', '.svelte', '.astro',
]);
// Where UI source typically lives. The detector walks these and skips
// node_modules / dist / build / .next / .nuxt automatically.
const SOURCE_DIRS = ['src', 'app', 'components', 'pages', 'public'];
/**
* Local paths the agent should point the bundled detector at — never a URL.
* A URL means a costly Puppeteer browser render, and a probed dev-server port
* may not even belong to this project. An HTML *file* or a source tree is
* scanned by the cheap, jsdom-free static engine. This script does NOT run the
* detector; it just surfaces the target(s) so the agent can run
* `node <scripts>/detect.mjs --json <targets>` and fold the hits in.
*/
function scanTargets(cwd, git) {
// 1. Dirty tree wins: scan exactly the markup/style files in flight. It's
// what the user is working on, it's a small set, and it's local.
if (git.isRepo && git.changedFiles.length) {
const changed = git.changedFiles
.filter((f) => SCANNABLE_EXT.has(path.extname(f).toLowerCase()))
.filter((f) => fs.existsSync(path.join(cwd, f)));
if (changed.length) return { targets: changed.slice(0, 50), via: 'git-changes' };
}
// 2. Otherwise scan the local source dirs that exist.
const dirs = SOURCE_DIRS.filter((d) => fs.existsSync(path.join(cwd, d)));
if (dirs.length) return { targets: dirs, via: 'source-dir' };
// 3. A root HTML entry, or the project root as a last resort when there's
// code but no conventional source dir (walkDir still skips heavy dirs).
if (fs.existsSync(path.join(cwd, 'index.html'))) return { targets: ['index.html'], via: 'html' };
if (hasCode(cwd)) return { targets: ['.'], via: 'root' };
return { targets: [], via: null };
}
export async function gatherSignals(cwd = process.cwd()) {
const ctx = loadContext(cwd);
const git = gitSignals(cwd);
return {
setup: {
hasProduct: ctx.hasProduct,
productPath: ctx.productPath,
hasDesign: ctx.hasDesign,
designPath: ctx.designPath,
hasCode: hasCode(cwd),
register: extractRegister(ctx.product),
},
critique: { latest: latestCritique(cwd) },
git,
devServer: await devServerSignals(),
scan: scanTargets(cwd, git),
};
}
async function cli() {
const signals = await gatherSignals(process.cwd());
process.stdout.write(`${JSON.stringify(signals, null, 2)}\n`);
}
function invokedAsScript() {
const arg = process.argv[1];
if (!arg) return false;
try {
return fs.realpathSync(arg) === fs.realpathSync(fileURLToPath(import.meta.url));
} catch {
return false;
}
}
if (invokedAsScript()) {
cli();
}