@epoint-testtech/ep-stage-skill 0.0.3-alpha.2 → 0.0.4-alpha.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/SKILL.md +2 -2
- package/codex-skill/ep-stage/glue-create-project/SKILL.md +186 -0
- package/codex-skill/ep-stage/glue-generate-testcase/SKILL.md +199 -0
- package/codex-skill/ep-stage/glue-generate-testcase/references/testcase-schema.md +112 -0
- package/codex-skill/ep-stage/glue-run-test/SKILL.md +249 -0
- package/codex-skill/ep-stage/glue-run-test/references/crud-pipeline.md +145 -0
- package/codex-skill/ep-stage/{glue-test → glue-run-test}/scripts/generate-crud-spec.mjs +3 -3
- package/codex-skill/ep-stage/recording-to-glue/SKILL.md +1 -0
- package/codex-skill/ep-stage/scripts/validate-skill.mjs +29 -7
- package/dist/src/cli/dev/extract-contract.d.ts +14 -0
- package/dist/src/cli/dev/extract-contract.d.ts.map +1 -0
- package/dist/src/cli/dev/extract-contract.js +114 -0
- package/dist/src/cli/generate-crud-contract.js +7 -77
- package/dist/src/cli/generate-playwright-tests.d.ts +0 -28
- package/dist/src/cli/generate-playwright-tests.d.ts.map +1 -1
- package/dist/src/cli/generate-playwright-tests.js +4 -81
- package/dist/src/cli/generate-testcase.d.ts +83 -0
- package/dist/src/cli/generate-testcase.d.ts.map +1 -0
- package/dist/src/cli/generate-testcase.js +197 -0
- package/dist/src/cli/index.d.ts +18 -0
- package/dist/src/cli/index.d.ts.map +1 -0
- package/dist/src/cli/index.js +55 -0
- package/dist/src/cli/probe.d.ts +44 -0
- package/dist/src/cli/probe.d.ts.map +1 -0
- package/dist/src/cli/probe.js +221 -0
- package/dist/src/cli/run-gap-pipeline.js +4 -0
- package/dist/src/cli/run.d.ts +63 -0
- package/dist/src/cli/run.d.ts.map +1 -0
- package/dist/src/cli/run.js +116 -0
- package/dist/src/cli/spec.d.ts +45 -0
- package/dist/src/cli/spec.d.ts.map +1 -0
- package/dist/src/cli/spec.js +74 -0
- package/dist/src/context/stage-context.d.ts +72 -8
- package/dist/src/context/stage-context.d.ts.map +1 -1
- package/dist/src/context/stage-context.js +61 -15
- package/dist/src/index.d.ts +2 -2
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +1 -1
- package/dist/src/testcase/testcase-generator.d.ts.map +1 -1
- package/dist/src/testcase/testcase-generator.js +4 -0
- package/dist/src/testcase/testcase-v2.d.ts +50 -0
- package/dist/src/testcase/testcase-v2.d.ts.map +1 -0
- package/dist/src/testcase/testcase-v2.js +1 -0
- package/dist/src/util/credentials.d.ts +12 -0
- package/dist/src/util/credentials.d.ts.map +1 -0
- package/dist/src/util/credentials.js +19 -0
- package/dist/src/util/i18n-testcase.d.ts +8 -0
- package/dist/src/util/i18n-testcase.d.ts.map +1 -0
- package/dist/src/util/i18n-testcase.js +55 -0
- package/dist/src/util/softlink.d.ts +33 -0
- package/dist/src/util/softlink.d.ts.map +1 -0
- package/dist/src/util/softlink.js +43 -0
- package/dist/src/validation/credentials.d.ts +19 -0
- package/dist/src/validation/credentials.d.ts.map +1 -0
- package/dist/src/validation/credentials.js +38 -0
- package/dist/src/validation/index.d.ts +5 -0
- package/dist/src/validation/index.d.ts.map +1 -0
- package/dist/src/validation/index.js +3 -0
- package/dist/src/validation/projects-index.d.ts +13 -0
- package/dist/src/validation/projects-index.d.ts.map +1 -0
- package/dist/src/validation/projects-index.js +37 -0
- package/dist/src/validation/testcase.d.ts +13 -0
- package/dist/src/validation/testcase.d.ts.map +1 -0
- package/dist/src/validation/testcase.js +53 -0
- package/dist/test/cli/extract-contract.test.d.ts +2 -0
- package/dist/test/cli/extract-contract.test.d.ts.map +1 -0
- package/dist/test/cli/extract-contract.test.js +32 -0
- package/dist/test/cli/generate-testcase.test.d.ts +2 -0
- package/dist/test/cli/generate-testcase.test.d.ts.map +1 -0
- package/dist/test/cli/generate-testcase.test.js +130 -0
- package/dist/test/cli/index.test.d.ts +2 -0
- package/dist/test/cli/index.test.d.ts.map +1 -0
- package/dist/test/cli/index.test.js +93 -0
- package/dist/test/cli/run.test.d.ts +2 -0
- package/dist/test/cli/run.test.d.ts.map +1 -0
- package/dist/test/cli/run.test.js +149 -0
- package/dist/test/cli/spec.test.d.ts +2 -0
- package/dist/test/cli/spec.test.d.ts.map +1 -0
- package/dist/test/cli/spec.test.js +196 -0
- package/dist/test/stage-context.test.js +145 -13
- package/dist/test/util/credentials.test.d.ts +2 -0
- package/dist/test/util/credentials.test.d.ts.map +1 -0
- package/dist/test/util/credentials.test.js +64 -0
- package/dist/test/util/i18n-testcase.test.d.ts +2 -0
- package/dist/test/util/i18n-testcase.test.d.ts.map +1 -0
- package/dist/test/util/i18n-testcase.test.js +119 -0
- package/dist/test/util/softlink.test.d.ts +2 -0
- package/dist/test/util/softlink.test.d.ts.map +1 -0
- package/dist/test/util/softlink.test.js +82 -0
- package/dist/test/validation/credentials.test.d.ts +2 -0
- package/dist/test/validation/credentials.test.d.ts.map +1 -0
- package/dist/test/validation/credentials.test.js +72 -0
- package/dist/test/validation/projects-index.test.d.ts +2 -0
- package/dist/test/validation/projects-index.test.d.ts.map +1 -0
- package/dist/test/validation/projects-index.test.js +48 -0
- package/dist/test/validation/testcase.test.d.ts +2 -0
- package/dist/test/validation/testcase.test.d.ts.map +1 -0
- package/dist/test/validation/testcase.test.js +129 -0
- package/docs/README.md +6 -6
- package/docs/mvp-usage-guide.md +3 -3
- package/package.json +9 -4
- package/codex-skill/ep-stage/create-project/SKILL.md +0 -59
- package/codex-skill/ep-stage/glue-test/SKILL.md +0 -258
- package/codex-skill/ep-stage/glue-test/references/crud-pipeline.md +0 -139
- package/codex-skill/ep-stage/glue-testcase/SKILL.md +0 -31
- package/codex-skill/ep-stage/glue-testcase/references/testcase-schema.md +0 -67
- /package/codex-skill/ep-stage/{glue-testcase → glue-generate-testcase}/examples/observable-testcase.json +0 -0
- /package/codex-skill/ep-stage/{glue-test → glue-run-test}/references/gap-review-protocol.md +0 -0
- /package/codex-skill/ep-stage/{glue-test → glue-run-test}/references/harness-principles.md +0 -0
|
@@ -1,77 +1,7 @@
|
|
|
1
|
-
import {
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
const key = normalizedArgv[index];
|
|
9
|
-
const value = normalizedArgv[index + 1];
|
|
10
|
-
if (!key?.startsWith('--') || !value) {
|
|
11
|
-
throw new Error(`Invalid argument pair near ${key ?? '<empty>'}`);
|
|
12
|
-
}
|
|
13
|
-
result[key.slice(2)] = value;
|
|
14
|
-
}
|
|
15
|
-
if (result.module) {
|
|
16
|
-
throw new Error('已废弃 --module,请使用 --module-id。' +
|
|
17
|
-
'例: pnpm generate:crud-contract -- --module-id zwplace ...');
|
|
18
|
-
}
|
|
19
|
-
const required = ['module-id', 'docs', 'code-list', 'webapp', 'java-actions', 'out'];
|
|
20
|
-
for (const key of required) {
|
|
21
|
-
if (!result[key]) {
|
|
22
|
-
throw new Error(`Missing required argument --${key}`);
|
|
23
|
-
}
|
|
24
|
-
}
|
|
25
|
-
return {
|
|
26
|
-
module: result['module-id'],
|
|
27
|
-
docs: result.docs,
|
|
28
|
-
codeList: result['code-list'],
|
|
29
|
-
webapp: result.webapp,
|
|
30
|
-
javaActions: result['java-actions'],
|
|
31
|
-
hints: result.hints,
|
|
32
|
-
out: result.out
|
|
33
|
-
};
|
|
34
|
-
}
|
|
35
|
-
function htmlFiles(dir) {
|
|
36
|
-
return readdirSync(dir)
|
|
37
|
-
.filter((fileName) => fileName.endsWith('.html'))
|
|
38
|
-
.sort()
|
|
39
|
-
.map((fileName) => path.join(dir, fileName));
|
|
40
|
-
}
|
|
41
|
-
function javaFiles(dir) {
|
|
42
|
-
return readdirSync(dir)
|
|
43
|
-
.filter((fileName) => fileName.endsWith('Action.java'))
|
|
44
|
-
.sort()
|
|
45
|
-
.map((fileName) => path.join(dir, fileName));
|
|
46
|
-
}
|
|
47
|
-
const args = parseArgs(process.argv.slice(2));
|
|
48
|
-
const hints = args.hints
|
|
49
|
-
? JSON.parse(readFileSync(args.hints, 'utf8'))
|
|
50
|
-
: undefined;
|
|
51
|
-
const codeList = extractCodeListSummary(args.codeList);
|
|
52
|
-
function resolveSpec(docsDir, moduleId, codeListSummary) {
|
|
53
|
-
const specPath = path.join(docsDir, 'spec.yaml');
|
|
54
|
-
if (existsSync(specPath)) {
|
|
55
|
-
return extractSpecYaml(specPath);
|
|
56
|
-
}
|
|
57
|
-
return {
|
|
58
|
-
path: specPath,
|
|
59
|
-
module: {
|
|
60
|
-
id: moduleId,
|
|
61
|
-
label: codeListSummary.moduleLabel
|
|
62
|
-
},
|
|
63
|
-
fields: [],
|
|
64
|
-
businessRules: []
|
|
65
|
-
};
|
|
66
|
-
}
|
|
67
|
-
const contract = buildCrudBusinessModuleContract({
|
|
68
|
-
moduleId: args.module,
|
|
69
|
-
spec: resolveSpec(args.docs, args.module, codeList),
|
|
70
|
-
codeList,
|
|
71
|
-
pages: htmlFiles(args.webapp).map(extractHtmlPage),
|
|
72
|
-
actions: javaFiles(args.javaActions).map(extractJavaAction),
|
|
73
|
-
hints
|
|
74
|
-
});
|
|
75
|
-
mkdirSync(path.dirname(args.out), { recursive: true });
|
|
76
|
-
writeFileSync(args.out, `${JSON.stringify(contract, null, 2)}\n`, 'utf8');
|
|
77
|
-
console.log(`Wrote ${args.out}`);
|
|
1
|
+
import { runExtractContract } from './dev/extract-contract.js';
|
|
2
|
+
/**
|
|
3
|
+
* 旧 pnpm script 入口薄壳(REQ-CLI-03 修订):所有逻辑搬到 dev/extract-contract.ts。
|
|
4
|
+
* 本文件不删除,保持旧 pnpm script `generate:crud-contract` 零回归;新代码请直接
|
|
5
|
+
* 调用 `runExtractContract`。
|
|
6
|
+
*/
|
|
7
|
+
runExtractContract(process.argv.slice(2));
|
|
@@ -1,30 +1,2 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Playwright 测试脚本生成 CLI 入口
|
|
3
|
-
*
|
|
4
|
-
* 从 CRUD 业务模块契约 JSON 文件生成基于 glue 模板骨架的 Playwright 测试脚本。
|
|
5
|
-
*
|
|
6
|
-
* 使用方式:
|
|
7
|
-
* ```bash
|
|
8
|
-
* pnpm generate:playwright-tests -- \
|
|
9
|
-
* --contract <contract-json-path> \
|
|
10
|
-
* --out <output-spec-ts-path> \
|
|
11
|
-
* --menu <menu-path> \
|
|
12
|
-
* --testcase <glue-testcase-json-path>
|
|
13
|
-
* ```
|
|
14
|
-
*
|
|
15
|
-
* 参数说明:
|
|
16
|
-
* - --contract (必填): 契约 JSON 文件路径
|
|
17
|
-
* - --out (必填): 输出的 .spec.ts 文件路径
|
|
18
|
-
* - --menu (必填): 菜单路径,传给 MenuPage.navigateToMenu()
|
|
19
|
-
* - --testcase (可选): 已确认的 glue testcase JSON;传入后会校验用例与 contract 粒度一致。
|
|
20
|
-
*
|
|
21
|
-
* 示例:
|
|
22
|
-
* ```bash
|
|
23
|
-
* pnpm generate:playwright-tests -- \
|
|
24
|
-
* --contract output/zwplace.crud.contract.json \
|
|
25
|
-
* --out ../stage-project/src/tests/generated-zwplace.skeleton.spec.ts \
|
|
26
|
-
* --menu "场所窗口信息管理>场所窗口信息列表"
|
|
27
|
-
* ```
|
|
28
|
-
*/
|
|
29
1
|
export {};
|
|
30
2
|
//# sourceMappingURL=generate-playwright-tests.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"generate-playwright-tests.d.ts","sourceRoot":"","sources":["../../../src/cli/generate-playwright-tests.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"generate-playwright-tests.d.ts","sourceRoot":"","sources":["../../../src/cli/generate-playwright-tests.ts"],"names":[],"mappings":""}
|
|
@@ -1,81 +1,4 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
*
|
|
6
|
-
* 使用方式:
|
|
7
|
-
* ```bash
|
|
8
|
-
* pnpm generate:playwright-tests -- \
|
|
9
|
-
* --contract <contract-json-path> \
|
|
10
|
-
* --out <output-spec-ts-path> \
|
|
11
|
-
* --menu <menu-path> \
|
|
12
|
-
* --testcase <glue-testcase-json-path>
|
|
13
|
-
* ```
|
|
14
|
-
*
|
|
15
|
-
* 参数说明:
|
|
16
|
-
* - --contract (必填): 契约 JSON 文件路径
|
|
17
|
-
* - --out (必填): 输出的 .spec.ts 文件路径
|
|
18
|
-
* - --menu (必填): 菜单路径,传给 MenuPage.navigateToMenu()
|
|
19
|
-
* - --testcase (可选): 已确认的 glue testcase JSON;传入后会校验用例与 contract 粒度一致。
|
|
20
|
-
*
|
|
21
|
-
* 示例:
|
|
22
|
-
* ```bash
|
|
23
|
-
* pnpm generate:playwright-tests -- \
|
|
24
|
-
* --contract output/zwplace.crud.contract.json \
|
|
25
|
-
* --out ../stage-project/src/tests/generated-zwplace.skeleton.spec.ts \
|
|
26
|
-
* --menu "场所窗口信息管理>场所窗口信息列表"
|
|
27
|
-
* ```
|
|
28
|
-
*/
|
|
29
|
-
import { readFileSync, writeFileSync } from 'node:fs';
|
|
30
|
-
import { generateStageSkeletonCrudSpec } from '../generators/stage-skeleton-script.js';
|
|
31
|
-
import { validateGlueTestcaseDocumentForSpecAssembly } from '../testcase/testcase-spec-assembly.js';
|
|
32
|
-
/**
|
|
33
|
-
* 解析命令行参数
|
|
34
|
-
*
|
|
35
|
-
* 将 process.argv 转换为结构化的参数对象。
|
|
36
|
-
* 支持 `--key value` 格式的参数对,自动过滤 `--` 分隔符。
|
|
37
|
-
*
|
|
38
|
-
* @param argv - 命令行参数数组(不含 node 和脚本路径)
|
|
39
|
-
* @returns 解析后的参数对象
|
|
40
|
-
* @throws 当参数格式无效或缺少必填参数时抛出错误
|
|
41
|
-
*/
|
|
42
|
-
function parseArgs(argv) {
|
|
43
|
-
const result = {};
|
|
44
|
-
// 过滤 `--` 分隔符
|
|
45
|
-
const normalizedArgv = argv.filter((arg) => arg !== '--');
|
|
46
|
-
// 按键值对解析
|
|
47
|
-
for (let index = 0; index < normalizedArgv.length; index += 2) {
|
|
48
|
-
const key = normalizedArgv[index];
|
|
49
|
-
const value = normalizedArgv[index + 1];
|
|
50
|
-
if (!key?.startsWith('--') || !value) {
|
|
51
|
-
throw new Error(`无效的参数对,靠近 ${key ?? '<empty>'}`);
|
|
52
|
-
}
|
|
53
|
-
result[key.slice(2)] = value;
|
|
54
|
-
}
|
|
55
|
-
// 验证必填参数
|
|
56
|
-
if (!result.contract || !result.out || !result.menu) {
|
|
57
|
-
throw new Error('用法: tsx generate-playwright-tests.ts --contract <path> --out <path> --menu <menu-path>');
|
|
58
|
-
}
|
|
59
|
-
return {
|
|
60
|
-
contract: result.contract,
|
|
61
|
-
out: result.out,
|
|
62
|
-
menu: result.menu,
|
|
63
|
-
testcase: result.testcase,
|
|
64
|
-
};
|
|
65
|
-
}
|
|
66
|
-
// 执行主流程
|
|
67
|
-
const args = parseArgs(process.argv.slice(2));
|
|
68
|
-
// 读取契约 JSON 文件
|
|
69
|
-
const contract = JSON.parse(readFileSync(args.contract, 'utf8'));
|
|
70
|
-
if (args.testcase) {
|
|
71
|
-
const testcaseDocument = JSON.parse(readFileSync(args.testcase, 'utf8'));
|
|
72
|
-
validateGlueTestcaseDocumentForSpecAssembly(testcaseDocument, contract);
|
|
73
|
-
}
|
|
74
|
-
// 生成基于 glue 模板骨架的 Playwright 测试脚本
|
|
75
|
-
const script = generateStageSkeletonCrudSpec({
|
|
76
|
-
contract,
|
|
77
|
-
menuNavigation: args.menu
|
|
78
|
-
});
|
|
79
|
-
// 写入输出文件
|
|
80
|
-
writeFileSync(args.out, script, 'utf8');
|
|
81
|
-
console.log(`已生成 ${args.out}`);
|
|
1
|
+
// 薄壳:保留旧入口零回归,实际逻辑见 ./spec.ts(src/cli/index.ts spec 子命令统一入口)。
|
|
2
|
+
// 新参数:--testcase <p> --out <p>(旧的 --contract / --menu 已迁移到 testcase 子命令 + spec 子命令链路)。
|
|
3
|
+
import { parseSpecArgs, runSpec } from './spec.js';
|
|
4
|
+
runSpec(parseSpecArgs(process.argv.slice(2)));
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import type { TestcaseV2 } from '../testcase/testcase-v2.js';
|
|
2
|
+
/**
|
|
3
|
+
* testcase 子命令解析后的参数(REQ-CHAIN-01 / REQ-SCENE-01)。
|
|
4
|
+
*
|
|
5
|
+
* - codeList:上游 code_list.md 绝对路径(必填);
|
|
6
|
+
* - testsDir:<projectDir>/src/tests/<需求名> 绝对路径(必填);
|
|
7
|
+
* - scenario:①档优先级(显式 CLI 参数);
|
|
8
|
+
* - hints:ModuleHints JSON 路径(可选;推断 scenario 用);
|
|
9
|
+
* - projectDir / requirementDir:可选覆盖,默认从 codeList 路径推断。
|
|
10
|
+
*/
|
|
11
|
+
export type GenerateTestcaseArgs = {
|
|
12
|
+
codeList: string;
|
|
13
|
+
testsDir: string;
|
|
14
|
+
scenario?: string;
|
|
15
|
+
hints?: string;
|
|
16
|
+
projectDir?: string;
|
|
17
|
+
requirementDir?: string;
|
|
18
|
+
};
|
|
19
|
+
/**
|
|
20
|
+
* testcase 子命令产物 + 推断结果(供 skill Step 5 交互确认)。
|
|
21
|
+
*/
|
|
22
|
+
export type GenerateTestcaseResult = {
|
|
23
|
+
testcasePath: string;
|
|
24
|
+
testcaseMdPath: string;
|
|
25
|
+
credentialsTemplatePath: string;
|
|
26
|
+
inference: {
|
|
27
|
+
scenario: string;
|
|
28
|
+
requiredSystems: TestcaseV2['requiredSystems'];
|
|
29
|
+
requiredRoles: TestcaseV2['requiredRoles'];
|
|
30
|
+
credentialsTemplatePath: string;
|
|
31
|
+
credentialsActualPath: string;
|
|
32
|
+
cachedSystemsHit: string[];
|
|
33
|
+
};
|
|
34
|
+
};
|
|
35
|
+
/**
|
|
36
|
+
* 解析 testcase 子命令参数。
|
|
37
|
+
*
|
|
38
|
+
* @param argv - 已剥离子命令名的命令行参数。
|
|
39
|
+
* @returns 解析后的 GenerateTestcaseArgs。
|
|
40
|
+
* @throws 缺少 --code-list 或 --tests-dir 时抛错。
|
|
41
|
+
*/
|
|
42
|
+
export declare function parseGenerateTestcaseArgs(argv: string[]): GenerateTestcaseArgs;
|
|
43
|
+
/**
|
|
44
|
+
* scenario 3 档决策(REQ-SCENE-01):
|
|
45
|
+
* 1. --scenario CLI 参数(最高优先级)
|
|
46
|
+
* 2. ModuleHints 文件路径包含 `scenario=<value>` 字串(粗推断)
|
|
47
|
+
* 3. 默认 "crud.single-page"(与 i18n 表 / skeleton scenario 一致)
|
|
48
|
+
*
|
|
49
|
+
* @param args - 解析后的参数。
|
|
50
|
+
* @returns 决策结果。
|
|
51
|
+
*/
|
|
52
|
+
export declare function decideScenario(args: GenerateTestcaseArgs): string;
|
|
53
|
+
/**
|
|
54
|
+
* 把 TestcaseV2 渲染成中文 testcase.md(REQ-UX-03)。
|
|
55
|
+
*
|
|
56
|
+
* 设计:
|
|
57
|
+
* - 字段名 / 状态值通过 i18nTestcase() 翻译;
|
|
58
|
+
* - `unresolved` 字段标题硬编码「未确认项」(i18n 表中 `unresolved` 解析为状态值「未解决」);
|
|
59
|
+
* - 每次重跑覆盖(REQ-DATA-03 由调用方 writeFileSync 触发)。
|
|
60
|
+
*
|
|
61
|
+
* @param testcase - TestcaseV2 顶层结构。
|
|
62
|
+
* @returns Markdown 文本。
|
|
63
|
+
*/
|
|
64
|
+
export declare function renderTestcaseMarkdown(testcase: TestcaseV2): string;
|
|
65
|
+
/**
|
|
66
|
+
* testcase 子命令主体(REQ-CHAIN-01 / REQ-CHAIN-02 / REQ-SCENE-01 / REQ-DATA-02)。
|
|
67
|
+
*
|
|
68
|
+
* 流程:
|
|
69
|
+
* 1. 决定 scenario(3 档优先级,写入 testcase 顶层 scenario 字段);
|
|
70
|
+
* 2. 调 runProbeAndCoverage 钻探上游物料 + 装配 TestcaseV2;
|
|
71
|
+
* 3. 落盘 testcase.json5 + testcase.md(中文 i18n) + credentials.template.json5;
|
|
72
|
+
* 4. 软链上游 code_list.md → <tests-dir>/code_list.md;
|
|
73
|
+
* 5. 返回推断结果(供 skill Step 5 交互确认 systemUrl / credentialsCache 命中)。
|
|
74
|
+
*
|
|
75
|
+
* 注意:contract 提取通过 probe 内部完成,不直接复用 dev/extract-contract.ts 的 CLI 入口
|
|
76
|
+
* (probe + extract-contract 共享同一组 buildCrudBusinessModuleContract + extractor 依赖,
|
|
77
|
+
* 二者各自负责自己的 I/O 边界,保持职责单一)。
|
|
78
|
+
*
|
|
79
|
+
* @param args - 已解析参数。
|
|
80
|
+
* @returns 产物路径 + 推断结果。
|
|
81
|
+
*/
|
|
82
|
+
export declare function runGenerateTestcase(args: GenerateTestcaseArgs): GenerateTestcaseResult;
|
|
83
|
+
//# sourceMappingURL=generate-testcase.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generate-testcase.d.ts","sourceRoot":"","sources":["../../../src/cli/generate-testcase.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,4BAA4B,CAAC;AAE7D;;;;;;;;GAQG;AACH,MAAM,MAAM,oBAAoB,GAAG;IACjC,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,cAAc,CAAC,EAAE,MAAM,CAAC;CACzB,CAAC;AAEF;;GAEG;AACH,MAAM,MAAM,sBAAsB,GAAG;IACnC,YAAY,EAAE,MAAM,CAAC;IACrB,cAAc,EAAE,MAAM,CAAC;IACvB,uBAAuB,EAAE,MAAM,CAAC;IAChC,SAAS,EAAE;QACT,QAAQ,EAAE,MAAM,CAAC;QACjB,eAAe,EAAE,UAAU,CAAC,iBAAiB,CAAC,CAAC;QAC/C,aAAa,EAAE,UAAU,CAAC,eAAe,CAAC,CAAC;QAC3C,uBAAuB,EAAE,MAAM,CAAC;QAChC,qBAAqB,EAAE,MAAM,CAAC;QAC9B,gBAAgB,EAAE,MAAM,EAAE,CAAC;KAC5B,CAAC;CACH,CAAC;AAEF;;;;;;GAMG;AACH,wBAAgB,yBAAyB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,oBAAoB,CAqB9E;AAED;;;;;;;;GAQG;AACH,wBAAgB,cAAc,CAAC,IAAI,EAAE,oBAAoB,GAAG,MAAM,CAOjE;AAED;;;;;;;;;;GAUG;AACH,wBAAgB,sBAAsB,CAAC,QAAQ,EAAE,UAAU,GAAG,MAAM,CAgDnE;AAsBD;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,oBAAoB,GAAG,sBAAsB,CA+CtF"}
|
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
import { mkdirSync, writeFileSync } from 'node:fs';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
import JSON5 from 'json5';
|
|
4
|
+
import { runProbeAndCoverage } from './probe.js';
|
|
5
|
+
import { createSoftlinkWithFallback } from '../util/softlink.js';
|
|
6
|
+
import { i18nTestcase } from '../util/i18n-testcase.js';
|
|
7
|
+
/**
|
|
8
|
+
* 解析 testcase 子命令参数。
|
|
9
|
+
*
|
|
10
|
+
* @param argv - 已剥离子命令名的命令行参数。
|
|
11
|
+
* @returns 解析后的 GenerateTestcaseArgs。
|
|
12
|
+
* @throws 缺少 --code-list 或 --tests-dir 时抛错。
|
|
13
|
+
*/
|
|
14
|
+
export function parseGenerateTestcaseArgs(argv) {
|
|
15
|
+
const result = {};
|
|
16
|
+
const normalizedArgv = argv.filter((arg) => arg !== '--');
|
|
17
|
+
for (let index = 0; index < normalizedArgv.length; index += 2) {
|
|
18
|
+
const key = normalizedArgv[index];
|
|
19
|
+
const value = normalizedArgv[index + 1];
|
|
20
|
+
if (!key?.startsWith('--') || !value) {
|
|
21
|
+
throw new Error(`无效参数对: ${key ?? '<empty>'}`);
|
|
22
|
+
}
|
|
23
|
+
result[key.slice(2)] = value;
|
|
24
|
+
}
|
|
25
|
+
if (!result['code-list'])
|
|
26
|
+
throw new Error('缺少 --code-list');
|
|
27
|
+
if (!result['tests-dir'])
|
|
28
|
+
throw new Error('缺少 --tests-dir');
|
|
29
|
+
return {
|
|
30
|
+
codeList: result['code-list'],
|
|
31
|
+
testsDir: result['tests-dir'],
|
|
32
|
+
scenario: result.scenario,
|
|
33
|
+
hints: result.hints,
|
|
34
|
+
projectDir: result['project-dir'],
|
|
35
|
+
requirementDir: result['requirement-dir'],
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* scenario 3 档决策(REQ-SCENE-01):
|
|
40
|
+
* 1. --scenario CLI 参数(最高优先级)
|
|
41
|
+
* 2. ModuleHints 文件路径包含 `scenario=<value>` 字串(粗推断)
|
|
42
|
+
* 3. 默认 "crud.single-page"(与 i18n 表 / skeleton scenario 一致)
|
|
43
|
+
*
|
|
44
|
+
* @param args - 解析后的参数。
|
|
45
|
+
* @returns 决策结果。
|
|
46
|
+
*/
|
|
47
|
+
export function decideScenario(args) {
|
|
48
|
+
if (args.scenario)
|
|
49
|
+
return args.scenario;
|
|
50
|
+
if (args.hints && args.hints.includes('scenario=')) {
|
|
51
|
+
const inferred = args.hints.split('scenario=')[1]?.split(/\s|,/)[0];
|
|
52
|
+
if (inferred)
|
|
53
|
+
return inferred;
|
|
54
|
+
}
|
|
55
|
+
return 'crud.single-page';
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* 把 TestcaseV2 渲染成中文 testcase.md(REQ-UX-03)。
|
|
59
|
+
*
|
|
60
|
+
* 设计:
|
|
61
|
+
* - 字段名 / 状态值通过 i18nTestcase() 翻译;
|
|
62
|
+
* - `unresolved` 字段标题硬编码「未确认项」(i18n 表中 `unresolved` 解析为状态值「未解决」);
|
|
63
|
+
* - 每次重跑覆盖(REQ-DATA-03 由调用方 writeFileSync 触发)。
|
|
64
|
+
*
|
|
65
|
+
* @param testcase - TestcaseV2 顶层结构。
|
|
66
|
+
* @returns Markdown 文本。
|
|
67
|
+
*/
|
|
68
|
+
export function renderTestcaseMarkdown(testcase) {
|
|
69
|
+
const lines = [];
|
|
70
|
+
lines.push(`# ${testcase.moduleName} 胶水测试用例`);
|
|
71
|
+
lines.push('');
|
|
72
|
+
lines.push(`- 模块 ID:${testcase.moduleId}`);
|
|
73
|
+
lines.push(`- ${i18nTestcase('scenario')}:${i18nTestcase(testcase.scenario)}`);
|
|
74
|
+
lines.push(`- ${i18nTestcase('menuPath')}:${testcase.menuPath || '(未填写)'}`);
|
|
75
|
+
lines.push(`- ${i18nTestcase('requiredSystems')}:${testcase.requiredSystems.map((s) => s.url || '(待填写)').join(',')}`);
|
|
76
|
+
lines.push(`- ${i18nTestcase('requiredRoles')}:${testcase.requiredRoles.map((r) => r.role).join(',')}`);
|
|
77
|
+
lines.push('');
|
|
78
|
+
lines.push(`## ${i18nTestcase('coveredActions')}`);
|
|
79
|
+
if (testcase.coveredActions.length === 0)
|
|
80
|
+
lines.push('(无)');
|
|
81
|
+
for (const action of testcase.coveredActions) {
|
|
82
|
+
lines.push(`- ${action.actionId}(${action.kind})证据:${action.evidence.join(' / ')}`);
|
|
83
|
+
}
|
|
84
|
+
lines.push('');
|
|
85
|
+
lines.push(`## ${i18nTestcase('uncoveredCandidates')}`);
|
|
86
|
+
if (testcase.uncoveredCandidates.length === 0)
|
|
87
|
+
lines.push('(无)');
|
|
88
|
+
for (const cand of testcase.uncoveredCandidates) {
|
|
89
|
+
lines.push(`- ${cand.actionId} ${cand.label}|状态:${i18nTestcase(cand.status)}|${i18nTestcase('requiredRole')}:${cand.requiredRole}`);
|
|
90
|
+
lines.push(` - ${i18nTestcase('businessIntent')}:${cand.businessIntent}`);
|
|
91
|
+
lines.push(` - ${i18nTestcase('assertionExpectation')}:${cand.assertionExpectation}`);
|
|
92
|
+
lines.push(` - ${i18nTestcase('evidence')}:${cand.evidence.join(' / ')}`);
|
|
93
|
+
}
|
|
94
|
+
lines.push('');
|
|
95
|
+
lines.push('## 用例');
|
|
96
|
+
for (const c of testcase.cases) {
|
|
97
|
+
lines.push(`### ${c.title}(${i18nTestcase('caseId')}:${c.caseId})`);
|
|
98
|
+
lines.push(`- ${i18nTestcase('scenario')}:${i18nTestcase(c.scenario)}|${i18nTestcase('reviewStatus')}:${i18nTestcase(c.reviewStatus)}|${i18nTestcase('requiredRole')}:${c.requiredRole}`);
|
|
99
|
+
lines.push(`- ${i18nTestcase('evidence')}:${c.evidence.join(' / ')}`);
|
|
100
|
+
lines.push('- 步骤:');
|
|
101
|
+
for (const s of c.steps)
|
|
102
|
+
lines.push(` 1. ${s}`);
|
|
103
|
+
lines.push('- 断言:');
|
|
104
|
+
for (const a of c.assertions)
|
|
105
|
+
lines.push(` - ${a}`);
|
|
106
|
+
// 硬编码标题「未确认项」:i18nTestcase('unresolved') 返回状态值「未解决」(spec §4.6 trade-off)。
|
|
107
|
+
lines.push(`- 未确认项:${c.unresolved.length === 0 ? '(无)' : c.unresolved.join(';')}`);
|
|
108
|
+
lines.push('');
|
|
109
|
+
}
|
|
110
|
+
lines.push('## 推理摘要');
|
|
111
|
+
lines.push(testcase.reasoningSummary.conclusion);
|
|
112
|
+
lines.push('');
|
|
113
|
+
lines.push('证据链:');
|
|
114
|
+
for (const ev of testcase.reasoningSummary.evidenceChain)
|
|
115
|
+
lines.push(`- ${ev}`);
|
|
116
|
+
return `${lines.join('\n').trimEnd()}\n`;
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* 渲染 credentials.template.json5 数组内容。每条对应 requiredRoles[i],
|
|
120
|
+
* url/systemName 复用 requiredSystems[0](多系统场景由用户手工补齐)。
|
|
121
|
+
*
|
|
122
|
+
* @param testcase - TestcaseV2。
|
|
123
|
+
* @returns JSON5 字符串。
|
|
124
|
+
*/
|
|
125
|
+
function renderCredentialsTemplate(testcase) {
|
|
126
|
+
const baseUrl = testcase.requiredSystems[0]?.url ?? '';
|
|
127
|
+
const baseSystem = testcase.requiredSystems[0]?.systemName ?? '';
|
|
128
|
+
const entries = testcase.requiredRoles.map((roleEntry) => ({
|
|
129
|
+
url: baseUrl,
|
|
130
|
+
username: '',
|
|
131
|
+
password: '',
|
|
132
|
+
role: roleEntry.role,
|
|
133
|
+
systemName: baseSystem,
|
|
134
|
+
}));
|
|
135
|
+
return `${JSON5.stringify(entries, null, 2)}\n`;
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* testcase 子命令主体(REQ-CHAIN-01 / REQ-CHAIN-02 / REQ-SCENE-01 / REQ-DATA-02)。
|
|
139
|
+
*
|
|
140
|
+
* 流程:
|
|
141
|
+
* 1. 决定 scenario(3 档优先级,写入 testcase 顶层 scenario 字段);
|
|
142
|
+
* 2. 调 runProbeAndCoverage 钻探上游物料 + 装配 TestcaseV2;
|
|
143
|
+
* 3. 落盘 testcase.json5 + testcase.md(中文 i18n) + credentials.template.json5;
|
|
144
|
+
* 4. 软链上游 code_list.md → <tests-dir>/code_list.md;
|
|
145
|
+
* 5. 返回推断结果(供 skill Step 5 交互确认 systemUrl / credentialsCache 命中)。
|
|
146
|
+
*
|
|
147
|
+
* 注意:contract 提取通过 probe 内部完成,不直接复用 dev/extract-contract.ts 的 CLI 入口
|
|
148
|
+
* (probe + extract-contract 共享同一组 buildCrudBusinessModuleContract + extractor 依赖,
|
|
149
|
+
* 二者各自负责自己的 I/O 边界,保持职责单一)。
|
|
150
|
+
*
|
|
151
|
+
* @param args - 已解析参数。
|
|
152
|
+
* @returns 产物路径 + 推断结果。
|
|
153
|
+
*/
|
|
154
|
+
export function runGenerateTestcase(args) {
|
|
155
|
+
const scenario = decideScenario(args);
|
|
156
|
+
mkdirSync(args.testsDir, { recursive: true });
|
|
157
|
+
const requirementDir = args.requirementDir ?? path.dirname(args.codeList);
|
|
158
|
+
const docsDir = requirementDir;
|
|
159
|
+
const webappDir = path.join(requirementDir, 'src/main/webapp');
|
|
160
|
+
const javaActionsDir = path.join(requirementDir, 'src/main/java');
|
|
161
|
+
const moduleId = path.basename(args.testsDir);
|
|
162
|
+
const { contract, testcaseV2 } = runProbeAndCoverage({
|
|
163
|
+
moduleId,
|
|
164
|
+
codeListPath: args.codeList,
|
|
165
|
+
docsDir,
|
|
166
|
+
webappDir,
|
|
167
|
+
javaActionsDir,
|
|
168
|
+
hintsPath: args.hints,
|
|
169
|
+
scenario,
|
|
170
|
+
});
|
|
171
|
+
const epStageDir = path.join(args.testsDir, '.ep-stage');
|
|
172
|
+
mkdirSync(epStageDir, { recursive: true });
|
|
173
|
+
const contractPath = path.join(epStageDir, 'contract.json');
|
|
174
|
+
const testcasePath = path.join(args.testsDir, 'testcase.json5');
|
|
175
|
+
const testcaseMdPath = path.join(args.testsDir, 'testcase.md');
|
|
176
|
+
const credentialsTemplatePath = path.join(args.testsDir, 'credentials.template.json5');
|
|
177
|
+
const credentialsActualPath = path.join(args.testsDir, 'credentials.json5');
|
|
178
|
+
// 落盘 contract.json 供 spec / run 子命令复用(避免重复钻探)。
|
|
179
|
+
writeFileSync(contractPath, `${JSON.stringify(contract, null, 2)}\n`, 'utf8');
|
|
180
|
+
writeFileSync(testcasePath, `${JSON5.stringify(testcaseV2, null, 2)}\n`, 'utf8');
|
|
181
|
+
writeFileSync(credentialsTemplatePath, renderCredentialsTemplate(testcaseV2), 'utf8');
|
|
182
|
+
writeFileSync(testcaseMdPath, renderTestcaseMarkdown(testcaseV2), 'utf8');
|
|
183
|
+
createSoftlinkWithFallback(args.codeList, path.join(args.testsDir, 'code_list.md'));
|
|
184
|
+
return {
|
|
185
|
+
testcasePath,
|
|
186
|
+
testcaseMdPath,
|
|
187
|
+
credentialsTemplatePath,
|
|
188
|
+
inference: {
|
|
189
|
+
scenario,
|
|
190
|
+
requiredSystems: testcaseV2.requiredSystems,
|
|
191
|
+
requiredRoles: testcaseV2.requiredRoles,
|
|
192
|
+
credentialsTemplatePath,
|
|
193
|
+
credentialsActualPath,
|
|
194
|
+
cachedSystemsHit: [],
|
|
195
|
+
},
|
|
196
|
+
};
|
|
197
|
+
}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* ep-stage-skill CLI 统一入口(REQ-CLI-02 修订):1 bin + 3 子命令。
|
|
4
|
+
*
|
|
5
|
+
* 用法:
|
|
6
|
+
* ep-stage-skill testcase --code-list <p> --tests-dir <p> [--scenario <v>] [--hints <p>]
|
|
7
|
+
* ep-stage-skill spec --testcase <p> --out <p>
|
|
8
|
+
* ep-stage-skill run --spec <p> --testcase <p> [--headless false]
|
|
9
|
+
*
|
|
10
|
+
* 无 contract 子命令(contract 提取作为 testcase 子命令内部步骤;
|
|
11
|
+
* 开发期独立调试入口为 dev/extract-contract.ts,被 pnpm generate:crud-contract 调用)。
|
|
12
|
+
*/
|
|
13
|
+
export declare function main(argv?: string[]): Promise<unknown>;
|
|
14
|
+
/**
|
|
15
|
+
* 打印 CLI help(仅 testcase / spec / run 3 子命令,无 contract)。
|
|
16
|
+
*/
|
|
17
|
+
export declare function printHelp(): void;
|
|
18
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/cli/index.ts"],"names":[],"mappings":";AAKA;;;;;;;;;;GAUG;AACH,wBAAsB,IAAI,CAAC,IAAI,GAAE,MAAM,EAAiB,GAAG,OAAO,CAAC,OAAO,CAAC,CAmB1E;AAED;;GAEG;AACH,wBAAgB,SAAS,IAAI,IAAI,CAYhC"}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { parseGenerateTestcaseArgs, runGenerateTestcase } from './generate-testcase.js';
|
|
3
|
+
import { parseSpecArgs, runSpec } from './spec.js';
|
|
4
|
+
import { parseRunArgs, runRun } from './run.js';
|
|
5
|
+
/**
|
|
6
|
+
* ep-stage-skill CLI 统一入口(REQ-CLI-02 修订):1 bin + 3 子命令。
|
|
7
|
+
*
|
|
8
|
+
* 用法:
|
|
9
|
+
* ep-stage-skill testcase --code-list <p> --tests-dir <p> [--scenario <v>] [--hints <p>]
|
|
10
|
+
* ep-stage-skill spec --testcase <p> --out <p>
|
|
11
|
+
* ep-stage-skill run --spec <p> --testcase <p> [--headless false]
|
|
12
|
+
*
|
|
13
|
+
* 无 contract 子命令(contract 提取作为 testcase 子命令内部步骤;
|
|
14
|
+
* 开发期独立调试入口为 dev/extract-contract.ts,被 pnpm generate:crud-contract 调用)。
|
|
15
|
+
*/
|
|
16
|
+
export async function main(argv = process.argv) {
|
|
17
|
+
const [, , subcommand, ...rest] = argv;
|
|
18
|
+
const args = rest.filter((arg) => arg !== '--');
|
|
19
|
+
switch (subcommand) {
|
|
20
|
+
case 'testcase':
|
|
21
|
+
return runGenerateTestcase(parseGenerateTestcaseArgs(args));
|
|
22
|
+
case 'spec':
|
|
23
|
+
return runSpec(parseSpecArgs(args));
|
|
24
|
+
case 'run':
|
|
25
|
+
return runRun(parseRunArgs(args));
|
|
26
|
+
case undefined:
|
|
27
|
+
case '--help':
|
|
28
|
+
case '-h':
|
|
29
|
+
printHelp();
|
|
30
|
+
return;
|
|
31
|
+
default:
|
|
32
|
+
console.error(`未知子命令: ${subcommand}。可用: testcase / spec / run`);
|
|
33
|
+
process.exit(1);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* 打印 CLI help(仅 testcase / spec / run 3 子命令,无 contract)。
|
|
38
|
+
*/
|
|
39
|
+
export function printHelp() {
|
|
40
|
+
console.log(`ep-stage-skill — 胶水测试链路 CLI
|
|
41
|
+
|
|
42
|
+
用法:
|
|
43
|
+
ep-stage-skill testcase --code-list <path> --tests-dir <path> [--scenario <value>] [--hints <path>]
|
|
44
|
+
ep-stage-skill spec --testcase <path> --out <path>
|
|
45
|
+
ep-stage-skill run --spec <path> --testcase <path> [--headless false]
|
|
46
|
+
|
|
47
|
+
子命令:
|
|
48
|
+
testcase 钻探上游物料 + 生成 testcase v2 + credentials.template + 软链 code_list
|
|
49
|
+
spec 基于 testcase + contract 生成 v1 初始版 .spec.ts(只生成不执行)
|
|
50
|
+
run 启动 Playwright 跑胶水部分 + AI 推理追加 uncoveredCandidates → 产 v2 + 报告`);
|
|
51
|
+
}
|
|
52
|
+
// 直接执行时调用 main;作为模块 import 时由调用方决定是否执行。
|
|
53
|
+
if (import.meta.url === `file://${process.argv[1]}`) {
|
|
54
|
+
await main();
|
|
55
|
+
}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import type { CrudBusinessModuleContract } from '../contracts/crud-business-module.js';
|
|
2
|
+
import type { CoverageDiffTrace } from '../contracts/observable-chain.js';
|
|
3
|
+
import type { TestcaseV2 } from '../testcase/testcase-v2.js';
|
|
4
|
+
/**
|
|
5
|
+
* probe 子模块入参(REQ-CHAIN-01)。
|
|
6
|
+
*
|
|
7
|
+
* 把 run-gap-pipeline.ts 与 generate-testcase.ts 共用的「物料解析 → contract 构建 →
|
|
8
|
+
* skeleton coverage → coverage diff」段抽成单一入口,避免两侧复制实现。
|
|
9
|
+
*/
|
|
10
|
+
export type ProbeAndCoverageArgs = {
|
|
11
|
+
moduleId: string;
|
|
12
|
+
codeListPath: string;
|
|
13
|
+
docsDir: string;
|
|
14
|
+
webappDir: string;
|
|
15
|
+
javaActionsDir: string;
|
|
16
|
+
hintsPath?: string;
|
|
17
|
+
scenario: string;
|
|
18
|
+
menuPath?: string;
|
|
19
|
+
};
|
|
20
|
+
/**
|
|
21
|
+
* probe 子模块产物。
|
|
22
|
+
*
|
|
23
|
+
* @property contract - 标准化后的 CRUD 业务模块契约。
|
|
24
|
+
* @property coverageDiff - 骨架 vs gap 覆盖度差异 trace。
|
|
25
|
+
* @property testcaseV2 - 装配后的 testcase v2 顶层结构(待 generate-testcase 写入 .json5)。
|
|
26
|
+
*/
|
|
27
|
+
export type ProbeAndCoverageResult = {
|
|
28
|
+
contract: CrudBusinessModuleContract;
|
|
29
|
+
coverageDiff: CoverageDiffTrace;
|
|
30
|
+
testcaseV2: TestcaseV2;
|
|
31
|
+
};
|
|
32
|
+
/**
|
|
33
|
+
* probe + coverage + testcase v2 装配的统一入口(REQ-CHAIN-01)。
|
|
34
|
+
*
|
|
35
|
+
* 把 run-gap-pipeline.ts 与 generate-testcase.ts 共用的物料解析 → contract → skeleton coverage →
|
|
36
|
+
* coverage diff 段抽成单一函数,避免 1881 行 run-gap-pipeline.ts 与新 CLI 双份维护。
|
|
37
|
+
*
|
|
38
|
+
* 注意:本函数只做钻探与装配,不写文件;写盘由调用方负责(保持 I/O 边界清晰)。
|
|
39
|
+
*
|
|
40
|
+
* @param args - 物料路径 + 场景。
|
|
41
|
+
* @returns contract + coverageDiff + 装配好的 TestcaseV2。
|
|
42
|
+
*/
|
|
43
|
+
export declare function runProbeAndCoverage(args: ProbeAndCoverageArgs): ProbeAndCoverageResult;
|
|
44
|
+
//# sourceMappingURL=probe.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"probe.d.ts","sourceRoot":"","sources":["../../../src/cli/probe.ts"],"names":[],"mappings":"AAYA,OAAO,KAAK,EAAE,0BAA0B,EAAe,MAAM,sCAAsC,CAAC;AACpG,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,kCAAkC,CAAC;AAC1E,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,4BAA4B,CAAC;AAG7D;;;;;GAKG;AACH,MAAM,MAAM,oBAAoB,GAAG;IACjC,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;IAClB,cAAc,EAAE,MAAM,CAAC;IACvB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB,CAAC;AAEF;;;;;;GAMG;AACH,MAAM,MAAM,sBAAsB,GAAG;IACnC,QAAQ,EAAE,0BAA0B,CAAC;IACrC,YAAY,EAAE,iBAAiB,CAAC;IAChC,UAAU,EAAE,UAAU,CAAC;CACxB,CAAC;AA+LF;;;;;;;;;;GAUG;AACH,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,oBAAoB,GAAG,sBAAsB,CAkCtF"}
|