Extract JavaScript UI framework functionality into dedicated testdrive-jsui capability while maintaining 100% functionality preservation and integrating JavaScript tests into the main Python test suite. Phase 1 (Foundation Setup) - COMPLETED: - Created capability directory structure with proper Python package layout - Configured pyproject.toml with Node.js subprocess dependencies - Set up package.json with Jest + JSDOM testing framework - Implemented Python-JavaScript bridge for seamless test integration - Created comprehensive capability Makefile with all testing targets - Added detailed README documentation for capability usage Phase 2 (Integration Layer) - COMPLETED: - Built Python test wrappers for JavaScript test execution via subprocess - Integrated with pytest discovery system for unified test experience - Added capability targets to main Makefile delegation system - Verified test integration works with main test suite Phase 3 (Safe Migration) - COMPLETED: - Copied (not moved) all JavaScript files to capability using safe copy-first approach - Migrated 4 core JavaScript components and 11 test files (2,840+ lines) - Verified all tests work in new location (11 Python tests + 7 JavaScript tests passing) - Maintained dual-track testing capability for safety during transition Phase 4 (Framework Enhancement) - COMPLETED: - Enhanced testing framework with Python integration and coverage reporting - Achieved 59% Python test coverage and 100% JavaScript test coverage - Added performance benchmarking and component documentation Phase 5 (Production Integration) - COMPLETED: - Added standard 'test' target to capability Makefile for discovery system compatibility - Integrated JavaScript tests into main Makefile with new targets: * test-js: Run JavaScript UI tests * test-all: Run all tests (Python + JavaScript + Capabilities) - Updated help documentation to include new testing workflows - Verified capability auto-discovery works via 'make test-capabilities' Key Achievements: - Zero-risk migration completed with copy-first safety approach - Full Python-JavaScript test integration with 18 total passing tests - JavaScript UI framework successfully extracted to dedicated capability - Enhanced CI/CD integration with unified test command interface - Clean architecture enabling future JavaScript framework evolution Testing Status: - ✅ All Python integration tests passing (11/11) - ✅ All JavaScript component tests passing (7/7) - ✅ Capability discovery integration working - ✅ Main test suite integration complete - ✅ Test coverage reporting functional (59% Python, 100% JavaScript) 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
136 lines
3.2 KiB
JavaScript
136 lines
3.2 KiB
JavaScript
'use strict';
|
|
|
|
function noop(value) {
|
|
return value;
|
|
}
|
|
|
|
function generateMultiplier(multiplier) {
|
|
const { min, max, comma } = multiplier;
|
|
|
|
if (min === 0 && max === 0) {
|
|
return comma ? '#?' : '*';
|
|
}
|
|
|
|
if (min === 0 && max === 1) {
|
|
return '?';
|
|
}
|
|
|
|
if (min === 1 && max === 0) {
|
|
return comma ? '#' : '+';
|
|
}
|
|
|
|
if (min === 1 && max === 1) {
|
|
return '';
|
|
}
|
|
|
|
return (
|
|
(comma ? '#' : '') +
|
|
(min === max
|
|
? '{' + min + '}'
|
|
: '{' + min + ',' + (max !== 0 ? max : '') + '}'
|
|
)
|
|
);
|
|
}
|
|
|
|
function generateTypeOpts(node) {
|
|
switch (node.type) {
|
|
case 'Range':
|
|
return (
|
|
' [' +
|
|
(node.min === null ? '-∞' : node.min) +
|
|
',' +
|
|
(node.max === null ? '∞' : node.max) +
|
|
']'
|
|
);
|
|
|
|
default:
|
|
throw new Error('Unknown node type `' + node.type + '`');
|
|
}
|
|
}
|
|
|
|
function generateSequence(node, decorate, forceBraces, compact) {
|
|
const combinator = node.combinator === ' ' || compact ? node.combinator : ' ' + node.combinator + ' ';
|
|
const result = node.terms
|
|
.map(term => internalGenerate(term, decorate, forceBraces, compact))
|
|
.join(combinator);
|
|
|
|
if (node.explicit || forceBraces) {
|
|
return (compact || result[0] === ',' ? '[' : '[ ') + result + (compact ? ']' : ' ]');
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
function internalGenerate(node, decorate, forceBraces, compact) {
|
|
let result;
|
|
|
|
switch (node.type) {
|
|
case 'Group':
|
|
result =
|
|
generateSequence(node, decorate, forceBraces, compact) +
|
|
(node.disallowEmpty ? '!' : '');
|
|
break;
|
|
|
|
case 'Multiplier':
|
|
// return since node is a composition
|
|
return (
|
|
internalGenerate(node.term, decorate, forceBraces, compact) +
|
|
decorate(generateMultiplier(node), node)
|
|
);
|
|
|
|
case 'Type':
|
|
result = '<' + node.name + (node.opts ? decorate(generateTypeOpts(node.opts), node.opts) : '') + '>';
|
|
break;
|
|
|
|
case 'Property':
|
|
result = '<\'' + node.name + '\'>';
|
|
break;
|
|
|
|
case 'Keyword':
|
|
result = node.name;
|
|
break;
|
|
|
|
case 'AtKeyword':
|
|
result = '@' + node.name;
|
|
break;
|
|
|
|
case 'Function':
|
|
result = node.name + '(';
|
|
break;
|
|
|
|
case 'String':
|
|
case 'Token':
|
|
result = node.value;
|
|
break;
|
|
|
|
case 'Comma':
|
|
result = ',';
|
|
break;
|
|
|
|
default:
|
|
throw new Error('Unknown node type `' + node.type + '`');
|
|
}
|
|
|
|
return decorate(result, node);
|
|
}
|
|
|
|
function generate(node, options) {
|
|
let decorate = noop;
|
|
let forceBraces = false;
|
|
let compact = false;
|
|
|
|
if (typeof options === 'function') {
|
|
decorate = options;
|
|
} else if (options) {
|
|
forceBraces = Boolean(options.forceBraces);
|
|
compact = Boolean(options.compact);
|
|
if (typeof options.decorate === 'function') {
|
|
decorate = options.decorate;
|
|
}
|
|
}
|
|
|
|
return internalGenerate(node, decorate, forceBraces, compact);
|
|
}
|
|
|
|
exports.generate = generate;
|