mirror of
https://github.com/glittercowboy/get-shit-done
synced 2026-04-25 17:25:23 +02:00
* test(2130): add failing tests for frontmatter body --- sequence mis-parse Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * fix(2130): anchor extractFrontmatter regex to file start, preventing body --- mis-parse Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
536 lines
20 KiB
JavaScript
536 lines
20 KiB
JavaScript
/**
|
|
* GSD Tools Tests - frontmatter.cjs
|
|
*
|
|
* Tests for the hand-rolled YAML parser's pure function exports:
|
|
* extractFrontmatter, reconstructFrontmatter, spliceFrontmatter,
|
|
* parseMustHavesBlock, and FRONTMATTER_SCHEMAS.
|
|
*
|
|
* Includes REG-04 regression: quoted comma inline array edge case.
|
|
*/
|
|
|
|
const { test, describe } = require('node:test');
|
|
const assert = require('node:assert/strict');
|
|
|
|
const {
|
|
extractFrontmatter,
|
|
reconstructFrontmatter,
|
|
spliceFrontmatter,
|
|
parseMustHavesBlock,
|
|
FRONTMATTER_SCHEMAS,
|
|
} = require('../get-shit-done/bin/lib/frontmatter.cjs');
|
|
|
|
// ─── extractFrontmatter ─────────────────────────────────────────────────────
|
|
|
|
describe('extractFrontmatter', () => {
|
|
test('parses simple key-value pairs', () => {
|
|
const content = '---\nname: foo\ntype: execute\n---\nbody';
|
|
const result = extractFrontmatter(content);
|
|
assert.strictEqual(result.name, 'foo');
|
|
assert.strictEqual(result.type, 'execute');
|
|
});
|
|
|
|
test('strips quotes from values', () => {
|
|
const doubleQuoted = '---\nname: "foo"\n---\n';
|
|
const singleQuoted = '---\nname: \'foo\'\n---\n';
|
|
assert.strictEqual(extractFrontmatter(doubleQuoted).name, 'foo');
|
|
assert.strictEqual(extractFrontmatter(singleQuoted).name, 'foo');
|
|
});
|
|
|
|
test('parses nested objects', () => {
|
|
const content = '---\ntechstack:\n added: prisma\n patterns: repository\n---\n';
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result.techstack, { added: 'prisma', patterns: 'repository' });
|
|
});
|
|
|
|
test('parses block arrays', () => {
|
|
const content = '---\nitems:\n - alpha\n - beta\n - gamma\n---\n';
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result.items, ['alpha', 'beta', 'gamma']);
|
|
});
|
|
|
|
test('parses inline arrays', () => {
|
|
const content = '---\nkey: [a, b, c]\n---\n';
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result.key, ['a', 'b', 'c']);
|
|
});
|
|
|
|
test('handles quoted commas in inline arrays — REG-04 fixed', () => {
|
|
const content = '---\nkey: ["a, b", c]\n---\n';
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result.key, ['a, b', 'c']);
|
|
});
|
|
|
|
test('handles single-quoted commas in inline arrays', () => {
|
|
const content = "---\nkey: ['x, y', z]\n---\n";
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result.key, ['x, y', 'z']);
|
|
});
|
|
|
|
test('handles mixed quotes in inline arrays', () => {
|
|
const content = '---\nkey: ["a, b", \'c, d\', e]\n---\n';
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result.key, ['a, b', 'c, d', 'e']);
|
|
});
|
|
|
|
test('returns empty object for no frontmatter', () => {
|
|
const content = 'Just plain content, no frontmatter.';
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result, {});
|
|
});
|
|
|
|
test('returns empty object for empty frontmatter', () => {
|
|
const content = '---\n---\nBody text.';
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result, {});
|
|
});
|
|
|
|
test('parses frontmatter-only content', () => {
|
|
const content = '---\nkey: val\n---';
|
|
const result = extractFrontmatter(content);
|
|
assert.strictEqual(result.key, 'val');
|
|
});
|
|
|
|
test('handles emoji and non-ASCII in values', () => {
|
|
const content = '---\nname: "Hello World"\nlabel: "cafe"\n---\n';
|
|
const result = extractFrontmatter(content);
|
|
assert.strictEqual(result.name, 'Hello World');
|
|
assert.strictEqual(result.label, 'cafe');
|
|
});
|
|
|
|
test('converts empty-object placeholders to arrays when dash items follow', () => {
|
|
// When a key has no value, it gets an empty {} placeholder.
|
|
// When "- item" lines follow, the parser converts {} to [].
|
|
const content = '---\nrequirements:\n - REQ-01\n - REQ-02\n---\n';
|
|
const result = extractFrontmatter(content);
|
|
assert.ok(Array.isArray(result.requirements), 'should convert placeholder object to array');
|
|
assert.deepStrictEqual(result.requirements, ['REQ-01', 'REQ-02']);
|
|
});
|
|
|
|
test('skips empty lines in YAML body', () => {
|
|
const content = '---\nfirst: one\n\nsecond: two\n\nthird: three\n---\n';
|
|
const result = extractFrontmatter(content);
|
|
assert.strictEqual(result.first, 'one');
|
|
assert.strictEqual(result.second, 'two');
|
|
assert.strictEqual(result.third, 'three');
|
|
});
|
|
|
|
// ─── Bug #2130: body --- sequence mis-parse ──────────────────────────────
|
|
|
|
test('#2130: frontmatter at top with YAML example block in body — returns top frontmatter', () => {
|
|
const content = [
|
|
'---',
|
|
'name: my-agent',
|
|
'type: execute',
|
|
'---',
|
|
'',
|
|
'# Documentation',
|
|
'',
|
|
'Here is a YAML example:',
|
|
'',
|
|
'```yaml',
|
|
'---',
|
|
'key: value',
|
|
'other: stuff',
|
|
'---',
|
|
'```',
|
|
'',
|
|
'End of doc.',
|
|
].join('\n');
|
|
const result = extractFrontmatter(content);
|
|
assert.strictEqual(result.name, 'my-agent', 'should extract name from TOP frontmatter');
|
|
assert.strictEqual(result.type, 'execute', 'should extract type from TOP frontmatter');
|
|
assert.strictEqual(result.key, undefined, 'should NOT extract key from body YAML block');
|
|
assert.strictEqual(result.other, undefined, 'should NOT extract other from body YAML block');
|
|
});
|
|
|
|
test('#2130: frontmatter at top with horizontal rules in body — returns top frontmatter', () => {
|
|
const content = [
|
|
'---',
|
|
'title: My Doc',
|
|
'status: active',
|
|
'---',
|
|
'',
|
|
'# Section One',
|
|
'',
|
|
'Some text.',
|
|
'',
|
|
'---',
|
|
'',
|
|
'# Section Two',
|
|
'',
|
|
'More text.',
|
|
'',
|
|
'---',
|
|
'',
|
|
'# Section Three',
|
|
].join('\n');
|
|
const result = extractFrontmatter(content);
|
|
assert.strictEqual(result.title, 'My Doc', 'should extract title from TOP frontmatter');
|
|
assert.strictEqual(result.status, 'active', 'should extract status from TOP frontmatter');
|
|
});
|
|
|
|
test('#2130: body-only --- block with no frontmatter at byte 0 — returns empty', () => {
|
|
const content = [
|
|
'# My Document',
|
|
'',
|
|
'Some intro text.',
|
|
'',
|
|
'---',
|
|
'key: value',
|
|
'other: stuff',
|
|
'---',
|
|
'',
|
|
'End of doc.',
|
|
].join('\n');
|
|
const result = extractFrontmatter(content);
|
|
assert.deepStrictEqual(result, {}, 'should return empty object when --- block is not at byte 0');
|
|
});
|
|
|
|
test('#2130: valid frontmatter at byte 0 still works (regression guard)', () => {
|
|
const content = [
|
|
'---',
|
|
'phase: 01',
|
|
'plan: 03',
|
|
'type: execute',
|
|
'wave: 1',
|
|
'depends_on: ["01-01", "01-02"]',
|
|
'files_modified:',
|
|
' - src/auth.ts',
|
|
' - src/middleware.ts',
|
|
'autonomous: true',
|
|
'---',
|
|
'',
|
|
'# Plan body here',
|
|
].join('\n');
|
|
const result = extractFrontmatter(content);
|
|
assert.strictEqual(result.phase, '01');
|
|
assert.strictEqual(result.plan, '03');
|
|
assert.strictEqual(result.type, 'execute');
|
|
assert.strictEqual(result.wave, '1');
|
|
assert.deepStrictEqual(result.depends_on, ['01-01', '01-02']);
|
|
assert.deepStrictEqual(result.files_modified, ['src/auth.ts', 'src/middleware.ts']);
|
|
assert.strictEqual(result.autonomous, 'true');
|
|
});
|
|
});
|
|
|
|
// ─── reconstructFrontmatter ─────────────────────────────────────────────────
|
|
|
|
describe('reconstructFrontmatter', () => {
|
|
test('serializes simple key-value', () => {
|
|
const result = reconstructFrontmatter({ name: 'foo' });
|
|
assert.strictEqual(result, 'name: foo');
|
|
});
|
|
|
|
test('serializes empty array as inline []', () => {
|
|
const result = reconstructFrontmatter({ items: [] });
|
|
assert.strictEqual(result, 'items: []');
|
|
});
|
|
|
|
test('serializes short string arrays inline', () => {
|
|
const result = reconstructFrontmatter({ key: ['a', 'b', 'c'] });
|
|
assert.strictEqual(result, 'key: [a, b, c]');
|
|
});
|
|
|
|
test('serializes long arrays as block', () => {
|
|
const result = reconstructFrontmatter({ key: ['one', 'two', 'three', 'four'] });
|
|
assert.ok(result.includes('key:'), 'should have key header');
|
|
assert.ok(result.includes(' - one'), 'should have block array items');
|
|
assert.ok(result.includes(' - four'), 'should have last item');
|
|
});
|
|
|
|
test('quotes values containing colons or hashes', () => {
|
|
const result = reconstructFrontmatter({ url: 'http://example.com' });
|
|
assert.ok(result.includes('"http://example.com"'), 'should quote value with colon');
|
|
|
|
const hashResult = reconstructFrontmatter({ comment: 'value # note' });
|
|
assert.ok(hashResult.includes('"value # note"'), 'should quote value with hash');
|
|
});
|
|
|
|
test('serializes nested objects with proper indentation', () => {
|
|
const result = reconstructFrontmatter({ tech: { added: 'prisma', patterns: 'repo' } });
|
|
assert.ok(result.includes('tech:'), 'should have parent key');
|
|
assert.ok(result.includes(' added: prisma'), 'should have indented child');
|
|
assert.ok(result.includes(' patterns: repo'), 'should have indented child');
|
|
});
|
|
|
|
test('serializes nested arrays within objects', () => {
|
|
const result = reconstructFrontmatter({
|
|
tech: { added: ['prisma', 'jose'] },
|
|
});
|
|
assert.ok(result.includes('tech:'), 'should have parent key');
|
|
assert.ok(result.includes(' added: [prisma, jose]'), 'should serialize nested short array inline');
|
|
});
|
|
|
|
test('skips null and undefined values', () => {
|
|
const result = reconstructFrontmatter({ name: 'foo', skip: null, also: undefined, keep: 'bar' });
|
|
assert.ok(!result.includes('skip'), 'should not include null key');
|
|
assert.ok(!result.includes('also'), 'should not include undefined key');
|
|
assert.ok(result.includes('name: foo'), 'should include non-null key');
|
|
assert.ok(result.includes('keep: bar'), 'should include non-null key');
|
|
});
|
|
|
|
test('round-trip: simple frontmatter', () => {
|
|
const original = '---\nname: test\ntype: execute\nwave: 1\n---\n';
|
|
const extracted1 = extractFrontmatter(original);
|
|
const reconstructed = reconstructFrontmatter(extracted1);
|
|
const roundTrip = `---\n${reconstructed}\n---\n`;
|
|
const extracted2 = extractFrontmatter(roundTrip);
|
|
assert.deepStrictEqual(extracted2, extracted1, 'round-trip should preserve data identity');
|
|
});
|
|
|
|
test('round-trip: nested with arrays', () => {
|
|
const original = '---\nphase: 01\ntech:\n added:\n - prisma\n - jose\n patterns:\n - repository\n - jwt\n---\n';
|
|
const extracted1 = extractFrontmatter(original);
|
|
const reconstructed = reconstructFrontmatter(extracted1);
|
|
const roundTrip = `---\n${reconstructed}\n---\n`;
|
|
const extracted2 = extractFrontmatter(roundTrip);
|
|
assert.deepStrictEqual(extracted2, extracted1, 'round-trip should preserve nested structures');
|
|
});
|
|
|
|
test('round-trip: multiple data types', () => {
|
|
const original = '---\nname: testplan\nwave: 2\ntags: [auth, api, db]\ndeps:\n - dep1\n - dep2\nconfig:\n enabled: true\n count: 5\n---\n';
|
|
const extracted1 = extractFrontmatter(original);
|
|
const reconstructed = reconstructFrontmatter(extracted1);
|
|
const roundTrip = `---\n${reconstructed}\n---\n`;
|
|
const extracted2 = extractFrontmatter(roundTrip);
|
|
assert.deepStrictEqual(extracted2, extracted1, 'round-trip should preserve multiple data types');
|
|
});
|
|
});
|
|
|
|
// ─── spliceFrontmatter ──────────────────────────────────────────────────────
|
|
|
|
describe('spliceFrontmatter', () => {
|
|
test('replaces existing frontmatter preserving body', () => {
|
|
const content = '---\nphase: 01\ntype: execute\n---\n\n# Body Content\n\nParagraph here.';
|
|
const newObj = { phase: '02', type: 'tdd', wave: '1' };
|
|
const result = spliceFrontmatter(content, newObj);
|
|
|
|
// New frontmatter should be present
|
|
const extracted = extractFrontmatter(result);
|
|
assert.strictEqual(extracted.phase, '02');
|
|
assert.strictEqual(extracted.type, 'tdd');
|
|
assert.strictEqual(extracted.wave, '1');
|
|
|
|
// Body should be preserved
|
|
assert.ok(result.includes('# Body Content'), 'body heading should be preserved');
|
|
assert.ok(result.includes('Paragraph here.'), 'body paragraph should be preserved');
|
|
});
|
|
|
|
test('adds frontmatter to content without any', () => {
|
|
const content = 'Plain text with no frontmatter.';
|
|
const newObj = { phase: '01', plan: '01' };
|
|
const result = spliceFrontmatter(content, newObj);
|
|
|
|
// Should start with frontmatter delimiters
|
|
assert.ok(result.startsWith('---\n'), 'should start with opening delimiter');
|
|
assert.ok(result.includes('\n---\n'), 'should have closing delimiter');
|
|
|
|
// Original content should follow
|
|
assert.ok(result.includes('Plain text with no frontmatter.'), 'original content should be preserved');
|
|
|
|
// Frontmatter should be extractable
|
|
const extracted = extractFrontmatter(result);
|
|
assert.strictEqual(extracted.phase, '01');
|
|
assert.strictEqual(extracted.plan, '01');
|
|
});
|
|
|
|
test('preserves content after frontmatter delimiters exactly', () => {
|
|
const body = '\n\nExact content with special chars: $, %, &, <, >\nLine 2\nLine 3';
|
|
const content = '---\nold: value\n---' + body;
|
|
const newObj = { new: 'value' };
|
|
const result = spliceFrontmatter(content, newObj);
|
|
|
|
// The body after the closing --- should be exactly preserved
|
|
const closingIdx = result.indexOf('\n---', 4); // skip the opening ---
|
|
const resultBody = result.slice(closingIdx + 4); // skip \n---
|
|
assert.strictEqual(resultBody, body, 'body content after frontmatter should be exactly preserved');
|
|
});
|
|
});
|
|
|
|
// ─── parseMustHavesBlock ────────────────────────────────────────────────────
|
|
|
|
describe('parseMustHavesBlock', () => {
|
|
test('extracts truths as string array', () => {
|
|
const content = `---
|
|
phase: 01
|
|
must_haves:
|
|
truths:
|
|
- "All tests pass on CI"
|
|
- "Coverage exceeds 80%"
|
|
---
|
|
|
|
Body content.`;
|
|
const result = parseMustHavesBlock(content, 'truths');
|
|
assert.ok(Array.isArray(result), 'should return an array');
|
|
assert.strictEqual(result.length, 2);
|
|
assert.strictEqual(result[0], 'All tests pass on CI');
|
|
assert.strictEqual(result[1], 'Coverage exceeds 80%');
|
|
});
|
|
|
|
test('extracts artifacts as object array', () => {
|
|
const content = `---
|
|
phase: 01
|
|
must_haves:
|
|
artifacts:
|
|
- path: "src/auth.ts"
|
|
provides: "JWT authentication"
|
|
min_lines: 100
|
|
- path: "src/middleware.ts"
|
|
provides: "Route protection"
|
|
min_lines: 50
|
|
---
|
|
|
|
Body.`;
|
|
const result = parseMustHavesBlock(content, 'artifacts');
|
|
assert.ok(Array.isArray(result), 'should return an array');
|
|
assert.strictEqual(result.length, 2);
|
|
assert.strictEqual(result[0].path, 'src/auth.ts');
|
|
assert.strictEqual(result[0].provides, 'JWT authentication');
|
|
assert.strictEqual(result[0].min_lines, 100);
|
|
assert.strictEqual(result[1].path, 'src/middleware.ts');
|
|
assert.strictEqual(result[1].min_lines, 50);
|
|
});
|
|
|
|
test('extracts key_links with from/to/via/pattern fields', () => {
|
|
const content = `---
|
|
phase: 01
|
|
must_haves:
|
|
key_links:
|
|
- from: "tests/auth.test.ts"
|
|
to: "src/auth.ts"
|
|
via: "import statement"
|
|
pattern: "import.*auth"
|
|
---
|
|
`;
|
|
const result = parseMustHavesBlock(content, 'key_links');
|
|
assert.ok(Array.isArray(result), 'should return an array');
|
|
assert.strictEqual(result.length, 1);
|
|
assert.strictEqual(result[0].from, 'tests/auth.test.ts');
|
|
assert.strictEqual(result[0].to, 'src/auth.ts');
|
|
assert.strictEqual(result[0].via, 'import statement');
|
|
assert.strictEqual(result[0].pattern, 'import.*auth');
|
|
});
|
|
|
|
test('returns empty array when block not found', () => {
|
|
const content = `---
|
|
phase: 01
|
|
must_haves:
|
|
truths:
|
|
- "Some truth"
|
|
---
|
|
`;
|
|
const result = parseMustHavesBlock(content, 'nonexistent_block');
|
|
assert.deepStrictEqual(result, []);
|
|
});
|
|
|
|
test('returns empty array when no frontmatter', () => {
|
|
const content = 'Plain text without any frontmatter delimiters.';
|
|
const result = parseMustHavesBlock(content, 'truths');
|
|
assert.deepStrictEqual(result, []);
|
|
});
|
|
|
|
test('parses key_links with 2-space indentation — issue #1356', () => {
|
|
// Real-world YAML uses 2-space indentation, not 4-space.
|
|
// The parser was hardcoded to expect 4-space indentation which caused
|
|
// "No must_haves.key_links found in frontmatter" for valid YAML.
|
|
const content = `---
|
|
phase: 01-conversion-engine-iva-correctness
|
|
plan: 02
|
|
type: execute
|
|
wave: 2
|
|
depends_on: ["01-01"]
|
|
files_modified:
|
|
- src/features/currency/exchange-rate-store.ts
|
|
- src/features/currency/use-currency-config.ts
|
|
autonomous: true
|
|
requirements:
|
|
- CONV-02
|
|
- CONV-03
|
|
|
|
must_haves:
|
|
truths:
|
|
- "All tests pass"
|
|
artifacts:
|
|
- path: "src/features/currency/use-currency-config.ts"
|
|
key_links:
|
|
- from: "src/features/currency/use-currency-config.ts"
|
|
to: "src/api/generated/company-config/company-config.ts"
|
|
via: "getCompanyConfigControllerFindAllQueryOptions"
|
|
pattern: "getCompanyConfigControllerFindAllQueryOptions"
|
|
- from: "src/features/currency/use-currency-config.ts"
|
|
to: "src/features/currency/exchange-rate-store.ts"
|
|
via: "useExchangeRateStore for MMKV persist"
|
|
pattern: "useExchangeRateStore"
|
|
---
|
|
|
|
# Plan body
|
|
`;
|
|
const result = parseMustHavesBlock(content, 'key_links');
|
|
assert.ok(Array.isArray(result), 'should return an array');
|
|
assert.strictEqual(result.length, 2, `expected 2 key_links, got ${result.length}: ${JSON.stringify(result)}`);
|
|
assert.strictEqual(result[0].from, 'src/features/currency/use-currency-config.ts');
|
|
assert.strictEqual(result[0].to, 'src/api/generated/company-config/company-config.ts');
|
|
assert.strictEqual(result[0].via, 'getCompanyConfigControllerFindAllQueryOptions');
|
|
assert.strictEqual(result[0].pattern, 'getCompanyConfigControllerFindAllQueryOptions');
|
|
assert.strictEqual(result[1].from, 'src/features/currency/use-currency-config.ts');
|
|
assert.strictEqual(result[1].to, 'src/features/currency/exchange-rate-store.ts');
|
|
assert.strictEqual(result[1].via, 'useExchangeRateStore for MMKV persist');
|
|
assert.strictEqual(result[1].pattern, 'useExchangeRateStore');
|
|
});
|
|
|
|
test('parses truths with 2-space indentation — issue #1356', () => {
|
|
const content = `---
|
|
phase: 01
|
|
must_haves:
|
|
truths:
|
|
- "All tests pass on CI"
|
|
- "Coverage exceeds 80%"
|
|
---
|
|
`;
|
|
const result = parseMustHavesBlock(content, 'truths');
|
|
assert.ok(Array.isArray(result), 'should return an array');
|
|
assert.strictEqual(result.length, 2);
|
|
assert.strictEqual(result[0], 'All tests pass on CI');
|
|
assert.strictEqual(result[1], 'Coverage exceeds 80%');
|
|
});
|
|
|
|
test('parses artifacts with 2-space indentation — issue #1356', () => {
|
|
const content = `---
|
|
phase: 01
|
|
must_haves:
|
|
artifacts:
|
|
- path: "src/auth.ts"
|
|
provides: "JWT authentication"
|
|
min_lines: 100
|
|
---
|
|
`;
|
|
const result = parseMustHavesBlock(content, 'artifacts');
|
|
assert.ok(Array.isArray(result), 'should return an array');
|
|
assert.strictEqual(result.length, 1);
|
|
assert.strictEqual(result[0].path, 'src/auth.ts');
|
|
assert.strictEqual(result[0].provides, 'JWT authentication');
|
|
assert.strictEqual(result[0].min_lines, 100);
|
|
});
|
|
|
|
test('handles nested arrays within artifact objects', () => {
|
|
const content = `---
|
|
phase: 01
|
|
must_haves:
|
|
artifacts:
|
|
- path: "src/api.ts"
|
|
provides: "REST endpoints"
|
|
exports:
|
|
- "GET"
|
|
- "POST"
|
|
---
|
|
`;
|
|
const result = parseMustHavesBlock(content, 'artifacts');
|
|
assert.ok(Array.isArray(result), 'should return an array');
|
|
assert.strictEqual(result.length, 1);
|
|
assert.strictEqual(result[0].path, 'src/api.ts');
|
|
// The nested array should be captured
|
|
assert.ok(result[0].exports !== undefined, 'should have exports field');
|
|
});
|
|
});
|
|
|