@sun-asterisk/sungen 3.1.0 → 3.1.2-beta.100
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/README.md +4 -428
- package/dist/capabilities/builtins.d.ts +31 -0
- package/dist/capabilities/builtins.d.ts.map +1 -0
- package/dist/capabilities/builtins.js +84 -0
- package/dist/capabilities/builtins.js.map +1 -0
- package/dist/capabilities/context-router.d.ts +34 -0
- package/dist/capabilities/context-router.d.ts.map +1 -0
- package/dist/capabilities/context-router.js +49 -0
- package/dist/capabilities/context-router.js.map +1 -0
- package/dist/capabilities/context.d.ts +51 -0
- package/dist/capabilities/context.d.ts.map +1 -0
- package/dist/capabilities/context.js +17 -0
- package/dist/capabilities/context.js.map +1 -0
- package/dist/capabilities/discover.d.ts +2 -0
- package/dist/capabilities/discover.d.ts.map +1 -0
- package/dist/capabilities/discover.js +48 -0
- package/dist/capabilities/discover.js.map +1 -0
- package/dist/capabilities/registry.d.ts +90 -0
- package/dist/capabilities/registry.d.ts.map +1 -0
- package/dist/capabilities/registry.js +43 -0
- package/dist/capabilities/registry.js.map +1 -0
- package/dist/capabilities/sensor.d.ts +49 -0
- package/dist/capabilities/sensor.d.ts.map +1 -0
- package/dist/capabilities/sensor.js +3 -0
- package/dist/capabilities/sensor.js.map +1 -0
- package/dist/cli/commands/challenge.d.ts.map +1 -1
- package/dist/cli/commands/challenge.js +9 -2
- package/dist/cli/commands/challenge.js.map +1 -1
- package/dist/cli/commands/delivery.d.ts.map +1 -1
- package/dist/cli/commands/delivery.js +3 -2
- package/dist/cli/commands/delivery.js.map +1 -1
- package/dist/cli/commands/generate.d.ts.map +1 -1
- package/dist/cli/commands/generate.js +12 -0
- package/dist/cli/commands/generate.js.map +1 -1
- package/dist/cli/index.js +10 -1
- package/dist/cli/index.js.map +1 -1
- package/dist/exporters/csv-exporter.d.ts.map +1 -1
- package/dist/exporters/csv-exporter.js +92 -76
- package/dist/exporters/csv-exporter.js.map +1 -1
- package/dist/exporters/spec-parser.d.ts.map +1 -1
- package/dist/exporters/spec-parser.js +6 -1
- package/dist/exporters/spec-parser.js.map +1 -1
- package/dist/generators/test-generator/adapters/adapter-interface.d.ts +2 -0
- package/dist/generators/test-generator/adapters/adapter-interface.d.ts.map +1 -1
- package/dist/generators/test-generator/adapters/playwright/playwright-adapter.d.ts +1 -0
- package/dist/generators/test-generator/adapters/playwright/playwright-adapter.d.ts.map +1 -1
- package/dist/generators/test-generator/adapters/playwright/playwright-adapter.js.map +1 -1
- package/dist/generators/test-generator/adapters/playwright/templates/imports.hbs +3 -0
- package/dist/generators/test-generator/adapters/playwright/templates/scenario.hbs +19 -1
- package/dist/generators/test-generator/code-generator.d.ts +21 -4
- package/dist/generators/test-generator/code-generator.d.ts.map +1 -1
- package/dist/generators/test-generator/code-generator.js +118 -57
- package/dist/generators/test-generator/code-generator.js.map +1 -1
- package/dist/generators/test-generator/patterns/expect-patterns.d.ts +3 -0
- package/dist/generators/test-generator/patterns/expect-patterns.d.ts.map +1 -0
- package/dist/generators/test-generator/patterns/expect-patterns.js +54 -0
- package/dist/generators/test-generator/patterns/expect-patterns.js.map +1 -0
- package/dist/generators/test-generator/patterns/index.d.ts +0 -10
- package/dist/generators/test-generator/patterns/index.d.ts.map +1 -1
- package/dist/generators/test-generator/patterns/index.js +10 -45
- package/dist/generators/test-generator/patterns/index.js.map +1 -1
- package/dist/generators/test-generator/step-mapper.d.ts +6 -0
- package/dist/generators/test-generator/step-mapper.d.ts.map +1 -1
- package/dist/generators/test-generator/step-mapper.js +8 -0
- package/dist/generators/test-generator/step-mapper.js.map +1 -1
- package/dist/generators/test-generator/template-engine.d.ts +4 -0
- package/dist/generators/test-generator/template-engine.d.ts.map +1 -1
- package/dist/generators/test-generator/template-engine.js +1 -1
- package/dist/generators/test-generator/template-engine.js.map +1 -1
- package/dist/generators/test-generator/utils/runtime-data-transformer.d.ts +1 -1
- package/dist/generators/test-generator/utils/runtime-data-transformer.d.ts.map +1 -1
- package/dist/generators/test-generator/utils/runtime-data-transformer.js +5 -5
- package/dist/generators/test-generator/utils/runtime-data-transformer.js.map +1 -1
- package/dist/harness/annotation-overrides.d.ts +9 -0
- package/dist/harness/annotation-overrides.d.ts.map +1 -0
- package/dist/harness/annotation-overrides.js +36 -0
- package/dist/harness/annotation-overrides.js.map +1 -0
- package/dist/harness/audit.d.ts.map +1 -1
- package/dist/harness/audit.js +35 -7
- package/dist/harness/audit.js.map +1 -1
- package/dist/harness/catalog/drivers.yaml +35 -12
- package/dist/harness/challenge.d.ts +1 -0
- package/dist/harness/challenge.d.ts.map +1 -1
- package/dist/harness/challenge.js +49 -2
- package/dist/harness/challenge.js.map +1 -1
- package/dist/harness/data-driven-lint.d.ts +7 -0
- package/dist/harness/data-driven-lint.d.ts.map +1 -0
- package/dist/harness/data-driven-lint.js +153 -0
- package/dist/harness/data-driven-lint.js.map +1 -0
- package/dist/harness/parse.d.ts +3 -0
- package/dist/harness/parse.d.ts.map +1 -1
- package/dist/harness/parse.js +25 -0
- package/dist/harness/parse.js.map +1 -1
- package/dist/harness/query-catalog.d.ts +48 -0
- package/dist/harness/query-catalog.d.ts.map +1 -0
- package/dist/harness/query-catalog.js +0 -0
- package/dist/harness/query-catalog.js.map +1 -0
- package/dist/harness/script-check.d.ts.map +1 -1
- package/dist/harness/script-check.js +7 -4
- package/dist/harness/script-check.js.map +1 -1
- package/dist/index.d.ts +20 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +32 -0
- package/dist/index.js.map +1 -0
- package/dist/orchestrator/templates/ai-instructions/claude-agent-challenge.md +3 -2
- package/dist/orchestrator/templates/ai-instructions/claude-cmd-create-test.md +4 -3
- package/dist/orchestrator/templates/ai-instructions/claude-skill-gherkin-syntax.md +41 -0
- package/dist/orchestrator/templates/ai-instructions/claude-skill-tc-generation.md +22 -0
- package/dist/orchestrator/templates/ai-instructions/claude-skill-tc-review.md +1 -0
- package/dist/orchestrator/templates/ai-instructions/claude-skill-test-design-techniques.md +6 -0
- package/dist/orchestrator/templates/ai-instructions/copilot-cmd-create-test.md +4 -3
- package/dist/orchestrator/templates/ai-instructions/github-skill-sungen-gherkin-syntax.md +41 -0
- package/dist/orchestrator/templates/ai-instructions/github-skill-sungen-tc-generation.md +22 -0
- package/dist/orchestrator/templates/ai-instructions/github-skill-sungen-tc-review.md +1 -0
- package/dist/orchestrator/templates/ai-instructions/github-skill-sungen-test-design-techniques.md +6 -0
- package/dist/orchestrator/templates/specs-api.d.ts +19 -0
- package/dist/orchestrator/templates/specs-api.d.ts.map +1 -0
- package/dist/orchestrator/templates/specs-api.js +128 -0
- package/dist/orchestrator/templates/specs-api.js.map +1 -0
- package/dist/orchestrator/templates/specs-api.ts +101 -0
- package/dist/orchestrator/templates/specs-db.d.ts +8 -0
- package/dist/orchestrator/templates/specs-db.d.ts.map +1 -1
- package/dist/orchestrator/templates/specs-db.js +22 -0
- package/dist/orchestrator/templates/specs-db.js.map +1 -1
- package/dist/orchestrator/templates/specs-db.ts +22 -0
- package/dist/orchestrator/templates/specs-test-data.ts +76 -15
- package/package.json +7 -30
- package/src/capabilities/builtins.ts +85 -0
- package/src/capabilities/context-router.ts +66 -0
- package/src/capabilities/context.ts +46 -0
- package/src/capabilities/discover.ts +42 -0
- package/src/capabilities/registry.ts +111 -0
- package/src/capabilities/sensor.ts +47 -0
- package/src/cli/commands/challenge.ts +6 -2
- package/src/cli/commands/delivery.ts +3 -2
- package/src/cli/commands/generate.ts +12 -0
- package/src/cli/index.ts +10 -1
- package/src/exporters/csv-exporter.ts +22 -6
- package/src/exporters/spec-parser.ts +6 -1
- package/src/generators/test-generator/adapters/adapter-interface.ts +2 -1
- package/src/generators/test-generator/adapters/playwright/playwright-adapter.ts +1 -1
- package/src/generators/test-generator/adapters/playwright/templates/imports.hbs +3 -0
- package/src/generators/test-generator/adapters/playwright/templates/scenario.hbs +19 -1
- package/src/generators/test-generator/code-generator.ts +114 -59
- package/src/generators/test-generator/patterns/expect-patterns.ts +49 -0
- package/src/generators/test-generator/patterns/index.ts +9 -33
- package/src/generators/test-generator/step-mapper.ts +9 -0
- package/src/generators/test-generator/template-engine.ts +5 -2
- package/src/generators/test-generator/utils/runtime-data-transformer.ts +5 -5
- package/src/harness/annotation-overrides.ts +25 -0
- package/src/harness/audit.ts +37 -8
- package/src/harness/catalog/drivers.yaml +35 -12
- package/src/harness/challenge.ts +47 -2
- package/src/harness/data-driven-lint.ts +119 -0
- package/src/harness/parse.ts +17 -0
- package/src/harness/query-catalog.ts +0 -0
- package/src/harness/script-check.ts +8 -5
- package/src/index.ts +30 -0
- package/src/orchestrator/templates/ai-instructions/claude-agent-challenge.md +3 -2
- package/src/orchestrator/templates/ai-instructions/claude-cmd-create-test.md +4 -3
- package/src/orchestrator/templates/ai-instructions/claude-skill-gherkin-syntax.md +41 -0
- package/src/orchestrator/templates/ai-instructions/claude-skill-tc-generation.md +22 -0
- package/src/orchestrator/templates/ai-instructions/claude-skill-tc-review.md +1 -0
- package/src/orchestrator/templates/ai-instructions/claude-skill-test-design-techniques.md +6 -0
- package/src/orchestrator/templates/ai-instructions/copilot-cmd-create-test.md +4 -3
- package/src/orchestrator/templates/ai-instructions/github-skill-sungen-gherkin-syntax.md +41 -0
- package/src/orchestrator/templates/ai-instructions/github-skill-sungen-tc-generation.md +22 -0
- package/src/orchestrator/templates/ai-instructions/github-skill-sungen-tc-review.md +1 -0
- package/src/orchestrator/templates/ai-instructions/github-skill-sungen-test-design-techniques.md +6 -0
- package/src/orchestrator/templates/specs-api.ts +101 -0
- package/src/orchestrator/templates/specs-db.ts +22 -0
- package/src/orchestrator/templates/specs-test-data.ts +76 -15
- package/dist/generators/test-generator/patterns/assertion-patterns.d.ts +0 -7
- package/dist/generators/test-generator/patterns/assertion-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/assertion-patterns.js +0 -626
- package/dist/generators/test-generator/patterns/assertion-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/capture-patterns.d.ts +0 -21
- package/dist/generators/test-generator/patterns/capture-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/capture-patterns.js +0 -87
- package/dist/generators/test-generator/patterns/capture-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/database-patterns.d.ts +0 -5
- package/dist/generators/test-generator/patterns/database-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/database-patterns.js +0 -94
- package/dist/generators/test-generator/patterns/database-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/form-patterns.d.ts +0 -6
- package/dist/generators/test-generator/patterns/form-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/form-patterns.js +0 -160
- package/dist/generators/test-generator/patterns/form-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/interaction-patterns.d.ts +0 -6
- package/dist/generators/test-generator/patterns/interaction-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/interaction-patterns.js +0 -433
- package/dist/generators/test-generator/patterns/interaction-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/keyboard-patterns.d.ts +0 -7
- package/dist/generators/test-generator/patterns/keyboard-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/keyboard-patterns.js +0 -47
- package/dist/generators/test-generator/patterns/keyboard-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/navigation-patterns.d.ts +0 -6
- package/dist/generators/test-generator/patterns/navigation-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/navigation-patterns.js +0 -125
- package/dist/generators/test-generator/patterns/navigation-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/scope-patterns.d.ts +0 -7
- package/dist/generators/test-generator/patterns/scope-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/scope-patterns.js +0 -36
- package/dist/generators/test-generator/patterns/scope-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/scroll-patterns.d.ts +0 -7
- package/dist/generators/test-generator/patterns/scroll-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/scroll-patterns.js +0 -25
- package/dist/generators/test-generator/patterns/scroll-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/setup-patterns.d.ts +0 -6
- package/dist/generators/test-generator/patterns/setup-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/setup-patterns.js +0 -72
- package/dist/generators/test-generator/patterns/setup-patterns.js.map +0 -1
- package/dist/generators/test-generator/patterns/table-patterns.d.ts +0 -19
- package/dist/generators/test-generator/patterns/table-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/table-patterns.js +0 -239
- package/dist/generators/test-generator/patterns/table-patterns.js.map +0 -1
- package/docs/orchestration-spec.md +0 -267
- package/src/generators/test-generator/patterns/assertion-patterns.ts +0 -691
- package/src/generators/test-generator/patterns/capture-patterns.ts +0 -97
- package/src/generators/test-generator/patterns/database-patterns.ts +0 -95
- package/src/generators/test-generator/patterns/form-patterns.ts +0 -167
- package/src/generators/test-generator/patterns/interaction-patterns.ts +0 -465
- package/src/generators/test-generator/patterns/keyboard-patterns.ts +0 -51
- package/src/generators/test-generator/patterns/navigation-patterns.ts +0 -140
- package/src/generators/test-generator/patterns/scope-patterns.ts +0 -40
- package/src/generators/test-generator/patterns/scroll-patterns.ts +0 -27
- package/src/generators/test-generator/patterns/setup-patterns.ts +0 -76
- package/src/generators/test-generator/patterns/table-patterns.ts +0 -279
|
@@ -1,97 +0,0 @@
|
|
|
1
|
-
import { ParsedStep } from '../../gherkin-parser';
|
|
2
|
-
import { StepPattern, StepTemplateData } from './types';
|
|
3
|
-
|
|
4
|
-
/**
|
|
5
|
-
* Capture & collection patterns (P5) — enable cross-screen data consistency and
|
|
6
|
-
* filter-result correctness that plain single-element assertions can't express.
|
|
7
|
-
*
|
|
8
|
-
* 1. Capture: `User remember [Product Name] text as {{selected_product_name}}`
|
|
9
|
-
* → stores the element's text/value into a runtime variable so a
|
|
10
|
-
* later step (on another screen) can assert against it.
|
|
11
|
-
* REQUIRES runtime data mode (default) — emits `testData.set(...)`.
|
|
12
|
-
*
|
|
13
|
-
* 2. List: `User see all [Product Name] contain {{selected_category}}`
|
|
14
|
-
* → asserts EVERY matching element's text contains the value
|
|
15
|
-
* (e.g. all products belong to the selected category/brand).
|
|
16
|
-
*
|
|
17
|
-
* 3. All-card: `User see all [Product Card] contain [Add To Cart] button`
|
|
18
|
-
* → asserts EVERY container element holds the child element
|
|
19
|
-
* (structural per-card proof: every card has name/price/action).
|
|
20
|
-
* Roadmap Q2 — the fix for "each card exposes X" claims.
|
|
21
|
-
*/
|
|
22
|
-
export const capturePatterns: StepPattern[] = [
|
|
23
|
-
{
|
|
24
|
-
name: 'capture-variable',
|
|
25
|
-
matcher: (step: ParsedStep) =>
|
|
26
|
-
/\bremember\b/i.test(step.text) && /\bas\b/i.test(step.text) && !!step.selectorRef && !!step.dataRef,
|
|
27
|
-
resolver: (step, context): StepTemplateData => {
|
|
28
|
-
const resolved = context.selectorResolver.resolveSelector(
|
|
29
|
-
step.selectorRef!, undefined, step.elementType, step.nth,
|
|
30
|
-
);
|
|
31
|
-
const varName = step.dataRef!;
|
|
32
|
-
const isValue = /\bvalue\b/i.test(step.text);
|
|
33
|
-
// Register so later `{{varName}}` references resolve to testData.get(varName)
|
|
34
|
-
// and skip YAML validation.
|
|
35
|
-
context.dataResolver.registerCaptured(varName);
|
|
36
|
-
return {
|
|
37
|
-
templateName: 'capture-variable',
|
|
38
|
-
data: { ...resolved, varName, capture: isValue ? 'inputValue' : 'textContent' },
|
|
39
|
-
comment: `Remember ${step.selectorRef} ${isValue ? 'value' : 'text'} as ${varName}`,
|
|
40
|
-
};
|
|
41
|
-
},
|
|
42
|
-
priority: 35,
|
|
43
|
-
},
|
|
44
|
-
{
|
|
45
|
-
name: 'all-contain-assertion',
|
|
46
|
-
matcher: (step: ParsedStep) =>
|
|
47
|
-
/\b(see|sees)\b/i.test(step.text) &&
|
|
48
|
-
/\ball\b/i.test(step.text) &&
|
|
49
|
-
/(contain|contains|match|matches|belong)/i.test(step.text) &&
|
|
50
|
-
!!step.selectorRef && !!(step.value || step.dataRef),
|
|
51
|
-
resolver: (step, context): StepTemplateData => {
|
|
52
|
-
const resolved = context.selectorResolver.resolveSelector(
|
|
53
|
-
step.selectorRef!, undefined, step.elementType, step.nth,
|
|
54
|
-
);
|
|
55
|
-
const expectedText = step.value || context.dataResolver.resolveData(step.dataRef!, context.featureName);
|
|
56
|
-
return {
|
|
57
|
-
templateName: 'all-contain-assertion',
|
|
58
|
-
data: { ...resolved, expectedText, selectorRef: step.selectorRef, stepCounter: context.stepCounter },
|
|
59
|
-
comment: `Assert all ${step.selectorRef} contain "${step.value || step.dataRef}"`,
|
|
60
|
-
};
|
|
61
|
-
},
|
|
62
|
-
priority: 34,
|
|
63
|
-
},
|
|
64
|
-
{
|
|
65
|
-
// Q2 — all-card structural assertion: every container holds the child element.
|
|
66
|
-
// Distinguished from `all-contain-assertion` by a second [ref] (childRef) and the
|
|
67
|
-
// ABSENCE of a value/data (which would make it a text-contains assertion instead).
|
|
68
|
-
name: 'all-contain-element',
|
|
69
|
-
matcher: (step: ParsedStep) =>
|
|
70
|
-
/\b(see|sees)\b/i.test(step.text) &&
|
|
71
|
-
/\ball\b/i.test(step.text) &&
|
|
72
|
-
/(contain|contains|include|includes)/i.test(step.text) &&
|
|
73
|
-
!!step.selectorRef && !!step.childRef && !step.value && !step.dataRef,
|
|
74
|
-
resolver: (step, context): StepTemplateData => {
|
|
75
|
-
const container = context.selectorResolver.resolveSelector(
|
|
76
|
-
step.selectorRef!, undefined, step.elementType, step.nth,
|
|
77
|
-
);
|
|
78
|
-
const child = context.selectorResolver.resolveSelector(
|
|
79
|
-
step.childRef!, undefined, step.childElementType, 0,
|
|
80
|
-
);
|
|
81
|
-
const sc = context.stepCounter;
|
|
82
|
-
return {
|
|
83
|
-
templateName: 'all-contain-element',
|
|
84
|
-
data: {
|
|
85
|
-
...container,
|
|
86
|
-
selectorRef: step.selectorRef,
|
|
87
|
-
childRef: step.childRef,
|
|
88
|
-
stepCounter: sc,
|
|
89
|
-
// Render the child RELATIVE to each container row via locator.hbs's parentLocator branch.
|
|
90
|
-
child: { ...child, parentLocator: `__cards_${sc}.nth(__i_${sc})` },
|
|
91
|
-
},
|
|
92
|
-
comment: `Assert all ${step.selectorRef} contain [${step.childRef}]`,
|
|
93
|
-
};
|
|
94
|
-
},
|
|
95
|
-
priority: 36,
|
|
96
|
-
},
|
|
97
|
-
];
|
|
@@ -1,95 +0,0 @@
|
|
|
1
|
-
import { ParsedStep } from '../../gherkin-parser';
|
|
2
|
-
import { StepPattern, PatternContext } from './types';
|
|
3
|
-
import { MappedStep } from '../step-mapper';
|
|
4
|
-
|
|
5
|
-
/**
|
|
6
|
-
* Database verification patterns (Data Driver v1) — declarative, no-SQL DB assertions
|
|
7
|
-
* that compile to calls on the runtime `db` helper (specs/db.ts). Read-only.
|
|
8
|
-
*
|
|
9
|
-
* User see [users] row where [email] is {{reg_email}}
|
|
10
|
-
* User see [users] row where [email] is {{reg_email}} has [status] = "active"
|
|
11
|
-
* User see [users] no row where [email] is {{dup_email}}
|
|
12
|
-
* User see [orders] where [buyer] is {{buyer}} count is {{expected_count}}
|
|
13
|
-
*
|
|
14
|
-
* Identifiers ([table]/[column]) are validated by the helper; values bind as parameters.
|
|
15
|
-
*/
|
|
16
|
-
|
|
17
|
-
const TABLE = String.raw`\[([A-Za-z_][A-Za-z0-9_]*)\]`;
|
|
18
|
-
const VALUE = String.raw`\{\{[^}]+\}\}|"[^"]*"|'[^']*'|-?\d+(?:\.\d+)?`;
|
|
19
|
-
|
|
20
|
-
const reRow = new RegExp(`see\\s+${TABLE}\\s+row\\s+where\\b`, 'i');
|
|
21
|
-
const reNoRow = new RegExp(`see\\s+${TABLE}\\s+no\\s+row\\s+where\\b`, 'i');
|
|
22
|
-
const reCount = new RegExp(`see\\s+${TABLE}.*\\bcount\\s+is\\b`, 'i');
|
|
23
|
-
|
|
24
|
-
/** True when a step is a DB-verification step (used to wire the `db` import). */
|
|
25
|
-
export function isDbStep(text: string): boolean {
|
|
26
|
-
return reNoRow.test(text) || reRow.test(text) || reCount.test(text);
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
/** Render a value token (`{{var}}` | "literal" | 'literal' | number) as a JS expression. */
|
|
30
|
-
function valueExpr(token: string): string {
|
|
31
|
-
const t = token.trim();
|
|
32
|
-
const v = t.match(/^\{\{\s*([^}]+?)\s*\}\}$/);
|
|
33
|
-
if (v) return `testData.get(${JSON.stringify(v[1])})`;
|
|
34
|
-
const q = t.match(/^["'](.*)["']$/);
|
|
35
|
-
if (q) return JSON.stringify(q[1]);
|
|
36
|
-
if (/^-?\d+(?:\.\d+)?$/.test(t)) return t;
|
|
37
|
-
return JSON.stringify(t);
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
/** Parse a `[col] is VALUE [and [col2] is VALUE2]` segment into a JS object literal. */
|
|
41
|
-
function parseFilter(segment: string): string {
|
|
42
|
-
const re = new RegExp(`\\[([A-Za-z_][A-Za-z0-9_]*)\\]\\s+is\\s+(${VALUE})`, 'gi');
|
|
43
|
-
const parts: string[] = [];
|
|
44
|
-
let m: RegExpExecArray | null;
|
|
45
|
-
while ((m = re.exec(segment))) parts.push(`${JSON.stringify(m[1])}: ${valueExpr(m[2])}`);
|
|
46
|
-
return `{ ${parts.join(', ')} }`;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
/** Parse a `has [col] = VALUE [and [col2] = VALUE2]` segment into a JS object literal. */
|
|
50
|
-
function parseExpected(segment: string): string {
|
|
51
|
-
const re = new RegExp(`\\[([A-Za-z_][A-Za-z0-9_]*)\\]\\s*=\\s*(${VALUE})`, 'gi');
|
|
52
|
-
const parts: string[] = [];
|
|
53
|
-
let m: RegExpExecArray | null;
|
|
54
|
-
while ((m = re.exec(segment))) parts.push(`${JSON.stringify(m[1])}: ${valueExpr(m[2])}`);
|
|
55
|
-
return parts.length ? `{ ${parts.join(', ')} }` : '';
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
export const databasePatterns: StepPattern[] = [
|
|
59
|
-
{
|
|
60
|
-
name: 'db-no-row',
|
|
61
|
-
priority: 60, // above generic see-assertions
|
|
62
|
-
matcher: (step: ParsedStep) => reNoRow.test(step.text),
|
|
63
|
-
generator: (step: ParsedStep, _ctx: PatternContext): MappedStep => {
|
|
64
|
-
const m = step.text.match(new RegExp(`${TABLE}\\s+no\\s+row\\s+where\\s+(.+)$`, 'i'))!;
|
|
65
|
-
const table = m[1];
|
|
66
|
-
const filter = parseFilter(m[2]);
|
|
67
|
-
return { code: `await db.assertNoRow(${JSON.stringify(table)}, ${filter});`, comment: `DB: no row in ${table}` };
|
|
68
|
-
},
|
|
69
|
-
},
|
|
70
|
-
{
|
|
71
|
-
name: 'db-count',
|
|
72
|
-
priority: 60,
|
|
73
|
-
matcher: (step: ParsedStep) => reCount.test(step.text) && !reRow.test(step.text),
|
|
74
|
-
generator: (step: ParsedStep, _ctx: PatternContext): MappedStep => {
|
|
75
|
-
const m = step.text.match(new RegExp(`${TABLE}(?:\\s+where\\s+(.+?))?\\s+count\\s+is\\s+(${VALUE})`, 'i'))!;
|
|
76
|
-
const table = m[1];
|
|
77
|
-
const filter = m[2] ? parseFilter(m[2]) : '{}';
|
|
78
|
-
return { code: `await db.assertCount(${JSON.stringify(table)}, ${filter}, Number(${valueExpr(m[3])}));`, comment: `DB: count rows in ${table}` };
|
|
79
|
-
},
|
|
80
|
-
},
|
|
81
|
-
{
|
|
82
|
-
name: 'db-row',
|
|
83
|
-
priority: 60,
|
|
84
|
-
matcher: (step: ParsedStep) => reRow.test(step.text),
|
|
85
|
-
generator: (step: ParsedStep, _ctx: PatternContext): MappedStep => {
|
|
86
|
-
// [table] row where <filter> [has <expected>]
|
|
87
|
-
const m = step.text.match(new RegExp(`${TABLE}\\s+row\\s+where\\s+(.+?)(?:\\s+has\\s+(.+))?$`, 'i'))!;
|
|
88
|
-
const table = m[1];
|
|
89
|
-
const filter = parseFilter(m[2]);
|
|
90
|
-
const expected = m[3] ? parseExpected(m[3]) : '';
|
|
91
|
-
const args = expected ? `${JSON.stringify(table)}, ${filter}, ${expected}` : `${JSON.stringify(table)}, ${filter}`;
|
|
92
|
-
return { code: `await db.assertRow(${args});`, comment: `DB: row in ${table}` };
|
|
93
|
-
},
|
|
94
|
-
},
|
|
95
|
-
];
|
|
@@ -1,167 +0,0 @@
|
|
|
1
|
-
import { ParsedStep } from '../../gherkin-parser';
|
|
2
|
-
import { StepPattern, StepTemplateData } from './types';
|
|
3
|
-
|
|
4
|
-
/**
|
|
5
|
-
* Form action patterns: fill, type, select, check, uncheck
|
|
6
|
-
*/
|
|
7
|
-
export const formPatterns: StepPattern[] = [
|
|
8
|
-
{
|
|
9
|
-
name: 'upload-file',
|
|
10
|
-
matcher: (step: ParsedStep) =>
|
|
11
|
-
(step.text.includes('fill') || step.text.includes('fills')) &&
|
|
12
|
-
step.elementType === 'uploader' &&
|
|
13
|
-
!!step.selectorRef &&
|
|
14
|
-
!!(step.dataRef || step.value),
|
|
15
|
-
resolver: (step, context) => {
|
|
16
|
-
const resolved = context.selectorResolver.resolveSelector(step.selectorRef!, undefined, step.elementType, step.nth);
|
|
17
|
-
|
|
18
|
-
let fileName: string;
|
|
19
|
-
if (step.dataRef) {
|
|
20
|
-
try {
|
|
21
|
-
fileName = context.dataResolver.resolveData(step.dataRef, context.featureName);
|
|
22
|
-
} catch (error) {
|
|
23
|
-
fileName = `\${${step.dataRef}}`;
|
|
24
|
-
}
|
|
25
|
-
} else {
|
|
26
|
-
fileName = step.value!;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
return {
|
|
30
|
-
templateName: 'upload-action',
|
|
31
|
-
data: { ...resolved, selectorRef: step.selectorRef, fileName },
|
|
32
|
-
comment: `Upload file ${fileName} via ${step.selectorRef}`,
|
|
33
|
-
};
|
|
34
|
-
},
|
|
35
|
-
priority: 12,
|
|
36
|
-
},
|
|
37
|
-
{
|
|
38
|
-
name: 'fill-input',
|
|
39
|
-
matcher: (step: ParsedStep) =>
|
|
40
|
-
(step.text.includes('fills') || step.text.includes('fill') || step.text.includes('inputs') || step.text.includes('input')) && !!step.selectorRef && !!(step.dataRef || step.value),
|
|
41
|
-
resolver: (step, context) => {
|
|
42
|
-
const resolved = context.selectorResolver.resolveSelector(step.selectorRef!, undefined, step.elementType, step.nth);
|
|
43
|
-
|
|
44
|
-
let value: string;
|
|
45
|
-
if (step.dataRef) {
|
|
46
|
-
try {
|
|
47
|
-
value = context.dataResolver.resolveData(step.dataRef, context.featureName);
|
|
48
|
-
} catch (error) {
|
|
49
|
-
value = `\${${step.dataRef}}`;
|
|
50
|
-
}
|
|
51
|
-
} else {
|
|
52
|
-
value = step.value!;
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
// Use pressSequentially template for contenteditable/rich-text editors (auto-detected by live-scan)
|
|
56
|
-
// or when Gherkin explicitly uses 'editor' element type
|
|
57
|
-
const isEditor = resolved.inputMethod === 'pressSequentially' || step.elementType === 'editor';
|
|
58
|
-
const templateName = isEditor ? 'fill-editor-action' : 'fill-action';
|
|
59
|
-
|
|
60
|
-
return {
|
|
61
|
-
templateName,
|
|
62
|
-
data: { ...resolved, selectorRef: step.selectorRef, fillValue: value },
|
|
63
|
-
comment: isEditor
|
|
64
|
-
? `Fill rich text editor ${step.selectorRef} with ${step.dataRef || step.value}`
|
|
65
|
-
: `Fill ${step.selectorRef} with ${step.dataRef || step.value}`,
|
|
66
|
-
};
|
|
67
|
-
},
|
|
68
|
-
priority: 10,
|
|
69
|
-
},
|
|
70
|
-
{
|
|
71
|
-
name: 'check-checkbox',
|
|
72
|
-
matcher: (step: ParsedStep) =>
|
|
73
|
-
(step.text.includes('checks') || step.text.match(/\bcheck\b/)) && !!step.selectorRef,
|
|
74
|
-
resolver: (step, context) => {
|
|
75
|
-
const resolved = context.selectorResolver.resolveSelector(step.selectorRef!, undefined, step.elementType, step.nth);
|
|
76
|
-
return {
|
|
77
|
-
templateName: 'check-action',
|
|
78
|
-
data: { ...resolved, selectorRef: step.selectorRef },
|
|
79
|
-
comment: `Check ${step.selectorRef}`,
|
|
80
|
-
};
|
|
81
|
-
},
|
|
82
|
-
priority: 8,
|
|
83
|
-
},
|
|
84
|
-
{
|
|
85
|
-
name: 'uncheck-checkbox',
|
|
86
|
-
matcher: (step: ParsedStep) =>
|
|
87
|
-
(step.text.includes('unchecks') || step.text.match(/\buncheck\b/)) && !!step.selectorRef,
|
|
88
|
-
resolver: (step, context) => {
|
|
89
|
-
const resolved = context.selectorResolver.resolveSelector(step.selectorRef!, undefined, step.elementType, step.nth);
|
|
90
|
-
return {
|
|
91
|
-
templateName: 'uncheck-action',
|
|
92
|
-
data: { ...resolved, selectorRef: step.selectorRef },
|
|
93
|
-
comment: `Uncheck ${step.selectorRef}`,
|
|
94
|
-
};
|
|
95
|
-
},
|
|
96
|
-
priority: 8,
|
|
97
|
-
},
|
|
98
|
-
{
|
|
99
|
-
name: 'select-dropdown',
|
|
100
|
-
matcher: (step: ParsedStep) =>
|
|
101
|
-
(step.text.includes('selects') || step.text.match(/\bselect\b/)) &&
|
|
102
|
-
!!step.selectorRef &&
|
|
103
|
-
!!(step.dataRef || step.value),
|
|
104
|
-
resolver: (step, context): StepTemplateData => {
|
|
105
|
-
const resolved = context.selectorResolver.resolveSelector(step.selectorRef!, undefined, step.elementType, step.nth);
|
|
106
|
-
|
|
107
|
-
let value: string;
|
|
108
|
-
if (step.dataRef) {
|
|
109
|
-
try {
|
|
110
|
-
value = context.dataResolver.resolveData(step.dataRef, context.featureName);
|
|
111
|
-
} catch (error) {
|
|
112
|
-
value = `\${${step.dataRef}}`;
|
|
113
|
-
}
|
|
114
|
-
} else {
|
|
115
|
-
value = step.value!;
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
const isRadio = resolved.role === 'radio' || step.text.includes('radio');
|
|
119
|
-
const isCheckbox = resolved.role === 'checkbox' || step.text.includes('checkbox');
|
|
120
|
-
|
|
121
|
-
let templateName: string;
|
|
122
|
-
let data: Record<string, any>;
|
|
123
|
-
|
|
124
|
-
if (isRadio) {
|
|
125
|
-
templateName = 'radio-select-action';
|
|
126
|
-
data = { ...resolved, selectorRef: step.selectorRef, selectValue: value };
|
|
127
|
-
} else if (isCheckbox) {
|
|
128
|
-
templateName = 'check-action';
|
|
129
|
-
data = { ...resolved, selectorRef: step.selectorRef };
|
|
130
|
-
} else {
|
|
131
|
-
const nativeSelectRoles = ['combobox', 'listbox', 'select'];
|
|
132
|
-
// A native <select> is detected by its ARIA role, or by an explicit `variant: native`
|
|
133
|
-
// on the selector entry (lets a CSS/#id-located select opt into .selectOption()).
|
|
134
|
-
const isNativeSelect = nativeSelectRoles.includes(resolved.role) || resolved.variant === 'native';
|
|
135
|
-
|
|
136
|
-
if (isNativeSelect) {
|
|
137
|
-
templateName = 'select-action';
|
|
138
|
-
data = { ...resolved, selectorRef: step.selectorRef, selectValue: value };
|
|
139
|
-
} else {
|
|
140
|
-
templateName = 'click-select-action';
|
|
141
|
-
data = { ...resolved, selectorRef: step.selectorRef, selectValue: value };
|
|
142
|
-
}
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
return {
|
|
146
|
-
templateName,
|
|
147
|
-
data,
|
|
148
|
-
comment: `Select ${step.dataRef || step.value} in ${step.selectorRef}`,
|
|
149
|
-
};
|
|
150
|
-
},
|
|
151
|
-
priority: 8,
|
|
152
|
-
},
|
|
153
|
-
{
|
|
154
|
-
name: 'clear-input',
|
|
155
|
-
matcher: (step: ParsedStep) =>
|
|
156
|
-
(step.text.includes('clears') || step.text.match(/\bclear\b/)) && !!step.selectorRef,
|
|
157
|
-
resolver: (step, context) => {
|
|
158
|
-
const resolved = context.selectorResolver.resolveSelector(step.selectorRef!, undefined, step.elementType, step.nth);
|
|
159
|
-
return {
|
|
160
|
-
templateName: 'clear-action',
|
|
161
|
-
data: { ...resolved, selectorRef: step.selectorRef },
|
|
162
|
-
comment: `Clear ${step.selectorRef}`,
|
|
163
|
-
};
|
|
164
|
-
},
|
|
165
|
-
priority: 7,
|
|
166
|
-
},
|
|
167
|
-
];
|