@sun-asterisk/sungen 1.0.24 → 2.0.1
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 +198 -74
- package/dist/cli/commands/add.d.ts.map +1 -1
- package/dist/cli/commands/add.js +5 -3
- package/dist/cli/commands/add.js.map +1 -1
- package/dist/cli/commands/generate.d.ts.map +1 -1
- package/dist/cli/commands/generate.js +110 -35
- package/dist/cli/commands/generate.js.map +1 -1
- package/dist/cli/index.d.ts +2 -2
- package/dist/cli/index.js +5 -16
- package/dist/cli/index.js.map +1 -1
- package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/frame-enter-action.hbs +1 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/frame-exit-action.hbs +1 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/keyboard-global-action.hbs +1 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/scroll-action.hbs +1 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/actions/table-action-in-row.hbs +2 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-cell-by-filter.hbs +3 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-cell-by-index.hbs +3 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-column-exists.hbs +2 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-empty.hbs +2 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-count.hbs +2 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-exists.hbs +2 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-not-exists.hbs +2 -0
- package/dist/generators/test-generator/adapters/playwright/templates/steps/assertions/visible-with-value-assertion.hbs +1 -1
- package/dist/generators/test-generator/adapters/playwright/templates/steps/partials/locator-base.hbs +11 -2
- package/dist/generators/test-generator/adapters/playwright/templates/steps/partials/locator.hbs +11 -2
- package/dist/generators/test-generator/code-generator.d.ts +0 -1
- package/dist/generators/test-generator/code-generator.d.ts.map +1 -1
- package/dist/generators/test-generator/code-generator.js +10 -47
- package/dist/generators/test-generator/code-generator.js.map +1 -1
- package/dist/generators/test-generator/patterns/assertion-patterns.d.ts.map +1 -1
- package/dist/generators/test-generator/patterns/assertion-patterns.js +2 -0
- package/dist/generators/test-generator/patterns/assertion-patterns.js.map +1 -1
- package/dist/generators/test-generator/patterns/index.d.ts +4 -1
- package/dist/generators/test-generator/patterns/index.d.ts.map +1 -1
- package/dist/generators/test-generator/patterns/index.js +17 -5
- package/dist/generators/test-generator/patterns/index.js.map +1 -1
- package/dist/generators/test-generator/patterns/interaction-patterns.js +1 -1
- package/dist/generators/test-generator/patterns/interaction-patterns.js.map +1 -1
- package/dist/generators/test-generator/patterns/keyboard-patterns.d.ts +7 -0
- package/dist/generators/test-generator/patterns/keyboard-patterns.d.ts.map +1 -0
- package/dist/generators/test-generator/patterns/keyboard-patterns.js +47 -0
- package/dist/generators/test-generator/patterns/keyboard-patterns.js.map +1 -0
- package/dist/generators/test-generator/patterns/scope-patterns.d.ts +7 -0
- package/dist/generators/test-generator/patterns/scope-patterns.d.ts.map +1 -0
- package/dist/generators/test-generator/patterns/scope-patterns.js +36 -0
- package/dist/generators/test-generator/patterns/scope-patterns.js.map +1 -0
- package/dist/generators/test-generator/patterns/scroll-patterns.d.ts +7 -0
- package/dist/generators/test-generator/patterns/scroll-patterns.d.ts.map +1 -0
- package/dist/generators/test-generator/patterns/scroll-patterns.js +25 -0
- package/dist/generators/test-generator/patterns/scroll-patterns.js.map +1 -0
- package/dist/generators/test-generator/patterns/table-patterns.d.ts +7 -0
- package/dist/generators/test-generator/patterns/table-patterns.d.ts.map +1 -0
- package/dist/generators/test-generator/patterns/table-patterns.js +192 -0
- package/dist/generators/test-generator/patterns/table-patterns.js.map +1 -0
- package/dist/generators/test-generator/step-mapper.d.ts +5 -3
- package/dist/generators/test-generator/step-mapper.d.ts.map +1 -1
- package/dist/generators/test-generator/step-mapper.js +38 -27
- package/dist/generators/test-generator/step-mapper.js.map +1 -1
- package/dist/generators/test-generator/template-engine.d.ts.map +1 -1
- package/dist/generators/test-generator/template-engine.js +4 -1
- package/dist/generators/test-generator/template-engine.js.map +1 -1
- package/dist/generators/test-generator/types.d.ts +7 -24
- package/dist/generators/test-generator/types.d.ts.map +1 -1
- package/dist/generators/test-generator/types.js +2 -101
- package/dist/generators/test-generator/types.js.map +1 -1
- package/dist/generators/test-generator/utils/selector-resolver.d.ts +14 -0
- package/dist/generators/test-generator/utils/selector-resolver.d.ts.map +1 -1
- package/dist/generators/test-generator/utils/selector-resolver.js +37 -11
- package/dist/generators/test-generator/utils/selector-resolver.js.map +1 -1
- package/dist/orchestrator/project-initializer.d.ts +12 -0
- package/dist/orchestrator/project-initializer.d.ts.map +1 -1
- package/dist/orchestrator/project-initializer.js +72 -160
- package/dist/orchestrator/project-initializer.js.map +1 -1
- package/dist/orchestrator/screen-manager.d.ts +1 -32
- package/dist/orchestrator/screen-manager.d.ts.map +1 -1
- package/dist/orchestrator/screen-manager.js +55 -216
- package/dist/orchestrator/screen-manager.js.map +1 -1
- package/dist/orchestrator/templates/ai-rules.md +189 -0
- package/dist/orchestrator/templates/gitignore +16 -0
- package/dist/orchestrator/templates/playwright.config.d.ts +10 -0
- package/dist/orchestrator/templates/playwright.config.d.ts.map +1 -0
- package/dist/orchestrator/templates/playwright.config.js +77 -0
- package/dist/orchestrator/templates/playwright.config.js.map +1 -0
- package/dist/orchestrator/templates/playwright.config.ts +80 -0
- package/dist/orchestrator/templates/readme.md +197 -0
- package/dist/utils/selector-types.d.ts +1 -1
- package/dist/utils/selector-types.d.ts.map +1 -1
- package/dist/utils/selector-types.js +3 -0
- package/dist/utils/selector-types.js.map +1 -1
- package/docs/gherkin standards/gherkin-core-standard.md +377 -0
- package/docs/gherkin standards/gherkin-core-standard.vi.md +303 -0
- package/docs/gherkin-dictionary.md +1071 -0
- package/docs/makeauth.md +225 -0
- package/package.json +4 -3
- package/src/cli/commands/add.ts +5 -3
- package/src/cli/commands/generate.ts +90 -38
- package/src/cli/index.ts +5 -16
- package/src/generators/test-generator/adapters/playwright/templates/steps/actions/frame-enter-action.hbs +1 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/actions/frame-exit-action.hbs +1 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/actions/keyboard-global-action.hbs +1 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/actions/scroll-action.hbs +1 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/actions/table-action-in-row.hbs +2 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-cell-by-filter.hbs +3 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-cell-by-index.hbs +3 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-column-exists.hbs +2 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-empty.hbs +2 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-count.hbs +2 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-exists.hbs +2 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/table-row-not-exists.hbs +2 -0
- package/src/generators/test-generator/adapters/playwright/templates/steps/assertions/visible-with-value-assertion.hbs +1 -1
- package/src/generators/test-generator/adapters/playwright/templates/steps/partials/locator-base.hbs +11 -2
- package/src/generators/test-generator/adapters/playwright/templates/steps/partials/locator.hbs +11 -2
- package/src/generators/test-generator/code-generator.ts +11 -59
- package/src/generators/test-generator/patterns/assertion-patterns.ts +2 -0
- package/src/generators/test-generator/patterns/index.ts +12 -3
- package/src/generators/test-generator/patterns/interaction-patterns.ts +1 -1
- package/src/generators/test-generator/patterns/keyboard-patterns.ts +51 -0
- package/src/generators/test-generator/patterns/scope-patterns.ts +40 -0
- package/src/generators/test-generator/patterns/scroll-patterns.ts +27 -0
- package/src/generators/test-generator/patterns/table-patterns.ts +232 -0
- package/src/generators/test-generator/step-mapper.ts +42 -27
- package/src/generators/test-generator/template-engine.ts +3 -1
- package/src/generators/test-generator/types.ts +7 -112
- package/src/generators/test-generator/utils/selector-resolver.ts +70 -25
- package/src/orchestrator/project-initializer.ts +75 -160
- package/src/orchestrator/screen-manager.ts +61 -233
- package/src/orchestrator/templates/ai-rules.md +189 -0
- package/src/orchestrator/templates/gitignore +16 -0
- package/src/orchestrator/templates/playwright.config.ts +80 -0
- package/src/orchestrator/templates/readme.md +197 -0
- package/src/utils/selector-types.ts +3 -0
- package/dist/cli/commands/cache-clear.d.ts +0 -3
- package/dist/cli/commands/cache-clear.d.ts.map +0 -1
- package/dist/cli/commands/cache-clear.js +0 -24
- package/dist/cli/commands/cache-clear.js.map +0 -1
- package/dist/cli/commands/full.d.ts +0 -3
- package/dist/cli/commands/full.d.ts.map +0 -1
- package/dist/cli/commands/full.js +0 -37
- package/dist/cli/commands/full.js.map +0 -1
- package/dist/cli/commands/live-scan.d.ts +0 -3
- package/dist/cli/commands/live-scan.d.ts.map +0 -1
- package/dist/cli/commands/live-scan.js +0 -78
- package/dist/cli/commands/live-scan.js.map +0 -1
- package/dist/cli/commands/map.d.ts +0 -3
- package/dist/cli/commands/map.d.ts.map +0 -1
- package/dist/cli/commands/map.js +0 -93
- package/dist/cli/commands/map.js.map +0 -1
- package/dist/cli/commands/validate.d.ts +0 -3
- package/dist/cli/commands/validate.d.ts.map +0 -1
- package/dist/cli/commands/validate.js +0 -43
- package/dist/cli/commands/validate.js.map +0 -1
- package/dist/cli/utils.d.ts +0 -6
- package/dist/cli/utils.d.ts.map +0 -1
- package/dist/cli/utils.js +0 -101
- package/dist/cli/utils.js.map +0 -1
- package/dist/config/config-loader.d.ts +0 -51
- package/dist/config/config-loader.d.ts.map +0 -1
- package/dist/config/config-loader.js +0 -216
- package/dist/config/config-loader.js.map +0 -1
- package/dist/config/config-schema.d.ts +0 -121
- package/dist/config/config-schema.d.ts.map +0 -1
- package/dist/config/config-schema.js +0 -7
- package/dist/config/config-schema.js.map +0 -1
- package/dist/core/live-scanner/config-reader.d.ts +0 -10
- package/dist/core/live-scanner/config-reader.d.ts.map +0 -1
- package/dist/core/live-scanner/config-reader.js +0 -87
- package/dist/core/live-scanner/config-reader.js.map +0 -1
- package/dist/core/live-scanner/element-finder.d.ts +0 -20
- package/dist/core/live-scanner/element-finder.d.ts.map +0 -1
- package/dist/core/live-scanner/element-finder.js +0 -481
- package/dist/core/live-scanner/element-finder.js.map +0 -1
- package/dist/core/live-scanner/index.d.ts +0 -8
- package/dist/core/live-scanner/index.d.ts.map +0 -1
- package/dist/core/live-scanner/index.js +0 -33
- package/dist/core/live-scanner/index.js.map +0 -1
- package/dist/core/live-scanner/matrix-reader.d.ts +0 -17
- package/dist/core/live-scanner/matrix-reader.d.ts.map +0 -1
- package/dist/core/live-scanner/matrix-reader.js +0 -60
- package/dist/core/live-scanner/matrix-reader.js.map +0 -1
- package/dist/core/live-scanner/matrix-writer.d.ts +0 -7
- package/dist/core/live-scanner/matrix-writer.d.ts.map +0 -1
- package/dist/core/live-scanner/matrix-writer.js +0 -103
- package/dist/core/live-scanner/matrix-writer.js.map +0 -1
- package/dist/core/live-scanner/role-fallback.d.ts +0 -15
- package/dist/core/live-scanner/role-fallback.d.ts.map +0 -1
- package/dist/core/live-scanner/role-fallback.js +0 -46
- package/dist/core/live-scanner/role-fallback.js.map +0 -1
- package/dist/core/live-scanner/scanner.d.ts +0 -22
- package/dist/core/live-scanner/scanner.d.ts.map +0 -1
- package/dist/core/live-scanner/scanner.js +0 -303
- package/dist/core/live-scanner/scanner.js.map +0 -1
- package/dist/core/live-scanner/step-replayer.d.ts +0 -26
- package/dist/core/live-scanner/step-replayer.d.ts.map +0 -1
- package/dist/core/live-scanner/step-replayer.js +0 -473
- package/dist/core/live-scanner/step-replayer.js.map +0 -1
- package/dist/core/live-scanner/types.d.ts +0 -52
- package/dist/core/live-scanner/types.d.ts.map +0 -1
- package/dist/core/live-scanner/types.js +0 -14
- package/dist/core/live-scanner/types.js.map +0 -1
- package/dist/core/selector-base/annotation-handler.d.ts +0 -45
- package/dist/core/selector-base/annotation-handler.d.ts.map +0 -1
- package/dist/core/selector-base/annotation-handler.js +0 -102
- package/dist/core/selector-base/annotation-handler.js.map +0 -1
- package/dist/core/selector-base/base-generator.d.ts +0 -49
- package/dist/core/selector-base/base-generator.d.ts.map +0 -1
- package/dist/core/selector-base/base-generator.js +0 -214
- package/dist/core/selector-base/base-generator.js.map +0 -1
- package/dist/core/selector-base/gherkin-parser.d.ts +0 -24
- package/dist/core/selector-base/gherkin-parser.d.ts.map +0 -1
- package/dist/core/selector-base/gherkin-parser.js +0 -42
- package/dist/core/selector-base/gherkin-parser.js.map +0 -1
- package/dist/core/selector-mapper/priority-mapper.d.ts +0 -74
- package/dist/core/selector-mapper/priority-mapper.d.ts.map +0 -1
- package/dist/core/selector-mapper/priority-mapper.js +0 -477
- package/dist/core/selector-mapper/priority-mapper.js.map +0 -1
- package/dist/core/ui-scanner/heuristics/base-heuristic.d.ts +0 -91
- package/dist/core/ui-scanner/heuristics/base-heuristic.d.ts.map +0 -1
- package/dist/core/ui-scanner/heuristics/base-heuristic.js +0 -175
- package/dist/core/ui-scanner/heuristics/base-heuristic.js.map +0 -1
- package/dist/core/ui-scanner/react-scanner.d.ts +0 -32
- package/dist/core/ui-scanner/react-scanner.d.ts.map +0 -1
- package/dist/core/ui-scanner/react-scanner.js +0 -163
- package/dist/core/ui-scanner/react-scanner.js.map +0 -1
- package/dist/core/ui-scanner/scanner-interface.d.ts +0 -94
- package/dist/core/ui-scanner/scanner-interface.d.ts.map +0 -1
- package/dist/core/ui-scanner/scanner-interface.js +0 -33
- package/dist/core/ui-scanner/scanner-interface.js.map +0 -1
- package/dist/core/ui-scanner/strict-scanner.d.ts +0 -81
- package/dist/core/ui-scanner/strict-scanner.d.ts.map +0 -1
- package/dist/core/ui-scanner/strict-scanner.js +0 -511
- package/dist/core/ui-scanner/strict-scanner.js.map +0 -1
- package/dist/core/validator/data-validator.d.ts +0 -38
- package/dist/core/validator/data-validator.d.ts.map +0 -1
- package/dist/core/validator/data-validator.js +0 -212
- package/dist/core/validator/data-validator.js.map +0 -1
- package/dist/core/validator/feature-validator.d.ts +0 -27
- package/dist/core/validator/feature-validator.d.ts.map +0 -1
- package/dist/core/validator/feature-validator.js +0 -182
- package/dist/core/validator/feature-validator.js.map +0 -1
- package/dist/core/validator/index.d.ts +0 -46
- package/dist/core/validator/index.d.ts.map +0 -1
- package/dist/core/validator/index.js +0 -17
- package/dist/core/validator/index.js.map +0 -1
- package/dist/core/validator/screen-validator.d.ts +0 -35
- package/dist/core/validator/screen-validator.d.ts.map +0 -1
- package/dist/core/validator/screen-validator.js +0 -195
- package/dist/core/validator/screen-validator.js.map +0 -1
- package/dist/core/validator/selector-validator.d.ts +0 -36
- package/dist/core/validator/selector-validator.d.ts.map +0 -1
- package/dist/core/validator/selector-validator.js +0 -210
- package/dist/core/validator/selector-validator.js.map +0 -1
- package/dist/external/ai-provider.d.ts +0 -60
- package/dist/external/ai-provider.d.ts.map +0 -1
- package/dist/external/ai-provider.js +0 -30
- package/dist/external/ai-provider.js.map +0 -1
- package/dist/external/anthropic-provider.d.ts +0 -29
- package/dist/external/anthropic-provider.d.ts.map +0 -1
- package/dist/external/anthropic-provider.js +0 -85
- package/dist/external/anthropic-provider.js.map +0 -1
- package/dist/generators/cache/cache-manager.d.ts +0 -66
- package/dist/generators/cache/cache-manager.d.ts.map +0 -1
- package/dist/generators/cache/cache-manager.js +0 -286
- package/dist/generators/cache/cache-manager.js.map +0 -1
- package/dist/generators/dsl-writer/index.d.ts +0 -33
- package/dist/generators/dsl-writer/index.d.ts.map +0 -1
- package/dist/generators/dsl-writer/index.js +0 -226
- package/dist/generators/dsl-writer/index.js.map +0 -1
- package/dist/generators/scaffold-generator/index.d.ts +0 -162
- package/dist/generators/scaffold-generator/index.d.ts.map +0 -1
- package/dist/generators/scaffold-generator/index.js +0 -877
- package/dist/generators/scaffold-generator/index.js.map +0 -1
- package/dist/generators/selector-mapper/ai-mapper.d.ts +0 -56
- package/dist/generators/selector-mapper/ai-mapper.d.ts.map +0 -1
- package/dist/generators/selector-mapper/ai-mapper.js +0 -457
- package/dist/generators/selector-mapper/ai-mapper.js.map +0 -1
- package/dist/generators/selector-mapper/hybrid-mapper.d.ts +0 -67
- package/dist/generators/selector-mapper/hybrid-mapper.d.ts.map +0 -1
- package/dist/generators/selector-mapper/hybrid-mapper.js +0 -349
- package/dist/generators/selector-mapper/hybrid-mapper.js.map +0 -1
- package/dist/generators/selector-mapper/index.d.ts +0 -8
- package/dist/generators/selector-mapper/index.d.ts.map +0 -1
- package/dist/generators/selector-mapper/index.js +0 -12
- package/dist/generators/selector-mapper/index.js.map +0 -1
- package/dist/generators/selector-mapper/intelligent-mapper.d.ts +0 -125
- package/dist/generators/selector-mapper/intelligent-mapper.d.ts.map +0 -1
- package/dist/generators/selector-mapper/intelligent-mapper.js +0 -391
- package/dist/generators/selector-mapper/intelligent-mapper.js.map +0 -1
- package/dist/generators/test-generator/ai-step-mapper.d.ts +0 -27
- package/dist/generators/test-generator/ai-step-mapper.d.ts.map +0 -1
- package/dist/generators/test-generator/ai-step-mapper.js +0 -204
- package/dist/generators/test-generator/ai-step-mapper.js.map +0 -1
- package/dist/generators/test-generator/auth-setup-generator.d.ts +0 -18
- package/dist/generators/test-generator/auth-setup-generator.d.ts.map +0 -1
- package/dist/generators/test-generator/auth-setup-generator.js +0 -82
- package/dist/generators/test-generator/auth-setup-generator.js.map +0 -1
- package/dist/generators/test-generator/patterns/legacy-patterns.d.ts +0 -7
- package/dist/generators/test-generator/patterns/legacy-patterns.d.ts.map +0 -1
- package/dist/generators/test-generator/patterns/legacy-patterns.js +0 -98
- package/dist/generators/test-generator/patterns/legacy-patterns.js.map +0 -1
- package/dist/generators/test-generator/templates/auth-setup.ts.hbs +0 -36
- package/dist/generators/ui-model-builder/deep-scanner.d.ts +0 -121
- package/dist/generators/ui-model-builder/deep-scanner.d.ts.map +0 -1
- package/dist/generators/ui-model-builder/deep-scanner.js +0 -1113
- package/dist/generators/ui-model-builder/deep-scanner.js.map +0 -1
- package/dist/generators/ui-model-builder/enhanced-deep-scanner.d.ts +0 -110
- package/dist/generators/ui-model-builder/enhanced-deep-scanner.d.ts.map +0 -1
- package/dist/generators/ui-model-builder/enhanced-deep-scanner.js +0 -608
- package/dist/generators/ui-model-builder/enhanced-deep-scanner.js.map +0 -1
- package/dist/generators/ui-model-builder/react-scanner.d.ts +0 -107
- package/dist/generators/ui-model-builder/react-scanner.d.ts.map +0 -1
- package/dist/generators/ui-model-builder/react-scanner.js +0 -797
- package/dist/generators/ui-model-builder/react-scanner.js.map +0 -1
- package/dist/orchestrator/cache-manager.d.ts +0 -15
- package/dist/orchestrator/cache-manager.d.ts.map +0 -1
- package/dist/orchestrator/cache-manager.js +0 -62
- package/dist/orchestrator/cache-manager.js.map +0 -1
- package/dist/orchestrator/pipeline.d.ts +0 -56
- package/dist/orchestrator/pipeline.d.ts.map +0 -1
- package/dist/orchestrator/pipeline.js +0 -298
- package/dist/orchestrator/pipeline.js.map +0 -1
- package/dist/orchestrator/reporter.d.ts +0 -15
- package/dist/orchestrator/reporter.d.ts.map +0 -1
- package/dist/orchestrator/reporter.js +0 -30
- package/dist/orchestrator/reporter.js.map +0 -1
- package/src/cli/commands/cache-clear.ts +0 -22
- package/src/cli/commands/full.ts +0 -35
- package/src/cli/commands/live-scan.ts +0 -82
- package/src/cli/commands/map.ts +0 -97
- package/src/cli/commands/validate.ts +0 -43
- package/src/cli/utils.ts +0 -106
- package/src/config/ai-providers.yaml +0 -56
- package/src/config/config-loader.ts +0 -248
- package/src/config/config-schema.ts +0 -148
- package/src/config/default.config.yaml +0 -107
- package/src/config/framework.config.yaml +0 -52
- package/src/config/routes.yaml +0 -31
- package/src/core/live-scanner/config-reader.ts +0 -57
- package/src/core/live-scanner/element-finder.ts +0 -534
- package/src/core/live-scanner/index.ts +0 -7
- package/src/core/live-scanner/matrix-reader.ts +0 -65
- package/src/core/live-scanner/matrix-writer.ts +0 -77
- package/src/core/live-scanner/role-fallback.ts +0 -44
- package/src/core/live-scanner/scanner.ts +0 -321
- package/src/core/live-scanner/step-replayer.ts +0 -503
- package/src/core/live-scanner/types.ts +0 -58
- package/src/core/selector-base/annotation-handler.ts +0 -127
- package/src/core/selector-base/base-generator.ts +0 -234
- package/src/core/selector-base/gherkin-parser.ts +0 -57
- package/src/core/selector-mapper/priority-mapper.ts +0 -607
- package/src/core/ui-scanner/heuristics/base-heuristic.ts +0 -216
- package/src/core/ui-scanner/react-scanner.ts +0 -156
- package/src/core/ui-scanner/scanner-interface.ts +0 -133
- package/src/core/ui-scanner/strict-scanner.ts +0 -629
- package/src/core/validator/data-validator.ts +0 -202
- package/src/core/validator/feature-validator.ts +0 -176
- package/src/core/validator/index.ts +0 -57
- package/src/core/validator/screen-validator.ts +0 -209
- package/src/core/validator/selector-validator.ts +0 -209
- package/src/external/ai-provider.ts +0 -90
- package/src/external/anthropic-provider.ts +0 -114
- package/src/generators/README.md +0 -410
- package/src/generators/cache/cache-manager.ts +0 -322
- package/src/generators/dsl-writer/index.ts +0 -253
- package/src/generators/scaffold-generator/index.ts +0 -1029
- package/src/generators/selector-mapper/ai-mapper.ts +0 -528
- package/src/generators/selector-mapper/hybrid-mapper.ts +0 -427
- package/src/generators/selector-mapper/index.ts +0 -10
- package/src/generators/selector-mapper/intelligent-mapper.ts +0 -530
- package/src/generators/test-generator/ai-step-mapper.ts +0 -224
- package/src/generators/test-generator/auth-setup-generator.ts +0 -59
- package/src/generators/test-generator/patterns/legacy-patterns.ts +0 -104
- package/src/generators/test-generator/templates/auth-setup.ts.hbs +0 -36
- package/src/generators/ui-model-builder/deep-scanner.ts +0 -1244
- package/src/generators/ui-model-builder/enhanced-deep-scanner.ts +0 -731
- package/src/generators/ui-model-builder/react-scanner.ts +0 -959
- package/src/orchestrator/cache-manager.ts +0 -32
- package/src/orchestrator/pipeline.ts +0 -354
- package/src/orchestrator/reporter.ts +0 -36
|
@@ -9,6 +9,7 @@ import * as path from 'path';
|
|
|
9
9
|
export interface ScreenOptions {
|
|
10
10
|
name: string;
|
|
11
11
|
path?: string;
|
|
12
|
+
feature?: string;
|
|
12
13
|
description?: string;
|
|
13
14
|
}
|
|
14
15
|
|
|
@@ -25,20 +26,23 @@ export class ScreenManager {
|
|
|
25
26
|
* Add a new screen definition
|
|
26
27
|
*/
|
|
27
28
|
async addScreen(options: ScreenOptions): Promise<void> {
|
|
28
|
-
console.log(`📱 Adding screen: ${options.name}...\n`);
|
|
29
|
-
|
|
30
29
|
// Validate screen name
|
|
31
30
|
this.validateScreenName(options.name);
|
|
32
31
|
|
|
33
32
|
// Normalize screen name for directory
|
|
34
33
|
const screenName = this.normalizeScreenName(options.name);
|
|
35
34
|
const screenDir = path.join(this.screensDir, screenName);
|
|
36
|
-
|
|
37
|
-
// Derive filename
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
35
|
+
|
|
36
|
+
// Derive filename: --feature flag creates "<screen>-<feature>" name
|
|
37
|
+
let filename: string;
|
|
38
|
+
if (options.feature) {
|
|
39
|
+
filename = `${screenName}-${this.normalizeScreenName(options.feature)}`;
|
|
40
|
+
} else if (options.path) {
|
|
41
|
+
filename = this.extractFilenameFromPath(options.path);
|
|
42
|
+
} else {
|
|
43
|
+
filename = screenName;
|
|
44
|
+
}
|
|
45
|
+
|
|
42
46
|
// Create directory structure
|
|
43
47
|
const featuresDir = path.join(screenDir, 'features');
|
|
44
48
|
const selectorsDir = path.join(screenDir, 'selectors');
|
|
@@ -46,102 +50,92 @@ export class ScreenManager {
|
|
|
46
50
|
|
|
47
51
|
// File paths
|
|
48
52
|
const featurePath = path.join(featuresDir, `${filename}.feature`);
|
|
49
|
-
const
|
|
50
|
-
const
|
|
53
|
+
const selectorPath = path.join(selectorsDir, `${filename}.yaml`);
|
|
54
|
+
const testDataPath = path.join(testDataDir, `${filename}.yaml`);
|
|
51
55
|
|
|
52
56
|
// Smart behavior: Create dirs only if first file
|
|
53
57
|
const isFirstFile = !fs.existsSync(screenDir);
|
|
54
|
-
|
|
58
|
+
|
|
55
59
|
if (isFirstFile) {
|
|
56
|
-
console.log(
|
|
57
|
-
|
|
58
|
-
// Create directories
|
|
60
|
+
console.log(`Creating screen: ${options.name}\n`);
|
|
61
|
+
|
|
59
62
|
try {
|
|
60
63
|
if (!fs.existsSync(this.screensDir)) {
|
|
61
64
|
fs.mkdirSync(this.screensDir, { recursive: true });
|
|
62
|
-
console.log(`✓ Created directory: qa/screens/\n`);
|
|
63
65
|
}
|
|
64
66
|
fs.mkdirSync(featuresDir, { recursive: true });
|
|
65
67
|
fs.mkdirSync(selectorsDir, { recursive: true });
|
|
66
68
|
fs.mkdirSync(testDataDir, { recursive: true });
|
|
67
69
|
} catch (error) {
|
|
68
|
-
console.error(
|
|
69
|
-
console.error(`
|
|
70
|
-
console.error(' Check file permissions and try again.\n');
|
|
70
|
+
console.error(`Error: Failed to create directories`);
|
|
71
|
+
console.error(` ${error instanceof Error ? error.message : String(error)}`);
|
|
71
72
|
process.exit(1);
|
|
72
73
|
}
|
|
73
74
|
} else {
|
|
74
|
-
console.log(
|
|
75
|
+
console.log(`Adding feature to screen: ${options.name}\n`);
|
|
75
76
|
}
|
|
76
|
-
|
|
77
|
+
|
|
77
78
|
// Check file existence to prevent overwrite
|
|
78
79
|
if (fs.existsSync(featurePath)) {
|
|
79
|
-
console.error(
|
|
80
|
-
console.error(`
|
|
81
|
-
console.error(`
|
|
80
|
+
console.error(`Error: Feature file "${filename}.feature" already exists`);
|
|
81
|
+
console.error(` Path: ${featurePath}`);
|
|
82
|
+
console.error(` Use a different --path or --feature value.`);
|
|
82
83
|
process.exit(1);
|
|
83
84
|
}
|
|
84
85
|
|
|
85
86
|
// Generate and write feature file
|
|
86
87
|
const featureContent = this.generateFeatureTemplate(options, filename);
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
88
|
+
fs.writeFileSync(featurePath, featureContent, 'utf-8');
|
|
89
|
+
|
|
90
|
+
// Generate empty selector YAML (with page entry)
|
|
91
|
+
if (!fs.existsSync(selectorPath)) {
|
|
92
|
+
const pagePath = options.path || `/${screenName}`;
|
|
93
|
+
fs.writeFileSync(selectorPath, [
|
|
94
|
+
`# ${options.name} Screen Selectors`,
|
|
95
|
+
`# Fill via AI or manually`,
|
|
96
|
+
``,
|
|
97
|
+
`${screenName}:`,
|
|
98
|
+
` type: 'page'`,
|
|
99
|
+
` value: '${pagePath}'`,
|
|
100
|
+
` nth: 0`,
|
|
101
|
+
``,
|
|
102
|
+
].join('\n'), 'utf-8');
|
|
93
103
|
}
|
|
94
104
|
|
|
95
|
-
// Generate
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
105
|
+
// Generate empty test-data YAML
|
|
106
|
+
if (!fs.existsSync(testDataPath)) {
|
|
107
|
+
fs.writeFileSync(testDataPath, [
|
|
108
|
+
`# ${options.name} Screen Test Data`,
|
|
109
|
+
`# Fill via AI or manually`,
|
|
110
|
+
`# Reference in features using {{variable}} syntax`,
|
|
111
|
+
``,
|
|
112
|
+
].join('\n'), 'utf-8');
|
|
103
113
|
}
|
|
104
114
|
|
|
105
|
-
//
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
console.error(`❌ Error: Failed to create test data override file at ${testDataOverridePath}`);
|
|
111
|
-
console.error(` ${error instanceof Error ? error.message : String(error)}`);
|
|
112
|
-
process.exit(1);
|
|
113
|
-
}
|
|
115
|
+
// Display success
|
|
116
|
+
console.log(`Created files:`);
|
|
117
|
+
console.log(` ${path.relative(this.cwd, featurePath)}`);
|
|
118
|
+
console.log(` ${path.relative(this.cwd, selectorPath)}`);
|
|
119
|
+
console.log(` ${path.relative(this.cwd, testDataPath)}\n`);
|
|
114
120
|
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
this.displaySuccess(options.name, filename, screenDir);
|
|
121
|
+
console.log('Next steps:');
|
|
122
|
+
console.log(` 1. Use AI (Copilot/Claude) to generate Gherkin + selectors from the live page`);
|
|
123
|
+
console.log(` 2. Fill in test-data values`);
|
|
124
|
+
console.log(` 3. Compile: sungen generate --screen ${screenName}`);
|
|
125
|
+
console.log(` 4. Run: npx playwright test\n`);
|
|
123
126
|
}
|
|
124
127
|
|
|
125
|
-
/**
|
|
126
|
-
* Validate screen name
|
|
127
|
-
*/
|
|
128
128
|
private validateScreenName(name: string): void {
|
|
129
129
|
if (!name || name.trim().length === 0) {
|
|
130
|
-
console.error('
|
|
130
|
+
console.error('Error: Screen name cannot be empty');
|
|
131
131
|
process.exit(1);
|
|
132
132
|
}
|
|
133
|
-
|
|
134
|
-
// Check if name contains at least one alphanumeric character
|
|
135
133
|
if (!/[a-zA-Z0-9]/.test(name)) {
|
|
136
|
-
console.error('
|
|
137
|
-
console.error(' Valid examples: login, user-profile, Dashboard');
|
|
134
|
+
console.error('Error: Screen name must contain at least one alphanumeric character');
|
|
138
135
|
process.exit(1);
|
|
139
136
|
}
|
|
140
137
|
}
|
|
141
138
|
|
|
142
|
-
/**
|
|
143
|
-
* Normalize screen name to kebab-case for filename
|
|
144
|
-
*/
|
|
145
139
|
private normalizeScreenName(name: string): string {
|
|
146
140
|
return name
|
|
147
141
|
.toLowerCase()
|
|
@@ -150,29 +144,18 @@ export class ScreenManager {
|
|
|
150
144
|
.replace(/-+/g, '-');
|
|
151
145
|
}
|
|
152
146
|
|
|
153
|
-
/**
|
|
154
|
-
* Extract filename from path value
|
|
155
|
-
*/
|
|
156
147
|
private extractFilenameFromPath(pathString: string): string {
|
|
157
|
-
// Remove leading and trailing slashes
|
|
158
148
|
const cleanPath = pathString.replace(/^\/+/, '').replace(/\/+$/, '');
|
|
159
|
-
|
|
160
|
-
// Split by '/' and take last segment
|
|
161
149
|
const segments = cleanPath.split('/');
|
|
162
150
|
const lastSegment = segments[segments.length - 1];
|
|
163
|
-
|
|
164
|
-
// Normalize to kebab-case
|
|
165
151
|
return this.normalizeScreenName(lastSegment);
|
|
166
152
|
}
|
|
167
153
|
|
|
168
|
-
/**
|
|
169
|
-
* Generate feature file template
|
|
170
|
-
*/
|
|
171
154
|
private generateFeatureTemplate(options: ScreenOptions, filename: string): string {
|
|
172
155
|
const screenName = this.normalizeScreenName(options.name);
|
|
173
156
|
const featurePath = options.path || `/${screenName}`;
|
|
174
157
|
const description = options.description || `interact with the ${options.name} screen`;
|
|
175
|
-
|
|
158
|
+
|
|
176
159
|
return `Feature: ${options.name} Screen
|
|
177
160
|
|
|
178
161
|
As a user
|
|
@@ -182,164 +165,9 @@ export class ScreenManager {
|
|
|
182
165
|
|
|
183
166
|
@high
|
|
184
167
|
Scenario: Sample scenario for ${options.name}
|
|
185
|
-
Given User is on [${
|
|
168
|
+
Given User is on [${screenName}] page
|
|
186
169
|
When User click [element] button
|
|
187
170
|
Then User see [result] text with {{success}}
|
|
188
171
|
`;
|
|
189
172
|
}
|
|
190
|
-
|
|
191
|
-
/**
|
|
192
|
-
* Generate selector YAML template
|
|
193
|
-
*/
|
|
194
|
-
private generateSelectorTemplate(options: ScreenOptions, filename: string): string {
|
|
195
|
-
const timestamp = new Date().toISOString().split('T')[0];
|
|
196
|
-
const screenPath = options.path || '';
|
|
197
|
-
|
|
198
|
-
return `# ${options.name} Screen Selectors
|
|
199
|
-
# Created: ${timestamp}
|
|
200
|
-
${screenPath ? `# Path: ${screenPath}` : '# Path: (not specified - update as needed)'}
|
|
201
|
-
|
|
202
|
-
# Example element - replace with actual elements or use 'sungen map'
|
|
203
|
-
#
|
|
204
|
-
# Element structure:
|
|
205
|
-
# key: element-identifier
|
|
206
|
-
# locator: "" # Playwright locator
|
|
207
|
-
# type: text|role|placeholder|testid|xpath
|
|
208
|
-
# value: "text content" # Element value/text/label
|
|
209
|
-
# name: "Display Name" # Optional: readable name
|
|
210
|
-
# nth: 0 # Optional: element index if multiple matches
|
|
211
|
-
|
|
212
|
-
# element:
|
|
213
|
-
# locator: ""
|
|
214
|
-
# type: role
|
|
215
|
-
# value: "button"
|
|
216
|
-
# name: "element"
|
|
217
|
-
# nth: 0
|
|
218
|
-
|
|
219
|
-
# Next steps:
|
|
220
|
-
# 1. Edit the feature file: qa/screens/${filename}/features/${filename}.feature
|
|
221
|
-
# 2. Run: sungen map --screen ${filename}
|
|
222
|
-
# This will populate selectors from your feature file
|
|
223
|
-
#
|
|
224
|
-
# Documentation: https://github.com/sun-asterisk/sungen
|
|
225
|
-
`;
|
|
226
|
-
}
|
|
227
|
-
|
|
228
|
-
/**
|
|
229
|
-
* Generate test data YAML template
|
|
230
|
-
*/
|
|
231
|
-
private generateTestDataTemplate(options: ScreenOptions): string {
|
|
232
|
-
const timestamp = new Date().toISOString().split('T')[0];
|
|
233
|
-
|
|
234
|
-
return `# ${options.name} Screen Test Data
|
|
235
|
-
# Created: ${timestamp}
|
|
236
|
-
|
|
237
|
-
# Define test data variables here
|
|
238
|
-
# Example:
|
|
239
|
-
# success: "Operation completed successfully"
|
|
240
|
-
# username: "testuser@example.com"
|
|
241
|
-
# password: "SecurePassword123"
|
|
242
|
-
|
|
243
|
-
# Next steps:
|
|
244
|
-
# 1. Add test data variables referenced in your feature scenarios
|
|
245
|
-
# 2. Use {{variable}} syntax in your feature files to reference these values
|
|
246
|
-
`;
|
|
247
|
-
}
|
|
248
|
-
|
|
249
|
-
/**
|
|
250
|
-
* Generate selector override YAML template (empty with guidelines)
|
|
251
|
-
*/
|
|
252
|
-
private generateSelectorOverrideTemplate(filename: string): string {
|
|
253
|
-
return `# ${filename}.override.yaml - User Override File
|
|
254
|
-
# This file will NEVER be modified by 'sungen map' command
|
|
255
|
-
# Use this to override auto-generated selectors or add custom ones
|
|
256
|
-
#
|
|
257
|
-
# Format:
|
|
258
|
-
# element.key:
|
|
259
|
-
# locator: "css-locator-here" # Optional: custom CSS locator
|
|
260
|
-
# type: role|testid|placeholder|label|text
|
|
261
|
-
# value: "value-for-type"
|
|
262
|
-
# name: "Display Name" # Optional
|
|
263
|
-
# nth: 0 # Optional: element index
|
|
264
|
-
#
|
|
265
|
-
# Examples:
|
|
266
|
-
# email.field:
|
|
267
|
-
# type: testid
|
|
268
|
-
# value: "login-email"
|
|
269
|
-
#
|
|
270
|
-
# submit.button:
|
|
271
|
-
# type: role
|
|
272
|
-
# value: "button"
|
|
273
|
-
# name: "Sign In"
|
|
274
|
-
#
|
|
275
|
-
# custom.element:
|
|
276
|
-
# locator: "#my-custom-id"
|
|
277
|
-
# type: placeholder
|
|
278
|
-
# value: "Enter text"
|
|
279
|
-
#
|
|
280
|
-
# Leave empty if no overrides needed
|
|
281
|
-
# Base selectors will be auto-generated by: sungen map --screen ${filename}
|
|
282
|
-
`;
|
|
283
|
-
}
|
|
284
|
-
|
|
285
|
-
/**
|
|
286
|
-
* Generate test-data override YAML template (empty with guidelines)
|
|
287
|
-
*/
|
|
288
|
-
private generateTestDataOverrideTemplate(filename: string): string {
|
|
289
|
-
return `# ${filename}.override.yaml - User Override File
|
|
290
|
-
# This file will NEVER be modified by 'sungen map' command
|
|
291
|
-
# Use this to override auto-generated test data or add custom data
|
|
292
|
-
#
|
|
293
|
-
# Format:
|
|
294
|
-
# variable_name: "value"
|
|
295
|
-
#
|
|
296
|
-
# nested:
|
|
297
|
-
# variable: "value"
|
|
298
|
-
#
|
|
299
|
-
# array_example:
|
|
300
|
-
# - "item1"
|
|
301
|
-
# - "item2"
|
|
302
|
-
#
|
|
303
|
-
# Examples:
|
|
304
|
-
# # Simple string override
|
|
305
|
-
# valid_email: "custom@example.com"
|
|
306
|
-
#
|
|
307
|
-
# # Nested object override
|
|
308
|
-
# api:
|
|
309
|
-
# url: "https://staging.example.com"
|
|
310
|
-
# timeout: 5000
|
|
311
|
-
#
|
|
312
|
-
# # Array override
|
|
313
|
-
# test_users:
|
|
314
|
-
# - "user1@example.com"
|
|
315
|
-
# - "user2@example.com"
|
|
316
|
-
#
|
|
317
|
-
# Leave empty if no overrides needed
|
|
318
|
-
# Base test data will be auto-generated by: sungen map --screen ${filename}
|
|
319
|
-
`;
|
|
320
|
-
}
|
|
321
|
-
|
|
322
|
-
/**
|
|
323
|
-
* Display success message with next steps
|
|
324
|
-
*/
|
|
325
|
-
private displaySuccess(name: string, filename: string, screenDir: string): void {
|
|
326
|
-
console.log('✅ Screen definition created successfully!\n');
|
|
327
|
-
console.log(`Screen: ${name}`);
|
|
328
|
-
console.log(`Path: ${screenDir}\n`);
|
|
329
|
-
|
|
330
|
-
console.log('Created files:');
|
|
331
|
-
console.log(` 📄 ${path.join(screenDir, 'features', `${filename}.feature`)}`);
|
|
332
|
-
console.log(` 📝 ${path.join(screenDir, 'selectors', `${filename}.override.yaml`)} (user-editable)`);
|
|
333
|
-
console.log(` 📝 ${path.join(screenDir, 'test-data', `${filename}.override.yaml`)} (user-editable)\n`);
|
|
334
|
-
|
|
335
|
-
console.log('📌 Note: Override files will NEVER be modified by automation');
|
|
336
|
-
console.log(' Edit these files to customize selectors and test data\n');
|
|
337
|
-
|
|
338
|
-
console.log('Next steps:');
|
|
339
|
-
console.log(` 1. Edit the Gherkin feature file to define test scenarios`);
|
|
340
|
-
console.log(` 2. Run: sungen map --screen ${name}`);
|
|
341
|
-
console.log(` This will generate BASE selector and test-data files`);
|
|
342
|
-
console.log(` (Override files remain untouched)\n`);
|
|
343
|
-
console.log('Documentation: https://github.com/sun-asterisk/sungen');
|
|
344
|
-
}
|
|
345
173
|
}
|
|
@@ -0,0 +1,189 @@
|
|
|
1
|
+
# Sungen AI Rules
|
|
2
|
+
|
|
3
|
+
You generate 3 files for sungen — a Gherkin compiler that produces Playwright tests.
|
|
4
|
+
**You do NOT write Playwright code.** You only write `.feature`, `selectors.yaml`, and `test-data.yaml`.
|
|
5
|
+
|
|
6
|
+
## Complete Example
|
|
7
|
+
|
|
8
|
+
Given a login page, here are the 3 files you generate:
|
|
9
|
+
|
|
10
|
+
**qa/screens/login/features/login.feature**
|
|
11
|
+
```gherkin
|
|
12
|
+
@auth:admin
|
|
13
|
+
Feature: Login Screen
|
|
14
|
+
|
|
15
|
+
Scenario: Successful login
|
|
16
|
+
Given User is on [login] page
|
|
17
|
+
When User fill [Email] field with {{email}}
|
|
18
|
+
And User fill [Password] field with {{password}}
|
|
19
|
+
And User click [Submit] button
|
|
20
|
+
Then User see [Welcome] heading with {{welcome_text}}
|
|
21
|
+
And User see [Dashboard] link is visible
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
**qa/screens/login/selectors/login.yaml**
|
|
25
|
+
```yaml
|
|
26
|
+
login:
|
|
27
|
+
type: 'page'
|
|
28
|
+
value: '/login'
|
|
29
|
+
|
|
30
|
+
email:
|
|
31
|
+
type: 'placeholder'
|
|
32
|
+
value: 'Enter your email'
|
|
33
|
+
|
|
34
|
+
password:
|
|
35
|
+
type: 'placeholder'
|
|
36
|
+
value: 'Enter your password'
|
|
37
|
+
|
|
38
|
+
submit:
|
|
39
|
+
type: 'role'
|
|
40
|
+
value: 'button'
|
|
41
|
+
name: 'Submit'
|
|
42
|
+
|
|
43
|
+
welcome:
|
|
44
|
+
type: 'role'
|
|
45
|
+
value: 'heading'
|
|
46
|
+
name: 'Welcome'
|
|
47
|
+
|
|
48
|
+
dashboard:
|
|
49
|
+
type: 'role'
|
|
50
|
+
value: 'link'
|
|
51
|
+
name: 'Dashboard'
|
|
52
|
+
```
|
|
53
|
+
|
|
54
|
+
**qa/screens/login/test-data/login.yaml**
|
|
55
|
+
```yaml
|
|
56
|
+
email: 'admin@example.com'
|
|
57
|
+
password: 'password123'
|
|
58
|
+
welcome_text: 'Welcome back'
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
## Key Rules
|
|
62
|
+
|
|
63
|
+
### YAML key format (CRITICAL)
|
|
64
|
+
|
|
65
|
+
YAML keys = Gherkin `[Reference]` converted to: **lowercase, spaces→dots, no special chars**.
|
|
66
|
+
|
|
67
|
+
| Gherkin | YAML Key |
|
|
68
|
+
|---|---|
|
|
69
|
+
| `[Submit]` | `submit:` |
|
|
70
|
+
| `[Search Content]` | `search.content:` |
|
|
71
|
+
| `[Kudos Detail Modal]` | `kudos.detail.modal:` |
|
|
72
|
+
| `[Page 2]` | `page.2:` |
|
|
73
|
+
|
|
74
|
+
**NEVER use spaces in YAML keys.**
|
|
75
|
+
|
|
76
|
+
### Gherkin patterns (all 17)
|
|
77
|
+
|
|
78
|
+
```
|
|
79
|
+
User click [Target] button # click
|
|
80
|
+
User fill [Target] field with {{value}} # fill
|
|
81
|
+
User check [Target] checkbox # check/uncheck
|
|
82
|
+
User select [Target] dropdown with {{value}} # select
|
|
83
|
+
User upload [Target] file with {{filename}} # upload
|
|
84
|
+
User double click [Target] element # double click
|
|
85
|
+
User hover [Target] icon # hover
|
|
86
|
+
User clear [Target] field # clear
|
|
87
|
+
User press Escape key # global key
|
|
88
|
+
User press Enter on [Target] field # key on element
|
|
89
|
+
User is on [target] page # navigate (Given)
|
|
90
|
+
User see [target] page # URL assertion (Then)
|
|
91
|
+
User wait for 3 seconds # wait timeout
|
|
92
|
+
User wait for [Target] dialog is hidden # wait state
|
|
93
|
+
User scroll to [Target] section # scroll
|
|
94
|
+
User switch to [Target] frame # iframe
|
|
95
|
+
User see [Target] heading with {{value}} # visible assertion
|
|
96
|
+
User see [Target] button is disabled # state assertion
|
|
97
|
+
User see [Target] text contains {{partial}} # text contains
|
|
98
|
+
User see [Table] table has row with {{filter}} # table row
|
|
99
|
+
User see [Table] table row with {{f}} has [Col] with {{v}} # table cell
|
|
100
|
+
User click [Edit] in [Table] table row with {{name}} # action in row
|
|
101
|
+
```
|
|
102
|
+
|
|
103
|
+
States: `hidden`, `visible`, `disabled`, `enabled`, `checked`, `unchecked`, `focused`, `empty`
|
|
104
|
+
|
|
105
|
+
### Selector types
|
|
106
|
+
|
|
107
|
+
| type | value | name | when to use |
|
|
108
|
+
|---|---|---|---|
|
|
109
|
+
| `role` | `button`, `heading`, `link`, etc. | accessible name | **preferred** for interactive elements |
|
|
110
|
+
| `testid` | `data-testid` value | — | when `data-testid` exists |
|
|
111
|
+
| `placeholder` | placeholder text | — | input fields |
|
|
112
|
+
| `label` | label text | — | labeled inputs |
|
|
113
|
+
| `text` | — | — | static text content |
|
|
114
|
+
| `locator` | CSS selector | — | **last resort** |
|
|
115
|
+
| `page` | relative URL path | — | navigation targets |
|
|
116
|
+
| `upload` | — | — | file upload inputs |
|
|
117
|
+
| `frame` | iframe selector | — | iframe targets |
|
|
118
|
+
|
|
119
|
+
Priority: `testid` > `role+name` > `placeholder` > `label` > `text` > `locator`
|
|
120
|
+
|
|
121
|
+
### Tags
|
|
122
|
+
|
|
123
|
+
| Tag | Effect |
|
|
124
|
+
|---|---|
|
|
125
|
+
| `@auth:role` | Adds `test.use({ storageState: 'specs/.auth/role.json' })` |
|
|
126
|
+
| `@steps:name` | Define reusable step group |
|
|
127
|
+
| `@extend:name` | Inherit steps from another scenario |
|
|
128
|
+
| `@manual` | Skip in generation |
|
|
129
|
+
|
|
130
|
+
### Selector options
|
|
131
|
+
|
|
132
|
+
```yaml
|
|
133
|
+
element.key:
|
|
134
|
+
type: 'role'
|
|
135
|
+
value: 'button'
|
|
136
|
+
name: 'Submit' # accessible name
|
|
137
|
+
nth: 0 # element index (0=first)
|
|
138
|
+
exact: true # exact name match (avoid substring collisions)
|
|
139
|
+
scope: 'navigation' # parent landmark scope
|
|
140
|
+
match: 'exact' # exact text match for getByText
|
|
141
|
+
variant: 'native' # dropdown: 'native' (<select>) or 'custom' (div-based)
|
|
142
|
+
frame: '#iframe-id' # iframe scope
|
|
143
|
+
contenteditable: true # rich text editor
|
|
144
|
+
columns: # table columns
|
|
145
|
+
name: { index: 0, header: 'Name' }
|
|
146
|
+
```
|
|
147
|
+
|
|
148
|
+
### Navigation and baseURL
|
|
149
|
+
|
|
150
|
+
Page selectors use **relative paths only**. Playwright prepends `baseURL` from `playwright.config.ts`.
|
|
151
|
+
|
|
152
|
+
```yaml
|
|
153
|
+
login:
|
|
154
|
+
type: 'page'
|
|
155
|
+
value: '/login' # NOT https://example.com/login
|
|
156
|
+
```
|
|
157
|
+
|
|
158
|
+
### Auth
|
|
159
|
+
|
|
160
|
+
- `@auth:admin` on Feature or Scenario → compiler adds `test.use({ storageState: 'specs/.auth/admin.json' })`
|
|
161
|
+
- Auth JSON created by: `sungen makeauth admin` (opens browser for manual login)
|
|
162
|
+
- **Login pages**: no `@auth` tag needed
|
|
163
|
+
- **Authenticated pages**: always add `@auth:role`
|
|
164
|
+
|
|
165
|
+
## Using Playwright MCP to explore pages
|
|
166
|
+
|
|
167
|
+
When exploring a page to generate test files:
|
|
168
|
+
1. Read `playwright.config.ts` for `baseURL`
|
|
169
|
+
2. Use `browser_navigate` to open `baseURL + /path`
|
|
170
|
+
3. Use `browser_snapshot` to see all elements
|
|
171
|
+
4. Generate the 3 files from the snapshot
|
|
172
|
+
|
|
173
|
+
**NEVER use `browser_run_code`.** It fails with `require is not defined`.
|
|
174
|
+
Only use: `browser_navigate`, `browser_snapshot`, `browser_click`, `browser_fill_form`, `browser_evaluate`.
|
|
175
|
+
|
|
176
|
+
To browse authenticated pages via MCP:
|
|
177
|
+
1. Read `specs/.auth/role.json` with your file read tool
|
|
178
|
+
2. `browser_navigate` to the page
|
|
179
|
+
3. `browser_evaluate` to inject localStorage and cookies from the JSON
|
|
180
|
+
4. `browser_navigate` again to reload with auth
|
|
181
|
+
|
|
182
|
+
## Commands
|
|
183
|
+
|
|
184
|
+
```bash
|
|
185
|
+
sungen add --screen <name> --path <url-path> # Create screen
|
|
186
|
+
sungen generate --screen <name> # Compile to .spec.ts
|
|
187
|
+
sungen generate --all # Compile all
|
|
188
|
+
sungen makeauth <role> # Capture auth state
|
|
189
|
+
```
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import { defineConfig, devices } from '@playwright/test';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Read environment variables from file.
|
|
5
|
+
* https://github.com/motdotla/dotenv
|
|
6
|
+
*/
|
|
7
|
+
// import dotenv from 'dotenv';
|
|
8
|
+
// import path from 'path';
|
|
9
|
+
// dotenv.config({ path: path.resolve(__dirname, '.env') });
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* See https://playwright.dev/docs/test-configuration.
|
|
13
|
+
*/
|
|
14
|
+
export default defineConfig({
|
|
15
|
+
testDir: './specs/generated',
|
|
16
|
+
/* Run tests in files in parallel */
|
|
17
|
+
fullyParallel: true,
|
|
18
|
+
/* Fail the build on CI if you accidentally left test.only in the source code. */
|
|
19
|
+
forbidOnly: !!process.env.CI,
|
|
20
|
+
/* Retry on CI only */
|
|
21
|
+
retries: process.env.CI ? 2 : 0,
|
|
22
|
+
/* Opt out of parallel tests on CI. */
|
|
23
|
+
workers: process.env.CI ? 1 : undefined,
|
|
24
|
+
/* Reporter to use. See https://playwright.dev/docs/test-reporters */
|
|
25
|
+
reporter: 'html',
|
|
26
|
+
/* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */
|
|
27
|
+
use: {
|
|
28
|
+
/* Base URL to use in actions like `await page.goto('')`. */
|
|
29
|
+
baseURL: 'https://example.com',
|
|
30
|
+
|
|
31
|
+
/* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */
|
|
32
|
+
trace: 'on-first-retry',
|
|
33
|
+
},
|
|
34
|
+
|
|
35
|
+
/* Configure projects for major browsers */
|
|
36
|
+
projects: [
|
|
37
|
+
{
|
|
38
|
+
name: 'chromium',
|
|
39
|
+
use: { ...devices['Desktop Chrome'] },
|
|
40
|
+
},
|
|
41
|
+
|
|
42
|
+
/*
|
|
43
|
+
// {
|
|
44
|
+
// name: 'firefox',
|
|
45
|
+
// use: { ...devices['Desktop Firefox'] },
|
|
46
|
+
// },
|
|
47
|
+
|
|
48
|
+
// {
|
|
49
|
+
// name: 'webkit',
|
|
50
|
+
// use: { ...devices['Desktop Safari'] },
|
|
51
|
+
// },
|
|
52
|
+
|
|
53
|
+
/* Test against mobile viewports.
|
|
54
|
+
// {
|
|
55
|
+
// name: 'Mobile Chrome',
|
|
56
|
+
// use: { ...devices['Pixel 5'] },
|
|
57
|
+
// },
|
|
58
|
+
// {
|
|
59
|
+
// name: 'Mobile Safari',
|
|
60
|
+
// use: { ...devices['iPhone 12'] },
|
|
61
|
+
// },
|
|
62
|
+
|
|
63
|
+
/* Test against branded browsers. */
|
|
64
|
+
// {
|
|
65
|
+
// name: 'Microsoft Edge',
|
|
66
|
+
// use: { ...devices['Desktop Edge'], channel: 'msedge' },
|
|
67
|
+
// },
|
|
68
|
+
// {
|
|
69
|
+
// name: 'Google Chrome',
|
|
70
|
+
// use: { ...devices['Desktop Chrome'], channel: 'chrome' },
|
|
71
|
+
// },
|
|
72
|
+
],
|
|
73
|
+
|
|
74
|
+
/* Run your local dev server before starting the tests */
|
|
75
|
+
// webServer: {
|
|
76
|
+
// command: 'npm run start',
|
|
77
|
+
// url: 'http://localhost:3000',
|
|
78
|
+
// reuseExistingServer: !process.env.CI,
|
|
79
|
+
// },
|
|
80
|
+
});
|