@jaimevalasek/aioson 1.8.0 → 1.9.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/CHANGELOG.md +595 -595
- package/CODE_OF_CONDUCT.md +12 -12
- package/CONTRIBUTING.md +13 -13
- package/LICENSE +661 -661
- package/README.md +919 -919
- package/bin/aioson.js +4 -4
- package/docs/design-previews/aurora-command-ui-website.html +884 -884
- package/docs/design-previews/aurora-command-ui.html +682 -682
- package/docs/design-previews/bold-editorial-ui-website.html +658 -658
- package/docs/design-previews/bold-editorial-ui.html +717 -717
- package/docs/design-previews/clean-saas-ui-website.html +1202 -1202
- package/docs/design-previews/clean-saas-ui.html +549 -549
- package/docs/design-previews/cognitive-core-ui-website.html +1009 -1009
- package/docs/design-previews/cognitive-core-ui.html +463 -463
- package/docs/design-previews/glassmorphism-ui-website.html +572 -572
- package/docs/design-previews/glassmorphism-ui.html +886 -886
- package/docs/design-previews/index.html +699 -699
- package/docs/design-previews/interface-design-website.html +1187 -1187
- package/docs/design-previews/interface-design.html +513 -513
- package/docs/design-previews/neo-brutalist-ui-website.html +621 -621
- package/docs/design-previews/neo-brutalist-ui.html +797 -797
- package/docs/design-previews/premium-command-center-ui-website.html +1217 -1217
- package/docs/design-previews/premium-command-center-ui.html +552 -552
- package/docs/design-previews/pt.squarespace.com-homepage.html +889 -889
- package/docs/design-previews/warm-craft-ui-website.html +684 -684
- package/docs/design-previews/warm-craft-ui.html +739 -739
- package/docs/en/1-understand/ecosystem-map.md +228 -0
- package/docs/en/1-understand/glossary.md +288 -0
- package/docs/en/1-understand/what-is-aioson.md +94 -0
- package/docs/en/1-understand/why-it-exists.md +106 -0
- package/docs/en/2-start/existing-project.md +246 -0
- package/docs/en/2-start/first-project.md +307 -0
- package/docs/en/2-start/initial-decisions.md +223 -0
- package/docs/en/3-recipes/README.md +28 -0
- package/docs/en/3-recipes/continuity-between-sessions.md +303 -0
- package/docs/en/3-recipes/from-idea-to-prd-via-briefing.md +235 -0
- package/docs/en/3-recipes/full-feature-with-sheldon.md +338 -0
- package/docs/en/4-agents/README.md +56 -0
- package/docs/en/5-reference/README.md +60 -0
- package/docs/en/{cli-reference.md → 5-reference/cli-reference.md} +639 -464
- package/docs/en/{i18n.md → 5-reference/i18n.md} +52 -52
- package/docs/en/{json-schemas.md → 5-reference/json-schemas.md} +41 -41
- package/docs/en/{mcp.md → 5-reference/mcp.md} +56 -56
- package/docs/en/{parallel.md → 5-reference/parallel.md} +82 -82
- package/docs/en/{qa-browser.md → 5-reference/qa-browser.md} +339 -339
- package/docs/en/{release-flow.md → 5-reference/release-flow.md} +22 -22
- package/docs/en/{release-notes-template.md → 5-reference/release-notes-template.md} +41 -41
- package/docs/en/{release.md → 5-reference/release.md} +28 -28
- package/docs/en/{schemas → 5-reference/schemas}/agent-prompt.schema.json +17 -17
- package/docs/en/{schemas → 5-reference/schemas}/agents.schema.json +32 -32
- package/docs/en/{schemas → 5-reference/schemas}/context-validate.schema.json +36 -36
- package/docs/en/{schemas → 5-reference/schemas}/doctor.schema.json +89 -89
- package/docs/en/{schemas → 5-reference/schemas}/error.schema.json +24 -24
- package/docs/en/{schemas → 5-reference/schemas}/i18n-add.schema.json +15 -15
- package/docs/en/{schemas → 5-reference/schemas}/index.json +126 -126
- package/docs/en/{schemas → 5-reference/schemas}/info.schema.json +39 -39
- package/docs/en/{schemas → 5-reference/schemas}/init.schema.json +48 -48
- package/docs/en/{schemas → 5-reference/schemas}/install.schema.json +60 -60
- package/docs/en/{schemas → 5-reference/schemas}/locale-apply.schema.json +30 -30
- package/docs/en/{schemas → 5-reference/schemas}/mcp-doctor.schema.json +95 -95
- package/docs/en/{schemas → 5-reference/schemas}/mcp-init.schema.json +122 -122
- package/docs/en/{schemas → 5-reference/schemas}/package-test.schema.json +24 -24
- package/docs/en/{schemas → 5-reference/schemas}/parallel-assign.schema.json +66 -66
- package/docs/en/{schemas → 5-reference/schemas}/parallel-doctor.schema.json +122 -122
- package/docs/en/{schemas → 5-reference/schemas}/parallel-guard.schema.json +63 -63
- package/docs/en/{schemas → 5-reference/schemas}/parallel-init.schema.json +53 -53
- package/docs/en/{schemas → 5-reference/schemas}/parallel-merge.schema.json +84 -84
- package/docs/en/{schemas → 5-reference/schemas}/parallel-status.schema.json +184 -184
- package/docs/en/{schemas → 5-reference/schemas}/setup-context.schema.json +39 -39
- package/docs/en/{schemas → 5-reference/schemas}/smoke.schema.json +23 -23
- package/docs/en/{schemas → 5-reference/schemas}/update.schema.json +48 -48
- package/docs/en/{schemas → 5-reference/schemas}/workflow-plan.schema.json +30 -30
- package/docs/en/{squad-dashboard.md → 5-reference/squad-dashboard.md} +372 -372
- package/docs/en/{web3.md → 5-reference/web3.md} +54 -54
- package/docs/en/README.md +115 -0
- package/docs/en/active-learning-loop/README.md +117 -0
- package/docs/en/active-learning-loop/active-learning-loop.md +117 -0
- package/docs/en/active-learning-loop/cli-commands.md +320 -0
- package/docs/en/active-learning-loop/diagrams.md +225 -0
- package/docs/en/active-learning-loop/doctor-checks.md +151 -0
- package/docs/en/active-learning-loop/how-to-use.md +313 -0
- package/docs/en/active-learning-loop/troubleshooting.md +283 -0
- package/docs/en/deyvin-subtask-scout/README.md +109 -0
- package/docs/en/deyvin-subtask-scout/cli-commands.md +248 -0
- package/docs/en/deyvin-subtask-scout/diagrams.md +124 -0
- package/docs/en/deyvin-subtask-scout/how-to-use.md +221 -0
- package/docs/en/deyvin-subtask-scout/sub-task-scout.md +115 -0
- package/docs/en/deyvin-subtask-scout/troubleshooting.md +184 -0
- package/docs/integrations/apps-publish-marketplace.md +94 -94
- package/docs/integrations/sdlc-genius-boundary.md +76 -76
- package/docs/integrations/sdlc-genius-eval-matrix.md +75 -75
- package/docs/integrations/sdlc-genius-install-checklist.md +93 -93
- package/docs/integrations/sdlc-genius-review-samples.md +86 -86
- package/docs/openclaw-bridge.md +308 -308
- package/docs/pt/1-entender/glossario.md +288 -0
- package/docs/pt/1-entender/mapa-do-ecossistema.md +228 -0
- package/docs/pt/1-entender/o-que-e-aioson.md +94 -0
- package/docs/pt/1-entender/por-que-existe.md +107 -0
- package/docs/pt/2-comecar/decisoes-iniciais.md +223 -0
- package/docs/pt/2-comecar/primeiro-projeto.md +307 -0
- package/docs/pt/2-comecar/projeto-existente.md +245 -0
- package/docs/pt/3-receitas/README.md +28 -0
- package/docs/pt/3-receitas/app-saas-do-zero.md +324 -0
- package/docs/pt/3-receitas/auditoria-seguranca.md +254 -0
- package/docs/pt/3-receitas/clonar-design-de-site.md +211 -0
- package/docs/pt/3-receitas/continuidade-entre-sessoes.md +303 -0
- package/docs/pt/3-receitas/da-ideia-ao-prd-via-briefing.md +234 -0
- package/docs/pt/3-receitas/feature-completa-com-sheldon.md +338 -0
- package/docs/pt/3-receitas/integracao-em-codebase-grande.md +243 -0
- package/docs/pt/3-receitas/landing-page.md +281 -0
- package/docs/pt/3-receitas/plans-externos-para-product.md +191 -0
- package/docs/pt/3-receitas/publicar-no-aioson-com.md +219 -0
- package/docs/pt/3-receitas/refatoracao-grande.md +251 -0
- package/docs/pt/4-agentes/README.md +65 -0
- package/docs/pt/4-agentes/analyst.md +111 -0
- package/docs/pt/4-agentes/architect.md +113 -0
- package/docs/pt/4-agentes/briefing.md +95 -0
- package/docs/pt/4-agentes/committer.md +108 -0
- package/docs/pt/4-agentes/copywriter.md +279 -0
- package/docs/pt/4-agentes/design-hybrid-forge.md +116 -0
- package/docs/pt/4-agentes/dev.md +136 -0
- package/docs/pt/4-agentes/deyvin.md +99 -0
- package/docs/pt/4-agentes/discover.md +122 -0
- package/docs/pt/4-agentes/discovery-design-doc.md +91 -0
- package/docs/pt/4-agentes/genome.md +115 -0
- package/docs/pt/4-agentes/neo.md +93 -0
- package/docs/pt/4-agentes/orache.md +107 -0
- package/docs/pt/4-agentes/orchestrator.md +118 -0
- package/docs/pt/4-agentes/pentester.md +131 -0
- package/docs/pt/4-agentes/pm.md +97 -0
- package/docs/pt/4-agentes/product.md +114 -0
- package/docs/pt/4-agentes/profiler-enricher.md +93 -0
- package/docs/pt/4-agentes/profiler-forge.md +93 -0
- package/docs/pt/4-agentes/profiler-researcher.md +98 -0
- package/docs/pt/4-agentes/qa.md +124 -0
- package/docs/pt/4-agentes/setup.md +104 -0
- package/docs/pt/4-agentes/sheldon.md +95 -0
- package/docs/pt/4-agentes/site-forge.md +104 -0
- package/docs/pt/4-agentes/squad.md +127 -0
- package/docs/pt/4-agentes/tester.md +105 -0
- package/docs/pt/4-agentes/ux-ui.md +110 -0
- package/docs/pt/4-agentes/validator.md +118 -0
- package/docs/pt/5-referencia/README.md +88 -0
- package/docs/pt/5-referencia/agent-chain-continuity.md +124 -0
- package/docs/pt/{agent-sharding.md → 5-referencia/agent-sharding.md} +132 -132
- package/docs/pt/5-referencia/aioson-com-store.md +119 -0
- package/docs/pt/{automacao-squads.md → 5-referencia/automacao-squads.md} +407 -407
- package/docs/pt/{clientes-ai.md → 5-referencia/clientes-ai.md} +300 -290
- package/docs/pt/{comandos-cli.md → 5-referencia/comandos-cli.md} +1823 -1781
- package/docs/pt/{compress-agents.md → 5-referencia/compress-agents.md} +304 -304
- package/docs/pt/{design-docs-governance.md → 5-referencia/design-docs-governance.md} +59 -59
- package/docs/pt/{devlog-pipeline.md → 5-referencia/devlog-pipeline.md} +270 -270
- package/docs/pt/{feature-archive.md → 5-referencia/feature-archive.md} +199 -191
- package/docs/pt/5-referencia/feature-dossier.md +121 -0
- package/docs/pt/{fluxo-artefatos.md → 5-referencia/fluxo-artefatos.md} +179 -178
- package/docs/pt/{genome-3.0-spec.md → 5-referencia/genome-4.0-spec.md} +407 -407
- package/docs/pt/{genome-distribution.md → 5-referencia/genome-distribution.md} +232 -232
- package/docs/pt/{hooks-session-guard.md → 5-referencia/hooks-session-guard.md} +454 -454
- package/docs/pt/{inteligencia-adaptativa.md → 5-referencia/inteligencia-adaptativa.md} +324 -324
- package/docs/pt/5-referencia/live-sessions.md +144 -0
- package/docs/pt/5-referencia/memoria-e-contexto.md +340 -0
- package/docs/pt/{motor-hardening.md → 5-referencia/motor-hardening.md} +493 -492
- package/docs/pt/{output-strategy-delivery.md → 5-referencia/output-strategy-delivery.md} +655 -655
- package/docs/pt/{runner-system.md → 5-referencia/runner-system.md} +113 -113
- package/docs/pt/{runtime-observability.md → 5-referencia/runtime-observability.md} +76 -76
- package/docs/pt/{sandbox.md → 5-referencia/sandbox.md} +125 -125
- package/docs/pt/{sdd-automation-scripts.md → 5-referencia/sdd-automation-scripts.md} +559 -557
- package/docs/pt/5-referencia/sdd-framework.md +115 -0
- package/docs/pt/5-referencia/sdd-planos-e-estrutura.md +321 -0
- package/docs/pt/5-referencia/secure-by-default.md +117 -0
- package/docs/pt/{skills.md → 5-referencia/skills.md} +275 -267
- package/docs/pt/{spec-learnings-pipeline.md → 5-referencia/spec-learnings-pipeline.md} +265 -265
- package/docs/pt/{squad-dashboard.md → 5-referencia/squad-dashboard.md} +373 -373
- package/docs/pt/{web3.md → 5-referencia/web3.md} +797 -797
- package/docs/pt/README.md +111 -125
- package/docs/pt/_arquivo/README.md +130 -0
- package/docs/pt/{advisor-spec.md → _arquivo/advisor-spec.md} +343 -335
- package/docs/pt/{agentes-customizados.md → _arquivo/agentes-customizados.md} +678 -670
- package/docs/pt/{busca-de-contexto.md → _arquivo/busca-de-contexto.md} +136 -129
- package/docs/pt/{cache-de-contexto.md → _arquivo/cache-de-contexto.md} +163 -156
- package/docs/pt/{cenarios.md → _arquivo/cenarios.md} +1282 -1274
- package/docs/pt/{design-hybrid-forge.md → _arquivo/design-hybrid-forge.md} +365 -356
- package/docs/pt/{deyvin.md → _arquivo/deyvin.md} +123 -115
- package/docs/pt/{guia-engineer.md → _arquivo/guia-engineer.md} +234 -226
- package/docs/pt/{inicio-rapido.md → _arquivo/inicio-rapido.md} +261 -251
- package/docs/pt/{memoria-contexto.md → _arquivo/memoria-contexto.md} +262 -255
- package/docs/pt/{monitor-de-contexto.md → _arquivo/monitor-de-contexto.md} +165 -158
- package/docs/pt/{profiler-system.md → _arquivo/profiler-system.md} +222 -214
- package/docs/pt/{recuperacao-de-sessao.md → _arquivo/recuperacao-de-sessao.md} +134 -125
- package/docs/pt/{site-forge.md → _arquivo/site-forge.md} +318 -309
- package/docs/pt/{squad-genome.md → _arquivo/squad-genome.md} +793 -783
- package/docs/pt/active-learning-loop/README.md +117 -0
- package/docs/pt/active-learning-loop/ativo-learning-loop.md +117 -0
- package/docs/pt/active-learning-loop/comandos-cli.md +320 -0
- package/docs/pt/active-learning-loop/como-usar.md +313 -0
- package/docs/pt/active-learning-loop/diagramas.md +225 -0
- package/docs/pt/active-learning-loop/doctor-checks.md +151 -0
- package/docs/pt/active-learning-loop/troubleshooting.md +283 -0
- package/docs/pt/agentes.md +996 -993
- package/docs/pt/deyvin-subtask-scout/README.md +109 -0
- package/docs/pt/deyvin-subtask-scout/comandos-cli.md +248 -0
- package/docs/pt/deyvin-subtask-scout/como-usar.md +221 -0
- package/docs/pt/deyvin-subtask-scout/diagramas.md +124 -0
- package/docs/pt/deyvin-subtask-scout/sub-task-scout.md +113 -0
- package/docs/pt/deyvin-subtask-scout/troubleshooting.md +184 -0
- package/docs/pt/living-memory/README.md +81 -0
- package/docs/pt/living-memory/autonomy-contract.md +206 -0
- package/docs/pt/living-memory/diagramas.md +365 -0
- package/docs/pt/living-memory/memoria-viva.md +141 -0
- package/docs/pt/living-memory/notificacoes-info.md +142 -0
- package/docs/pt/living-memory/reflexao-in-harness.md +218 -0
- package/docs/pt/living-memory/troubleshooting.md +286 -0
- package/docs/testing/genome-2.0-manual-regression.md +23 -23
- package/docs/testing/genome-2.0-matrix.md +36 -36
- package/docs/testing/genome-2.0-rollout.md +184 -184
- package/package.json +51 -51
- package/src/a2a/client.js +165 -165
- package/src/a2a/server.js +223 -223
- package/src/agent-loader.js +280 -280
- package/src/agent-manifests.js +86 -66
- package/src/agents.js +92 -92
- package/src/autonomy-policy.js +163 -139
- package/src/backup-local.js +74 -74
- package/src/backup-provider.js +303 -303
- package/src/brain-query.js +171 -161
- package/src/cli.js +85 -5
- package/src/commands/agent-audit.js +397 -397
- package/src/commands/agent-export-skill.js +229 -229
- package/src/commands/agent-loader.js +85 -85
- package/src/commands/agents.js +273 -255
- package/src/commands/artifact-validate.js +218 -218
- package/src/commands/auth.js +298 -272
- package/src/commands/backup-local-cmd.js +25 -25
- package/src/commands/backup.js +533 -533
- package/src/commands/brain-query.js +44 -44
- package/src/commands/brief-gen.js +405 -405
- package/src/commands/brief-validate.js +65 -65
- package/src/commands/briefing.js +344 -344
- package/src/commands/classify.js +256 -256
- package/src/commands/cloud.js +1767 -1767
- package/src/commands/commit-prepare.js +610 -547
- package/src/commands/compress-agents.js +416 -416
- package/src/commands/config.js +90 -90
- package/src/commands/context-cache.js +90 -90
- package/src/commands/context-compact.js +49 -49
- package/src/commands/context-health.js +187 -177
- package/src/commands/context-load.js +219 -0
- package/src/commands/context-monitor.js +163 -163
- package/src/commands/context-pack.js +45 -45
- package/src/commands/context-search.js +66 -66
- package/src/commands/context-trim.js +183 -183
- package/src/commands/context-validate.js +91 -91
- package/src/commands/design-hybrid-options.js +385 -385
- package/src/commands/detect-test-runner.js +55 -55
- package/src/commands/dev-resume.js +32 -0
- package/src/commands/devlog-export-brains.js +27 -27
- package/src/commands/devlog-process.js +294 -294
- package/src/commands/devlog-watch.js +131 -131
- package/src/commands/doctor.js +123 -123
- package/src/commands/dossier-add-research.js +114 -0
- package/src/commands/dossier-audit.js +222 -0
- package/src/commands/dossier.js +423 -423
- package/src/commands/feature-archive.js +513 -513
- package/src/commands/feature-close.js +554 -270
- package/src/commands/gate-approve.js +198 -198
- package/src/commands/gate-check.js +247 -247
- package/src/commands/genome-doctor.js +489 -198
- package/src/commands/genome-migrate.js +49 -49
- package/src/commands/git-guard.js +170 -170
- package/src/commands/harness.js +307 -121
- package/src/commands/health.js +214 -214
- package/src/commands/hooks-emit.js +253 -253
- package/src/commands/hooks-install.js +347 -347
- package/src/commands/i18n-add.js +56 -56
- package/src/commands/implementation-plan.js +367 -367
- package/src/commands/info.js +41 -41
- package/src/commands/init.js +120 -120
- package/src/commands/install.js +162 -111
- package/src/commands/learning-auto-promote.js +197 -195
- package/src/commands/learning-evolve.js +364 -364
- package/src/commands/learning-export.js +103 -103
- package/src/commands/learning-rollback.js +164 -164
- package/src/commands/learning.js +134 -134
- package/src/commands/live.js +2101 -2082
- package/src/commands/locale-apply.js +54 -54
- package/src/commands/locale-diff.js +25 -25
- package/src/commands/mcp-doctor.js +407 -407
- package/src/commands/mcp-init.js +373 -373
- package/src/commands/memory-archive.js +193 -0
- package/src/commands/memory-reflect-commit.js +148 -0
- package/src/commands/memory-reflect-prepare.js +97 -0
- package/src/commands/memory-restore.js +177 -0
- package/src/commands/memory-search.js +135 -0
- package/src/commands/memory.js +299 -234
- package/src/commands/notify.js +68 -0
- package/src/commands/package-e2e.js +273 -273
- package/src/commands/parallel-assign.js +483 -483
- package/src/commands/parallel-doctor.js +850 -850
- package/src/commands/parallel-guard.js +241 -241
- package/src/commands/parallel-init.js +311 -311
- package/src/commands/parallel-merge.js +299 -299
- package/src/commands/parallel-status.js +434 -434
- package/src/commands/pattern-detect.js +33 -33
- package/src/commands/preflight-context.js +30 -30
- package/src/commands/preflight.js +267 -267
- package/src/commands/pulse-update.js +130 -130
- package/src/commands/qa-doctor.js +185 -185
- package/src/commands/qa-init.js +166 -166
- package/src/commands/qa-report.js +58 -58
- package/src/commands/qa-run.js +873 -873
- package/src/commands/qa-scan.js +337 -337
- package/src/commands/recovery.js +43 -43
- package/src/commands/revision.js +235 -235
- package/src/commands/runner-daemon.js +274 -274
- package/src/commands/runner-plan.js +70 -70
- package/src/commands/runner-queue-from-plan.js +166 -166
- package/src/commands/runner-queue.js +189 -189
- package/src/commands/runner-run.js +129 -129
- package/src/commands/runtime.js +2086 -2067
- package/src/commands/sandbox.js +37 -37
- package/src/commands/scaffold-complete.js +188 -188
- package/src/commands/scan-project.js +1371 -1371
- package/src/commands/scout-commit.js +163 -0
- package/src/commands/scout-prep.js +214 -0
- package/src/commands/scout-validate.js +112 -0
- package/src/commands/security-audit.js +275 -275
- package/src/commands/security-scan.js +376 -376
- package/src/commands/self-implement-loop.js +306 -300
- package/src/commands/session-guard.js +218 -218
- package/src/commands/setup-context.js +699 -699
- package/src/commands/setup.js +178 -178
- package/src/commands/sizing.js +165 -165
- package/src/commands/skill.js +670 -670
- package/src/commands/smoke.js +426 -426
- package/src/commands/spec-checkpoint.js +177 -177
- package/src/commands/spec-status.js +79 -79
- package/src/commands/spec-sync.js +190 -190
- package/src/commands/spec-tasks.js +288 -288
- package/src/commands/squad-agent-create.js +830 -830
- package/src/commands/squad-autorun.js +1220 -1220
- package/src/commands/squad-bus.js +217 -217
- package/src/commands/squad-card.js +149 -149
- package/src/commands/squad-daemon.js +343 -343
- package/src/commands/squad-dashboard.js +39 -39
- package/src/commands/squad-dependency-graph.js +164 -164
- package/src/commands/squad-deploy.js +64 -64
- package/src/commands/squad-doctor.js +460 -460
- package/src/commands/squad-export.js +77 -46
- package/src/commands/squad-investigate.js +314 -314
- package/src/commands/squad-learning.js +209 -209
- package/src/commands/squad-mcp.js +270 -270
- package/src/commands/squad-pipeline.js +343 -343
- package/src/commands/squad-plan.js +361 -361
- package/src/commands/squad-processes.js +56 -56
- package/src/commands/squad-recovery.js +42 -42
- package/src/commands/squad-repair-genomes.js +39 -39
- package/src/commands/squad-review.js +106 -106
- package/src/commands/squad-roi.js +291 -291
- package/src/commands/squad-scaffold.js +56 -56
- package/src/commands/squad-score.js +311 -307
- package/src/commands/squad-status.js +481 -481
- package/src/commands/squad-tool-register.js +157 -157
- package/src/commands/squad-validate.js +438 -438
- package/src/commands/squad-webhook.js +160 -160
- package/src/commands/squad-worker.js +191 -191
- package/src/commands/squad-worktrees.js +75 -75
- package/src/commands/state-save.js +220 -122
- package/src/commands/store-genome.js +667 -304
- package/src/commands/store-skill.js +247 -247
- package/src/commands/store-squad.js +431 -431
- package/src/commands/store-system.js +392 -392
- package/src/commands/sync-agents-preflight.js +176 -0
- package/src/commands/test-agents.js +199 -199
- package/src/commands/tool-capabilities.js +63 -63
- package/src/commands/tool-registry-cmd.js +232 -232
- package/src/commands/update.js +68 -64
- package/src/commands/verify-gate.js +612 -612
- package/src/commands/web-map.js +70 -70
- package/src/commands/web-scrape.js +71 -71
- package/src/commands/workflow-execute.js +730 -730
- package/src/commands/workflow-harden.js +231 -231
- package/src/commands/workflow-heal.js +136 -136
- package/src/commands/workflow-next.js +1279 -1039
- package/src/commands/workflow-plan.js +108 -108
- package/src/commands/workflow-status.js +440 -440
- package/src/commands/workspace.js +144 -144
- package/src/constants.js +417 -384
- package/src/context-cache.js +159 -159
- package/src/context-memory.js +975 -966
- package/src/context-parse-reason.js +22 -22
- package/src/context-search.js +326 -326
- package/src/context-writer.js +197 -197
- package/src/context.js +247 -247
- package/src/delivery-runner.js +319 -319
- package/src/design-variation-catalog.js +503 -503
- package/src/detector.js +261 -261
- package/src/doctor.js +812 -329
- package/src/dossier/codemap-store.js +267 -267
- package/src/dossier/dossier-bootstrap.js +222 -222
- package/src/dossier/dossier-compact.js +159 -159
- package/src/dossier/lock.js +128 -128
- package/src/dossier/research-index-store.js +233 -0
- package/src/dossier/revision-store.js +313 -313
- package/src/dossier/schema.js +162 -155
- package/src/dossier/scout-section.js +127 -0
- package/src/dossier/store.js +406 -400
- package/src/execution-gateway.js +464 -464
- package/src/friction-scanner.js +202 -202
- package/src/gateway-pointer-merge.js +101 -0
- package/src/genome-files.js +198 -198
- package/src/genome-format.js +442 -442
- package/src/genome-schema.js +238 -238
- package/src/genomes/bindings.js +281 -281
- package/src/genomes.js +500 -500
- package/src/handoff-contract.js +417 -363
- package/src/handoff-validator.js +45 -45
- package/src/harness/circuit-breaker.js +135 -135
- package/src/i18n/index.js +103 -103
- package/src/i18n/messages/en.js +1548 -1434
- package/src/i18n/messages/es.js +1332 -1221
- package/src/i18n/messages/fr.js +1340 -1229
- package/src/i18n/messages/pt-BR.js +1568 -1457
- package/src/i18n/scaffold.js +64 -64
- package/src/install-animation.js +260 -260
- package/src/install-profile.js +127 -127
- package/src/install-wizard.js +475 -475
- package/src/installer-config-merge.js +207 -0
- package/src/installer.js +487 -358
- package/src/jargon-leak-doctor.js +257 -0
- package/src/learning-loop-archive.js +595 -0
- package/src/learning-loop-doctor.js +217 -0
- package/src/learning-loop-engine.js +254 -0
- package/src/learning-loop-fts5.js +132 -0
- package/src/learning-loop-migration.js +163 -0
- package/src/lib/dev-resume.js +140 -0
- package/src/lib/dossier-telemetry.js +36 -0
- package/src/lib/genomes/compat.js +206 -206
- package/src/lib/genomes/migrate.js +90 -90
- package/src/lib/git-commit-guard.js +751 -691
- package/src/lib/health-check.js +158 -158
- package/src/lib/hook-protocol.js +76 -76
- package/src/lib/llm-content-sanitizer.js +44 -0
- package/src/lib/security/artifact-reader.js +167 -167
- package/src/lib/security/exit-codes.js +51 -51
- package/src/lib/security/findings-writer.js +176 -176
- package/src/lib/security/runtime-events.js +77 -77
- package/src/lib/security/secrets-regex.js +115 -115
- package/src/lib/squads/genome-repair.js +49 -49
- package/src/lib/store/security-scan.js +175 -173
- package/src/lib/terminal-checkbox.js +135 -130
- package/src/lib/terminal-picker.js +447 -0
- package/src/lib/tmux-launcher.js +163 -163
- package/src/lib/tool-capabilities.js +102 -102
- package/src/lib/webhook-server.js +328 -328
- package/src/locales.js +88 -88
- package/src/mcp/apps/squad-dashboard/app.js +163 -163
- package/src/mcp/apps/squad-dashboard/index.html +261 -261
- package/src/mcp/apps/squad-dashboard/mcp-manifest.json +23 -23
- package/src/mcp/resources/squad-state.js +130 -130
- package/src/mcp-connectors/registry.js +602 -602
- package/src/memory-reflect-engine.js +359 -0
- package/src/migrations/profile-rename.js +66 -0
- package/src/notify-renderer.js +32 -0
- package/src/onboarding.js +307 -305
- package/src/parallel-workspace.js +756 -756
- package/src/parser.js +74 -66
- package/src/path-guard.js +47 -47
- package/src/permissions-generator.js +400 -0
- package/src/preflight-engine.js +654 -654
- package/src/prompt-tool.js +20 -20
- package/src/qa-html-report.js +472 -472
- package/src/recovery-context-session.js +154 -154
- package/src/runner/cascade.js +97 -97
- package/src/runner/cli-launcher.js +109 -109
- package/src/runner/plan-importer.js +63 -63
- package/src/runner/queue-store.js +159 -159
- package/src/runtime-store.js +2720 -2676
- package/src/sandbox.js +194 -177
- package/src/self-healing.js +142 -142
- package/src/session-handoff.js +295 -187
- package/src/squad/agent-teams-adapter.js +270 -264
- package/src/squad/brief-validator.js +350 -350
- package/src/squad/bus-bridge.js +140 -140
- package/src/squad/context-compactor.js +265 -265
- package/src/squad/cross-ai-synthesizer.js +250 -250
- package/src/squad/external-session.js +180 -180
- package/src/squad/hooks-generator.js +196 -196
- package/src/squad/inter-squad-events.js +175 -175
- package/src/squad/inter-squad.js +74 -74
- package/src/squad/intra-bus.js +345 -345
- package/src/squad/learning-extractor.js +213 -213
- package/src/squad/pattern-detector.js +365 -365
- package/src/squad/preflight-context.js +296 -296
- package/src/squad/recovery-context.js +372 -372
- package/src/squad/reflection.js +365 -365
- package/src/squad/squad-scaffold.js +341 -341
- package/src/squad/state-manager.js +310 -310
- package/src/squad/task-decomposer.js +652 -652
- package/src/squad/verify-gate.js +303 -303
- package/src/squad/worktree-manager.js +114 -114
- package/src/squad-daemon.js +490 -490
- package/src/squad-dashboard/api.js +223 -223
- package/src/squad-dashboard/attachment-handler.js +93 -93
- package/src/squad-dashboard/context-monitor.js +157 -157
- package/src/squad-dashboard/execution-logs.js +115 -115
- package/src/squad-dashboard/hunk-review.js +209 -209
- package/src/squad-dashboard/metrics.js +133 -133
- package/src/squad-dashboard/process-monitor.js +125 -125
- package/src/squad-dashboard/renderer.js +858 -858
- package/src/squad-dashboard/server.js +232 -232
- package/src/squad-dashboard/styles.js +525 -525
- package/src/squad-dashboard/token-tracker.js +99 -99
- package/src/squads/apply-genome.js +21 -21
- package/src/squads/genome-binding-service.js +154 -154
- package/src/sub-task-engine.js +415 -0
- package/src/sub-task-schemas.js +150 -0
- package/src/sub-task-state.js +152 -0
- package/src/sub-task-telemetry.js +69 -0
- package/src/test-briefing.js +226 -226
- package/src/tool-executor.js +94 -94
- package/src/updater.js +52 -39
- package/src/utils.js +49 -49
- package/src/version.js +50 -50
- package/src/web.js +284 -284
- package/src/worker-runner.js +541 -524
- package/src/workflow-gates.js +185 -185
- package/template/.aioson/advisors/.gitkeep +1 -1
- package/template/.aioson/agents/analyst.md +345 -318
- package/template/.aioson/agents/architect.md +325 -305
- package/template/.aioson/agents/{cypher.md → briefing.md} +264 -252
- package/template/.aioson/agents/committer.md +161 -161
- package/template/.aioson/agents/copywriter.md +937 -463
- package/template/.aioson/agents/design-hybrid-forge.md +141 -141
- package/template/.aioson/agents/dev.md +298 -263
- package/template/.aioson/agents/deyvin.md +200 -87
- package/template/.aioson/agents/discover.md +235 -235
- package/template/.aioson/agents/discovery-design-doc.md +56 -29
- package/template/.aioson/agents/genome.md +1904 -364
- package/template/.aioson/agents/manifests/analyst.manifest.json +26 -26
- package/template/.aioson/agents/manifests/architect.manifest.json +23 -23
- package/template/.aioson/agents/manifests/committer.manifest.json +23 -23
- package/template/.aioson/agents/manifests/dev.manifest.json +54 -37
- package/template/.aioson/agents/manifests/deyvin.manifest.json +41 -0
- package/template/.aioson/agents/manifests/orchestrator.manifest.json +30 -30
- package/template/.aioson/agents/manifests/pentester.manifest.json +39 -39
- package/template/.aioson/agents/manifests/pm.manifest.json +26 -26
- package/template/.aioson/agents/manifests/product.manifest.json +23 -23
- package/template/.aioson/agents/manifests/qa.manifest.json +41 -25
- package/template/.aioson/agents/manifests/setup.manifest.json +20 -20
- package/template/.aioson/agents/manifests/ux-ui.manifest.json +24 -24
- package/template/.aioson/agents/neo.md +356 -231
- package/template/.aioson/agents/orache.md +430 -430
- package/template/.aioson/agents/orchestrator.md +274 -263
- package/template/.aioson/agents/pair.md +5 -5
- package/template/.aioson/agents/pentester.md +289 -235
- package/template/.aioson/agents/pm.md +141 -130
- package/template/.aioson/agents/product.md +367 -273
- package/template/.aioson/agents/profiler-enricher.md +331 -331
- package/template/.aioson/agents/profiler-forge.md +212 -212
- package/template/.aioson/agents/profiler-researcher.md +282 -282
- package/template/.aioson/agents/qa.md +432 -342
- package/template/.aioson/agents/setup.md +425 -423
- package/template/.aioson/agents/sheldon.md +259 -197
- package/template/.aioson/agents/site-forge.md +281 -281
- package/template/.aioson/agents/squad.md +160 -156
- package/template/.aioson/agents/tester.md +536 -473
- package/template/.aioson/agents/ux-ui.md +195 -162
- package/template/.aioson/agents/validator.md +101 -69
- package/template/.aioson/brains/README.md +132 -128
- package/template/.aioson/brains/_archived/.gitkeep +0 -0
- package/template/.aioson/brains/_index.json +34 -16
- package/template/.aioson/brains/dev/patterns.brain.json +79 -0
- package/template/.aioson/brains/scripts/query.js +107 -107
- package/template/.aioson/brains/sheldon/architecture-decisions.brain.json +79 -0
- package/template/.aioson/brains/site-forge/visual-patterns.brain.json +205 -205
- package/template/.aioson/config/autonomy-protocol.json +125 -43
- package/template/.aioson/config/learning-loop.json +10 -0
- package/template/.aioson/config/scout-engine.json +1 -0
- package/template/.aioson/config.md +410 -410
- package/template/.aioson/context/_archived/.gitkeep +0 -0
- package/template/.aioson/context/design-doc.md +136 -136
- package/template/.aioson/context/project-map.md +57 -57
- package/template/.aioson/context/project-pulse.md +34 -34
- package/template/.aioson/context/seeds/seed-example.md +27 -27
- package/template/.aioson/context/spec.md.template +54 -54
- package/template/.aioson/context/user-profile.md +42 -42
- package/template/.aioson/design-docs/code-reuse.md +48 -48
- package/template/.aioson/design-docs/componentization.md +47 -47
- package/template/.aioson/design-docs/file-size.md +52 -52
- package/template/.aioson/design-docs/folder-structure.md +51 -51
- package/template/.aioson/design-docs/naming.md +54 -54
- package/template/.aioson/docs/LAYERS.md +89 -89
- package/template/.aioson/docs/README.md +76 -76
- package/template/.aioson/docs/autonomy-protocol.md +80 -0
- package/template/.aioson/docs/briefing/briefing-craft.md +237 -0
- package/template/.aioson/docs/dev/execution-discipline.md +106 -106
- package/template/.aioson/docs/dev/stack-conventions.md +83 -83
- package/template/.aioson/docs/deyvin/continuity-recovery.md +57 -57
- package/template/.aioson/docs/deyvin/debugging-escalation.md +30 -30
- package/template/.aioson/docs/deyvin/pair-execution.md +44 -44
- package/template/.aioson/docs/deyvin/runtime-handoffs.md +42 -36
- package/template/.aioson/docs/example-external-api-context.md +72 -72
- package/template/.aioson/docs/handoff-persistence.md +94 -0
- package/template/.aioson/docs/pentester/app-playbooks.md +206 -0
- package/template/.aioson/docs/pentester/llm-supplychain.md +165 -0
- package/template/.aioson/docs/product/conversation-playbook.md +116 -116
- package/template/.aioson/docs/product/prd-contract.md +107 -107
- package/template/.aioson/docs/product/quality-lens.md +57 -57
- package/template/.aioson/docs/product/research-loop.md +65 -65
- package/template/.aioson/docs/sheldon/enrichment-paths.md +134 -134
- package/template/.aioson/docs/sheldon/harness-contract.md +118 -0
- package/template/.aioson/docs/sheldon/quality-lens.md +57 -57
- package/template/.aioson/docs/sheldon/research-loop.md +56 -56
- package/template/.aioson/docs/sheldon/web-intelligence.md +75 -75
- package/template/.aioson/docs/site-forge-build.md +195 -195
- package/template/.aioson/docs/site-forge-extraction.md +135 -135
- package/template/.aioson/docs/site-forge-qa.md +155 -155
- package/template/.aioson/docs/site-forge-recon.md +434 -434
- package/template/.aioson/docs/site-forge-transform.md +249 -249
- package/template/.aioson/docs/squad/content-output.md +91 -91
- package/template/.aioson/docs/squad/creation-flow.md +149 -135
- package/template/.aioson/docs/squad/domain-breadth.md +322 -0
- package/template/.aioson/docs/squad/domain-classification.md +117 -117
- package/template/.aioson/docs/squad/genome-bindings.md +47 -47
- package/template/.aioson/docs/squad/package-contract.md +260 -234
- package/template/.aioson/docs/squad/quality-lens.md +60 -56
- package/template/.aioson/docs/squad/research-loop.md +59 -59
- package/template/.aioson/docs/squad/session-operations.md +117 -117
- package/template/.aioson/docs/squad/workflow-quality.md +165 -165
- package/template/.aioson/docs/tester/coverage-quality.md +351 -0
- package/template/.aioson/docs/ux-ui/accessibility-audit.md +55 -55
- package/template/.aioson/docs/ux-ui/audit-mode.md +86 -86
- package/template/.aioson/docs/ux-ui/component-map.md +35 -35
- package/template/.aioson/docs/ux-ui/design-execution.md +111 -111
- package/template/.aioson/docs/ux-ui/design-gate.md +27 -27
- package/template/.aioson/docs/ux-ui/research-mode.md +39 -39
- package/template/.aioson/docs/ux-ui/site-delivery.md +156 -156
- package/template/.aioson/docs/ux-ui/token-contract.md +57 -57
- package/template/.aioson/genomes/INDEX.md +195 -0
- package/template/.aioson/genomes/copywriting/SKILL.md +137 -0
- package/template/.aioson/genomes/copywriting/manifest.json +140 -0
- package/template/.aioson/genomes/copywriting/references/application-notes.md +145 -0
- package/template/.aioson/genomes/copywriting/references/decision-weights.md +45 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/5-act-narrative.md +184 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/classical-formulas.md +164 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/offer-stack.md +195 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/one-belief.md +135 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/pms-research.md +211 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/two-paths-close.md +190 -0
- package/template/.aioson/genomes/copywriting/references/heuristics.md +114 -0
- package/template/.aioson/genomes/copywriting/references/meta-axioms.md +68 -0
- package/template/.aioson/genomes/copywriting/references/methodology.md +115 -0
- package/template/.aioson/genomes/copywriting-brunson/SKILL.md +133 -0
- package/template/.aioson/genomes/copywriting-brunson/manifest.json +152 -0
- package/template/.aioson/genomes/copywriting-brunson/references/application-notes.md +113 -0
- package/template/.aioson/genomes/copywriting-brunson/references/decision-weights.md +33 -0
- package/template/.aioson/genomes/copywriting-brunson/references/evidence-and-attribution.md +81 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/6-part-structure.md +136 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/origin-story.md +121 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/perfect-webinar-script.md +139 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/persuasive-storytelling-5-structures.md +164 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/value-stack.md +136 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/who-what-why-how.md +110 -0
- package/template/.aioson/genomes/copywriting-brunson/references/meta-axioms.md +36 -0
- package/template/.aioson/genomes/copywriting-brunson/references/methodology.md +112 -0
- package/template/.aioson/git-guard.json +12 -11
- package/template/.aioson/mcp/servers.md +23 -23
- package/template/.aioson/profiler-reports/.gitkeep +1 -1
- package/template/.aioson/rules/README.md +69 -69
- package/template/.aioson/rules/_archived/.gitkeep +0 -0
- package/template/.aioson/rules/agent-language-policy.md +93 -93
- package/template/.aioson/rules/aioson-context-boundary.md +63 -63
- package/template/.aioson/rules/canonical-path-contract.md +47 -47
- package/template/.aioson/rules/data-format-convention.md +74 -74
- package/template/.aioson/rules/disk-first-artifacts.md +44 -44
- package/template/.aioson/rules/example-monetary-values.md +30 -30
- package/template/.aioson/rules/output-brevity.md +44 -44
- package/template/.aioson/rules/prd-section-ownership.md +49 -49
- package/template/.aioson/rules/security-baseline.md +139 -139
- package/template/.aioson/rules/spec-level-ownership.md +61 -61
- package/template/.aioson/rules/squad/README.md +50 -50
- package/template/.aioson/rules/squad-driver-pattern.md +81 -81
- package/template/.aioson/schemas/content-blueprint.schema.json +30 -30
- package/template/.aioson/schemas/genome-meta.schema.json +150 -150
- package/template/.aioson/schemas/genome.schema.json +115 -115
- package/template/.aioson/schemas/readiness.schema.json +27 -27
- package/template/.aioson/schemas/squad-blueprint.schema.json +228 -228
- package/template/.aioson/schemas/squad-manifest.schema.json +874 -874
- package/template/.aioson/skills/design/aurora-command-ui/SKILL.md +243 -243
- package/template/.aioson/skills/design/aurora-command-ui/references/art-direction.md +293 -293
- package/template/.aioson/skills/design/aurora-command-ui/references/components.md +827 -827
- package/template/.aioson/skills/design/aurora-command-ui/references/dashboards.md +250 -250
- package/template/.aioson/skills/design/aurora-command-ui/references/design-tokens.md +585 -585
- package/template/.aioson/skills/design/aurora-command-ui/references/motion.md +365 -365
- package/template/.aioson/skills/design/aurora-command-ui/references/patterns.md +482 -482
- package/template/.aioson/skills/design/aurora-command-ui/references/websites.md +387 -387
- package/template/.aioson/skills/design/bold-editorial-ui/SKILL.md +205 -205
- package/template/.aioson/skills/design/bold-editorial-ui/references/art-direction.md +338 -338
- package/template/.aioson/skills/design/bold-editorial-ui/references/components.md +977 -977
- package/template/.aioson/skills/design/bold-editorial-ui/references/dashboards.md +218 -218
- package/template/.aioson/skills/design/bold-editorial-ui/references/design-tokens.md +326 -326
- package/template/.aioson/skills/design/bold-editorial-ui/references/motion.md +461 -461
- package/template/.aioson/skills/design/bold-editorial-ui/references/patterns.md +293 -293
- package/template/.aioson/skills/design/bold-editorial-ui/references/websites.md +352 -352
- package/template/.aioson/skills/design/clean-saas-ui/SKILL.md +210 -210
- package/template/.aioson/skills/design/clean-saas-ui/references/art-direction.md +319 -319
- package/template/.aioson/skills/design/clean-saas-ui/references/components.md +365 -365
- package/template/.aioson/skills/design/clean-saas-ui/references/dashboards.md +196 -196
- package/template/.aioson/skills/design/clean-saas-ui/references/design-tokens.md +244 -244
- package/template/.aioson/skills/design/clean-saas-ui/references/motion.md +235 -235
- package/template/.aioson/skills/design/clean-saas-ui/references/patterns.md +215 -215
- package/template/.aioson/skills/design/clean-saas-ui/references/websites.md +295 -295
- package/template/.aioson/skills/design/cognitive-core-ui/SKILL.md +203 -203
- package/template/.aioson/skills/design/cognitive-core-ui/references/art-direction.md +339 -339
- package/template/.aioson/skills/design/cognitive-core-ui/references/components.md +407 -407
- package/template/.aioson/skills/design/cognitive-core-ui/references/dashboards.md +272 -272
- package/template/.aioson/skills/design/cognitive-core-ui/references/design-tokens.md +524 -524
- package/template/.aioson/skills/design/cognitive-core-ui/references/motion.md +279 -279
- package/template/.aioson/skills/design/cognitive-core-ui/references/patterns.md +289 -289
- package/template/.aioson/skills/design/cognitive-core-ui/references/websites.md +437 -437
- package/template/.aioson/skills/design/glassmorphism-ui/SKILL.md +222 -222
- package/template/.aioson/skills/design/glassmorphism-ui/references/art-direction.md +159 -159
- package/template/.aioson/skills/design/glassmorphism-ui/references/components.md +498 -498
- package/template/.aioson/skills/design/glassmorphism-ui/references/dashboards.md +236 -236
- package/template/.aioson/skills/design/glassmorphism-ui/references/design-tokens.md +274 -274
- package/template/.aioson/skills/design/glassmorphism-ui/references/motion.md +355 -355
- package/template/.aioson/skills/design/glassmorphism-ui/references/patterns.md +198 -198
- package/template/.aioson/skills/design/glassmorphism-ui/references/websites.md +307 -307
- package/template/.aioson/skills/design/interface-design/SKILL.md +47 -47
- package/template/.aioson/skills/design/interface-design/references/components-and-states.md +105 -105
- package/template/.aioson/skills/design/interface-design/references/design-directions.md +101 -101
- package/template/.aioson/skills/design/interface-design/references/handoff-and-quality.md +71 -71
- package/template/.aioson/skills/design/interface-design/references/intent-and-domain.md +74 -74
- package/template/.aioson/skills/design/interface-design/references/tokens-and-depth.md +173 -173
- package/template/.aioson/skills/design/neo-brutalist-ui/SKILL.md +213 -213
- package/template/.aioson/skills/design/neo-brutalist-ui/references/art-direction.md +228 -228
- package/template/.aioson/skills/design/neo-brutalist-ui/references/components.md +855 -855
- package/template/.aioson/skills/design/neo-brutalist-ui/references/dashboards.md +334 -334
- package/template/.aioson/skills/design/neo-brutalist-ui/references/design-tokens.md +342 -342
- package/template/.aioson/skills/design/neo-brutalist-ui/references/motion.md +286 -286
- package/template/.aioson/skills/design/neo-brutalist-ui/references/patterns.md +458 -458
- package/template/.aioson/skills/design/neo-brutalist-ui/references/websites.md +723 -723
- package/template/.aioson/skills/design/premium-command-center-ui/SKILL.md +62 -62
- package/template/.aioson/skills/design/premium-command-center-ui/references/operations.md +74 -74
- package/template/.aioson/skills/design/premium-command-center-ui/references/patterns.md +116 -116
- package/template/.aioson/skills/design/premium-command-center-ui/references/validation.md +47 -47
- package/template/.aioson/skills/design/premium-command-center-ui/references/visual-system.md +215 -215
- package/template/.aioson/skills/design/pt.squarespace.com/.skill-meta.json +31 -31
- package/template/.aioson/skills/design/pt.squarespace.com/SKILL.md +66 -66
- package/template/.aioson/skills/design/pt.squarespace.com/references/components.md +368 -368
- package/template/.aioson/skills/design/pt.squarespace.com/references/design-tokens.md +150 -150
- package/template/.aioson/skills/design/pt.squarespace.com/references/motion.md +270 -270
- package/template/.aioson/skills/design/pt.squarespace.com/references/patterns.md +189 -189
- package/template/.aioson/skills/design/pt.squarespace.com/references/websites.md +165 -165
- package/template/.aioson/skills/design/warm-craft-ui/SKILL.md +209 -209
- package/template/.aioson/skills/design/warm-craft-ui/references/art-direction.md +324 -324
- package/template/.aioson/skills/design/warm-craft-ui/references/components.md +508 -508
- package/template/.aioson/skills/design/warm-craft-ui/references/dashboards.md +223 -223
- package/template/.aioson/skills/design/warm-craft-ui/references/design-tokens.md +374 -374
- package/template/.aioson/skills/design/warm-craft-ui/references/motion.md +356 -356
- package/template/.aioson/skills/design/warm-craft-ui/references/patterns.md +288 -288
- package/template/.aioson/skills/design/warm-craft-ui/references/websites.md +289 -289
- package/template/.aioson/skills/design-system/SKILL.md +92 -92
- package/template/.aioson/skills/design-system/components/SKILL.md +274 -274
- package/template/.aioson/skills/design-system/dashboards/SKILL.md +184 -184
- package/template/.aioson/skills/design-system/foundations/SKILL.md +250 -250
- package/template/.aioson/skills/design-system/motion/SKILL.md +197 -197
- package/template/.aioson/skills/design-system/patterns/SKILL.md +231 -231
- package/template/.aioson/skills/dynamic/README.md +30 -30
- package/template/.aioson/skills/dynamic/cardano-docs.md +16 -16
- package/template/.aioson/skills/dynamic/ethereum-docs.md +17 -17
- package/template/.aioson/skills/dynamic/flux-ui-docs.md +13 -13
- package/template/.aioson/skills/dynamic/laravel-docs.md +41 -41
- package/template/.aioson/skills/dynamic/npm-packages.md +16 -16
- package/template/.aioson/skills/dynamic/solana-docs.md +16 -16
- package/template/.aioson/skills/marketing/references/anti-patterns.md +254 -254
- package/template/.aioson/skills/marketing/references/cta-matrix.md +361 -0
- package/template/.aioson/skills/marketing/references/fascinations.md +192 -192
- package/template/.aioson/skills/marketing/references/five-acts.md +248 -248
- package/template/.aioson/skills/marketing/references/headline-matrix.md +358 -0
- package/template/.aioson/skills/marketing/references/market-intelligence.md +198 -198
- package/template/.aioson/skills/marketing/references/offer-structure.md +203 -203
- package/template/.aioson/skills/marketing/references/one-belief.md +149 -149
- package/template/.aioson/skills/marketing/references/patterns.md +218 -218
- package/template/.aioson/skills/marketing/references/platform-constraints.md +337 -0
- package/template/.aioson/skills/marketing/references/pms-research.md +193 -193
- package/template/.aioson/skills/marketing/vsl-craft.md +385 -385
- package/template/.aioson/skills/premium-visual-design/SKILL.md +83 -83
- package/template/.aioson/skills/premium-visual-design/components/agent-badge.md +92 -92
- package/template/.aioson/skills/premium-visual-design/components/dependency-node.md +102 -102
- package/template/.aioson/skills/premium-visual-design/components/mention-autocomplete.md +136 -136
- package/template/.aioson/skills/premium-visual-design/components/notification-center.md +136 -136
- package/template/.aioson/skills/premium-visual-design/components/review-action-bar.md +188 -188
- package/template/.aioson/skills/premium-visual-design/components/team-switcher.md +131 -131
- package/template/.aioson/skills/premium-visual-design/patterns/agent-message-thread.md +198 -198
- package/template/.aioson/skills/premium-visual-design/patterns/notification-panel.md +275 -275
- package/template/.aioson/skills/premium-visual-design/patterns/review-workflow-ui.md +234 -234
- package/template/.aioson/skills/premium-visual-design/patterns/task-dependency-graph.md +147 -147
- package/template/.aioson/skills/premium-visual-design/tokens/status-extended.md +142 -142
- package/template/.aioson/skills/process/aioson-spec-driven/SKILL.md +46 -46
- package/template/.aioson/skills/process/aioson-spec-driven/references/analyst.md +30 -30
- package/template/.aioson/skills/process/aioson-spec-driven/references/approval-gates.md +109 -109
- package/template/.aioson/skills/process/aioson-spec-driven/references/architect.md +23 -23
- package/template/.aioson/skills/process/aioson-spec-driven/references/artifact-map.md +44 -44
- package/template/.aioson/skills/process/aioson-spec-driven/references/classification-map.md +37 -37
- package/template/.aioson/skills/process/aioson-spec-driven/references/dev.md +47 -47
- package/template/.aioson/skills/process/aioson-spec-driven/references/deyvin.md +27 -27
- package/template/.aioson/skills/process/aioson-spec-driven/references/hardening-lane.md +49 -49
- package/template/.aioson/skills/process/aioson-spec-driven/references/maintenance-and-state.md +101 -101
- package/template/.aioson/skills/process/aioson-spec-driven/references/pm.md +30 -30
- package/template/.aioson/skills/process/aioson-spec-driven/references/product.md +25 -25
- package/template/.aioson/skills/process/aioson-spec-driven/references/qa.md +30 -30
- package/template/.aioson/skills/process/aioson-spec-driven/references/sheldon.md +25 -25
- package/template/.aioson/skills/process/aioson-spec-driven/references/ui-language.md +75 -75
- package/template/.aioson/skills/process/decision-presentation/SKILL.md +119 -0
- package/template/.aioson/skills/process/decision-presentation/references/jargon-map.en.yaml +108 -0
- package/template/.aioson/skills/process/decision-presentation/references/jargon-map.pt-BR.yaml +108 -0
- package/template/.aioson/skills/process/design-hybrid-forge/SKILL.md +147 -147
- package/template/.aioson/skills/process/design-hybrid-forge/references/crossover-protocol.md +221 -221
- package/template/.aioson/skills/process/design-hybrid-forge/references/naming-registry.md +88 -88
- package/template/.aioson/skills/process/design-hybrid-forge/references/output-contract.md +306 -306
- package/template/.aioson/skills/process/design-hybrid-forge/references/pair-compatibility.md +149 -149
- package/template/.aioson/skills/process/design-hybrid-forge/references/quality-gates.md +208 -208
- package/template/.aioson/skills/process/design-hybrid-forge/references/variation-library.md +125 -125
- package/template/.aioson/skills/process/secure-tdd/SKILL.md +97 -97
- package/template/.aioson/skills/process/simplify/SKILL.md +173 -173
- package/template/.aioson/skills/references/premium-command-center-ui/master-application-prompt.md +79 -79
- package/template/.aioson/skills/references/premium-command-center-ui/operational-ux-playbook.md +253 -253
- package/template/.aioson/skills/references/premium-command-center-ui/quality-validation-checklist.md +82 -82
- package/template/.aioson/skills/references/premium-command-center-ui/visual-system-and-component-patterns.md +270 -270
- package/template/.aioson/skills/squad/SKILL.md +58 -58
- package/template/.aioson/skills/squad/formats/catalog.json +15 -15
- package/template/.aioson/skills/squad/formats/content/blog-post.md +47 -47
- package/template/.aioson/skills/squad/formats/content/newsletter.md +47 -47
- package/template/.aioson/skills/squad/formats/creative/podcast-script.md +43 -43
- package/template/.aioson/skills/squad/formats/creative/video-script.md +41 -41
- package/template/.aioson/skills/squad/formats/social/instagram-feed.md +42 -42
- package/template/.aioson/skills/squad/formats/social/linkedin-post.md +42 -42
- package/template/.aioson/skills/squad/formats/social/tiktok.md +39 -39
- package/template/.aioson/skills/squad/formats/social/twitter-thread.md +39 -39
- package/template/.aioson/skills/squad/formats/social/youtube-long.md +47 -47
- package/template/.aioson/skills/squad/formats/social/youtube-shorts.md +39 -39
- package/template/.aioson/skills/squad/patterns/multi-platform-pattern.md +108 -108
- package/template/.aioson/skills/squad/patterns/persona-based-pattern.md +98 -98
- package/template/.aioson/skills/squad/patterns/pipeline-pattern.md +106 -106
- package/template/.aioson/skills/squad/patterns/review-loop-pattern.md +81 -81
- package/template/.aioson/skills/squad/references/checklist-templates.md +122 -122
- package/template/.aioson/skills/squad/references/executor-archetypes.md +123 -123
- package/template/.aioson/skills/squad/references/workflow-templates.md +169 -169
- package/template/.aioson/skills/static/context-budget-guide.md +46 -46
- package/template/.aioson/skills/static/debugging-protocol.md +42 -42
- package/template/.aioson/skills/static/django-patterns.md +342 -342
- package/template/.aioson/skills/static/fastapi-patterns.md +344 -344
- package/template/.aioson/skills/static/filament-patterns.md +267 -267
- package/template/.aioson/skills/static/flux-ui-components.md +262 -262
- package/template/.aioson/skills/static/git-conventions.md +227 -227
- package/template/.aioson/skills/static/git-worktrees.md +36 -36
- package/template/.aioson/skills/static/harness-sensors.md +74 -74
- package/template/.aioson/skills/static/harness-validate/SKILL.md +46 -46
- package/template/.aioson/skills/static/jetstream-setup.md +200 -200
- package/template/.aioson/skills/static/landing-page-deploy.md +192 -192
- package/template/.aioson/skills/static/landing-page-forge.md +730 -730
- package/template/.aioson/skills/static/laravel-conventions.md +491 -491
- package/template/.aioson/skills/static/multi-agent-patterns.md +43 -43
- package/template/.aioson/skills/static/nextjs-patterns.md +321 -321
- package/template/.aioson/skills/static/node-express-patterns.md +317 -317
- package/template/.aioson/skills/static/node-typescript-patterns.md +282 -282
- package/template/.aioson/skills/static/rails-conventions.md +307 -307
- package/template/.aioson/skills/static/react-motion-patterns.md +599 -599
- package/template/.aioson/skills/static/static-html-patterns/checklists.md +43 -43
- package/template/.aioson/skills/static/static-html-patterns/css-tokens.md +609 -609
- package/template/.aioson/skills/static/static-html-patterns/motion.md +193 -193
- package/template/.aioson/skills/static/static-html-patterns/premium.md +711 -711
- package/template/.aioson/skills/static/static-html-patterns/structure.md +209 -209
- package/template/.aioson/skills/static/static-html-patterns/utilities.md +190 -190
- package/template/.aioson/skills/static/static-html-patterns.md +80 -80
- package/template/.aioson/skills/static/tall-stack-patterns.md +286 -286
- package/template/.aioson/skills/static/threejs-patterns.md +929 -929
- package/template/.aioson/skills/static/ui-ux-modern.md +76 -76
- package/template/.aioson/skills/static/web-research-cache.md +115 -115
- package/template/.aioson/skills/static/web3-cardano-patterns.md +337 -337
- package/template/.aioson/skills/static/web3-ethereum-patterns.md +310 -310
- package/template/.aioson/skills/static/web3-security-checklist.md +284 -284
- package/template/.aioson/skills/static/web3-solana-patterns.md +324 -324
- package/template/.aioson/squads/memory.md +5 -5
- package/template/.aioson/tasks/implementation-plan.md +327 -327
- package/template/.aioson/tasks/squad-analyze.md +83 -83
- package/template/.aioson/tasks/squad-create.md +148 -148
- package/template/.aioson/tasks/squad-design.md +206 -206
- package/template/.aioson/tasks/squad-execution-plan.md +279 -279
- package/template/.aioson/tasks/squad-export.md +20 -20
- package/template/.aioson/tasks/squad-extend.md +68 -68
- package/template/.aioson/tasks/squad-investigate.md +57 -57
- package/template/.aioson/tasks/squad-learning-review.md +44 -44
- package/template/.aioson/tasks/squad-output-config.md +177 -177
- package/template/.aioson/tasks/squad-pipeline.md +122 -122
- package/template/.aioson/tasks/squad-profile.md +48 -48
- package/template/.aioson/tasks/squad-refresh.md +236 -0
- package/template/.aioson/tasks/squad-repair.md +85 -85
- package/template/.aioson/tasks/squad-review.md +61 -61
- package/template/.aioson/tasks/squad-task-decompose.md +66 -66
- package/template/.aioson/tasks/squad-validate.md +58 -58
- package/template/.aioson/templates/reflect-prompts/current-state.md +36 -0
- package/template/.aioson/templates/reflect-prompts/how-it-works.md +23 -0
- package/template/.aioson/templates/reflect-prompts/what-it-does.md +21 -0
- package/template/.aioson/templates/squads/content-basic/template.json +21 -21
- package/template/.aioson/templates/squads/digital-marketing-agency/template.json +96 -96
- package/template/.aioson/templates/squads/media-channel/template.json +24 -24
- package/template/.aioson/templates/squads/research-analysis/template.json +22 -22
- package/template/.aioson/templates/squads/software-delivery/template.json +21 -21
- package/template/.claude/commands/aioson/agent/analyst.md +5 -5
- package/template/.claude/commands/aioson/agent/architect.md +5 -5
- package/template/.claude/commands/aioson/agent/briefing.md +5 -0
- package/template/.claude/commands/aioson/agent/committer.md +5 -5
- package/template/.claude/commands/aioson/agent/copywriter.md +5 -5
- package/template/.claude/commands/aioson/agent/design-hybrid-forge.md +5 -5
- package/template/.claude/commands/aioson/agent/dev.md +5 -5
- package/template/.claude/commands/aioson/agent/deyvin.md +5 -5
- package/template/.claude/commands/aioson/agent/discover.md +5 -0
- package/template/.claude/commands/aioson/agent/discovery-design-doc.md +5 -5
- package/template/.claude/commands/aioson/agent/genome.md +5 -5
- package/template/.claude/commands/aioson/agent/neo.md +5 -5
- package/template/.claude/commands/aioson/agent/orache.md +5 -5
- package/template/.claude/commands/aioson/agent/orchestrator.md +5 -5
- package/template/.claude/commands/aioson/agent/pair.md +5 -5
- package/template/.claude/commands/aioson/agent/pentester.md +5 -0
- package/template/.claude/commands/aioson/agent/pm.md +5 -5
- package/template/.claude/commands/aioson/agent/product.md +5 -5
- package/template/.claude/commands/aioson/agent/profiler-enricher.md +5 -5
- package/template/.claude/commands/aioson/agent/profiler-forge.md +5 -5
- package/template/.claude/commands/aioson/agent/profiler-researcher.md +5 -5
- package/template/.claude/commands/aioson/agent/qa.md +5 -5
- package/template/.claude/commands/aioson/agent/setup.md +5 -5
- package/template/.claude/commands/aioson/agent/sheldon.md +5 -5
- package/template/.claude/commands/aioson/agent/site-forge.md +5 -5
- package/template/.claude/commands/aioson/agent/squad.md +5 -5
- package/template/.claude/commands/aioson/agent/tester.md +5 -5
- package/template/.claude/commands/aioson/agent/ux-ui.md +5 -5
- package/template/.claude/commands/aioson/agent/validator.md +5 -5
- package/template/.gemini/GEMINI.md +13 -13
- package/template/.gemini/commands/aios-analyst.toml +7 -7
- package/template/.gemini/commands/aios-architect.toml +8 -8
- package/template/.gemini/commands/aios-committer.toml +7 -7
- package/template/.gemini/commands/aios-copywriter.toml +7 -7
- package/template/.gemini/commands/aios-cypher.toml +7 -7
- package/template/.gemini/commands/aios-dev.toml +9 -9
- package/template/.gemini/commands/aios-deyvin.toml +7 -7
- package/template/.gemini/commands/aios-discover.toml +6 -0
- package/template/.gemini/commands/aios-discovery-design-doc.toml +7 -7
- package/template/.gemini/commands/aios-genome.toml +7 -7
- package/template/.gemini/commands/aios-neo.toml +6 -6
- package/template/.gemini/commands/aios-orache.toml +7 -7
- package/template/.gemini/commands/aios-orchestrator.toml +9 -9
- package/template/.gemini/commands/aios-pair.toml +7 -7
- package/template/.gemini/commands/aios-pm.toml +9 -9
- package/template/.gemini/commands/aios-product.toml +6 -6
- package/template/.gemini/commands/aios-qa.toml +7 -7
- package/template/.gemini/commands/aios-setup.toml +6 -6
- package/template/.gemini/commands/aios-sheldon.toml +7 -7
- package/template/.gemini/commands/aios-site-forge.toml +7 -7
- package/template/.gemini/commands/aios-squad.toml +7 -7
- package/template/.gemini/commands/aios-tester.toml +7 -7
- package/template/.gemini/commands/aios-ux-ui.toml +9 -9
- package/template/.gemini/commands/aios-validator.toml +7 -7
- package/template/AGENTS.md +184 -183
- package/template/CLAUDE.md +98 -97
- package/template/OPENCODE.md +35 -34
- package/template/aioson-models.json +40 -40
- package/template/.aioson/genomes/copywriting.md +0 -204
- package/template/.aioson/genomes/copywriting.meta.json +0 -48
- package/template/.aioson/skills/process/secure-tdd/references/nextjs.md +0 -81
- package/template/.aioson/skills/process/secure-tdd/references/node-express.md +0 -91
- package/template/.aioson/skills/process/secure-tdd/references/planned-stacks.md +0 -33
- package/template/.claude/commands/aioson/agent/cypher.md +0 -5
package/src/commands/qa-run.js
CHANGED
|
@@ -1,873 +1,873 @@
|
|
|
1
|
-
'use strict';
|
|
2
|
-
|
|
3
|
-
const path = require('node:path');
|
|
4
|
-
const fs = require('node:fs/promises');
|
|
5
|
-
const { readTextIfExists, ensureDir } = require('../utils');
|
|
6
|
-
|
|
7
|
-
// --- Secret patterns for exposure detection ---
|
|
8
|
-
const SECRET_PATTERNS = [
|
|
9
|
-
{ name: 'OpenAI key', regex: /sk-[a-zA-Z0-9]{20,}/ },
|
|
10
|
-
{ name: 'Stripe live key', regex: /pk_live_[a-zA-Z0-9]{20,}/ },
|
|
11
|
-
{ name: 'Stripe test key', regex: /pk_test_[a-zA-Z0-9]{20,}/ },
|
|
12
|
-
{ name: 'AWS access key', regex: /AKIA[A-Z0-9]{16}/ },
|
|
13
|
-
{ name: 'Google API key', regex: /AIzaSy[a-zA-Z0-9_-]{33}/ },
|
|
14
|
-
{ name: 'GitHub token', regex: /gh[ps]_[a-zA-Z0-9]{36}/ },
|
|
15
|
-
{ name: 'Slack token', regex: /xox[bpa]-[a-zA-Z0-9-]+/ },
|
|
16
|
-
{ name: 'Generic secret', regex: /(SECRET|TOKEN|PASSWORD|PRIVATE_KEY)\s*[:=]\s*['"]?[a-zA-Z0-9_/+=-]{16,}/i }
|
|
17
|
-
];
|
|
18
|
-
|
|
19
|
-
const SENSITIVE_FILE_PATHS = [
|
|
20
|
-
'/.env', '/.env.local', '/.env.production', '/.env.development',
|
|
21
|
-
'/.git/config', '/config.js', '/api/config', '/wp-config.php',
|
|
22
|
-
'/application.yml', '/application.properties'
|
|
23
|
-
];
|
|
24
|
-
|
|
25
|
-
const DEBUG_ROUTES = [
|
|
26
|
-
'/admin', '/debug', '/phpinfo.php', '/_debug',
|
|
27
|
-
'/api/health', '/api/debug', '/__nextjs_original-stack-frame',
|
|
28
|
-
'/api/env', '/server-status'
|
|
29
|
-
];
|
|
30
|
-
|
|
31
|
-
// --- Playwright gate ---
|
|
32
|
-
function requirePlaywright() {
|
|
33
|
-
try { return require('playwright'); } catch { return null; }
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
// --- Config ---
|
|
37
|
-
async function loadConfig(targetDir) {
|
|
38
|
-
try {
|
|
39
|
-
const raw = await fs.readFile(path.join(targetDir, 'aios-qa.config.json'), 'utf8');
|
|
40
|
-
return JSON.parse(raw);
|
|
41
|
-
} catch { return null; }
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
// --- Finding factory ---
|
|
45
|
-
let _counter = 0;
|
|
46
|
-
function makeFinding(severity, category, title, location, risk, fix) {
|
|
47
|
-
_counter++;
|
|
48
|
-
const prefix = severity[0].toUpperCase();
|
|
49
|
-
const id = `${prefix}-${String(_counter).padStart(2, '0')}`;
|
|
50
|
-
return { id, severity, category, title, location, risk, fix, screenshot: '' };
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
// --- Screenshot helper ---
|
|
54
|
-
async function takeScreenshot(page, screenshotsDir, id) {
|
|
55
|
-
try {
|
|
56
|
-
const file = path.join(screenshotsDir, `${id}.png`);
|
|
57
|
-
await page.screenshot({ path: file, fullPage: false });
|
|
58
|
-
return file;
|
|
59
|
-
} catch { return ''; }
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
// ============================================================
|
|
63
|
-
// SECURITY PROBES
|
|
64
|
-
// ============================================================
|
|
65
|
-
|
|
66
|
-
async function probeExposedSecrets(page, findings, screenshotsDir) {
|
|
67
|
-
// Check window globals: Next.js __NEXT_DATA__, window.ENV, etc.
|
|
68
|
-
const exposed = await page.evaluate((patterns) => {
|
|
69
|
-
const sources = {
|
|
70
|
-
'__NEXT_DATA__': window.__NEXT_DATA__,
|
|
71
|
-
'__env__': window.__env__,
|
|
72
|
-
'ENV': window.ENV,
|
|
73
|
-
'_env': window._env,
|
|
74
|
-
'CONFIG': window.CONFIG,
|
|
75
|
-
'APP_CONFIG': window.APP_CONFIG
|
|
76
|
-
};
|
|
77
|
-
const found = [];
|
|
78
|
-
for (const [src, val] of Object.entries(sources)) {
|
|
79
|
-
if (!val) continue;
|
|
80
|
-
const str = JSON.stringify(val);
|
|
81
|
-
for (const { name, regex } of patterns) {
|
|
82
|
-
if (new RegExp(regex).test(str)) found.push({ source: src, keyType: name });
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
return found;
|
|
86
|
-
}, SECRET_PATTERNS.map((p) => ({ name: p.name, regex: p.regex.source }))).catch(() => []);
|
|
87
|
-
|
|
88
|
-
for (const item of exposed) {
|
|
89
|
-
const f = makeFinding(
|
|
90
|
-
'critical', 'security',
|
|
91
|
-
`${item.keyType} exposed in window.${item.source}`,
|
|
92
|
-
`window.${item.source}`,
|
|
93
|
-
`${item.keyType} is visible to any browser user via the global object. Direct financial or account compromise exposure.`,
|
|
94
|
-
`Move to server-side only. Never use NEXT_PUBLIC_ or client-side globals for secrets.`
|
|
95
|
-
);
|
|
96
|
-
f.screenshot = await takeScreenshot(page, screenshotsDir, f.id);
|
|
97
|
-
findings.push(f);
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
// Also scan the rendered HTML source
|
|
101
|
-
const html = await page.content().catch(() => '');
|
|
102
|
-
for (const { name, regex } of SECRET_PATTERNS) {
|
|
103
|
-
if (regex.test(html)) {
|
|
104
|
-
findings.push(makeFinding(
|
|
105
|
-
'critical', 'security',
|
|
106
|
-
`${name} found in rendered HTML source`,
|
|
107
|
-
'Page HTML source',
|
|
108
|
-
`${name} is embedded in the HTML sent to the browser. Visible to anyone using DevTools.`,
|
|
109
|
-
'Remove from client-side rendering. Serve secrets only from server-side APIs.'
|
|
110
|
-
));
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
async function probeSensitiveFiles(page, baseUrl, findings) {
|
|
116
|
-
for (const filePath of SENSITIVE_FILE_PATHS) {
|
|
117
|
-
try {
|
|
118
|
-
const response = await page.goto(`${baseUrl}${filePath}`, {
|
|
119
|
-
waitUntil: 'commit', timeout: 5000
|
|
120
|
-
});
|
|
121
|
-
if (response && response.status() === 200) {
|
|
122
|
-
const body = await response.text().catch(() => '');
|
|
123
|
-
const looksLikeSensitive =
|
|
124
|
-
/[A-Z_]{3,}=/.test(body) ||
|
|
125
|
-
/<\?php/.test(body) ||
|
|
126
|
-
/(SECRET|PASSWORD|TOKEN|KEY|PRIVATE)/i.test(body);
|
|
127
|
-
if (looksLikeSensitive) {
|
|
128
|
-
findings.push(makeFinding(
|
|
129
|
-
'critical', 'security',
|
|
130
|
-
`Sensitive file publicly accessible: ${filePath}`,
|
|
131
|
-
`${baseUrl}${filePath}`,
|
|
132
|
-
`Configuration file is reachable by any internet user. May expose credentials, connection strings, or infrastructure details.`,
|
|
133
|
-
`Block ${filePath} in your web server (nginx/vercel/apache). Never deploy .env files to public directories.`
|
|
134
|
-
));
|
|
135
|
-
}
|
|
136
|
-
}
|
|
137
|
-
} catch { /* not accessible — good */ }
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
async function probeXss(page, baseUrl, findings, screenshotsDir) {
|
|
142
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
143
|
-
|
|
144
|
-
let xssFired = false;
|
|
145
|
-
page.on('dialog', async (dialog) => {
|
|
146
|
-
xssFired = true;
|
|
147
|
-
await dialog.dismiss().catch(() => {});
|
|
148
|
-
});
|
|
149
|
-
|
|
150
|
-
const inputs = await page.$$('input[type="text"],input[type="search"],input[type="email"],input[type="url"],textarea').catch(() => []);
|
|
151
|
-
for (const input of inputs.slice(0, 10)) {
|
|
152
|
-
await input.fill('<img src=x onerror="window.__xss=1">').catch(() => {});
|
|
153
|
-
}
|
|
154
|
-
await page.keyboard.press('Tab').catch(() => {});
|
|
155
|
-
await page.waitForTimeout(500).catch(() => {});
|
|
156
|
-
|
|
157
|
-
const xssEval = await page.evaluate(() => window.__xss === 1).catch(() => false);
|
|
158
|
-
if (xssFired || xssEval) {
|
|
159
|
-
const f = makeFinding(
|
|
160
|
-
'critical', 'security',
|
|
161
|
-
'XSS (Cross-Site Scripting) — injected script executed',
|
|
162
|
-
'Form text inputs',
|
|
163
|
-
'User input is rendered as HTML without sanitization. Attacker can steal sessions, redirect users, or deface the page.',
|
|
164
|
-
'Sanitize all user input before rendering. Use textContent instead of innerHTML. Apply a strict Content-Security-Policy header.'
|
|
165
|
-
);
|
|
166
|
-
f.screenshot = await takeScreenshot(page, screenshotsDir, f.id);
|
|
167
|
-
findings.push(f);
|
|
168
|
-
}
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
async function probeOpenRedirect(page, baseUrl, findings) {
|
|
172
|
-
const params = ['redirect', 'next', 'return', 'returnUrl', 'return_url', 'goto', 'url', 'dest'];
|
|
173
|
-
const evil = 'https://evil-phishing-example.com';
|
|
174
|
-
for (const param of params) {
|
|
175
|
-
try {
|
|
176
|
-
const response = await page.goto(`${baseUrl}?${param}=${encodeURIComponent(evil)}`, {
|
|
177
|
-
waitUntil: 'commit', timeout: 5000
|
|
178
|
-
});
|
|
179
|
-
const finalUrl = page.url();
|
|
180
|
-
const redirected = finalUrl.startsWith(evil) ||
|
|
181
|
-
(response && [301, 302, 303, 307, 308].includes(response.status()) &&
|
|
182
|
-
String(response.headers()['location'] || '').startsWith(evil));
|
|
183
|
-
if (redirected) {
|
|
184
|
-
findings.push(makeFinding(
|
|
185
|
-
'high', 'security',
|
|
186
|
-
`Open redirect via ?${param}= parameter`,
|
|
187
|
-
`${baseUrl}?${param}=`,
|
|
188
|
-
'Attacker can use your trusted domain to redirect users to phishing sites. Bypasses browser warnings.',
|
|
189
|
-
'Validate redirect targets against an allowlist of trusted paths. Reject external URLs unconditionally.'
|
|
190
|
-
));
|
|
191
|
-
break;
|
|
192
|
-
}
|
|
193
|
-
} catch { /* not redirected */ }
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
async function probeInjectionInputs(page, baseUrl, findings) {
|
|
198
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
199
|
-
const inputs = await page.$$('input[type="text"],input[type="search"],textarea').catch(() => []);
|
|
200
|
-
const sqlPayload = `' OR '1'='1' -- `;
|
|
201
|
-
for (const input of inputs.slice(0, 5)) {
|
|
202
|
-
await input.fill(sqlPayload).catch(() => {});
|
|
203
|
-
}
|
|
204
|
-
await page.waitForTimeout(500).catch(() => {});
|
|
205
|
-
const html = await page.content().catch(() => '');
|
|
206
|
-
if (/(SQL syntax|mysql_fetch|ORA-|pg_query|sqlite_|SQLSTATE)/i.test(html)) {
|
|
207
|
-
findings.push(makeFinding(
|
|
208
|
-
'critical', 'security',
|
|
209
|
-
'SQL error message exposed after injection probe',
|
|
210
|
-
'Form inputs',
|
|
211
|
-
'A SQL error was returned when input contained single quotes. Indicates raw string interpolation in queries.',
|
|
212
|
-
'Use parameterized queries or ORM exclusively. Never build SQL strings with user input. Disable detailed DB error messages in production.'
|
|
213
|
-
));
|
|
214
|
-
}
|
|
215
|
-
}
|
|
216
|
-
|
|
217
|
-
async function probeDebugRoutes(page, baseUrl, findings) {
|
|
218
|
-
for (const route of DEBUG_ROUTES) {
|
|
219
|
-
try {
|
|
220
|
-
const response = await page.goto(`${baseUrl}${route}`, { waitUntil: 'commit', timeout: 4000 });
|
|
221
|
-
if (response && response.status() === 200) {
|
|
222
|
-
const title = await page.title().catch(() => '');
|
|
223
|
-
if (!/404|not found/i.test(title)) {
|
|
224
|
-
findings.push(makeFinding(
|
|
225
|
-
'medium', 'security',
|
|
226
|
-
`Debug/admin route accessible without authentication: ${route}`,
|
|
227
|
-
`${baseUrl}${route}`,
|
|
228
|
-
'Unauthenticated access to debug/admin endpoints may expose metrics, logs, internal state, or admin controls.',
|
|
229
|
-
`Require authentication for ${route}. Restrict to internal network or remove in production.`
|
|
230
|
-
));
|
|
231
|
-
}
|
|
232
|
-
}
|
|
233
|
-
} catch { /* not accessible */ }
|
|
234
|
-
}
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
async function probeInjectionInjection(page, findings, consoleLog, networkRequests) {
|
|
238
|
-
// Console error leakage
|
|
239
|
-
const stackTraceRx = /at\s+\w+\s+\(/;
|
|
240
|
-
const sensitiveRx = /(token|secret|password|key|authorization)/i;
|
|
241
|
-
const leaks = consoleLog.filter((l) =>
|
|
242
|
-
l.type === 'error' && (stackTraceRx.test(l.text) || sensitiveRx.test(l.text))
|
|
243
|
-
);
|
|
244
|
-
if (leaks.length > 0) {
|
|
245
|
-
findings.push(makeFinding(
|
|
246
|
-
'medium', 'security',
|
|
247
|
-
`Browser console exposes ${leaks.length} error(s) with stack traces or sensitive keywords`,
|
|
248
|
-
'Browser DevTools console',
|
|
249
|
-
'Stack traces reveal file paths, library versions, and logic. Sensitive keywords may expose credentials or tokens.',
|
|
250
|
-
'Disable verbose error logging in production. Use a centralized error service (Sentry, Datadog) instead of console.error.'
|
|
251
|
-
));
|
|
252
|
-
}
|
|
253
|
-
|
|
254
|
-
// Sensitive data in GET params
|
|
255
|
-
const sensitiveParamRx = /(token|secret|password|api_key|apikey|auth)/i;
|
|
256
|
-
for (const req of networkRequests) {
|
|
257
|
-
try {
|
|
258
|
-
const url = new URL(req.url);
|
|
259
|
-
for (const [key] of url.searchParams) {
|
|
260
|
-
if (sensitiveParamRx.test(key)) {
|
|
261
|
-
findings.push(makeFinding(
|
|
262
|
-
'high', 'security',
|
|
263
|
-
`Sensitive parameter "${key}" transmitted in GET URL`,
|
|
264
|
-
req.url.substring(0, 120),
|
|
265
|
-
'Sensitive data in URLs is logged by web servers, proxies, CDNs, and browser history. Leaks through Referer headers.',
|
|
266
|
-
'Move sensitive parameters to POST body or Authorization header. Never pass secrets in query strings.'
|
|
267
|
-
));
|
|
268
|
-
break;
|
|
269
|
-
}
|
|
270
|
-
}
|
|
271
|
-
} catch { /* invalid URL */ }
|
|
272
|
-
}
|
|
273
|
-
|
|
274
|
-
// HTTP requests from HTTPS page (mixed content)
|
|
275
|
-
const mixed = networkRequests.filter((r) =>
|
|
276
|
-
r.url.startsWith('http://') && !r.url.startsWith('http://localhost') && !r.url.startsWith('http://127.')
|
|
277
|
-
);
|
|
278
|
-
if (mixed.length > 0) {
|
|
279
|
-
findings.push(makeFinding(
|
|
280
|
-
'medium', 'security',
|
|
281
|
-
`${mixed.length} mixed content request(s) — HTTP resources on page`,
|
|
282
|
-
mixed[0].url.substring(0, 120),
|
|
283
|
-
'HTTP requests from an HTTPS page expose data in transit. Modern browsers block or warn on mixed content.',
|
|
284
|
-
'Upgrade all resource references to HTTPS. Use protocol-relative URLs (//) or absolute HTTPS URLs.'
|
|
285
|
-
));
|
|
286
|
-
}
|
|
287
|
-
}
|
|
288
|
-
|
|
289
|
-
// ============================================================
|
|
290
|
-
// PERSONA — NAIVE USER
|
|
291
|
-
// ============================================================
|
|
292
|
-
async function runNaivePersona(page, baseUrl, findings, screenshotsDir) {
|
|
293
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
294
|
-
|
|
295
|
-
// Submit all empty forms
|
|
296
|
-
const forms = await page.$$('form').catch(() => []);
|
|
297
|
-
for (const form of forms.slice(0, 5)) {
|
|
298
|
-
const beforeUrl = page.url();
|
|
299
|
-
await page.evaluate((f) => { try { f.submit(); } catch (_) {} }, form).catch(() => {});
|
|
300
|
-
await page.waitForTimeout(800).catch(() => {});
|
|
301
|
-
const title = await page.title().catch(() => '');
|
|
302
|
-
const html = await page.content().catch(() => '');
|
|
303
|
-
if (/error|exception|stacktrace|500/i.test(title) || /Internal Server Error|Uncaught Exception/i.test(html)) {
|
|
304
|
-
const f = makeFinding(
|
|
305
|
-
'high', 'reliability',
|
|
306
|
-
'Empty form submission causes server error (5xx)',
|
|
307
|
-
`Form on ${beforeUrl}`,
|
|
308
|
-
'Server returns 5xx when form is submitted empty. Missing or bypassed server-side validation.',
|
|
309
|
-
'Add server-side validation before processing. Return 422 with field-specific errors instead of throwing 500.'
|
|
310
|
-
);
|
|
311
|
-
f.screenshot = await takeScreenshot(page, screenshotsDir, f.id);
|
|
312
|
-
findings.push(f);
|
|
313
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 10000 }).catch(() => {});
|
|
314
|
-
}
|
|
315
|
-
}
|
|
316
|
-
|
|
317
|
-
// Type very long strings (buffer overflow / DoS potential)
|
|
318
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 10000 }).catch(() => {});
|
|
319
|
-
const inputs = await page.$$('input[type="text"],input[type="email"],input[type="search"],textarea').catch(() => []);
|
|
320
|
-
const longStr = 'A'.repeat(10000);
|
|
321
|
-
for (const input of inputs.slice(0, 5)) {
|
|
322
|
-
await input.fill(longStr).catch(() => {});
|
|
323
|
-
}
|
|
324
|
-
await page.waitForTimeout(500).catch(() => {});
|
|
325
|
-
const srcAfterLong = await page.content().catch(() => '');
|
|
326
|
-
if (/maximum call stack|out of memory|RangeError|Cannot read/i.test(srcAfterLong)) {
|
|
327
|
-
findings.push(makeFinding(
|
|
328
|
-
'high', 'reliability',
|
|
329
|
-
'Application crashes on very long input (10,000 chars)',
|
|
330
|
-
'Text input fields',
|
|
331
|
-
'No length validation at the UI boundary causes a client-side crash on extreme input.',
|
|
332
|
-
'Add maxlength attribute to inputs and validate length server-side before any processing.'
|
|
333
|
-
));
|
|
334
|
-
}
|
|
335
|
-
|
|
336
|
-
// Detect ghost clickables: cursor:pointer with no handler
|
|
337
|
-
const deadClicks = await page.evaluate(() => {
|
|
338
|
-
const els = document.querySelectorAll('[style*="cursor: pointer"],[class*="cursor-pointer"],[class*="cursor_pointer"]');
|
|
339
|
-
const dead = [];
|
|
340
|
-
for (const el of els) {
|
|
341
|
-
const tag = el.tagName.toLowerCase();
|
|
342
|
-
const isInteractive = ['a', 'button', 'input', 'select', 'textarea'].includes(tag);
|
|
343
|
-
const hasRole = el.getAttribute('role');
|
|
344
|
-
const hasClick = el.onclick !== null || el.getAttribute('data-action');
|
|
345
|
-
if (!isInteractive && !hasRole && !hasClick) {
|
|
346
|
-
dead.push((el.className || el.id || tag).toString().substring(0, 60));
|
|
347
|
-
}
|
|
348
|
-
}
|
|
349
|
-
return dead.slice(0, 10);
|
|
350
|
-
}).catch(() => []);
|
|
351
|
-
|
|
352
|
-
if (deadClicks.length > 0) {
|
|
353
|
-
findings.push(makeFinding(
|
|
354
|
-
'low', 'ux',
|
|
355
|
-
`${deadClicks.length} element(s) appear clickable (cursor:pointer) but have no action`,
|
|
356
|
-
`Elements: ${deadClicks.slice(0, 3).join(' | ')}`,
|
|
357
|
-
'Cursor changes to pointer but clicking does nothing. Confuses users and erodes trust.',
|
|
358
|
-
'Remove cursor:pointer from non-interactive elements, or add the appropriate click handler or ARIA role.'
|
|
359
|
-
));
|
|
360
|
-
}
|
|
361
|
-
}
|
|
362
|
-
|
|
363
|
-
// ============================================================
|
|
364
|
-
// PERSONA — HACKER
|
|
365
|
-
// ============================================================
|
|
366
|
-
async function runHackerPersona(page, baseUrl, findings, screenshotsDir) {
|
|
367
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
368
|
-
|
|
369
|
-
await probeExposedSecrets(page, findings, screenshotsDir);
|
|
370
|
-
await probeSensitiveFiles(page, baseUrl, findings);
|
|
371
|
-
await probeXss(page, baseUrl, findings, screenshotsDir);
|
|
372
|
-
await probeOpenRedirect(page, baseUrl, findings);
|
|
373
|
-
await probeInjectionInputs(page, baseUrl, findings);
|
|
374
|
-
await probeDebugRoutes(page, baseUrl, findings);
|
|
375
|
-
|
|
376
|
-
// IDOR probe: detect numeric IDs in current URL, try ±1
|
|
377
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 10000 }).catch(() => {});
|
|
378
|
-
const currentUrl = page.url();
|
|
379
|
-
const idMatch = currentUrl.match(/\/(\d{1,9})(\/|$|\?)/);
|
|
380
|
-
if (idMatch) {
|
|
381
|
-
const id = parseInt(idMatch[1], 10);
|
|
382
|
-
for (const delta of [-1, 1, 9999]) {
|
|
383
|
-
try {
|
|
384
|
-
const probe = currentUrl.replace(`/${id}`, `/${id + delta}`);
|
|
385
|
-
const response = await page.goto(probe, { waitUntil: 'commit', timeout: 5000 });
|
|
386
|
-
if (response && response.status() === 200) {
|
|
387
|
-
findings.push(makeFinding(
|
|
388
|
-
'high', 'security',
|
|
389
|
-
`Potential IDOR: resource /${id} → /${id + delta} returns 200`,
|
|
390
|
-
probe,
|
|
391
|
-
'Incrementing the resource ID in the URL returns a valid response with no authorization rejection. May expose other users\' data.',
|
|
392
|
-
'Implement per-resource authorization: verify the authenticated user owns or is permitted to access the requested ID.'
|
|
393
|
-
));
|
|
394
|
-
break;
|
|
395
|
-
}
|
|
396
|
-
} catch { /* not accessible */ }
|
|
397
|
-
}
|
|
398
|
-
}
|
|
399
|
-
}
|
|
400
|
-
|
|
401
|
-
// ============================================================
|
|
402
|
-
// PERSONA — POWER USER
|
|
403
|
-
// ============================================================
|
|
404
|
-
async function runPowerPersona(page, baseUrl, findings) {
|
|
405
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
406
|
-
|
|
407
|
-
// Keyboard navigation — visible focus indicator
|
|
408
|
-
let missingFocus = 0;
|
|
409
|
-
for (let i = 0; i < 25; i++) {
|
|
410
|
-
await page.keyboard.press('Tab').catch(() => {});
|
|
411
|
-
const focusOk = await page.evaluate(() => {
|
|
412
|
-
const el = document.activeElement;
|
|
413
|
-
if (!el || el === document.body) return true;
|
|
414
|
-
const style = window.getComputedStyle(el);
|
|
415
|
-
const outline = style.outline || '';
|
|
416
|
-
return outline !== 'none' && !outline.startsWith('0px');
|
|
417
|
-
}).catch(() => true);
|
|
418
|
-
if (!focusOk) missingFocus++;
|
|
419
|
-
}
|
|
420
|
-
|
|
421
|
-
if (missingFocus > 2) {
|
|
422
|
-
findings.push(makeFinding(
|
|
423
|
-
'medium', 'accessibility',
|
|
424
|
-
`Keyboard focus indicator missing on ${missingFocus} interactive element(s)`,
|
|
425
|
-
'Tab key navigation',
|
|
426
|
-
'Users relying on keyboard cannot see which element is focused. WCAG 2.1 SC 2.4.7 (Level AA) violation.',
|
|
427
|
-
'Never use outline: none without an equivalent visible :focus-visible alternative. Ensure contrast ratio ≥ 3:1.'
|
|
428
|
-
));
|
|
429
|
-
}
|
|
430
|
-
|
|
431
|
-
// Boundary values on number inputs
|
|
432
|
-
const numberInputs = await page.$$('input[type="number"],input[type="range"]').catch(() => []);
|
|
433
|
-
for (const input of numberInputs.slice(0, 5)) {
|
|
434
|
-
for (const value of ['-999999999', '0', '999999999999999', '9007199254740992']) {
|
|
435
|
-
await input.fill(value).catch(() => {});
|
|
436
|
-
await page.keyboard.press('Tab').catch(() => {});
|
|
437
|
-
await page.waitForTimeout(200).catch(() => {});
|
|
438
|
-
}
|
|
439
|
-
}
|
|
440
|
-
|
|
441
|
-
// Date boundary values
|
|
442
|
-
const dateInputs = await page.$$('input[type="date"]').catch(() => []);
|
|
443
|
-
for (const input of dateInputs.slice(0, 3)) {
|
|
444
|
-
for (const value of ['1900-01-01', '9999-12-31', '2000-02-29']) {
|
|
445
|
-
await input.fill(value).catch(() => {});
|
|
446
|
-
await page.keyboard.press('Tab').catch(() => {});
|
|
447
|
-
}
|
|
448
|
-
}
|
|
449
|
-
}
|
|
450
|
-
|
|
451
|
-
// ============================================================
|
|
452
|
-
// PERSONA — MOBILE
|
|
453
|
-
// ============================================================
|
|
454
|
-
async function runMobilePersona(browser, baseUrl, findings, screenshotsDir) {
|
|
455
|
-
const context = await browser.newContext({
|
|
456
|
-
viewport: { width: 375, height: 667 },
|
|
457
|
-
userAgent: 'Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.0 Mobile/15E148 Safari/604.1',
|
|
458
|
-
hasTouch: true,
|
|
459
|
-
isMobile: true
|
|
460
|
-
}).catch(() => null);
|
|
461
|
-
if (!context) return;
|
|
462
|
-
|
|
463
|
-
const page = await context.newPage().catch(() => null);
|
|
464
|
-
if (!page) { await context.close().catch(() => {}); return; }
|
|
465
|
-
|
|
466
|
-
try {
|
|
467
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
468
|
-
|
|
469
|
-
// Horizontal overflow
|
|
470
|
-
const overflow = await page.evaluate(() => document.body.scrollWidth > window.innerWidth + 5).catch(() => false);
|
|
471
|
-
if (overflow) {
|
|
472
|
-
const f = makeFinding(
|
|
473
|
-
'medium', 'ux',
|
|
474
|
-
'Horizontal overflow on mobile viewport (375px)',
|
|
475
|
-
`${baseUrl} — iPhone SE viewport`,
|
|
476
|
-
'Content overflows horizontally on small screens. Forces unwanted horizontal scrolling. Breaks layout.',
|
|
477
|
-
'Audit for fixed-width elements wider than 375px. Use max-width: 100%, flexbox, or CSS grid for responsive layouts.'
|
|
478
|
-
);
|
|
479
|
-
f.screenshot = await takeScreenshot(page, screenshotsDir, f.id);
|
|
480
|
-
findings.push(f);
|
|
481
|
-
}
|
|
482
|
-
|
|
483
|
-
// Touch target size (< 44px = WCAG 2.5.5 violation)
|
|
484
|
-
const smallTargets = await page.evaluate(() => {
|
|
485
|
-
const els = document.querySelectorAll('a, button, [role="button"], input[type="submit"], input[type="checkbox"], input[type="radio"]');
|
|
486
|
-
const small = [];
|
|
487
|
-
for (const el of els) {
|
|
488
|
-
const r = el.getBoundingClientRect();
|
|
489
|
-
if (r.width > 0 && r.height > 0 && (r.width < 44 || r.height < 44)) {
|
|
490
|
-
small.push({ tag: el.tagName, text: (el.textContent || '').trim().substring(0, 30), w: Math.round(r.width), h: Math.round(r.height) });
|
|
491
|
-
}
|
|
492
|
-
}
|
|
493
|
-
return small.slice(0, 8);
|
|
494
|
-
}).catch(() => []);
|
|
495
|
-
|
|
496
|
-
if (smallTargets.length > 0) {
|
|
497
|
-
findings.push(makeFinding(
|
|
498
|
-
'low', 'accessibility',
|
|
499
|
-
`${smallTargets.length} touch target(s) smaller than 44×44px (WCAG 2.5.5)`,
|
|
500
|
-
smallTargets.map((t) => `${t.tag} "${t.text}" ${t.w}×${t.h}px`).slice(0, 3).join(', '),
|
|
501
|
-
'Small touch targets cause mis-taps, frustrate mobile users, and fail WCAG 2.5.5 (AAA) and Apple HIG guidelines.',
|
|
502
|
-
'Set min-height: 44px; min-width: 44px on all interactive elements. Increase padding rather than element size if needed.'
|
|
503
|
-
));
|
|
504
|
-
}
|
|
505
|
-
|
|
506
|
-
// Tiny fonts
|
|
507
|
-
const tinyFonts = await page.evaluate(() => {
|
|
508
|
-
const tiny = [];
|
|
509
|
-
for (const el of document.querySelectorAll('p, span, li, td, label, a')) {
|
|
510
|
-
if (!el.textContent.trim()) continue;
|
|
511
|
-
const size = parseFloat(window.getComputedStyle(el).fontSize);
|
|
512
|
-
if (size > 0 && size < 12) tiny.push({ tag: el.tagName, size, text: el.textContent.trim().substring(0, 40) });
|
|
513
|
-
}
|
|
514
|
-
return tiny.slice(0, 5);
|
|
515
|
-
}).catch(() => []);
|
|
516
|
-
|
|
517
|
-
if (tinyFonts.length > 0) {
|
|
518
|
-
findings.push(makeFinding(
|
|
519
|
-
'low', 'accessibility',
|
|
520
|
-
`${tinyFonts.length} text element(s) with font size below 12px`,
|
|
521
|
-
tinyFonts.map((t) => `${t.tag} (${t.size}px)`).slice(0, 3).join(', '),
|
|
522
|
-
'Text smaller than 12px triggers automatic zoom on iOS, breaking layout. Very hard to read without zooming.',
|
|
523
|
-
'Set a minimum font size of 12px. Use rem/em units for scalable typography across screen sizes.'
|
|
524
|
-
));
|
|
525
|
-
}
|
|
526
|
-
} finally {
|
|
527
|
-
await page.close().catch(() => {});
|
|
528
|
-
await context.close().catch(() => {});
|
|
529
|
-
}
|
|
530
|
-
}
|
|
531
|
-
|
|
532
|
-
// ============================================================
|
|
533
|
-
// ACCESSIBILITY AUDIT
|
|
534
|
-
// ============================================================
|
|
535
|
-
async function checkAccessibility(page, findings) {
|
|
536
|
-
const issues = await page.evaluate(() => {
|
|
537
|
-
const result = [];
|
|
538
|
-
|
|
539
|
-
const imgs = document.querySelectorAll('img:not([alt])');
|
|
540
|
-
if (imgs.length) result.push({ type: 'img_no_alt', count: imgs.length });
|
|
541
|
-
|
|
542
|
-
let unlabeled = 0;
|
|
543
|
-
for (const input of document.querySelectorAll('input:not([type="hidden"]):not([type="submit"]):not([type="button"]):not([type="reset"])')) {
|
|
544
|
-
const hasLabel = input.id && document.querySelector(`label[for="${input.id}"]`);
|
|
545
|
-
const hasAria = input.getAttribute('aria-label') || input.getAttribute('aria-labelledby');
|
|
546
|
-
if (!hasLabel && !hasAria) unlabeled++;
|
|
547
|
-
}
|
|
548
|
-
if (unlabeled) result.push({ type: 'input_no_label', count: unlabeled });
|
|
549
|
-
|
|
550
|
-
let unnamed = 0;
|
|
551
|
-
for (const btn of document.querySelectorAll('button,[role="button"]')) {
|
|
552
|
-
const hasText = (btn.textContent || '').trim().length > 0;
|
|
553
|
-
const hasAria = btn.getAttribute('aria-label') || btn.getAttribute('aria-labelledby') || btn.getAttribute('title');
|
|
554
|
-
if (!hasText && !hasAria) unnamed++;
|
|
555
|
-
}
|
|
556
|
-
if (unnamed) result.push({ type: 'button_no_name', count: unnamed });
|
|
557
|
-
|
|
558
|
-
const headings = [...document.querySelectorAll('h1,h2,h3,h4,h5,h6')];
|
|
559
|
-
for (let i = 1; i < headings.length; i++) {
|
|
560
|
-
if (parseInt(headings[i].tagName[1]) - parseInt(headings[i - 1].tagName[1]) > 1) {
|
|
561
|
-
result.push({ type: 'heading_skip' });
|
|
562
|
-
break;
|
|
563
|
-
}
|
|
564
|
-
}
|
|
565
|
-
|
|
566
|
-
if (!document.querySelector('html[lang]')) result.push({ type: 'no_lang' });
|
|
567
|
-
|
|
568
|
-
return result;
|
|
569
|
-
}).catch(() => []);
|
|
570
|
-
|
|
571
|
-
const defs = {
|
|
572
|
-
img_no_alt: { sev: 'medium', title: '{count} image(s) missing alt attribute', location: '<img> elements', risk: 'Screen readers cannot describe images to visually impaired users. WCAG 1.1.1 (Level A) violation.', fix: 'Add descriptive alt text to all informative images. Use alt="" for decorative images.' },
|
|
573
|
-
input_no_label: { sev: 'medium', title: '{count} form input(s) with no accessible label', location: '<input> elements', risk: 'Screen readers announce only the input type with no context. WCAG 1.3.1 (Level A) violation.', fix: 'Add <label for="..."> or aria-label to every form input.' },
|
|
574
|
-
button_no_name: { sev: 'medium', title: '{count} button(s) with no accessible name', location: '<button> elements', risk: 'Screen readers say "button" with no indication of what action it triggers. WCAG 4.1.2 violation.', fix: 'Add visible text or aria-label to every button.' },
|
|
575
|
-
heading_skip: { sev: 'low', title: 'Heading level skipped (e.g. h1 → h3)', location: 'Document heading structure', risk: 'Screen reader users rely on heading hierarchy for page navigation. Skipped levels break their mental model.', fix: 'Use sequential heading levels. Never choose heading levels for visual size — use CSS instead.' },
|
|
576
|
-
no_lang: { sev: 'low', title: '<html> element missing lang attribute', location: '<html> tag', risk: 'Screen readers guess language for pronunciation. Wrong language causes incorrect speech. WCAG 3.1.1 (Level A).', fix: 'Add lang="en" (or appropriate BCP-47 code) to the <html> element.' }
|
|
577
|
-
};
|
|
578
|
-
|
|
579
|
-
for (const issue of issues) {
|
|
580
|
-
const d = defs[issue.type];
|
|
581
|
-
if (!d) continue;
|
|
582
|
-
findings.push(makeFinding(d.sev, 'accessibility', d.title.replace('{count}', issue.count || ''), d.location, d.risk, d.fix));
|
|
583
|
-
}
|
|
584
|
-
}
|
|
585
|
-
|
|
586
|
-
// ============================================================
|
|
587
|
-
// PERFORMANCE
|
|
588
|
-
// ============================================================
|
|
589
|
-
async function capturePerformance(page, thresholds, findings) {
|
|
590
|
-
const perf = await page.evaluate(() => {
|
|
591
|
-
const nav = performance.getEntriesByType('navigation')[0];
|
|
592
|
-
if (!nav) return null;
|
|
593
|
-
const resources = performance.getEntriesByType('resource');
|
|
594
|
-
return {
|
|
595
|
-
domContentLoaded: Math.round(nav.domContentLoadedEventEnd),
|
|
596
|
-
loadComplete: Math.round(nav.loadEventEnd),
|
|
597
|
-
ttfb: Math.round(nav.responseStart - nav.requestStart),
|
|
598
|
-
resourceCount: resources.length,
|
|
599
|
-
resourceSizeKb: Math.round(resources.reduce((acc, r) => acc + (r.transferSize || 0), 0) / 1024)
|
|
600
|
-
};
|
|
601
|
-
}).catch(() => null);
|
|
602
|
-
|
|
603
|
-
if (!perf) return null;
|
|
604
|
-
|
|
605
|
-
if (perf.loadComplete > (thresholds.page_load_ms || 3000)) {
|
|
606
|
-
findings.push(makeFinding(
|
|
607
|
-
'medium', 'performance',
|
|
608
|
-
`Page load time exceeds threshold (${perf.loadComplete}ms > ${thresholds.page_load_ms || 3000}ms)`,
|
|
609
|
-
'Page load',
|
|
610
|
-
'Slow page load degrades UX and Core Web Vitals score. Google Search penalizes pages with poor LCP.',
|
|
611
|
-
'Analyze network waterfall. Defer non-critical JS. Enable Gzip/Brotli compression. Use a CDN for static assets.'
|
|
612
|
-
));
|
|
613
|
-
}
|
|
614
|
-
|
|
615
|
-
if (perf.ttfb > (thresholds.ttfb_ms || 800)) {
|
|
616
|
-
findings.push(makeFinding(
|
|
617
|
-
'medium', 'performance',
|
|
618
|
-
`High Time to First Byte — TTFB ${perf.ttfb}ms`,
|
|
619
|
-
'Server response time',
|
|
620
|
-
'TTFB > 800ms means the server is slow to respond. Users see a blank page for too long.',
|
|
621
|
-
'Optimize database queries, add server-side caching, or review server infrastructure capacity.'
|
|
622
|
-
));
|
|
623
|
-
}
|
|
624
|
-
|
|
625
|
-
if (perf.resourceCount > (thresholds.requests_max || 80)) {
|
|
626
|
-
findings.push(makeFinding(
|
|
627
|
-
'low', 'performance',
|
|
628
|
-
`High request count: ${perf.resourceCount} network requests on load`,
|
|
629
|
-
'Network requests',
|
|
630
|
-
`${perf.resourceCount} requests slow down page load and increases server load.`,
|
|
631
|
-
'Bundle JavaScript and CSS files. Use HTTP/2 multiplexing. Lazy-load images and below-the-fold content.'
|
|
632
|
-
));
|
|
633
|
-
}
|
|
634
|
-
|
|
635
|
-
if (perf.resourceSizeKb > (thresholds.transfer_max_kb || 2048)) {
|
|
636
|
-
findings.push(makeFinding(
|
|
637
|
-
'low', 'performance',
|
|
638
|
-
`Total transfer size ${perf.resourceSizeKb}KB exceeds threshold`,
|
|
639
|
-
'Network transfer',
|
|
640
|
-
`Large payload increases load time on slow connections and mobile data.`,
|
|
641
|
-
'Enable compression. Audit and tree-shake large JS bundles. Optimize images (WebP, lazy loading).'
|
|
642
|
-
));
|
|
643
|
-
}
|
|
644
|
-
|
|
645
|
-
return perf;
|
|
646
|
-
}
|
|
647
|
-
|
|
648
|
-
// ============================================================
|
|
649
|
-
// AC COVERAGE
|
|
650
|
-
// ============================================================
|
|
651
|
-
function parseAcItems(prdContent) {
|
|
652
|
-
if (!prdContent) return [];
|
|
653
|
-
const items = [];
|
|
654
|
-
for (const match of String(prdContent).matchAll(/\|\s*(AC-\d+)\s*\|\s*([^|]+)\|/g)) {
|
|
655
|
-
items.push({ id: match[1].trim(), description: match[2].trim() });
|
|
656
|
-
}
|
|
657
|
-
for (const match of String(prdContent).matchAll(/🔴\s*([^\n]{10,100})/g)) {
|
|
658
|
-
if (items.length >= 20) break;
|
|
659
|
-
items.push({ id: `AC-${String(items.length + 1).padStart(2, '0')}`, description: match[1].trim() });
|
|
660
|
-
}
|
|
661
|
-
return items.slice(0, 20);
|
|
662
|
-
}
|
|
663
|
-
|
|
664
|
-
async function runAcCoverage(page, baseUrl, prdPath, screenshotsDir) {
|
|
665
|
-
const prdContent = await readTextIfExists(prdPath);
|
|
666
|
-
const acItems = parseAcItems(prdContent);
|
|
667
|
-
if (acItems.length === 0) return [];
|
|
668
|
-
|
|
669
|
-
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
670
|
-
|
|
671
|
-
const coverage = [];
|
|
672
|
-
for (const ac of acItems) {
|
|
673
|
-
const screenshotFile = path.join(screenshotsDir, `${ac.id}.png`);
|
|
674
|
-
await page.screenshot({ path: screenshotFile, fullPage: false }).catch(() => {});
|
|
675
|
-
coverage.push({
|
|
676
|
-
id: ac.id,
|
|
677
|
-
description: ac.description,
|
|
678
|
-
status: 'Documented',
|
|
679
|
-
screenshot: screenshotFile
|
|
680
|
-
});
|
|
681
|
-
}
|
|
682
|
-
return coverage;
|
|
683
|
-
}
|
|
684
|
-
|
|
685
|
-
// ============================================================
|
|
686
|
-
// REPORT GENERATION
|
|
687
|
-
// ============================================================
|
|
688
|
-
function buildMarkdownReport(projectName, url, findings, acCoverage, perf, mode) {
|
|
689
|
-
const sorted = [...findings].sort((a, b) => {
|
|
690
|
-
const o = { critical: 0, high: 1, medium: 2, low: 3 };
|
|
691
|
-
return (o[a.severity] ?? 4) - (o[b.severity] ?? 4);
|
|
692
|
-
});
|
|
693
|
-
const bySev = (s) => sorted.filter((f) => f.severity === s);
|
|
694
|
-
const date = new Date().toISOString().split('T')[0];
|
|
695
|
-
|
|
696
|
-
let md = `## QA Browser Report — ${projectName} — ${date}\n\n`;
|
|
697
|
-
md += `> Generated by: \`aioson qa:${mode}\` \n`;
|
|
698
|
-
md += `> Browser: Chromium | Viewport: 1280×720 \n`;
|
|
699
|
-
md += `> URL: ${url}\n\n`;
|
|
700
|
-
|
|
701
|
-
if (acCoverage.length > 0) {
|
|
702
|
-
md += `### Acceptance criteria coverage\n| AC | Description | Status |\n|---|---|---|\n`;
|
|
703
|
-
for (const ac of acCoverage) md += `| ${ac.id} | ${ac.description} | ${ac.status} |\n`;
|
|
704
|
-
md += '\n';
|
|
705
|
-
}
|
|
706
|
-
|
|
707
|
-
md += `### Findings\n\n`;
|
|
708
|
-
for (const [label, group] of [['Critical', bySev('critical')], ['High', bySev('high')], ['Medium', bySev('medium')], ['Low', bySev('low')]]) {
|
|
709
|
-
if (group.length === 0) continue;
|
|
710
|
-
md += `#### ${label}\n`;
|
|
711
|
-
for (const f of group) {
|
|
712
|
-
md += `**[${f.id}] ${f.title}** \n`;
|
|
713
|
-
md += `Location: \`${f.location}\` \n`;
|
|
714
|
-
md += `Risk: ${f.risk} \n`;
|
|
715
|
-
md += `Fix: ${f.fix} \n`;
|
|
716
|
-
if (f.screenshot) md += `Screenshot: ${f.screenshot} \n`;
|
|
717
|
-
md += '\n';
|
|
718
|
-
}
|
|
719
|
-
}
|
|
720
|
-
|
|
721
|
-
if (perf) {
|
|
722
|
-
md += `### Performance\n| Metric | Value |\n|---|---|\n`;
|
|
723
|
-
md += `| DOM Content Loaded | ${perf.domContentLoaded}ms |\n`;
|
|
724
|
-
md += `| Page Load Complete | ${perf.loadComplete}ms |\n`;
|
|
725
|
-
md += `| Time to First Byte | ${perf.ttfb}ms |\n`;
|
|
726
|
-
md += `| Network requests | ${perf.resourceCount} |\n`;
|
|
727
|
-
md += `| Total transfer | ${perf.resourceSizeKb}KB |\n\n`;
|
|
728
|
-
}
|
|
729
|
-
|
|
730
|
-
md += `### Residual risks\n`;
|
|
731
|
-
md += `- Tests run against a running instance; production environment may differ (headers, CSP, CDN).\n`;
|
|
732
|
-
md += `- Content behind authentication was not tested — no credentials were provided.\n`;
|
|
733
|
-
md += `- JavaScript-heavy interactions may need additional manual verification.\n\n`;
|
|
734
|
-
|
|
735
|
-
const c = bySev('critical').length, h = bySev('high').length, m = bySev('medium').length, l = bySev('low').length;
|
|
736
|
-
md += `### Summary\n`;
|
|
737
|
-
md += `- Critical: ${c} | High: ${h} | Medium: ${m} | Low: ${l}\n`;
|
|
738
|
-
if (acCoverage.length > 0) md += `- AC documented: ${acCoverage.length}\n`;
|
|
739
|
-
|
|
740
|
-
return md;
|
|
741
|
-
}
|
|
742
|
-
|
|
743
|
-
async function writeReports(targetDir, projectName, url, findings, acCoverage, perf, mode) {
|
|
744
|
-
const mdPath = path.join(targetDir, 'aios-qa-report.md');
|
|
745
|
-
const jsonPath = path.join(targetDir, 'aios-qa-report.json');
|
|
746
|
-
const md = buildMarkdownReport(projectName, url, findings, acCoverage, perf, mode);
|
|
747
|
-
const bySev = (s) => findings.filter((f) => f.severity === s).length;
|
|
748
|
-
const json = {
|
|
749
|
-
generated_at: new Date().toISOString(),
|
|
750
|
-
project: projectName, url, mode,
|
|
751
|
-
summary: { critical: bySev('critical'), high: bySev('high'), medium: bySev('medium'), low: bySev('low') },
|
|
752
|
-
ac_coverage: acCoverage,
|
|
753
|
-
performance: perf,
|
|
754
|
-
findings
|
|
755
|
-
};
|
|
756
|
-
await fs.writeFile(mdPath, md, 'utf8');
|
|
757
|
-
await fs.writeFile(jsonPath, `${JSON.stringify(json, null, 2)}\n`, 'utf8');
|
|
758
|
-
return { mdPath, jsonPath };
|
|
759
|
-
}
|
|
760
|
-
|
|
761
|
-
// ============================================================
|
|
762
|
-
// MAIN
|
|
763
|
-
// ============================================================
|
|
764
|
-
async function runQaRun({ args, options = {}, logger, t }) {
|
|
765
|
-
const targetDir = path.resolve(process.cwd(), args[0] || '.');
|
|
766
|
-
|
|
767
|
-
const pw = requirePlaywright();
|
|
768
|
-
if (!pw) {
|
|
769
|
-
logger.error(t('qa_run.playwright_missing'));
|
|
770
|
-
process.exitCode = 1;
|
|
771
|
-
return { ok: false, error: 'playwright_not_installed' };
|
|
772
|
-
}
|
|
773
|
-
|
|
774
|
-
const config = await loadConfig(targetDir);
|
|
775
|
-
if (!config) {
|
|
776
|
-
logger.error(t('qa_run.config_missing'));
|
|
777
|
-
process.exitCode = 1;
|
|
778
|
-
return { ok: false, error: 'config_not_found' };
|
|
779
|
-
}
|
|
780
|
-
|
|
781
|
-
const url = String(options.url || config.url || '');
|
|
782
|
-
if (!url) {
|
|
783
|
-
logger.error(t('qa_run.url_missing'));
|
|
784
|
-
process.exitCode = 1;
|
|
785
|
-
return { ok: false, error: 'url_not_configured' };
|
|
786
|
-
}
|
|
787
|
-
|
|
788
|
-
const projectName = config.project_name || path.basename(targetDir) || 'Project';
|
|
789
|
-
const selectedPersona = String(options.persona || '').toLowerCase() || null;
|
|
790
|
-
const headed = Boolean(options.headed);
|
|
791
|
-
const screenshotsDir = path.join(targetDir, 'aios-qa-screenshots');
|
|
792
|
-
const prdPath = path.join(targetDir, '.aioson/context/prd.md');
|
|
793
|
-
const thresholds = config.performance_thresholds || {};
|
|
794
|
-
|
|
795
|
-
_counter = 0;
|
|
796
|
-
const findings = [];
|
|
797
|
-
const consoleLogs = [];
|
|
798
|
-
const networkRequests = [];
|
|
799
|
-
|
|
800
|
-
logger.log(t('qa_run.starting', { url }));
|
|
801
|
-
await ensureDir(screenshotsDir);
|
|
802
|
-
|
|
803
|
-
const browser = await pw.chromium.launch({ headless: !headed });
|
|
804
|
-
const context = await browser.newContext({ viewport: { width: 1280, height: 720 } });
|
|
805
|
-
const page = await context.newPage();
|
|
806
|
-
|
|
807
|
-
page.on('console', (msg) => consoleLogs.push({ type: msg.type(), text: msg.text() }));
|
|
808
|
-
page.on('request', (req) => networkRequests.push({ url: req.url(), method: req.method() }));
|
|
809
|
-
|
|
810
|
-
try {
|
|
811
|
-
const personas = config.personas || ['naive', 'hacker', 'power', 'mobile'];
|
|
812
|
-
|
|
813
|
-
for (const persona of personas) {
|
|
814
|
-
if (selectedPersona && persona !== selectedPersona) continue;
|
|
815
|
-
logger.log(t('qa_run.persona_start', { persona }));
|
|
816
|
-
const before = findings.length;
|
|
817
|
-
|
|
818
|
-
if (persona === 'naive') await runNaivePersona(page, url, findings, screenshotsDir).catch(() => {});
|
|
819
|
-
else if (persona === 'hacker') await runHackerPersona(page, url, findings, screenshotsDir).catch(() => {});
|
|
820
|
-
else if (persona === 'power') await runPowerPersona(page, url, findings).catch(() => {});
|
|
821
|
-
else if (persona === 'mobile') await runMobilePersona(browser, url, findings, screenshotsDir).catch(() => {});
|
|
822
|
-
|
|
823
|
-
logger.log(t('qa_run.persona_done', { persona, count: findings.length - before }));
|
|
824
|
-
}
|
|
825
|
-
|
|
826
|
-
// Network + console analysis
|
|
827
|
-
await probeInjectionInjection(page, findings, consoleLogs, networkRequests).catch(() => {});
|
|
828
|
-
|
|
829
|
-
// Accessibility
|
|
830
|
-
logger.log(t('qa_run.accessibility'));
|
|
831
|
-
await page.goto(url, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
832
|
-
await checkAccessibility(page, findings).catch(() => {});
|
|
833
|
-
|
|
834
|
-
// Performance
|
|
835
|
-
logger.log(t('qa_run.performance'));
|
|
836
|
-
await page.goto(url, { waitUntil: 'load', timeout: 20000 }).catch(() => {});
|
|
837
|
-
const perf = await capturePerformance(page, thresholds, findings).catch(() => null);
|
|
838
|
-
|
|
839
|
-
// AC coverage
|
|
840
|
-
logger.log(t('qa_run.ac_scenarios'));
|
|
841
|
-
const acCoverage = await runAcCoverage(page, url, prdPath, screenshotsDir).catch(() => []);
|
|
842
|
-
|
|
843
|
-
// Write reports
|
|
844
|
-
const { mdPath, jsonPath } = await writeReports(targetDir, projectName, url, findings, acCoverage, perf, 'run');
|
|
845
|
-
|
|
846
|
-
logger.log(t('qa_run.done'));
|
|
847
|
-
logger.log(t('qa_run.report_written', { path: mdPath }));
|
|
848
|
-
logger.log(t('qa_run.json_written', { path: jsonPath }));
|
|
849
|
-
logger.log(t('qa_run.screenshots_dir', { path: screenshotsDir }));
|
|
850
|
-
|
|
851
|
-
const bySev = (s) => findings.filter((f) => f.severity === s).length;
|
|
852
|
-
const summary = { critical: bySev('critical'), high: bySev('high'), medium: bySev('medium'), low: bySev('low') };
|
|
853
|
-
logger.log(t('qa_run.findings_summary', summary));
|
|
854
|
-
|
|
855
|
-
// HTML report (optional, additive — does not replace MD/JSON)
|
|
856
|
-
let htmlPath, htmlDir;
|
|
857
|
-
if (options.html) {
|
|
858
|
-
const { writeHtmlReport } = require('../qa-html-report');
|
|
859
|
-
const result = await writeHtmlReport(targetDir, projectName, url, findings, acCoverage, perf, 'run', screenshotsDir, { thresholds });
|
|
860
|
-
htmlPath = result.htmlPath;
|
|
861
|
-
htmlDir = result.runDir;
|
|
862
|
-
logger.log(t('qa_run.html_report_written', { path: htmlPath }));
|
|
863
|
-
}
|
|
864
|
-
|
|
865
|
-
const output = { ok: true, targetDir, url, summary, mdPath, jsonPath, screenshotsDir, findings, acCoverage, ...(htmlPath ? { htmlPath, htmlDir } : {}) };
|
|
866
|
-
if (options.json) return output;
|
|
867
|
-
return output;
|
|
868
|
-
} finally {
|
|
869
|
-
await browser.close().catch(() => {});
|
|
870
|
-
}
|
|
871
|
-
}
|
|
872
|
-
|
|
873
|
-
module.exports = { runQaRun };
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
const path = require('node:path');
|
|
4
|
+
const fs = require('node:fs/promises');
|
|
5
|
+
const { readTextIfExists, ensureDir } = require('../utils');
|
|
6
|
+
|
|
7
|
+
// --- Secret patterns for exposure detection ---
|
|
8
|
+
const SECRET_PATTERNS = [
|
|
9
|
+
{ name: 'OpenAI key', regex: /sk-[a-zA-Z0-9]{20,}/ },
|
|
10
|
+
{ name: 'Stripe live key', regex: /pk_live_[a-zA-Z0-9]{20,}/ },
|
|
11
|
+
{ name: 'Stripe test key', regex: /pk_test_[a-zA-Z0-9]{20,}/ },
|
|
12
|
+
{ name: 'AWS access key', regex: /AKIA[A-Z0-9]{16}/ },
|
|
13
|
+
{ name: 'Google API key', regex: /AIzaSy[a-zA-Z0-9_-]{33}/ },
|
|
14
|
+
{ name: 'GitHub token', regex: /gh[ps]_[a-zA-Z0-9]{36}/ },
|
|
15
|
+
{ name: 'Slack token', regex: /xox[bpa]-[a-zA-Z0-9-]+/ },
|
|
16
|
+
{ name: 'Generic secret', regex: /(SECRET|TOKEN|PASSWORD|PRIVATE_KEY)\s*[:=]\s*['"]?[a-zA-Z0-9_/+=-]{16,}/i }
|
|
17
|
+
];
|
|
18
|
+
|
|
19
|
+
const SENSITIVE_FILE_PATHS = [
|
|
20
|
+
'/.env', '/.env.local', '/.env.production', '/.env.development',
|
|
21
|
+
'/.git/config', '/config.js', '/api/config', '/wp-config.php',
|
|
22
|
+
'/application.yml', '/application.properties'
|
|
23
|
+
];
|
|
24
|
+
|
|
25
|
+
const DEBUG_ROUTES = [
|
|
26
|
+
'/admin', '/debug', '/phpinfo.php', '/_debug',
|
|
27
|
+
'/api/health', '/api/debug', '/__nextjs_original-stack-frame',
|
|
28
|
+
'/api/env', '/server-status'
|
|
29
|
+
];
|
|
30
|
+
|
|
31
|
+
// --- Playwright gate ---
|
|
32
|
+
function requirePlaywright() {
|
|
33
|
+
try { return require('playwright'); } catch { return null; }
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
// --- Config ---
|
|
37
|
+
async function loadConfig(targetDir) {
|
|
38
|
+
try {
|
|
39
|
+
const raw = await fs.readFile(path.join(targetDir, 'aios-qa.config.json'), 'utf8');
|
|
40
|
+
return JSON.parse(raw);
|
|
41
|
+
} catch { return null; }
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// --- Finding factory ---
|
|
45
|
+
let _counter = 0;
|
|
46
|
+
function makeFinding(severity, category, title, location, risk, fix) {
|
|
47
|
+
_counter++;
|
|
48
|
+
const prefix = severity[0].toUpperCase();
|
|
49
|
+
const id = `${prefix}-${String(_counter).padStart(2, '0')}`;
|
|
50
|
+
return { id, severity, category, title, location, risk, fix, screenshot: '' };
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// --- Screenshot helper ---
|
|
54
|
+
async function takeScreenshot(page, screenshotsDir, id) {
|
|
55
|
+
try {
|
|
56
|
+
const file = path.join(screenshotsDir, `${id}.png`);
|
|
57
|
+
await page.screenshot({ path: file, fullPage: false });
|
|
58
|
+
return file;
|
|
59
|
+
} catch { return ''; }
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
// ============================================================
|
|
63
|
+
// SECURITY PROBES
|
|
64
|
+
// ============================================================
|
|
65
|
+
|
|
66
|
+
async function probeExposedSecrets(page, findings, screenshotsDir) {
|
|
67
|
+
// Check window globals: Next.js __NEXT_DATA__, window.ENV, etc.
|
|
68
|
+
const exposed = await page.evaluate((patterns) => {
|
|
69
|
+
const sources = {
|
|
70
|
+
'__NEXT_DATA__': window.__NEXT_DATA__,
|
|
71
|
+
'__env__': window.__env__,
|
|
72
|
+
'ENV': window.ENV,
|
|
73
|
+
'_env': window._env,
|
|
74
|
+
'CONFIG': window.CONFIG,
|
|
75
|
+
'APP_CONFIG': window.APP_CONFIG
|
|
76
|
+
};
|
|
77
|
+
const found = [];
|
|
78
|
+
for (const [src, val] of Object.entries(sources)) {
|
|
79
|
+
if (!val) continue;
|
|
80
|
+
const str = JSON.stringify(val);
|
|
81
|
+
for (const { name, regex } of patterns) {
|
|
82
|
+
if (new RegExp(regex).test(str)) found.push({ source: src, keyType: name });
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
return found;
|
|
86
|
+
}, SECRET_PATTERNS.map((p) => ({ name: p.name, regex: p.regex.source }))).catch(() => []);
|
|
87
|
+
|
|
88
|
+
for (const item of exposed) {
|
|
89
|
+
const f = makeFinding(
|
|
90
|
+
'critical', 'security',
|
|
91
|
+
`${item.keyType} exposed in window.${item.source}`,
|
|
92
|
+
`window.${item.source}`,
|
|
93
|
+
`${item.keyType} is visible to any browser user via the global object. Direct financial or account compromise exposure.`,
|
|
94
|
+
`Move to server-side only. Never use NEXT_PUBLIC_ or client-side globals for secrets.`
|
|
95
|
+
);
|
|
96
|
+
f.screenshot = await takeScreenshot(page, screenshotsDir, f.id);
|
|
97
|
+
findings.push(f);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// Also scan the rendered HTML source
|
|
101
|
+
const html = await page.content().catch(() => '');
|
|
102
|
+
for (const { name, regex } of SECRET_PATTERNS) {
|
|
103
|
+
if (regex.test(html)) {
|
|
104
|
+
findings.push(makeFinding(
|
|
105
|
+
'critical', 'security',
|
|
106
|
+
`${name} found in rendered HTML source`,
|
|
107
|
+
'Page HTML source',
|
|
108
|
+
`${name} is embedded in the HTML sent to the browser. Visible to anyone using DevTools.`,
|
|
109
|
+
'Remove from client-side rendering. Serve secrets only from server-side APIs.'
|
|
110
|
+
));
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
async function probeSensitiveFiles(page, baseUrl, findings) {
|
|
116
|
+
for (const filePath of SENSITIVE_FILE_PATHS) {
|
|
117
|
+
try {
|
|
118
|
+
const response = await page.goto(`${baseUrl}${filePath}`, {
|
|
119
|
+
waitUntil: 'commit', timeout: 5000
|
|
120
|
+
});
|
|
121
|
+
if (response && response.status() === 200) {
|
|
122
|
+
const body = await response.text().catch(() => '');
|
|
123
|
+
const looksLikeSensitive =
|
|
124
|
+
/[A-Z_]{3,}=/.test(body) ||
|
|
125
|
+
/<\?php/.test(body) ||
|
|
126
|
+
/(SECRET|PASSWORD|TOKEN|KEY|PRIVATE)/i.test(body);
|
|
127
|
+
if (looksLikeSensitive) {
|
|
128
|
+
findings.push(makeFinding(
|
|
129
|
+
'critical', 'security',
|
|
130
|
+
`Sensitive file publicly accessible: ${filePath}`,
|
|
131
|
+
`${baseUrl}${filePath}`,
|
|
132
|
+
`Configuration file is reachable by any internet user. May expose credentials, connection strings, or infrastructure details.`,
|
|
133
|
+
`Block ${filePath} in your web server (nginx/vercel/apache). Never deploy .env files to public directories.`
|
|
134
|
+
));
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
} catch { /* not accessible — good */ }
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
async function probeXss(page, baseUrl, findings, screenshotsDir) {
|
|
142
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
143
|
+
|
|
144
|
+
let xssFired = false;
|
|
145
|
+
page.on('dialog', async (dialog) => {
|
|
146
|
+
xssFired = true;
|
|
147
|
+
await dialog.dismiss().catch(() => {});
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
const inputs = await page.$$('input[type="text"],input[type="search"],input[type="email"],input[type="url"],textarea').catch(() => []);
|
|
151
|
+
for (const input of inputs.slice(0, 10)) {
|
|
152
|
+
await input.fill('<img src=x onerror="window.__xss=1">').catch(() => {});
|
|
153
|
+
}
|
|
154
|
+
await page.keyboard.press('Tab').catch(() => {});
|
|
155
|
+
await page.waitForTimeout(500).catch(() => {});
|
|
156
|
+
|
|
157
|
+
const xssEval = await page.evaluate(() => window.__xss === 1).catch(() => false);
|
|
158
|
+
if (xssFired || xssEval) {
|
|
159
|
+
const f = makeFinding(
|
|
160
|
+
'critical', 'security',
|
|
161
|
+
'XSS (Cross-Site Scripting) — injected script executed',
|
|
162
|
+
'Form text inputs',
|
|
163
|
+
'User input is rendered as HTML without sanitization. Attacker can steal sessions, redirect users, or deface the page.',
|
|
164
|
+
'Sanitize all user input before rendering. Use textContent instead of innerHTML. Apply a strict Content-Security-Policy header.'
|
|
165
|
+
);
|
|
166
|
+
f.screenshot = await takeScreenshot(page, screenshotsDir, f.id);
|
|
167
|
+
findings.push(f);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
async function probeOpenRedirect(page, baseUrl, findings) {
|
|
172
|
+
const params = ['redirect', 'next', 'return', 'returnUrl', 'return_url', 'goto', 'url', 'dest'];
|
|
173
|
+
const evil = 'https://evil-phishing-example.com';
|
|
174
|
+
for (const param of params) {
|
|
175
|
+
try {
|
|
176
|
+
const response = await page.goto(`${baseUrl}?${param}=${encodeURIComponent(evil)}`, {
|
|
177
|
+
waitUntil: 'commit', timeout: 5000
|
|
178
|
+
});
|
|
179
|
+
const finalUrl = page.url();
|
|
180
|
+
const redirected = finalUrl.startsWith(evil) ||
|
|
181
|
+
(response && [301, 302, 303, 307, 308].includes(response.status()) &&
|
|
182
|
+
String(response.headers()['location'] || '').startsWith(evil));
|
|
183
|
+
if (redirected) {
|
|
184
|
+
findings.push(makeFinding(
|
|
185
|
+
'high', 'security',
|
|
186
|
+
`Open redirect via ?${param}= parameter`,
|
|
187
|
+
`${baseUrl}?${param}=`,
|
|
188
|
+
'Attacker can use your trusted domain to redirect users to phishing sites. Bypasses browser warnings.',
|
|
189
|
+
'Validate redirect targets against an allowlist of trusted paths. Reject external URLs unconditionally.'
|
|
190
|
+
));
|
|
191
|
+
break;
|
|
192
|
+
}
|
|
193
|
+
} catch { /* not redirected */ }
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
async function probeInjectionInputs(page, baseUrl, findings) {
|
|
198
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
199
|
+
const inputs = await page.$$('input[type="text"],input[type="search"],textarea').catch(() => []);
|
|
200
|
+
const sqlPayload = `' OR '1'='1' -- `;
|
|
201
|
+
for (const input of inputs.slice(0, 5)) {
|
|
202
|
+
await input.fill(sqlPayload).catch(() => {});
|
|
203
|
+
}
|
|
204
|
+
await page.waitForTimeout(500).catch(() => {});
|
|
205
|
+
const html = await page.content().catch(() => '');
|
|
206
|
+
if (/(SQL syntax|mysql_fetch|ORA-|pg_query|sqlite_|SQLSTATE)/i.test(html)) {
|
|
207
|
+
findings.push(makeFinding(
|
|
208
|
+
'critical', 'security',
|
|
209
|
+
'SQL error message exposed after injection probe',
|
|
210
|
+
'Form inputs',
|
|
211
|
+
'A SQL error was returned when input contained single quotes. Indicates raw string interpolation in queries.',
|
|
212
|
+
'Use parameterized queries or ORM exclusively. Never build SQL strings with user input. Disable detailed DB error messages in production.'
|
|
213
|
+
));
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
async function probeDebugRoutes(page, baseUrl, findings) {
|
|
218
|
+
for (const route of DEBUG_ROUTES) {
|
|
219
|
+
try {
|
|
220
|
+
const response = await page.goto(`${baseUrl}${route}`, { waitUntil: 'commit', timeout: 4000 });
|
|
221
|
+
if (response && response.status() === 200) {
|
|
222
|
+
const title = await page.title().catch(() => '');
|
|
223
|
+
if (!/404|not found/i.test(title)) {
|
|
224
|
+
findings.push(makeFinding(
|
|
225
|
+
'medium', 'security',
|
|
226
|
+
`Debug/admin route accessible without authentication: ${route}`,
|
|
227
|
+
`${baseUrl}${route}`,
|
|
228
|
+
'Unauthenticated access to debug/admin endpoints may expose metrics, logs, internal state, or admin controls.',
|
|
229
|
+
`Require authentication for ${route}. Restrict to internal network or remove in production.`
|
|
230
|
+
));
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
} catch { /* not accessible */ }
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
async function probeInjectionInjection(page, findings, consoleLog, networkRequests) {
|
|
238
|
+
// Console error leakage
|
|
239
|
+
const stackTraceRx = /at\s+\w+\s+\(/;
|
|
240
|
+
const sensitiveRx = /(token|secret|password|key|authorization)/i;
|
|
241
|
+
const leaks = consoleLog.filter((l) =>
|
|
242
|
+
l.type === 'error' && (stackTraceRx.test(l.text) || sensitiveRx.test(l.text))
|
|
243
|
+
);
|
|
244
|
+
if (leaks.length > 0) {
|
|
245
|
+
findings.push(makeFinding(
|
|
246
|
+
'medium', 'security',
|
|
247
|
+
`Browser console exposes ${leaks.length} error(s) with stack traces or sensitive keywords`,
|
|
248
|
+
'Browser DevTools console',
|
|
249
|
+
'Stack traces reveal file paths, library versions, and logic. Sensitive keywords may expose credentials or tokens.',
|
|
250
|
+
'Disable verbose error logging in production. Use a centralized error service (Sentry, Datadog) instead of console.error.'
|
|
251
|
+
));
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
// Sensitive data in GET params
|
|
255
|
+
const sensitiveParamRx = /(token|secret|password|api_key|apikey|auth)/i;
|
|
256
|
+
for (const req of networkRequests) {
|
|
257
|
+
try {
|
|
258
|
+
const url = new URL(req.url);
|
|
259
|
+
for (const [key] of url.searchParams) {
|
|
260
|
+
if (sensitiveParamRx.test(key)) {
|
|
261
|
+
findings.push(makeFinding(
|
|
262
|
+
'high', 'security',
|
|
263
|
+
`Sensitive parameter "${key}" transmitted in GET URL`,
|
|
264
|
+
req.url.substring(0, 120),
|
|
265
|
+
'Sensitive data in URLs is logged by web servers, proxies, CDNs, and browser history. Leaks through Referer headers.',
|
|
266
|
+
'Move sensitive parameters to POST body or Authorization header. Never pass secrets in query strings.'
|
|
267
|
+
));
|
|
268
|
+
break;
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
} catch { /* invalid URL */ }
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
// HTTP requests from HTTPS page (mixed content)
|
|
275
|
+
const mixed = networkRequests.filter((r) =>
|
|
276
|
+
r.url.startsWith('http://') && !r.url.startsWith('http://localhost') && !r.url.startsWith('http://127.')
|
|
277
|
+
);
|
|
278
|
+
if (mixed.length > 0) {
|
|
279
|
+
findings.push(makeFinding(
|
|
280
|
+
'medium', 'security',
|
|
281
|
+
`${mixed.length} mixed content request(s) — HTTP resources on page`,
|
|
282
|
+
mixed[0].url.substring(0, 120),
|
|
283
|
+
'HTTP requests from an HTTPS page expose data in transit. Modern browsers block or warn on mixed content.',
|
|
284
|
+
'Upgrade all resource references to HTTPS. Use protocol-relative URLs (//) or absolute HTTPS URLs.'
|
|
285
|
+
));
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
// ============================================================
|
|
290
|
+
// PERSONA — NAIVE USER
|
|
291
|
+
// ============================================================
|
|
292
|
+
async function runNaivePersona(page, baseUrl, findings, screenshotsDir) {
|
|
293
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
294
|
+
|
|
295
|
+
// Submit all empty forms
|
|
296
|
+
const forms = await page.$$('form').catch(() => []);
|
|
297
|
+
for (const form of forms.slice(0, 5)) {
|
|
298
|
+
const beforeUrl = page.url();
|
|
299
|
+
await page.evaluate((f) => { try { f.submit(); } catch (_) {} }, form).catch(() => {});
|
|
300
|
+
await page.waitForTimeout(800).catch(() => {});
|
|
301
|
+
const title = await page.title().catch(() => '');
|
|
302
|
+
const html = await page.content().catch(() => '');
|
|
303
|
+
if (/error|exception|stacktrace|500/i.test(title) || /Internal Server Error|Uncaught Exception/i.test(html)) {
|
|
304
|
+
const f = makeFinding(
|
|
305
|
+
'high', 'reliability',
|
|
306
|
+
'Empty form submission causes server error (5xx)',
|
|
307
|
+
`Form on ${beforeUrl}`,
|
|
308
|
+
'Server returns 5xx when form is submitted empty. Missing or bypassed server-side validation.',
|
|
309
|
+
'Add server-side validation before processing. Return 422 with field-specific errors instead of throwing 500.'
|
|
310
|
+
);
|
|
311
|
+
f.screenshot = await takeScreenshot(page, screenshotsDir, f.id);
|
|
312
|
+
findings.push(f);
|
|
313
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 10000 }).catch(() => {});
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
// Type very long strings (buffer overflow / DoS potential)
|
|
318
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 10000 }).catch(() => {});
|
|
319
|
+
const inputs = await page.$$('input[type="text"],input[type="email"],input[type="search"],textarea').catch(() => []);
|
|
320
|
+
const longStr = 'A'.repeat(10000);
|
|
321
|
+
for (const input of inputs.slice(0, 5)) {
|
|
322
|
+
await input.fill(longStr).catch(() => {});
|
|
323
|
+
}
|
|
324
|
+
await page.waitForTimeout(500).catch(() => {});
|
|
325
|
+
const srcAfterLong = await page.content().catch(() => '');
|
|
326
|
+
if (/maximum call stack|out of memory|RangeError|Cannot read/i.test(srcAfterLong)) {
|
|
327
|
+
findings.push(makeFinding(
|
|
328
|
+
'high', 'reliability',
|
|
329
|
+
'Application crashes on very long input (10,000 chars)',
|
|
330
|
+
'Text input fields',
|
|
331
|
+
'No length validation at the UI boundary causes a client-side crash on extreme input.',
|
|
332
|
+
'Add maxlength attribute to inputs and validate length server-side before any processing.'
|
|
333
|
+
));
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
// Detect ghost clickables: cursor:pointer with no handler
|
|
337
|
+
const deadClicks = await page.evaluate(() => {
|
|
338
|
+
const els = document.querySelectorAll('[style*="cursor: pointer"],[class*="cursor-pointer"],[class*="cursor_pointer"]');
|
|
339
|
+
const dead = [];
|
|
340
|
+
for (const el of els) {
|
|
341
|
+
const tag = el.tagName.toLowerCase();
|
|
342
|
+
const isInteractive = ['a', 'button', 'input', 'select', 'textarea'].includes(tag);
|
|
343
|
+
const hasRole = el.getAttribute('role');
|
|
344
|
+
const hasClick = el.onclick !== null || el.getAttribute('data-action');
|
|
345
|
+
if (!isInteractive && !hasRole && !hasClick) {
|
|
346
|
+
dead.push((el.className || el.id || tag).toString().substring(0, 60));
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
return dead.slice(0, 10);
|
|
350
|
+
}).catch(() => []);
|
|
351
|
+
|
|
352
|
+
if (deadClicks.length > 0) {
|
|
353
|
+
findings.push(makeFinding(
|
|
354
|
+
'low', 'ux',
|
|
355
|
+
`${deadClicks.length} element(s) appear clickable (cursor:pointer) but have no action`,
|
|
356
|
+
`Elements: ${deadClicks.slice(0, 3).join(' | ')}`,
|
|
357
|
+
'Cursor changes to pointer but clicking does nothing. Confuses users and erodes trust.',
|
|
358
|
+
'Remove cursor:pointer from non-interactive elements, or add the appropriate click handler or ARIA role.'
|
|
359
|
+
));
|
|
360
|
+
}
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
// ============================================================
|
|
364
|
+
// PERSONA — HACKER
|
|
365
|
+
// ============================================================
|
|
366
|
+
async function runHackerPersona(page, baseUrl, findings, screenshotsDir) {
|
|
367
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
368
|
+
|
|
369
|
+
await probeExposedSecrets(page, findings, screenshotsDir);
|
|
370
|
+
await probeSensitiveFiles(page, baseUrl, findings);
|
|
371
|
+
await probeXss(page, baseUrl, findings, screenshotsDir);
|
|
372
|
+
await probeOpenRedirect(page, baseUrl, findings);
|
|
373
|
+
await probeInjectionInputs(page, baseUrl, findings);
|
|
374
|
+
await probeDebugRoutes(page, baseUrl, findings);
|
|
375
|
+
|
|
376
|
+
// IDOR probe: detect numeric IDs in current URL, try ±1
|
|
377
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 10000 }).catch(() => {});
|
|
378
|
+
const currentUrl = page.url();
|
|
379
|
+
const idMatch = currentUrl.match(/\/(\d{1,9})(\/|$|\?)/);
|
|
380
|
+
if (idMatch) {
|
|
381
|
+
const id = parseInt(idMatch[1], 10);
|
|
382
|
+
for (const delta of [-1, 1, 9999]) {
|
|
383
|
+
try {
|
|
384
|
+
const probe = currentUrl.replace(`/${id}`, `/${id + delta}`);
|
|
385
|
+
const response = await page.goto(probe, { waitUntil: 'commit', timeout: 5000 });
|
|
386
|
+
if (response && response.status() === 200) {
|
|
387
|
+
findings.push(makeFinding(
|
|
388
|
+
'high', 'security',
|
|
389
|
+
`Potential IDOR: resource /${id} → /${id + delta} returns 200`,
|
|
390
|
+
probe,
|
|
391
|
+
'Incrementing the resource ID in the URL returns a valid response with no authorization rejection. May expose other users\' data.',
|
|
392
|
+
'Implement per-resource authorization: verify the authenticated user owns or is permitted to access the requested ID.'
|
|
393
|
+
));
|
|
394
|
+
break;
|
|
395
|
+
}
|
|
396
|
+
} catch { /* not accessible */ }
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
// ============================================================
|
|
402
|
+
// PERSONA — POWER USER
|
|
403
|
+
// ============================================================
|
|
404
|
+
async function runPowerPersona(page, baseUrl, findings) {
|
|
405
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
406
|
+
|
|
407
|
+
// Keyboard navigation — visible focus indicator
|
|
408
|
+
let missingFocus = 0;
|
|
409
|
+
for (let i = 0; i < 25; i++) {
|
|
410
|
+
await page.keyboard.press('Tab').catch(() => {});
|
|
411
|
+
const focusOk = await page.evaluate(() => {
|
|
412
|
+
const el = document.activeElement;
|
|
413
|
+
if (!el || el === document.body) return true;
|
|
414
|
+
const style = window.getComputedStyle(el);
|
|
415
|
+
const outline = style.outline || '';
|
|
416
|
+
return outline !== 'none' && !outline.startsWith('0px');
|
|
417
|
+
}).catch(() => true);
|
|
418
|
+
if (!focusOk) missingFocus++;
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
if (missingFocus > 2) {
|
|
422
|
+
findings.push(makeFinding(
|
|
423
|
+
'medium', 'accessibility',
|
|
424
|
+
`Keyboard focus indicator missing on ${missingFocus} interactive element(s)`,
|
|
425
|
+
'Tab key navigation',
|
|
426
|
+
'Users relying on keyboard cannot see which element is focused. WCAG 2.1 SC 2.4.7 (Level AA) violation.',
|
|
427
|
+
'Never use outline: none without an equivalent visible :focus-visible alternative. Ensure contrast ratio ≥ 3:1.'
|
|
428
|
+
));
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
// Boundary values on number inputs
|
|
432
|
+
const numberInputs = await page.$$('input[type="number"],input[type="range"]').catch(() => []);
|
|
433
|
+
for (const input of numberInputs.slice(0, 5)) {
|
|
434
|
+
for (const value of ['-999999999', '0', '999999999999999', '9007199254740992']) {
|
|
435
|
+
await input.fill(value).catch(() => {});
|
|
436
|
+
await page.keyboard.press('Tab').catch(() => {});
|
|
437
|
+
await page.waitForTimeout(200).catch(() => {});
|
|
438
|
+
}
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
// Date boundary values
|
|
442
|
+
const dateInputs = await page.$$('input[type="date"]').catch(() => []);
|
|
443
|
+
for (const input of dateInputs.slice(0, 3)) {
|
|
444
|
+
for (const value of ['1900-01-01', '9999-12-31', '2000-02-29']) {
|
|
445
|
+
await input.fill(value).catch(() => {});
|
|
446
|
+
await page.keyboard.press('Tab').catch(() => {});
|
|
447
|
+
}
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
// ============================================================
|
|
452
|
+
// PERSONA — MOBILE
|
|
453
|
+
// ============================================================
|
|
454
|
+
async function runMobilePersona(browser, baseUrl, findings, screenshotsDir) {
|
|
455
|
+
const context = await browser.newContext({
|
|
456
|
+
viewport: { width: 375, height: 667 },
|
|
457
|
+
userAgent: 'Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.0 Mobile/15E148 Safari/604.1',
|
|
458
|
+
hasTouch: true,
|
|
459
|
+
isMobile: true
|
|
460
|
+
}).catch(() => null);
|
|
461
|
+
if (!context) return;
|
|
462
|
+
|
|
463
|
+
const page = await context.newPage().catch(() => null);
|
|
464
|
+
if (!page) { await context.close().catch(() => {}); return; }
|
|
465
|
+
|
|
466
|
+
try {
|
|
467
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
468
|
+
|
|
469
|
+
// Horizontal overflow
|
|
470
|
+
const overflow = await page.evaluate(() => document.body.scrollWidth > window.innerWidth + 5).catch(() => false);
|
|
471
|
+
if (overflow) {
|
|
472
|
+
const f = makeFinding(
|
|
473
|
+
'medium', 'ux',
|
|
474
|
+
'Horizontal overflow on mobile viewport (375px)',
|
|
475
|
+
`${baseUrl} — iPhone SE viewport`,
|
|
476
|
+
'Content overflows horizontally on small screens. Forces unwanted horizontal scrolling. Breaks layout.',
|
|
477
|
+
'Audit for fixed-width elements wider than 375px. Use max-width: 100%, flexbox, or CSS grid for responsive layouts.'
|
|
478
|
+
);
|
|
479
|
+
f.screenshot = await takeScreenshot(page, screenshotsDir, f.id);
|
|
480
|
+
findings.push(f);
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
// Touch target size (< 44px = WCAG 2.5.5 violation)
|
|
484
|
+
const smallTargets = await page.evaluate(() => {
|
|
485
|
+
const els = document.querySelectorAll('a, button, [role="button"], input[type="submit"], input[type="checkbox"], input[type="radio"]');
|
|
486
|
+
const small = [];
|
|
487
|
+
for (const el of els) {
|
|
488
|
+
const r = el.getBoundingClientRect();
|
|
489
|
+
if (r.width > 0 && r.height > 0 && (r.width < 44 || r.height < 44)) {
|
|
490
|
+
small.push({ tag: el.tagName, text: (el.textContent || '').trim().substring(0, 30), w: Math.round(r.width), h: Math.round(r.height) });
|
|
491
|
+
}
|
|
492
|
+
}
|
|
493
|
+
return small.slice(0, 8);
|
|
494
|
+
}).catch(() => []);
|
|
495
|
+
|
|
496
|
+
if (smallTargets.length > 0) {
|
|
497
|
+
findings.push(makeFinding(
|
|
498
|
+
'low', 'accessibility',
|
|
499
|
+
`${smallTargets.length} touch target(s) smaller than 44×44px (WCAG 2.5.5)`,
|
|
500
|
+
smallTargets.map((t) => `${t.tag} "${t.text}" ${t.w}×${t.h}px`).slice(0, 3).join(', '),
|
|
501
|
+
'Small touch targets cause mis-taps, frustrate mobile users, and fail WCAG 2.5.5 (AAA) and Apple HIG guidelines.',
|
|
502
|
+
'Set min-height: 44px; min-width: 44px on all interactive elements. Increase padding rather than element size if needed.'
|
|
503
|
+
));
|
|
504
|
+
}
|
|
505
|
+
|
|
506
|
+
// Tiny fonts
|
|
507
|
+
const tinyFonts = await page.evaluate(() => {
|
|
508
|
+
const tiny = [];
|
|
509
|
+
for (const el of document.querySelectorAll('p, span, li, td, label, a')) {
|
|
510
|
+
if (!el.textContent.trim()) continue;
|
|
511
|
+
const size = parseFloat(window.getComputedStyle(el).fontSize);
|
|
512
|
+
if (size > 0 && size < 12) tiny.push({ tag: el.tagName, size, text: el.textContent.trim().substring(0, 40) });
|
|
513
|
+
}
|
|
514
|
+
return tiny.slice(0, 5);
|
|
515
|
+
}).catch(() => []);
|
|
516
|
+
|
|
517
|
+
if (tinyFonts.length > 0) {
|
|
518
|
+
findings.push(makeFinding(
|
|
519
|
+
'low', 'accessibility',
|
|
520
|
+
`${tinyFonts.length} text element(s) with font size below 12px`,
|
|
521
|
+
tinyFonts.map((t) => `${t.tag} (${t.size}px)`).slice(0, 3).join(', '),
|
|
522
|
+
'Text smaller than 12px triggers automatic zoom on iOS, breaking layout. Very hard to read without zooming.',
|
|
523
|
+
'Set a minimum font size of 12px. Use rem/em units for scalable typography across screen sizes.'
|
|
524
|
+
));
|
|
525
|
+
}
|
|
526
|
+
} finally {
|
|
527
|
+
await page.close().catch(() => {});
|
|
528
|
+
await context.close().catch(() => {});
|
|
529
|
+
}
|
|
530
|
+
}
|
|
531
|
+
|
|
532
|
+
// ============================================================
|
|
533
|
+
// ACCESSIBILITY AUDIT
|
|
534
|
+
// ============================================================
|
|
535
|
+
async function checkAccessibility(page, findings) {
|
|
536
|
+
const issues = await page.evaluate(() => {
|
|
537
|
+
const result = [];
|
|
538
|
+
|
|
539
|
+
const imgs = document.querySelectorAll('img:not([alt])');
|
|
540
|
+
if (imgs.length) result.push({ type: 'img_no_alt', count: imgs.length });
|
|
541
|
+
|
|
542
|
+
let unlabeled = 0;
|
|
543
|
+
for (const input of document.querySelectorAll('input:not([type="hidden"]):not([type="submit"]):not([type="button"]):not([type="reset"])')) {
|
|
544
|
+
const hasLabel = input.id && document.querySelector(`label[for="${input.id}"]`);
|
|
545
|
+
const hasAria = input.getAttribute('aria-label') || input.getAttribute('aria-labelledby');
|
|
546
|
+
if (!hasLabel && !hasAria) unlabeled++;
|
|
547
|
+
}
|
|
548
|
+
if (unlabeled) result.push({ type: 'input_no_label', count: unlabeled });
|
|
549
|
+
|
|
550
|
+
let unnamed = 0;
|
|
551
|
+
for (const btn of document.querySelectorAll('button,[role="button"]')) {
|
|
552
|
+
const hasText = (btn.textContent || '').trim().length > 0;
|
|
553
|
+
const hasAria = btn.getAttribute('aria-label') || btn.getAttribute('aria-labelledby') || btn.getAttribute('title');
|
|
554
|
+
if (!hasText && !hasAria) unnamed++;
|
|
555
|
+
}
|
|
556
|
+
if (unnamed) result.push({ type: 'button_no_name', count: unnamed });
|
|
557
|
+
|
|
558
|
+
const headings = [...document.querySelectorAll('h1,h2,h3,h4,h5,h6')];
|
|
559
|
+
for (let i = 1; i < headings.length; i++) {
|
|
560
|
+
if (parseInt(headings[i].tagName[1]) - parseInt(headings[i - 1].tagName[1]) > 1) {
|
|
561
|
+
result.push({ type: 'heading_skip' });
|
|
562
|
+
break;
|
|
563
|
+
}
|
|
564
|
+
}
|
|
565
|
+
|
|
566
|
+
if (!document.querySelector('html[lang]')) result.push({ type: 'no_lang' });
|
|
567
|
+
|
|
568
|
+
return result;
|
|
569
|
+
}).catch(() => []);
|
|
570
|
+
|
|
571
|
+
const defs = {
|
|
572
|
+
img_no_alt: { sev: 'medium', title: '{count} image(s) missing alt attribute', location: '<img> elements', risk: 'Screen readers cannot describe images to visually impaired users. WCAG 1.1.1 (Level A) violation.', fix: 'Add descriptive alt text to all informative images. Use alt="" for decorative images.' },
|
|
573
|
+
input_no_label: { sev: 'medium', title: '{count} form input(s) with no accessible label', location: '<input> elements', risk: 'Screen readers announce only the input type with no context. WCAG 1.3.1 (Level A) violation.', fix: 'Add <label for="..."> or aria-label to every form input.' },
|
|
574
|
+
button_no_name: { sev: 'medium', title: '{count} button(s) with no accessible name', location: '<button> elements', risk: 'Screen readers say "button" with no indication of what action it triggers. WCAG 4.1.2 violation.', fix: 'Add visible text or aria-label to every button.' },
|
|
575
|
+
heading_skip: { sev: 'low', title: 'Heading level skipped (e.g. h1 → h3)', location: 'Document heading structure', risk: 'Screen reader users rely on heading hierarchy for page navigation. Skipped levels break their mental model.', fix: 'Use sequential heading levels. Never choose heading levels for visual size — use CSS instead.' },
|
|
576
|
+
no_lang: { sev: 'low', title: '<html> element missing lang attribute', location: '<html> tag', risk: 'Screen readers guess language for pronunciation. Wrong language causes incorrect speech. WCAG 3.1.1 (Level A).', fix: 'Add lang="en" (or appropriate BCP-47 code) to the <html> element.' }
|
|
577
|
+
};
|
|
578
|
+
|
|
579
|
+
for (const issue of issues) {
|
|
580
|
+
const d = defs[issue.type];
|
|
581
|
+
if (!d) continue;
|
|
582
|
+
findings.push(makeFinding(d.sev, 'accessibility', d.title.replace('{count}', issue.count || ''), d.location, d.risk, d.fix));
|
|
583
|
+
}
|
|
584
|
+
}
|
|
585
|
+
|
|
586
|
+
// ============================================================
|
|
587
|
+
// PERFORMANCE
|
|
588
|
+
// ============================================================
|
|
589
|
+
async function capturePerformance(page, thresholds, findings) {
|
|
590
|
+
const perf = await page.evaluate(() => {
|
|
591
|
+
const nav = performance.getEntriesByType('navigation')[0];
|
|
592
|
+
if (!nav) return null;
|
|
593
|
+
const resources = performance.getEntriesByType('resource');
|
|
594
|
+
return {
|
|
595
|
+
domContentLoaded: Math.round(nav.domContentLoadedEventEnd),
|
|
596
|
+
loadComplete: Math.round(nav.loadEventEnd),
|
|
597
|
+
ttfb: Math.round(nav.responseStart - nav.requestStart),
|
|
598
|
+
resourceCount: resources.length,
|
|
599
|
+
resourceSizeKb: Math.round(resources.reduce((acc, r) => acc + (r.transferSize || 0), 0) / 1024)
|
|
600
|
+
};
|
|
601
|
+
}).catch(() => null);
|
|
602
|
+
|
|
603
|
+
if (!perf) return null;
|
|
604
|
+
|
|
605
|
+
if (perf.loadComplete > (thresholds.page_load_ms || 3000)) {
|
|
606
|
+
findings.push(makeFinding(
|
|
607
|
+
'medium', 'performance',
|
|
608
|
+
`Page load time exceeds threshold (${perf.loadComplete}ms > ${thresholds.page_load_ms || 3000}ms)`,
|
|
609
|
+
'Page load',
|
|
610
|
+
'Slow page load degrades UX and Core Web Vitals score. Google Search penalizes pages with poor LCP.',
|
|
611
|
+
'Analyze network waterfall. Defer non-critical JS. Enable Gzip/Brotli compression. Use a CDN for static assets.'
|
|
612
|
+
));
|
|
613
|
+
}
|
|
614
|
+
|
|
615
|
+
if (perf.ttfb > (thresholds.ttfb_ms || 800)) {
|
|
616
|
+
findings.push(makeFinding(
|
|
617
|
+
'medium', 'performance',
|
|
618
|
+
`High Time to First Byte — TTFB ${perf.ttfb}ms`,
|
|
619
|
+
'Server response time',
|
|
620
|
+
'TTFB > 800ms means the server is slow to respond. Users see a blank page for too long.',
|
|
621
|
+
'Optimize database queries, add server-side caching, or review server infrastructure capacity.'
|
|
622
|
+
));
|
|
623
|
+
}
|
|
624
|
+
|
|
625
|
+
if (perf.resourceCount > (thresholds.requests_max || 80)) {
|
|
626
|
+
findings.push(makeFinding(
|
|
627
|
+
'low', 'performance',
|
|
628
|
+
`High request count: ${perf.resourceCount} network requests on load`,
|
|
629
|
+
'Network requests',
|
|
630
|
+
`${perf.resourceCount} requests slow down page load and increases server load.`,
|
|
631
|
+
'Bundle JavaScript and CSS files. Use HTTP/2 multiplexing. Lazy-load images and below-the-fold content.'
|
|
632
|
+
));
|
|
633
|
+
}
|
|
634
|
+
|
|
635
|
+
if (perf.resourceSizeKb > (thresholds.transfer_max_kb || 2048)) {
|
|
636
|
+
findings.push(makeFinding(
|
|
637
|
+
'low', 'performance',
|
|
638
|
+
`Total transfer size ${perf.resourceSizeKb}KB exceeds threshold`,
|
|
639
|
+
'Network transfer',
|
|
640
|
+
`Large payload increases load time on slow connections and mobile data.`,
|
|
641
|
+
'Enable compression. Audit and tree-shake large JS bundles. Optimize images (WebP, lazy loading).'
|
|
642
|
+
));
|
|
643
|
+
}
|
|
644
|
+
|
|
645
|
+
return perf;
|
|
646
|
+
}
|
|
647
|
+
|
|
648
|
+
// ============================================================
|
|
649
|
+
// AC COVERAGE
|
|
650
|
+
// ============================================================
|
|
651
|
+
function parseAcItems(prdContent) {
|
|
652
|
+
if (!prdContent) return [];
|
|
653
|
+
const items = [];
|
|
654
|
+
for (const match of String(prdContent).matchAll(/\|\s*(AC-\d+)\s*\|\s*([^|]+)\|/g)) {
|
|
655
|
+
items.push({ id: match[1].trim(), description: match[2].trim() });
|
|
656
|
+
}
|
|
657
|
+
for (const match of String(prdContent).matchAll(/🔴\s*([^\n]{10,100})/g)) {
|
|
658
|
+
if (items.length >= 20) break;
|
|
659
|
+
items.push({ id: `AC-${String(items.length + 1).padStart(2, '0')}`, description: match[1].trim() });
|
|
660
|
+
}
|
|
661
|
+
return items.slice(0, 20);
|
|
662
|
+
}
|
|
663
|
+
|
|
664
|
+
async function runAcCoverage(page, baseUrl, prdPath, screenshotsDir) {
|
|
665
|
+
const prdContent = await readTextIfExists(prdPath);
|
|
666
|
+
const acItems = parseAcItems(prdContent);
|
|
667
|
+
if (acItems.length === 0) return [];
|
|
668
|
+
|
|
669
|
+
await page.goto(baseUrl, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
670
|
+
|
|
671
|
+
const coverage = [];
|
|
672
|
+
for (const ac of acItems) {
|
|
673
|
+
const screenshotFile = path.join(screenshotsDir, `${ac.id}.png`);
|
|
674
|
+
await page.screenshot({ path: screenshotFile, fullPage: false }).catch(() => {});
|
|
675
|
+
coverage.push({
|
|
676
|
+
id: ac.id,
|
|
677
|
+
description: ac.description,
|
|
678
|
+
status: 'Documented',
|
|
679
|
+
screenshot: screenshotFile
|
|
680
|
+
});
|
|
681
|
+
}
|
|
682
|
+
return coverage;
|
|
683
|
+
}
|
|
684
|
+
|
|
685
|
+
// ============================================================
|
|
686
|
+
// REPORT GENERATION
|
|
687
|
+
// ============================================================
|
|
688
|
+
function buildMarkdownReport(projectName, url, findings, acCoverage, perf, mode) {
|
|
689
|
+
const sorted = [...findings].sort((a, b) => {
|
|
690
|
+
const o = { critical: 0, high: 1, medium: 2, low: 3 };
|
|
691
|
+
return (o[a.severity] ?? 4) - (o[b.severity] ?? 4);
|
|
692
|
+
});
|
|
693
|
+
const bySev = (s) => sorted.filter((f) => f.severity === s);
|
|
694
|
+
const date = new Date().toISOString().split('T')[0];
|
|
695
|
+
|
|
696
|
+
let md = `## QA Browser Report — ${projectName} — ${date}\n\n`;
|
|
697
|
+
md += `> Generated by: \`aioson qa:${mode}\` \n`;
|
|
698
|
+
md += `> Browser: Chromium | Viewport: 1280×720 \n`;
|
|
699
|
+
md += `> URL: ${url}\n\n`;
|
|
700
|
+
|
|
701
|
+
if (acCoverage.length > 0) {
|
|
702
|
+
md += `### Acceptance criteria coverage\n| AC | Description | Status |\n|---|---|---|\n`;
|
|
703
|
+
for (const ac of acCoverage) md += `| ${ac.id} | ${ac.description} | ${ac.status} |\n`;
|
|
704
|
+
md += '\n';
|
|
705
|
+
}
|
|
706
|
+
|
|
707
|
+
md += `### Findings\n\n`;
|
|
708
|
+
for (const [label, group] of [['Critical', bySev('critical')], ['High', bySev('high')], ['Medium', bySev('medium')], ['Low', bySev('low')]]) {
|
|
709
|
+
if (group.length === 0) continue;
|
|
710
|
+
md += `#### ${label}\n`;
|
|
711
|
+
for (const f of group) {
|
|
712
|
+
md += `**[${f.id}] ${f.title}** \n`;
|
|
713
|
+
md += `Location: \`${f.location}\` \n`;
|
|
714
|
+
md += `Risk: ${f.risk} \n`;
|
|
715
|
+
md += `Fix: ${f.fix} \n`;
|
|
716
|
+
if (f.screenshot) md += `Screenshot: ${f.screenshot} \n`;
|
|
717
|
+
md += '\n';
|
|
718
|
+
}
|
|
719
|
+
}
|
|
720
|
+
|
|
721
|
+
if (perf) {
|
|
722
|
+
md += `### Performance\n| Metric | Value |\n|---|---|\n`;
|
|
723
|
+
md += `| DOM Content Loaded | ${perf.domContentLoaded}ms |\n`;
|
|
724
|
+
md += `| Page Load Complete | ${perf.loadComplete}ms |\n`;
|
|
725
|
+
md += `| Time to First Byte | ${perf.ttfb}ms |\n`;
|
|
726
|
+
md += `| Network requests | ${perf.resourceCount} |\n`;
|
|
727
|
+
md += `| Total transfer | ${perf.resourceSizeKb}KB |\n\n`;
|
|
728
|
+
}
|
|
729
|
+
|
|
730
|
+
md += `### Residual risks\n`;
|
|
731
|
+
md += `- Tests run against a running instance; production environment may differ (headers, CSP, CDN).\n`;
|
|
732
|
+
md += `- Content behind authentication was not tested — no credentials were provided.\n`;
|
|
733
|
+
md += `- JavaScript-heavy interactions may need additional manual verification.\n\n`;
|
|
734
|
+
|
|
735
|
+
const c = bySev('critical').length, h = bySev('high').length, m = bySev('medium').length, l = bySev('low').length;
|
|
736
|
+
md += `### Summary\n`;
|
|
737
|
+
md += `- Critical: ${c} | High: ${h} | Medium: ${m} | Low: ${l}\n`;
|
|
738
|
+
if (acCoverage.length > 0) md += `- AC documented: ${acCoverage.length}\n`;
|
|
739
|
+
|
|
740
|
+
return md;
|
|
741
|
+
}
|
|
742
|
+
|
|
743
|
+
async function writeReports(targetDir, projectName, url, findings, acCoverage, perf, mode) {
|
|
744
|
+
const mdPath = path.join(targetDir, 'aios-qa-report.md');
|
|
745
|
+
const jsonPath = path.join(targetDir, 'aios-qa-report.json');
|
|
746
|
+
const md = buildMarkdownReport(projectName, url, findings, acCoverage, perf, mode);
|
|
747
|
+
const bySev = (s) => findings.filter((f) => f.severity === s).length;
|
|
748
|
+
const json = {
|
|
749
|
+
generated_at: new Date().toISOString(),
|
|
750
|
+
project: projectName, url, mode,
|
|
751
|
+
summary: { critical: bySev('critical'), high: bySev('high'), medium: bySev('medium'), low: bySev('low') },
|
|
752
|
+
ac_coverage: acCoverage,
|
|
753
|
+
performance: perf,
|
|
754
|
+
findings
|
|
755
|
+
};
|
|
756
|
+
await fs.writeFile(mdPath, md, 'utf8');
|
|
757
|
+
await fs.writeFile(jsonPath, `${JSON.stringify(json, null, 2)}\n`, 'utf8');
|
|
758
|
+
return { mdPath, jsonPath };
|
|
759
|
+
}
|
|
760
|
+
|
|
761
|
+
// ============================================================
|
|
762
|
+
// MAIN
|
|
763
|
+
// ============================================================
|
|
764
|
+
async function runQaRun({ args, options = {}, logger, t }) {
|
|
765
|
+
const targetDir = path.resolve(process.cwd(), args[0] || '.');
|
|
766
|
+
|
|
767
|
+
const pw = requirePlaywright();
|
|
768
|
+
if (!pw) {
|
|
769
|
+
logger.error(t('qa_run.playwright_missing'));
|
|
770
|
+
process.exitCode = 1;
|
|
771
|
+
return { ok: false, error: 'playwright_not_installed' };
|
|
772
|
+
}
|
|
773
|
+
|
|
774
|
+
const config = await loadConfig(targetDir);
|
|
775
|
+
if (!config) {
|
|
776
|
+
logger.error(t('qa_run.config_missing'));
|
|
777
|
+
process.exitCode = 1;
|
|
778
|
+
return { ok: false, error: 'config_not_found' };
|
|
779
|
+
}
|
|
780
|
+
|
|
781
|
+
const url = String(options.url || config.url || '');
|
|
782
|
+
if (!url) {
|
|
783
|
+
logger.error(t('qa_run.url_missing'));
|
|
784
|
+
process.exitCode = 1;
|
|
785
|
+
return { ok: false, error: 'url_not_configured' };
|
|
786
|
+
}
|
|
787
|
+
|
|
788
|
+
const projectName = config.project_name || path.basename(targetDir) || 'Project';
|
|
789
|
+
const selectedPersona = String(options.persona || '').toLowerCase() || null;
|
|
790
|
+
const headed = Boolean(options.headed);
|
|
791
|
+
const screenshotsDir = path.join(targetDir, 'aios-qa-screenshots');
|
|
792
|
+
const prdPath = path.join(targetDir, '.aioson/context/prd.md');
|
|
793
|
+
const thresholds = config.performance_thresholds || {};
|
|
794
|
+
|
|
795
|
+
_counter = 0;
|
|
796
|
+
const findings = [];
|
|
797
|
+
const consoleLogs = [];
|
|
798
|
+
const networkRequests = [];
|
|
799
|
+
|
|
800
|
+
logger.log(t('qa_run.starting', { url }));
|
|
801
|
+
await ensureDir(screenshotsDir);
|
|
802
|
+
|
|
803
|
+
const browser = await pw.chromium.launch({ headless: !headed });
|
|
804
|
+
const context = await browser.newContext({ viewport: { width: 1280, height: 720 } });
|
|
805
|
+
const page = await context.newPage();
|
|
806
|
+
|
|
807
|
+
page.on('console', (msg) => consoleLogs.push({ type: msg.type(), text: msg.text() }));
|
|
808
|
+
page.on('request', (req) => networkRequests.push({ url: req.url(), method: req.method() }));
|
|
809
|
+
|
|
810
|
+
try {
|
|
811
|
+
const personas = config.personas || ['naive', 'hacker', 'power', 'mobile'];
|
|
812
|
+
|
|
813
|
+
for (const persona of personas) {
|
|
814
|
+
if (selectedPersona && persona !== selectedPersona) continue;
|
|
815
|
+
logger.log(t('qa_run.persona_start', { persona }));
|
|
816
|
+
const before = findings.length;
|
|
817
|
+
|
|
818
|
+
if (persona === 'naive') await runNaivePersona(page, url, findings, screenshotsDir).catch(() => {});
|
|
819
|
+
else if (persona === 'hacker') await runHackerPersona(page, url, findings, screenshotsDir).catch(() => {});
|
|
820
|
+
else if (persona === 'power') await runPowerPersona(page, url, findings).catch(() => {});
|
|
821
|
+
else if (persona === 'mobile') await runMobilePersona(browser, url, findings, screenshotsDir).catch(() => {});
|
|
822
|
+
|
|
823
|
+
logger.log(t('qa_run.persona_done', { persona, count: findings.length - before }));
|
|
824
|
+
}
|
|
825
|
+
|
|
826
|
+
// Network + console analysis
|
|
827
|
+
await probeInjectionInjection(page, findings, consoleLogs, networkRequests).catch(() => {});
|
|
828
|
+
|
|
829
|
+
// Accessibility
|
|
830
|
+
logger.log(t('qa_run.accessibility'));
|
|
831
|
+
await page.goto(url, { waitUntil: 'domcontentloaded', timeout: 15000 }).catch(() => {});
|
|
832
|
+
await checkAccessibility(page, findings).catch(() => {});
|
|
833
|
+
|
|
834
|
+
// Performance
|
|
835
|
+
logger.log(t('qa_run.performance'));
|
|
836
|
+
await page.goto(url, { waitUntil: 'load', timeout: 20000 }).catch(() => {});
|
|
837
|
+
const perf = await capturePerformance(page, thresholds, findings).catch(() => null);
|
|
838
|
+
|
|
839
|
+
// AC coverage
|
|
840
|
+
logger.log(t('qa_run.ac_scenarios'));
|
|
841
|
+
const acCoverage = await runAcCoverage(page, url, prdPath, screenshotsDir).catch(() => []);
|
|
842
|
+
|
|
843
|
+
// Write reports
|
|
844
|
+
const { mdPath, jsonPath } = await writeReports(targetDir, projectName, url, findings, acCoverage, perf, 'run');
|
|
845
|
+
|
|
846
|
+
logger.log(t('qa_run.done'));
|
|
847
|
+
logger.log(t('qa_run.report_written', { path: mdPath }));
|
|
848
|
+
logger.log(t('qa_run.json_written', { path: jsonPath }));
|
|
849
|
+
logger.log(t('qa_run.screenshots_dir', { path: screenshotsDir }));
|
|
850
|
+
|
|
851
|
+
const bySev = (s) => findings.filter((f) => f.severity === s).length;
|
|
852
|
+
const summary = { critical: bySev('critical'), high: bySev('high'), medium: bySev('medium'), low: bySev('low') };
|
|
853
|
+
logger.log(t('qa_run.findings_summary', summary));
|
|
854
|
+
|
|
855
|
+
// HTML report (optional, additive — does not replace MD/JSON)
|
|
856
|
+
let htmlPath, htmlDir;
|
|
857
|
+
if (options.html) {
|
|
858
|
+
const { writeHtmlReport } = require('../qa-html-report');
|
|
859
|
+
const result = await writeHtmlReport(targetDir, projectName, url, findings, acCoverage, perf, 'run', screenshotsDir, { thresholds });
|
|
860
|
+
htmlPath = result.htmlPath;
|
|
861
|
+
htmlDir = result.runDir;
|
|
862
|
+
logger.log(t('qa_run.html_report_written', { path: htmlPath }));
|
|
863
|
+
}
|
|
864
|
+
|
|
865
|
+
const output = { ok: true, targetDir, url, summary, mdPath, jsonPath, screenshotsDir, findings, acCoverage, ...(htmlPath ? { htmlPath, htmlDir } : {}) };
|
|
866
|
+
if (options.json) return output;
|
|
867
|
+
return output;
|
|
868
|
+
} finally {
|
|
869
|
+
await browser.close().catch(() => {});
|
|
870
|
+
}
|
|
871
|
+
}
|
|
872
|
+
|
|
873
|
+
module.exports = { runQaRun };
|