7 nye moduler:
- lib/util/result.mjs — Result-shape m/ ok/fail/combine helpers
- lib/util/frontmatter.mjs — håndruller YAML-frontmatter-parser (subset, zero deps)
- lib/parsers/plan-schema.mjs — v1.7 step-regex + forbidden-heading-deteksjon (Fase/Phase/Stage/Steg)
- lib/parsers/manifest-yaml.mjs — per-step Manifest YAML-ekstraksjon m/ regex-validering
- lib/parsers/project-discovery.mjs — finn brief/research/architecture/plan/progress i prosjektmappe
- lib/parsers/arg-parser.mjs — $ARGUMENTS for alle 4 commands m/ flag-schema
- lib/parsers/bash-normalize.mjs — løftet fra hooks/scripts/pre-bash-executor.mjs
6 test-filer (66 tester totalt) — alle grønn:
- frontmatter (CRLF/BOM, scalars, lister, indent-rejection)
- plan-schema (positive Step-form, negative Fase/Phase/Stage/Steg, numbering, slicing)
- manifest-yaml (extraction, parsing, regex-validering, missing-key detection)
- project-discovery (sortert research, architecture-detection, phase-requirements)
- arg-parser (boolean/valued/multi-value flags, kvotert positional, ukjente flag)
- bash-normalize (\${x}/\\\\evasion, ANSI-stripping, full canonicalize-pipeline)
Forbereder Wave 2 (validators) og Spor 1-wiring inn i commands.
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
117 lines
2.9 KiB
JavaScript
117 lines
2.9 KiB
JavaScript
// lib/parsers/arg-parser.mjs
|
|
// Parse $ARGUMENTS strings for the four ultra commands.
|
|
//
|
|
// Each command has its own valid-flag set; passing flags from another command
|
|
// produces an `unknown_flags` array but does not error — the caller decides.
|
|
|
|
const FLAG_SCHEMA = {
|
|
ultrabrief: {
|
|
boolean: ['--quick', '--fg'],
|
|
valued: [],
|
|
aliases: {},
|
|
},
|
|
ultraresearch: {
|
|
boolean: ['--quick', '--local', '--external', '--fg'],
|
|
valued: ['--project'],
|
|
aliases: {},
|
|
},
|
|
ultraplan: {
|
|
boolean: ['--quick', '--fg'],
|
|
valued: ['--project', '--brief', '--export', '--decompose'],
|
|
multi: ['--research'],
|
|
aliases: {},
|
|
},
|
|
ultraexecute: {
|
|
boolean: ['--resume', '--dry-run', '--validate', '--fg'],
|
|
valued: ['--project', '--step', '--session'],
|
|
aliases: {},
|
|
},
|
|
};
|
|
|
|
/**
|
|
* @param {string} argString Raw $ARGUMENTS as the command sees it.
|
|
* @param {keyof FLAG_SCHEMA} command
|
|
* @returns {{
|
|
* command: string,
|
|
* flags: Record<string, true | string | string[]>,
|
|
* positional: string[],
|
|
* unknown: string[],
|
|
* errors: Array<{code: string, message: string}>,
|
|
* }}
|
|
*/
|
|
export function parseArgs(argString, command) {
|
|
const schema = FLAG_SCHEMA[command];
|
|
if (!schema) {
|
|
return {
|
|
command,
|
|
flags: {},
|
|
positional: [],
|
|
unknown: [],
|
|
errors: [{ code: 'ARG_UNKNOWN_COMMAND', message: `Unknown command: ${command}` }],
|
|
};
|
|
}
|
|
|
|
const tokens = tokenize(argString);
|
|
const flags = {};
|
|
const positional = [];
|
|
const unknown = [];
|
|
const errors = [];
|
|
|
|
for (let i = 0; i < tokens.length; i++) {
|
|
const tok = tokens[i];
|
|
|
|
if (!tok.startsWith('--')) {
|
|
positional.push(tok);
|
|
continue;
|
|
}
|
|
|
|
if (schema.boolean.includes(tok)) {
|
|
flags[tok] = true;
|
|
continue;
|
|
}
|
|
|
|
if (schema.valued.includes(tok)) {
|
|
const next = tokens[i + 1];
|
|
if (next === undefined || next.startsWith('--')) {
|
|
errors.push({ code: 'ARG_MISSING_VALUE', message: `Flag ${tok} requires a value` });
|
|
} else {
|
|
flags[tok] = next;
|
|
i++;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (schema.multi && schema.multi.includes(tok)) {
|
|
const collected = [];
|
|
while (i + 1 < tokens.length && !tokens[i + 1].startsWith('--')) {
|
|
collected.push(tokens[i + 1]);
|
|
i++;
|
|
}
|
|
if (collected.length === 0) {
|
|
errors.push({ code: 'ARG_MISSING_VALUE', message: `Flag ${tok} requires at least one value` });
|
|
} else {
|
|
flags[tok] = collected;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
unknown.push(tok);
|
|
}
|
|
|
|
return { command, flags, positional, unknown, errors };
|
|
}
|
|
|
|
function tokenize(s) {
|
|
if (typeof s !== 'string') return [];
|
|
const trimmed = s.trim();
|
|
if (trimmed === '') return [];
|
|
const out = [];
|
|
const re = /"([^"]*)"|'([^']*)'|(\S+)/g;
|
|
let m;
|
|
while ((m = re.exec(trimmed)) !== null) {
|
|
out.push(m[1] !== undefined ? m[1] : m[2] !== undefined ? m[2] : m[3]);
|
|
}
|
|
return out;
|
|
}
|
|
|
|
export { FLAG_SCHEMA };
|