47 lines
1.7 KiB
JavaScript
47 lines
1.7 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
import fs from 'fs'
|
|
import path from 'path'
|
|
|
|
const root = process.cwd()
|
|
const taskTypesPath = path.join(root, 'src', 'lib', 'task', 'types.ts')
|
|
const catalogPath = path.join(root, 'tests', 'contracts', 'task-type-catalog.ts')
|
|
|
|
function fail(title, details = []) {
|
|
console.error(`\n[test-tasktype-coverage-guard] ${title}`)
|
|
for (const detail of details) {
|
|
console.error(` - ${detail}`)
|
|
}
|
|
process.exit(1)
|
|
}
|
|
|
|
if (!fs.existsSync(taskTypesPath)) {
|
|
fail('Task type source file is missing', ['src/lib/task/types.ts'])
|
|
}
|
|
if (!fs.existsSync(catalogPath)) {
|
|
fail('Task type catalog file is missing', ['tests/contracts/task-type-catalog.ts'])
|
|
}
|
|
|
|
const taskTypesText = fs.readFileSync(taskTypesPath, 'utf8')
|
|
const catalogText = fs.readFileSync(catalogPath, 'utf8')
|
|
|
|
const taskTypeBlockMatch = taskTypesText.match(/export const TASK_TYPE = \{([\s\S]*?)\n\} as const/)
|
|
if (!taskTypeBlockMatch) {
|
|
fail('Unable to parse TASK_TYPE block from src/lib/task/types.ts')
|
|
}
|
|
const taskTypeBlock = taskTypeBlockMatch ? taskTypeBlockMatch[1] : ''
|
|
const taskTypeKeys = Array.from(taskTypeBlock.matchAll(/^\s+([A-Z_]+):\s'[^']+',?$/gm)).map((match) => match[1])
|
|
const catalogKeys = Array.from(catalogText.matchAll(/\[TASK_TYPE\.([A-Z_]+)\]/g)).map((match) => match[1])
|
|
|
|
const missingKeys = taskTypeKeys.filter((key) => !catalogKeys.includes(key))
|
|
const staleKeys = catalogKeys.filter((key) => !taskTypeKeys.includes(key))
|
|
|
|
if (missingKeys.length > 0) {
|
|
fail('Missing TASK_TYPE owners in tests/contracts/task-type-catalog.ts', missingKeys)
|
|
}
|
|
if (staleKeys.length > 0) {
|
|
fail('Stale TASK_TYPE keys in tests/contracts/task-type-catalog.ts', staleKeys)
|
|
}
|
|
|
|
console.log(`[test-tasktype-coverage-guard] OK taskTypes=${taskTypeKeys.length}`)
|