Files
gstack/test/readme-throughput.test.ts
Garry Tan 00a7a65026 test: V1 gate coverage — writing-style resolver + config + jargon + migration + dormancy
Six new gate-tier test files:

- test/writing-style-resolver.test.ts — asserts Writing Style section
  is injected into tier-≥2 preamble, all 6 rules present, jargon list
  inlined, terse-mode gate condition present, Codex output uses
  \$GSTACK_BIN (not ~/.claude/), tier-1 does NOT get the section,
  migration-prompt block present.

- test/explain-level-config.test.ts — gstack-config set/get round-trip
  for default + terse, unknown-value warns + defaults to default,
  header documents the key, round-trip across set→set→get.

- test/jargon-list.test.ts — shape + ~50 terms + no duplicates
  (case-insensitive) + includes canonical high-signal terms.

- test/v0-dormancy.test.ts — 5D dimension names + archetype names
  forbidden in default-mode tier-≥2 SKILL.md output, except for
  plan-tune and office-hours where they're load-bearing.

- test/readme-throughput.test.ts — script replaces anchor with number
  on happy path, writes PENDING marker when JSON missing, CI gate
  asserts committed README contains no PENDING string.

- test/upgrade-migration-v1.test.ts — fresh run writes pending flag,
  idempotent after user-answered, pre-existing explain_level counts
  as answered.

All 95 V1 test-expect() calls pass. Full suite: 0 failures.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-18 11:39:24 +08:00

114 lines
3.9 KiB
TypeScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
/**
* scripts/update-readme-throughput.ts + README anchor + CI pending-marker gate.
*
* Coverage:
* - Happy path: JSON present, anchor gets replaced with number + anchor preserved
* - Missing JSON: script writes PENDING marker, CI would reject
* - Invalid JSON: script errors, README untouched
* - CI gate: committed README must not contain PENDING marker
*/
import { describe, test, expect, beforeEach, afterEach } from 'bun:test';
import * as fs from 'fs';
import * as path from 'path';
import * as os from 'os';
import { spawnSync } from 'child_process';
const ROOT = path.resolve(import.meta.dir, '..');
const SCRIPT = path.join(ROOT, 'scripts', 'update-readme-throughput.ts');
const ANCHOR = '<!-- GSTACK-THROUGHPUT-PLACEHOLDER -->';
const PENDING = 'GSTACK-THROUGHPUT-PENDING';
let tmpDir: string;
let tmpReadme: string;
let tmpJsonPath: string;
beforeEach(() => {
tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'gstack-readme-test-'));
tmpReadme = path.join(tmpDir, 'README.md');
fs.mkdirSync(path.join(tmpDir, 'docs'), { recursive: true });
tmpJsonPath = path.join(tmpDir, 'docs', 'throughput-2013-vs-2026.json');
});
afterEach(() => {
fs.rmSync(tmpDir, { recursive: true, force: true });
});
function runScript(cwd: string): { stdout: string; stderr: string; status: number } {
const res = spawnSync('bun', ['run', SCRIPT], {
encoding: 'utf-8',
cwd,
env: { ...process.env },
});
return {
stdout: (res.stdout ?? '').trim(),
stderr: (res.stderr ?? '').trim(),
status: res.status ?? -1,
};
}
describe('update-readme-throughput script', () => {
test('happy path: JSON present → anchor replaced with number', () => {
fs.writeFileSync(tmpReadme, `gstack hero: ${ANCHOR} 2013 pro-rata.\n`);
fs.writeFileSync(tmpJsonPath, JSON.stringify({
multiples: { logical_lines_added: 12.3 },
}));
const result = runScript(tmpDir);
expect(result.status).toBe(0);
const updated = fs.readFileSync(tmpReadme, 'utf-8');
expect(updated).toContain('12.3×');
expect(updated).toContain(ANCHOR); // anchor stays for next run
expect(updated).not.toContain(PENDING);
});
test('missing JSON: PENDING marker written (CI rejects)', () => {
fs.writeFileSync(tmpReadme, `gstack hero: ${ANCHOR} 2013 pro-rata.\n`);
// No JSON written
const result = runScript(tmpDir);
expect(result.status).toBe(0);
const updated = fs.readFileSync(tmpReadme, 'utf-8');
expect(updated).toContain(PENDING);
expect(updated).toContain(ANCHOR); // anchor preserved for next run
});
test('JSON with null multiple: PENDING marker written (honest missing state)', () => {
fs.writeFileSync(tmpReadme, `gstack hero: ${ANCHOR} 2013 pro-rata.\n`);
fs.writeFileSync(tmpJsonPath, JSON.stringify({
multiples: { logical_lines_added: null },
}));
const result = runScript(tmpDir);
expect(result.status).toBe(0);
const updated = fs.readFileSync(tmpReadme, 'utf-8');
expect(updated).toContain(PENDING);
expect(updated).not.toMatch(/null×/);
});
test('anchor already replaced: script is a no-op', () => {
fs.writeFileSync(tmpReadme, 'gstack hero: 7.0× already set.\n');
// No anchor in README → nothing to replace
const result = runScript(tmpDir);
expect(result.status).toBe(0);
const updated = fs.readFileSync(tmpReadme, 'utf-8');
expect(updated).toBe('gstack hero: 7.0× already set.\n');
});
});
describe('CI gate: committed README must not contain PENDING marker', () => {
// This is the core reason the PENDING marker exists. A commit that lands
// the README with the pending string means the build didn't run.
test('real README.md does not contain GSTACK-THROUGHPUT-PENDING', () => {
const readmePath = path.join(ROOT, 'README.md');
if (!fs.existsSync(readmePath)) return; // Fresh clone edge-case
const content = fs.readFileSync(readmePath, 'utf-8');
expect(content).not.toContain(PENDING);
});
});