@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.
Files changed (228) hide show
  1. package/README.md +4 -428
  2. package/dist/capabilities/builtins.d.ts +31 -0
  3. package/dist/capabilities/builtins.d.ts.map +1 -0
  4. package/dist/capabilities/builtins.js +84 -0
  5. package/dist/capabilities/builtins.js.map +1 -0
  6. package/dist/capabilities/context-router.d.ts +34 -0
  7. package/dist/capabilities/context-router.d.ts.map +1 -0
  8. package/dist/capabilities/context-router.js +49 -0
  9. package/dist/capabilities/context-router.js.map +1 -0
  10. package/dist/capabilities/context.d.ts +51 -0
  11. package/dist/capabilities/context.d.ts.map +1 -0
  12. package/dist/capabilities/context.js +17 -0
  13. package/dist/capabilities/context.js.map +1 -0
  14. package/dist/capabilities/discover.d.ts +2 -0
  15. package/dist/capabilities/discover.d.ts.map +1 -0
  16. package/dist/capabilities/discover.js +48 -0
  17. package/dist/capabilities/discover.js.map +1 -0
  18. package/dist/capabilities/registry.d.ts +90 -0
  19. package/dist/capabilities/registry.d.ts.map +1 -0
  20. package/dist/capabilities/registry.js +43 -0
  21. package/dist/capabilities/registry.js.map +1 -0
  22. package/dist/capabilities/sensor.d.ts +49 -0
  23. package/dist/capabilities/sensor.d.ts.map +1 -0
  24. package/dist/capabilities/sensor.js +3 -0
  25. package/dist/capabilities/sensor.js.map +1 -0
  26. package/dist/cli/commands/challenge.d.ts.map +1 -1
  27. package/dist/cli/commands/challenge.js +9 -2
  28. package/dist/cli/commands/challenge.js.map +1 -1
  29. package/dist/cli/commands/delivery.d.ts.map +1 -1
  30. package/dist/cli/commands/delivery.js +3 -2
  31. package/dist/cli/commands/delivery.js.map +1 -1
  32. package/dist/cli/commands/generate.d.ts.map +1 -1
  33. package/dist/cli/commands/generate.js +12 -0
  34. package/dist/cli/commands/generate.js.map +1 -1
  35. package/dist/cli/index.js +10 -1
  36. package/dist/cli/index.js.map +1 -1
  37. package/dist/exporters/csv-exporter.d.ts.map +1 -1
  38. package/dist/exporters/csv-exporter.js +92 -76
  39. package/dist/exporters/csv-exporter.js.map +1 -1
  40. package/dist/exporters/spec-parser.d.ts.map +1 -1
  41. package/dist/exporters/spec-parser.js +6 -1
  42. package/dist/exporters/spec-parser.js.map +1 -1
  43. package/dist/generators/test-generator/adapters/adapter-interface.d.ts +2 -0
  44. package/dist/generators/test-generator/adapters/adapter-interface.d.ts.map +1 -1
  45. package/dist/generators/test-generator/adapters/playwright/playwright-adapter.d.ts +1 -0
  46. package/dist/generators/test-generator/adapters/playwright/playwright-adapter.d.ts.map +1 -1
  47. package/dist/generators/test-generator/adapters/playwright/playwright-adapter.js.map +1 -1
  48. package/dist/generators/test-generator/adapters/playwright/templates/imports.hbs +3 -0
  49. package/dist/generators/test-generator/adapters/playwright/templates/scenario.hbs +19 -1
  50. package/dist/generators/test-generator/code-generator.d.ts +21 -4
  51. package/dist/generators/test-generator/code-generator.d.ts.map +1 -1
  52. package/dist/generators/test-generator/code-generator.js +118 -57
  53. package/dist/generators/test-generator/code-generator.js.map +1 -1
  54. package/dist/generators/test-generator/patterns/expect-patterns.d.ts +3 -0
  55. package/dist/generators/test-generator/patterns/expect-patterns.d.ts.map +1 -0
  56. package/dist/generators/test-generator/patterns/expect-patterns.js +54 -0
  57. package/dist/generators/test-generator/patterns/expect-patterns.js.map +1 -0
  58. package/dist/generators/test-generator/patterns/index.d.ts +0 -10
  59. package/dist/generators/test-generator/patterns/index.d.ts.map +1 -1
  60. package/dist/generators/test-generator/patterns/index.js +10 -45
  61. package/dist/generators/test-generator/patterns/index.js.map +1 -1
  62. package/dist/generators/test-generator/step-mapper.d.ts +6 -0
  63. package/dist/generators/test-generator/step-mapper.d.ts.map +1 -1
  64. package/dist/generators/test-generator/step-mapper.js +8 -0
  65. package/dist/generators/test-generator/step-mapper.js.map +1 -1
  66. package/dist/generators/test-generator/template-engine.d.ts +4 -0
  67. package/dist/generators/test-generator/template-engine.d.ts.map +1 -1
  68. package/dist/generators/test-generator/template-engine.js +1 -1
  69. package/dist/generators/test-generator/template-engine.js.map +1 -1
  70. package/dist/generators/test-generator/utils/runtime-data-transformer.d.ts +1 -1
  71. package/dist/generators/test-generator/utils/runtime-data-transformer.d.ts.map +1 -1
  72. package/dist/generators/test-generator/utils/runtime-data-transformer.js +5 -5
  73. package/dist/generators/test-generator/utils/runtime-data-transformer.js.map +1 -1
  74. package/dist/harness/annotation-overrides.d.ts +9 -0
  75. package/dist/harness/annotation-overrides.d.ts.map +1 -0
  76. package/dist/harness/annotation-overrides.js +36 -0
  77. package/dist/harness/annotation-overrides.js.map +1 -0
  78. package/dist/harness/audit.d.ts.map +1 -1
  79. package/dist/harness/audit.js +35 -7
  80. package/dist/harness/audit.js.map +1 -1
  81. package/dist/harness/catalog/drivers.yaml +35 -12
  82. package/dist/harness/challenge.d.ts +1 -0
  83. package/dist/harness/challenge.d.ts.map +1 -1
  84. package/dist/harness/challenge.js +49 -2
  85. package/dist/harness/challenge.js.map +1 -1
  86. package/dist/harness/data-driven-lint.d.ts +7 -0
  87. package/dist/harness/data-driven-lint.d.ts.map +1 -0
  88. package/dist/harness/data-driven-lint.js +153 -0
  89. package/dist/harness/data-driven-lint.js.map +1 -0
  90. package/dist/harness/parse.d.ts +3 -0
  91. package/dist/harness/parse.d.ts.map +1 -1
  92. package/dist/harness/parse.js +25 -0
  93. package/dist/harness/parse.js.map +1 -1
  94. package/dist/harness/query-catalog.d.ts +48 -0
  95. package/dist/harness/query-catalog.d.ts.map +1 -0
  96. package/dist/harness/query-catalog.js +0 -0
  97. package/dist/harness/query-catalog.js.map +1 -0
  98. package/dist/harness/script-check.d.ts.map +1 -1
  99. package/dist/harness/script-check.js +7 -4
  100. package/dist/harness/script-check.js.map +1 -1
  101. package/dist/index.d.ts +20 -0
  102. package/dist/index.d.ts.map +1 -0
  103. package/dist/index.js +32 -0
  104. package/dist/index.js.map +1 -0
  105. package/dist/orchestrator/templates/ai-instructions/claude-agent-challenge.md +3 -2
  106. package/dist/orchestrator/templates/ai-instructions/claude-cmd-create-test.md +4 -3
  107. package/dist/orchestrator/templates/ai-instructions/claude-skill-gherkin-syntax.md +41 -0
  108. package/dist/orchestrator/templates/ai-instructions/claude-skill-tc-generation.md +22 -0
  109. package/dist/orchestrator/templates/ai-instructions/claude-skill-tc-review.md +1 -0
  110. package/dist/orchestrator/templates/ai-instructions/claude-skill-test-design-techniques.md +6 -0
  111. package/dist/orchestrator/templates/ai-instructions/copilot-cmd-create-test.md +4 -3
  112. package/dist/orchestrator/templates/ai-instructions/github-skill-sungen-gherkin-syntax.md +41 -0
  113. package/dist/orchestrator/templates/ai-instructions/github-skill-sungen-tc-generation.md +22 -0
  114. package/dist/orchestrator/templates/ai-instructions/github-skill-sungen-tc-review.md +1 -0
  115. package/dist/orchestrator/templates/ai-instructions/github-skill-sungen-test-design-techniques.md +6 -0
  116. package/dist/orchestrator/templates/specs-api.d.ts +19 -0
  117. package/dist/orchestrator/templates/specs-api.d.ts.map +1 -0
  118. package/dist/orchestrator/templates/specs-api.js +128 -0
  119. package/dist/orchestrator/templates/specs-api.js.map +1 -0
  120. package/dist/orchestrator/templates/specs-api.ts +101 -0
  121. package/dist/orchestrator/templates/specs-db.d.ts +8 -0
  122. package/dist/orchestrator/templates/specs-db.d.ts.map +1 -1
  123. package/dist/orchestrator/templates/specs-db.js +22 -0
  124. package/dist/orchestrator/templates/specs-db.js.map +1 -1
  125. package/dist/orchestrator/templates/specs-db.ts +22 -0
  126. package/dist/orchestrator/templates/specs-test-data.ts +76 -15
  127. package/package.json +7 -30
  128. package/src/capabilities/builtins.ts +85 -0
  129. package/src/capabilities/context-router.ts +66 -0
  130. package/src/capabilities/context.ts +46 -0
  131. package/src/capabilities/discover.ts +42 -0
  132. package/src/capabilities/registry.ts +111 -0
  133. package/src/capabilities/sensor.ts +47 -0
  134. package/src/cli/commands/challenge.ts +6 -2
  135. package/src/cli/commands/delivery.ts +3 -2
  136. package/src/cli/commands/generate.ts +12 -0
  137. package/src/cli/index.ts +10 -1
  138. package/src/exporters/csv-exporter.ts +22 -6
  139. package/src/exporters/spec-parser.ts +6 -1
  140. package/src/generators/test-generator/adapters/adapter-interface.ts +2 -1
  141. package/src/generators/test-generator/adapters/playwright/playwright-adapter.ts +1 -1
  142. package/src/generators/test-generator/adapters/playwright/templates/imports.hbs +3 -0
  143. package/src/generators/test-generator/adapters/playwright/templates/scenario.hbs +19 -1
  144. package/src/generators/test-generator/code-generator.ts +114 -59
  145. package/src/generators/test-generator/patterns/expect-patterns.ts +49 -0
  146. package/src/generators/test-generator/patterns/index.ts +9 -33
  147. package/src/generators/test-generator/step-mapper.ts +9 -0
  148. package/src/generators/test-generator/template-engine.ts +5 -2
  149. package/src/generators/test-generator/utils/runtime-data-transformer.ts +5 -5
  150. package/src/harness/annotation-overrides.ts +25 -0
  151. package/src/harness/audit.ts +37 -8
  152. package/src/harness/catalog/drivers.yaml +35 -12
  153. package/src/harness/challenge.ts +47 -2
  154. package/src/harness/data-driven-lint.ts +119 -0
  155. package/src/harness/parse.ts +17 -0
  156. package/src/harness/query-catalog.ts +0 -0
  157. package/src/harness/script-check.ts +8 -5
  158. package/src/index.ts +30 -0
  159. package/src/orchestrator/templates/ai-instructions/claude-agent-challenge.md +3 -2
  160. package/src/orchestrator/templates/ai-instructions/claude-cmd-create-test.md +4 -3
  161. package/src/orchestrator/templates/ai-instructions/claude-skill-gherkin-syntax.md +41 -0
  162. package/src/orchestrator/templates/ai-instructions/claude-skill-tc-generation.md +22 -0
  163. package/src/orchestrator/templates/ai-instructions/claude-skill-tc-review.md +1 -0
  164. package/src/orchestrator/templates/ai-instructions/claude-skill-test-design-techniques.md +6 -0
  165. package/src/orchestrator/templates/ai-instructions/copilot-cmd-create-test.md +4 -3
  166. package/src/orchestrator/templates/ai-instructions/github-skill-sungen-gherkin-syntax.md +41 -0
  167. package/src/orchestrator/templates/ai-instructions/github-skill-sungen-tc-generation.md +22 -0
  168. package/src/orchestrator/templates/ai-instructions/github-skill-sungen-tc-review.md +1 -0
  169. package/src/orchestrator/templates/ai-instructions/github-skill-sungen-test-design-techniques.md +6 -0
  170. package/src/orchestrator/templates/specs-api.ts +101 -0
  171. package/src/orchestrator/templates/specs-db.ts +22 -0
  172. package/src/orchestrator/templates/specs-test-data.ts +76 -15
  173. package/dist/generators/test-generator/patterns/assertion-patterns.d.ts +0 -7
  174. package/dist/generators/test-generator/patterns/assertion-patterns.d.ts.map +0 -1
  175. package/dist/generators/test-generator/patterns/assertion-patterns.js +0 -626
  176. package/dist/generators/test-generator/patterns/assertion-patterns.js.map +0 -1
  177. package/dist/generators/test-generator/patterns/capture-patterns.d.ts +0 -21
  178. package/dist/generators/test-generator/patterns/capture-patterns.d.ts.map +0 -1
  179. package/dist/generators/test-generator/patterns/capture-patterns.js +0 -87
  180. package/dist/generators/test-generator/patterns/capture-patterns.js.map +0 -1
  181. package/dist/generators/test-generator/patterns/database-patterns.d.ts +0 -5
  182. package/dist/generators/test-generator/patterns/database-patterns.d.ts.map +0 -1
  183. package/dist/generators/test-generator/patterns/database-patterns.js +0 -94
  184. package/dist/generators/test-generator/patterns/database-patterns.js.map +0 -1
  185. package/dist/generators/test-generator/patterns/form-patterns.d.ts +0 -6
  186. package/dist/generators/test-generator/patterns/form-patterns.d.ts.map +0 -1
  187. package/dist/generators/test-generator/patterns/form-patterns.js +0 -160
  188. package/dist/generators/test-generator/patterns/form-patterns.js.map +0 -1
  189. package/dist/generators/test-generator/patterns/interaction-patterns.d.ts +0 -6
  190. package/dist/generators/test-generator/patterns/interaction-patterns.d.ts.map +0 -1
  191. package/dist/generators/test-generator/patterns/interaction-patterns.js +0 -433
  192. package/dist/generators/test-generator/patterns/interaction-patterns.js.map +0 -1
  193. package/dist/generators/test-generator/patterns/keyboard-patterns.d.ts +0 -7
  194. package/dist/generators/test-generator/patterns/keyboard-patterns.d.ts.map +0 -1
  195. package/dist/generators/test-generator/patterns/keyboard-patterns.js +0 -47
  196. package/dist/generators/test-generator/patterns/keyboard-patterns.js.map +0 -1
  197. package/dist/generators/test-generator/patterns/navigation-patterns.d.ts +0 -6
  198. package/dist/generators/test-generator/patterns/navigation-patterns.d.ts.map +0 -1
  199. package/dist/generators/test-generator/patterns/navigation-patterns.js +0 -125
  200. package/dist/generators/test-generator/patterns/navigation-patterns.js.map +0 -1
  201. package/dist/generators/test-generator/patterns/scope-patterns.d.ts +0 -7
  202. package/dist/generators/test-generator/patterns/scope-patterns.d.ts.map +0 -1
  203. package/dist/generators/test-generator/patterns/scope-patterns.js +0 -36
  204. package/dist/generators/test-generator/patterns/scope-patterns.js.map +0 -1
  205. package/dist/generators/test-generator/patterns/scroll-patterns.d.ts +0 -7
  206. package/dist/generators/test-generator/patterns/scroll-patterns.d.ts.map +0 -1
  207. package/dist/generators/test-generator/patterns/scroll-patterns.js +0 -25
  208. package/dist/generators/test-generator/patterns/scroll-patterns.js.map +0 -1
  209. package/dist/generators/test-generator/patterns/setup-patterns.d.ts +0 -6
  210. package/dist/generators/test-generator/patterns/setup-patterns.d.ts.map +0 -1
  211. package/dist/generators/test-generator/patterns/setup-patterns.js +0 -72
  212. package/dist/generators/test-generator/patterns/setup-patterns.js.map +0 -1
  213. package/dist/generators/test-generator/patterns/table-patterns.d.ts +0 -19
  214. package/dist/generators/test-generator/patterns/table-patterns.d.ts.map +0 -1
  215. package/dist/generators/test-generator/patterns/table-patterns.js +0 -239
  216. package/dist/generators/test-generator/patterns/table-patterns.js.map +0 -1
  217. package/docs/orchestration-spec.md +0 -267
  218. package/src/generators/test-generator/patterns/assertion-patterns.ts +0 -691
  219. package/src/generators/test-generator/patterns/capture-patterns.ts +0 -97
  220. package/src/generators/test-generator/patterns/database-patterns.ts +0 -95
  221. package/src/generators/test-generator/patterns/form-patterns.ts +0 -167
  222. package/src/generators/test-generator/patterns/interaction-patterns.ts +0 -465
  223. package/src/generators/test-generator/patterns/keyboard-patterns.ts +0 -51
  224. package/src/generators/test-generator/patterns/navigation-patterns.ts +0 -140
  225. package/src/generators/test-generator/patterns/scope-patterns.ts +0 -40
  226. package/src/generators/test-generator/patterns/scroll-patterns.ts +0 -27
  227. package/src/generators/test-generator/patterns/setup-patterns.ts +0 -76
  228. 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
- ];