This commit is contained in:
parent
739a534ac2
commit
f237916291
165 changed files with 79237 additions and 0 deletions
225
.codex/skills/impeccable/scripts/context-signals.mjs
Normal file
225
.codex/skills/impeccable/scripts/context-signals.mjs
Normal file
|
|
@ -0,0 +1,225 @@
|
|||
#!/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();
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue