@jaimevalasek/aioson 1.7.2 → 1.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (1049) hide show
  1. package/CHANGELOG.md +595 -560
  2. package/CODE_OF_CONDUCT.md +12 -12
  3. package/CONTRIBUTING.md +13 -13
  4. package/LICENSE +661 -661
  5. package/README.md +919 -776
  6. package/bin/aioson.js +4 -4
  7. package/docs/design-previews/aurora-command-ui-website.html +884 -884
  8. package/docs/design-previews/aurora-command-ui.html +682 -682
  9. package/docs/design-previews/bold-editorial-ui-website.html +658 -658
  10. package/docs/design-previews/bold-editorial-ui.html +717 -717
  11. package/docs/design-previews/clean-saas-ui-website.html +1202 -1202
  12. package/docs/design-previews/clean-saas-ui.html +549 -549
  13. package/docs/design-previews/cognitive-core-ui-website.html +1009 -1009
  14. package/docs/design-previews/cognitive-core-ui.html +463 -463
  15. package/docs/design-previews/glassmorphism-ui-website.html +572 -572
  16. package/docs/design-previews/glassmorphism-ui.html +886 -886
  17. package/docs/design-previews/index.html +699 -699
  18. package/docs/design-previews/interface-design-website.html +1187 -1187
  19. package/docs/design-previews/interface-design.html +513 -513
  20. package/docs/design-previews/neo-brutalist-ui-website.html +621 -621
  21. package/docs/design-previews/neo-brutalist-ui.html +797 -797
  22. package/docs/design-previews/premium-command-center-ui-website.html +1217 -1217
  23. package/docs/design-previews/premium-command-center-ui.html +552 -552
  24. package/docs/design-previews/pt.squarespace.com-homepage.html +889 -889
  25. package/docs/design-previews/warm-craft-ui-website.html +684 -684
  26. package/docs/design-previews/warm-craft-ui.html +739 -739
  27. package/docs/en/1-understand/ecosystem-map.md +228 -0
  28. package/docs/en/1-understand/glossary.md +288 -0
  29. package/docs/en/1-understand/what-is-aioson.md +94 -0
  30. package/docs/en/1-understand/why-it-exists.md +106 -0
  31. package/docs/en/2-start/existing-project.md +246 -0
  32. package/docs/en/2-start/first-project.md +307 -0
  33. package/docs/en/2-start/initial-decisions.md +223 -0
  34. package/docs/en/3-recipes/README.md +28 -0
  35. package/docs/en/3-recipes/continuity-between-sessions.md +303 -0
  36. package/docs/en/3-recipes/from-idea-to-prd-via-briefing.md +235 -0
  37. package/docs/en/3-recipes/full-feature-with-sheldon.md +338 -0
  38. package/docs/en/4-agents/README.md +56 -0
  39. package/docs/en/5-reference/README.md +60 -0
  40. package/docs/en/{cli-reference.md → 5-reference/cli-reference.md} +639 -409
  41. package/docs/en/5-reference/i18n.md +52 -0
  42. package/docs/en/{json-schemas.md → 5-reference/json-schemas.md} +41 -41
  43. package/docs/en/{mcp.md → 5-reference/mcp.md} +56 -56
  44. package/docs/en/{parallel.md → 5-reference/parallel.md} +82 -82
  45. package/docs/en/{qa-browser.md → 5-reference/qa-browser.md} +339 -339
  46. package/docs/en/{release-flow.md → 5-reference/release-flow.md} +22 -22
  47. package/docs/en/{release-notes-template.md → 5-reference/release-notes-template.md} +41 -41
  48. package/docs/en/{release.md → 5-reference/release.md} +28 -28
  49. package/docs/en/{schemas → 5-reference/schemas}/agent-prompt.schema.json +17 -17
  50. package/docs/en/{schemas → 5-reference/schemas}/agents.schema.json +32 -32
  51. package/docs/en/{schemas → 5-reference/schemas}/context-validate.schema.json +36 -36
  52. package/docs/en/{schemas → 5-reference/schemas}/doctor.schema.json +89 -89
  53. package/docs/en/{schemas → 5-reference/schemas}/error.schema.json +24 -24
  54. package/docs/en/{schemas → 5-reference/schemas}/i18n-add.schema.json +15 -15
  55. package/docs/en/{schemas → 5-reference/schemas}/index.json +126 -116
  56. package/docs/en/{schemas → 5-reference/schemas}/info.schema.json +39 -39
  57. package/docs/en/{schemas → 5-reference/schemas}/init.schema.json +48 -48
  58. package/docs/en/{schemas → 5-reference/schemas}/install.schema.json +60 -60
  59. package/docs/en/{schemas → 5-reference/schemas}/locale-apply.schema.json +30 -30
  60. package/docs/en/{schemas → 5-reference/schemas}/mcp-doctor.schema.json +95 -95
  61. package/docs/en/{schemas → 5-reference/schemas}/mcp-init.schema.json +122 -122
  62. package/docs/en/{schemas → 5-reference/schemas}/package-test.schema.json +24 -24
  63. package/docs/en/{schemas → 5-reference/schemas}/parallel-assign.schema.json +66 -57
  64. package/docs/en/{schemas → 5-reference/schemas}/parallel-doctor.schema.json +122 -86
  65. package/docs/en/5-reference/schemas/parallel-guard.schema.json +63 -0
  66. package/docs/en/{schemas → 5-reference/schemas}/parallel-init.schema.json +53 -53
  67. package/docs/en/5-reference/schemas/parallel-merge.schema.json +84 -0
  68. package/docs/en/5-reference/schemas/parallel-status.schema.json +184 -0
  69. package/docs/en/{schemas → 5-reference/schemas}/setup-context.schema.json +39 -39
  70. package/docs/en/{schemas → 5-reference/schemas}/smoke.schema.json +23 -23
  71. package/docs/en/{schemas → 5-reference/schemas}/update.schema.json +48 -48
  72. package/docs/en/{schemas → 5-reference/schemas}/workflow-plan.schema.json +30 -30
  73. package/docs/en/{squad-dashboard.md → 5-reference/squad-dashboard.md} +372 -372
  74. package/docs/en/{web3.md → 5-reference/web3.md} +54 -54
  75. package/docs/en/README.md +115 -0
  76. package/docs/en/active-learning-loop/README.md +117 -0
  77. package/docs/en/active-learning-loop/active-learning-loop.md +117 -0
  78. package/docs/en/active-learning-loop/cli-commands.md +320 -0
  79. package/docs/en/active-learning-loop/diagrams.md +225 -0
  80. package/docs/en/active-learning-loop/doctor-checks.md +151 -0
  81. package/docs/en/active-learning-loop/how-to-use.md +313 -0
  82. package/docs/en/active-learning-loop/troubleshooting.md +283 -0
  83. package/docs/en/deyvin-subtask-scout/README.md +109 -0
  84. package/docs/en/deyvin-subtask-scout/cli-commands.md +248 -0
  85. package/docs/en/deyvin-subtask-scout/diagrams.md +124 -0
  86. package/docs/en/deyvin-subtask-scout/how-to-use.md +221 -0
  87. package/docs/en/deyvin-subtask-scout/sub-task-scout.md +115 -0
  88. package/docs/en/deyvin-subtask-scout/troubleshooting.md +184 -0
  89. package/docs/integrations/apps-publish-marketplace.md +94 -0
  90. package/docs/integrations/sdlc-genius-boundary.md +76 -76
  91. package/docs/integrations/sdlc-genius-eval-matrix.md +75 -75
  92. package/docs/integrations/sdlc-genius-install-checklist.md +93 -93
  93. package/docs/integrations/sdlc-genius-review-samples.md +86 -86
  94. package/docs/openclaw-bridge.md +308 -308
  95. package/docs/pt/1-entender/glossario.md +288 -0
  96. package/docs/pt/1-entender/mapa-do-ecossistema.md +228 -0
  97. package/docs/pt/1-entender/o-que-e-aioson.md +94 -0
  98. package/docs/pt/1-entender/por-que-existe.md +107 -0
  99. package/docs/pt/2-comecar/decisoes-iniciais.md +223 -0
  100. package/docs/pt/2-comecar/primeiro-projeto.md +307 -0
  101. package/docs/pt/2-comecar/projeto-existente.md +245 -0
  102. package/docs/pt/3-receitas/README.md +28 -0
  103. package/docs/pt/3-receitas/app-saas-do-zero.md +324 -0
  104. package/docs/pt/3-receitas/auditoria-seguranca.md +254 -0
  105. package/docs/pt/3-receitas/clonar-design-de-site.md +211 -0
  106. package/docs/pt/3-receitas/continuidade-entre-sessoes.md +303 -0
  107. package/docs/pt/3-receitas/da-ideia-ao-prd-via-briefing.md +234 -0
  108. package/docs/pt/3-receitas/feature-completa-com-sheldon.md +338 -0
  109. package/docs/pt/3-receitas/integracao-em-codebase-grande.md +243 -0
  110. package/docs/pt/3-receitas/landing-page.md +281 -0
  111. package/docs/pt/3-receitas/plans-externos-para-product.md +191 -0
  112. package/docs/pt/3-receitas/publicar-no-aioson-com.md +219 -0
  113. package/docs/pt/3-receitas/refatoracao-grande.md +251 -0
  114. package/docs/pt/4-agentes/README.md +65 -0
  115. package/docs/pt/4-agentes/analyst.md +111 -0
  116. package/docs/pt/4-agentes/architect.md +113 -0
  117. package/docs/pt/4-agentes/briefing.md +95 -0
  118. package/docs/pt/4-agentes/committer.md +108 -0
  119. package/docs/pt/4-agentes/copywriter.md +279 -0
  120. package/docs/pt/4-agentes/design-hybrid-forge.md +116 -0
  121. package/docs/pt/4-agentes/dev.md +136 -0
  122. package/docs/pt/4-agentes/deyvin.md +99 -0
  123. package/docs/pt/4-agentes/discover.md +122 -0
  124. package/docs/pt/4-agentes/discovery-design-doc.md +91 -0
  125. package/docs/pt/4-agentes/genome.md +115 -0
  126. package/docs/pt/4-agentes/neo.md +93 -0
  127. package/docs/pt/4-agentes/orache.md +107 -0
  128. package/docs/pt/4-agentes/orchestrator.md +118 -0
  129. package/docs/pt/4-agentes/pentester.md +131 -0
  130. package/docs/pt/4-agentes/pm.md +97 -0
  131. package/docs/pt/4-agentes/product.md +114 -0
  132. package/docs/pt/4-agentes/profiler-enricher.md +93 -0
  133. package/docs/pt/4-agentes/profiler-forge.md +93 -0
  134. package/docs/pt/4-agentes/profiler-researcher.md +98 -0
  135. package/docs/pt/4-agentes/qa.md +124 -0
  136. package/docs/pt/4-agentes/setup.md +104 -0
  137. package/docs/pt/4-agentes/sheldon.md +95 -0
  138. package/docs/pt/4-agentes/site-forge.md +104 -0
  139. package/docs/pt/4-agentes/squad.md +127 -0
  140. package/docs/pt/4-agentes/tester.md +105 -0
  141. package/docs/pt/4-agentes/ux-ui.md +110 -0
  142. package/docs/pt/4-agentes/validator.md +118 -0
  143. package/docs/pt/5-referencia/README.md +88 -0
  144. package/docs/pt/5-referencia/agent-chain-continuity.md +124 -0
  145. package/docs/pt/{agent-sharding.md → 5-referencia/agent-sharding.md} +132 -132
  146. package/docs/pt/5-referencia/aioson-com-store.md +119 -0
  147. package/docs/pt/{automacao-squads.md → 5-referencia/automacao-squads.md} +407 -407
  148. package/docs/pt/{clientes-ai.md → 5-referencia/clientes-ai.md} +300 -286
  149. package/docs/pt/{comandos-cli.md → 5-referencia/comandos-cli.md} +1823 -1634
  150. package/docs/pt/5-referencia/compress-agents.md +304 -0
  151. package/docs/pt/5-referencia/design-docs-governance.md +59 -0
  152. package/docs/pt/{devlog-pipeline.md → 5-referencia/devlog-pipeline.md} +270 -270
  153. package/docs/pt/5-referencia/feature-archive.md +199 -0
  154. package/docs/pt/5-referencia/feature-dossier.md +121 -0
  155. package/docs/pt/{fluxo-artefatos.md → 5-referencia/fluxo-artefatos.md} +179 -178
  156. package/docs/pt/{genome-3.0-spec.md → 5-referencia/genome-4.0-spec.md} +407 -296
  157. package/docs/pt/5-referencia/genome-distribution.md +232 -0
  158. package/docs/pt/{hooks-session-guard.md → 5-referencia/hooks-session-guard.md} +454 -454
  159. package/docs/pt/{inteligencia-adaptativa.md → 5-referencia/inteligencia-adaptativa.md} +324 -324
  160. package/docs/pt/5-referencia/live-sessions.md +144 -0
  161. package/docs/pt/5-referencia/memoria-e-contexto.md +340 -0
  162. package/docs/pt/5-referencia/motor-hardening.md +493 -0
  163. package/docs/pt/{output-strategy-delivery.md → 5-referencia/output-strategy-delivery.md} +655 -655
  164. package/docs/pt/5-referencia/runner-system.md +113 -0
  165. package/docs/pt/{runtime-observability.md → 5-referencia/runtime-observability.md} +76 -76
  166. package/docs/pt/{sandbox.md → 5-referencia/sandbox.md} +125 -125
  167. package/docs/pt/{sdd-automation-scripts.md → 5-referencia/sdd-automation-scripts.md} +559 -557
  168. package/docs/pt/5-referencia/sdd-framework.md +115 -0
  169. package/docs/pt/5-referencia/sdd-planos-e-estrutura.md +321 -0
  170. package/docs/pt/5-referencia/secure-by-default.md +117 -0
  171. package/docs/pt/{skills.md → 5-referencia/skills.md} +275 -267
  172. package/docs/pt/{spec-learnings-pipeline.md → 5-referencia/spec-learnings-pipeline.md} +265 -265
  173. package/docs/pt/{squad-dashboard.md → 5-referencia/squad-dashboard.md} +373 -373
  174. package/docs/pt/{web3.md → 5-referencia/web3.md} +797 -797
  175. package/docs/pt/README.md +111 -116
  176. package/docs/pt/_arquivo/README.md +130 -0
  177. package/docs/pt/{advisor-spec.md → _arquivo/advisor-spec.md} +343 -335
  178. package/docs/pt/{agentes-customizados.md → _arquivo/agentes-customizados.md} +678 -670
  179. package/docs/pt/{busca-de-contexto.md → _arquivo/busca-de-contexto.md} +136 -129
  180. package/docs/pt/{cache-de-contexto.md → _arquivo/cache-de-contexto.md} +163 -156
  181. package/docs/pt/{cenarios.md → _arquivo/cenarios.md} +1282 -1274
  182. package/docs/pt/{design-hybrid-forge.md → _arquivo/design-hybrid-forge.md} +365 -356
  183. package/docs/pt/{deyvin.md → _arquivo/deyvin.md} +123 -115
  184. package/docs/pt/{guia-engineer.md → _arquivo/guia-engineer.md} +234 -226
  185. package/docs/pt/{inicio-rapido.md → _arquivo/inicio-rapido.md} +261 -250
  186. package/docs/pt/{memoria-contexto.md → _arquivo/memoria-contexto.md} +262 -255
  187. package/docs/pt/{monitor-de-contexto.md → _arquivo/monitor-de-contexto.md} +165 -158
  188. package/docs/pt/{profiler-system.md → _arquivo/profiler-system.md} +222 -214
  189. package/docs/pt/{recuperacao-de-sessao.md → _arquivo/recuperacao-de-sessao.md} +134 -125
  190. package/docs/pt/{site-forge.md → _arquivo/site-forge.md} +318 -309
  191. package/docs/pt/{squad-genome.md → _arquivo/squad-genome.md} +793 -783
  192. package/docs/pt/active-learning-loop/README.md +117 -0
  193. package/docs/pt/active-learning-loop/ativo-learning-loop.md +117 -0
  194. package/docs/pt/active-learning-loop/comandos-cli.md +320 -0
  195. package/docs/pt/active-learning-loop/como-usar.md +313 -0
  196. package/docs/pt/active-learning-loop/diagramas.md +225 -0
  197. package/docs/pt/active-learning-loop/doctor-checks.md +151 -0
  198. package/docs/pt/active-learning-loop/troubleshooting.md +283 -0
  199. package/docs/pt/agentes.md +996 -672
  200. package/docs/pt/deyvin-subtask-scout/README.md +109 -0
  201. package/docs/pt/deyvin-subtask-scout/comandos-cli.md +248 -0
  202. package/docs/pt/deyvin-subtask-scout/como-usar.md +221 -0
  203. package/docs/pt/deyvin-subtask-scout/diagramas.md +124 -0
  204. package/docs/pt/deyvin-subtask-scout/sub-task-scout.md +113 -0
  205. package/docs/pt/deyvin-subtask-scout/troubleshooting.md +184 -0
  206. package/docs/pt/living-memory/README.md +81 -0
  207. package/docs/pt/living-memory/autonomy-contract.md +206 -0
  208. package/docs/pt/living-memory/diagramas.md +365 -0
  209. package/docs/pt/living-memory/memoria-viva.md +141 -0
  210. package/docs/pt/living-memory/notificacoes-info.md +142 -0
  211. package/docs/pt/living-memory/reflexao-in-harness.md +218 -0
  212. package/docs/pt/living-memory/troubleshooting.md +286 -0
  213. package/docs/testing/genome-2.0-manual-regression.md +23 -23
  214. package/docs/testing/genome-2.0-matrix.md +36 -36
  215. package/docs/testing/genome-2.0-rollout.md +184 -184
  216. package/package.json +51 -50
  217. package/src/a2a/client.js +165 -165
  218. package/src/a2a/server.js +223 -223
  219. package/src/agent-loader.js +280 -280
  220. package/src/agent-manifests.js +86 -0
  221. package/src/agents.js +92 -72
  222. package/src/autonomy-policy.js +163 -0
  223. package/src/backup-local.js +74 -74
  224. package/src/backup-provider.js +303 -303
  225. package/src/brain-query.js +171 -0
  226. package/src/cli.js +1450 -1099
  227. package/src/commands/agent-audit.js +397 -397
  228. package/src/commands/agent-export-skill.js +229 -229
  229. package/src/commands/agent-loader.js +85 -85
  230. package/src/commands/agents.js +273 -160
  231. package/src/commands/artifact-validate.js +218 -189
  232. package/src/commands/auth.js +298 -0
  233. package/src/commands/backup-local-cmd.js +25 -25
  234. package/src/commands/backup.js +533 -533
  235. package/src/commands/brain-query.js +44 -0
  236. package/src/commands/brief-gen.js +405 -405
  237. package/src/commands/brief-validate.js +65 -65
  238. package/src/commands/briefing.js +344 -0
  239. package/src/commands/classify.js +256 -256
  240. package/src/commands/cloud.js +1767 -1767
  241. package/src/commands/commit-prepare.js +610 -0
  242. package/src/commands/compress-agents.js +416 -0
  243. package/src/commands/config.js +90 -90
  244. package/src/commands/context-cache.js +90 -90
  245. package/src/commands/context-compact.js +49 -49
  246. package/src/commands/context-health.js +187 -175
  247. package/src/commands/context-load.js +219 -0
  248. package/src/commands/context-monitor.js +163 -163
  249. package/src/commands/context-pack.js +45 -45
  250. package/src/commands/context-search.js +66 -66
  251. package/src/commands/context-trim.js +183 -177
  252. package/src/commands/context-validate.js +91 -91
  253. package/src/commands/design-hybrid-options.js +385 -385
  254. package/src/commands/detect-test-runner.js +55 -55
  255. package/src/commands/dev-resume.js +32 -0
  256. package/src/commands/devlog-export-brains.js +27 -27
  257. package/src/commands/devlog-process.js +294 -292
  258. package/src/commands/devlog-watch.js +131 -131
  259. package/src/commands/doctor.js +123 -123
  260. package/src/commands/dossier-add-research.js +114 -0
  261. package/src/commands/dossier-audit.js +222 -0
  262. package/src/commands/dossier.js +423 -0
  263. package/src/commands/feature-archive.js +513 -0
  264. package/src/commands/feature-close.js +554 -165
  265. package/src/commands/gate-approve.js +198 -0
  266. package/src/commands/gate-check.js +247 -228
  267. package/src/commands/genome-doctor.js +489 -41
  268. package/src/commands/genome-migrate.js +49 -49
  269. package/src/commands/git-guard.js +170 -0
  270. package/src/commands/harness.js +307 -0
  271. package/src/commands/health.js +214 -214
  272. package/src/commands/hooks-emit.js +253 -253
  273. package/src/commands/hooks-install.js +347 -347
  274. package/src/commands/i18n-add.js +56 -56
  275. package/src/commands/implementation-plan.js +367 -340
  276. package/src/commands/info.js +41 -41
  277. package/src/commands/init.js +120 -116
  278. package/src/commands/install.js +162 -107
  279. package/src/commands/learning-auto-promote.js +197 -195
  280. package/src/commands/learning-evolve.js +364 -364
  281. package/src/commands/learning-export.js +103 -103
  282. package/src/commands/learning-rollback.js +164 -164
  283. package/src/commands/learning.js +134 -134
  284. package/src/commands/live.js +2101 -1641
  285. package/src/commands/locale-apply.js +54 -51
  286. package/src/commands/locale-diff.js +25 -126
  287. package/src/commands/mcp-doctor.js +407 -406
  288. package/src/commands/mcp-init.js +373 -379
  289. package/src/commands/memory-archive.js +193 -0
  290. package/src/commands/memory-reflect-commit.js +148 -0
  291. package/src/commands/memory-reflect-prepare.js +97 -0
  292. package/src/commands/memory-restore.js +177 -0
  293. package/src/commands/memory-search.js +135 -0
  294. package/src/commands/memory.js +299 -0
  295. package/src/commands/notify.js +68 -0
  296. package/src/commands/package-e2e.js +273 -273
  297. package/src/commands/parallel-assign.js +483 -403
  298. package/src/commands/parallel-doctor.js +850 -437
  299. package/src/commands/parallel-guard.js +241 -0
  300. package/src/commands/parallel-init.js +311 -249
  301. package/src/commands/parallel-merge.js +299 -0
  302. package/src/commands/parallel-status.js +434 -290
  303. package/src/commands/pattern-detect.js +33 -33
  304. package/src/commands/preflight-context.js +30 -30
  305. package/src/commands/preflight.js +267 -208
  306. package/src/commands/pulse-update.js +130 -130
  307. package/src/commands/qa-doctor.js +185 -185
  308. package/src/commands/qa-init.js +166 -161
  309. package/src/commands/qa-report.js +58 -58
  310. package/src/commands/qa-run.js +873 -873
  311. package/src/commands/qa-scan.js +337 -337
  312. package/src/commands/recovery.js +43 -43
  313. package/src/commands/revision.js +235 -0
  314. package/src/commands/runner-daemon.js +274 -274
  315. package/src/commands/runner-plan.js +70 -70
  316. package/src/commands/runner-queue-from-plan.js +166 -166
  317. package/src/commands/runner-queue.js +189 -189
  318. package/src/commands/runner-run.js +129 -129
  319. package/src/commands/runtime.js +2086 -2067
  320. package/src/commands/sandbox.js +37 -37
  321. package/src/commands/scaffold-complete.js +188 -0
  322. package/src/commands/scan-project.js +1371 -1371
  323. package/src/commands/scout-commit.js +163 -0
  324. package/src/commands/scout-prep.js +214 -0
  325. package/src/commands/scout-validate.js +112 -0
  326. package/src/commands/security-audit.js +275 -0
  327. package/src/commands/security-scan.js +376 -0
  328. package/src/commands/self-implement-loop.js +306 -256
  329. package/src/commands/session-guard.js +218 -218
  330. package/src/commands/setup-context.js +699 -698
  331. package/src/commands/setup.js +178 -178
  332. package/src/commands/sizing.js +165 -165
  333. package/src/commands/skill.js +670 -670
  334. package/src/commands/smoke.js +426 -426
  335. package/src/commands/spec-checkpoint.js +177 -177
  336. package/src/commands/spec-status.js +79 -79
  337. package/src/commands/spec-sync.js +190 -190
  338. package/src/commands/spec-tasks.js +288 -288
  339. package/src/commands/squad-agent-create.js +830 -788
  340. package/src/commands/squad-autorun.js +1220 -1220
  341. package/src/commands/squad-bus.js +217 -217
  342. package/src/commands/squad-card.js +149 -149
  343. package/src/commands/squad-daemon.js +343 -343
  344. package/src/commands/squad-dashboard.js +39 -39
  345. package/src/commands/squad-dependency-graph.js +164 -164
  346. package/src/commands/squad-deploy.js +64 -64
  347. package/src/commands/squad-doctor.js +460 -460
  348. package/src/commands/squad-export.js +77 -46
  349. package/src/commands/squad-investigate.js +314 -261
  350. package/src/commands/squad-learning.js +209 -209
  351. package/src/commands/squad-mcp.js +270 -270
  352. package/src/commands/squad-pipeline.js +343 -343
  353. package/src/commands/squad-plan.js +361 -329
  354. package/src/commands/squad-processes.js +56 -56
  355. package/src/commands/squad-recovery.js +42 -42
  356. package/src/commands/squad-repair-genomes.js +39 -39
  357. package/src/commands/squad-review.js +106 -106
  358. package/src/commands/squad-roi.js +291 -291
  359. package/src/commands/squad-scaffold.js +56 -55
  360. package/src/commands/squad-score.js +311 -250
  361. package/src/commands/squad-status.js +481 -460
  362. package/src/commands/squad-tool-register.js +157 -157
  363. package/src/commands/squad-validate.js +438 -347
  364. package/src/commands/squad-webhook.js +160 -160
  365. package/src/commands/squad-worker.js +191 -191
  366. package/src/commands/squad-worktrees.js +75 -75
  367. package/src/commands/state-save.js +122 -122
  368. package/src/commands/store-genome.js +667 -0
  369. package/src/commands/store-skill.js +247 -0
  370. package/src/commands/store-squad.js +431 -0
  371. package/src/commands/store-system.js +392 -0
  372. package/src/commands/sync-agents-preflight.js +176 -0
  373. package/src/commands/test-agents.js +199 -199
  374. package/src/commands/tool-capabilities.js +63 -0
  375. package/src/commands/tool-registry-cmd.js +232 -232
  376. package/src/commands/update.js +64 -64
  377. package/src/commands/verify-gate.js +612 -572
  378. package/src/commands/web-map.js +70 -70
  379. package/src/commands/web-scrape.js +71 -71
  380. package/src/commands/workflow-execute.js +730 -241
  381. package/src/commands/workflow-harden.js +231 -0
  382. package/src/commands/workflow-heal.js +136 -0
  383. package/src/commands/workflow-next.js +1279 -601
  384. package/src/commands/workflow-plan.js +108 -108
  385. package/src/commands/workflow-status.js +440 -250
  386. package/src/commands/workspace.js +144 -0
  387. package/src/constants.js +413 -417
  388. package/src/context-cache.js +159 -159
  389. package/src/context-memory.js +975 -837
  390. package/src/context-parse-reason.js +22 -22
  391. package/src/context-search.js +326 -326
  392. package/src/context-writer.js +197 -196
  393. package/src/context.js +247 -217
  394. package/src/delivery-runner.js +319 -319
  395. package/src/design-variation-catalog.js +503 -503
  396. package/src/detector.js +261 -261
  397. package/src/doctor.js +760 -289
  398. package/src/dossier/codemap-store.js +267 -0
  399. package/src/dossier/dossier-bootstrap.js +222 -0
  400. package/src/dossier/dossier-compact.js +159 -0
  401. package/src/dossier/lock.js +128 -0
  402. package/src/dossier/research-index-store.js +233 -0
  403. package/src/dossier/revision-store.js +313 -0
  404. package/src/dossier/schema.js +162 -0
  405. package/src/dossier/scout-section.js +127 -0
  406. package/src/dossier/store.js +406 -0
  407. package/src/execution-gateway.js +464 -461
  408. package/src/friction-scanner.js +202 -0
  409. package/src/genome-files.js +198 -198
  410. package/src/genome-format.js +442 -442
  411. package/src/genome-schema.js +238 -215
  412. package/src/genomes/bindings.js +281 -281
  413. package/src/genomes.js +500 -467
  414. package/src/handoff-contract.js +417 -0
  415. package/src/handoff-validator.js +45 -0
  416. package/src/harness/circuit-breaker.js +135 -0
  417. package/src/i18n/index.js +103 -103
  418. package/src/i18n/messages/en.js +1541 -1139
  419. package/src/i18n/messages/es.js +1325 -980
  420. package/src/i18n/messages/fr.js +1333 -987
  421. package/src/i18n/messages/pt-BR.js +1561 -1166
  422. package/src/i18n/scaffold.js +64 -64
  423. package/src/install-animation.js +260 -260
  424. package/src/install-profile.js +127 -143
  425. package/src/install-wizard.js +475 -475
  426. package/src/installer-config-merge.js +207 -0
  427. package/src/installer.js +449 -294
  428. package/src/learning-loop-archive.js +595 -0
  429. package/src/learning-loop-doctor.js +217 -0
  430. package/src/learning-loop-engine.js +254 -0
  431. package/src/learning-loop-fts5.js +132 -0
  432. package/src/learning-loop-migration.js +163 -0
  433. package/src/lib/dev-resume.js +140 -0
  434. package/src/lib/dossier-telemetry.js +36 -0
  435. package/src/lib/genomes/compat.js +206 -206
  436. package/src/lib/genomes/migrate.js +90 -90
  437. package/src/lib/git-commit-guard.js +751 -0
  438. package/src/lib/health-check.js +158 -158
  439. package/src/lib/hook-protocol.js +76 -76
  440. package/src/lib/llm-content-sanitizer.js +44 -0
  441. package/src/lib/security/artifact-reader.js +167 -0
  442. package/src/lib/security/exit-codes.js +51 -0
  443. package/src/lib/security/findings-writer.js +176 -0
  444. package/src/lib/security/runtime-events.js +77 -0
  445. package/src/lib/security/secrets-regex.js +115 -0
  446. package/src/lib/squads/genome-repair.js +49 -49
  447. package/src/lib/store/security-scan.js +175 -0
  448. package/src/lib/terminal-checkbox.js +135 -0
  449. package/src/lib/terminal-picker.js +447 -0
  450. package/src/lib/tmux-launcher.js +163 -0
  451. package/src/lib/tool-capabilities.js +102 -0
  452. package/src/lib/webhook-server.js +328 -328
  453. package/src/locales.js +88 -84
  454. package/src/mcp/apps/squad-dashboard/app.js +163 -163
  455. package/src/mcp/apps/squad-dashboard/index.html +261 -261
  456. package/src/mcp/apps/squad-dashboard/mcp-manifest.json +23 -23
  457. package/src/mcp/resources/squad-state.js +130 -130
  458. package/src/mcp-connectors/registry.js +602 -602
  459. package/src/memory-reflect-engine.js +359 -0
  460. package/src/notify-renderer.js +32 -0
  461. package/src/onboarding.js +305 -305
  462. package/src/parallel-workspace.js +756 -0
  463. package/src/parser.js +66 -59
  464. package/src/path-guard.js +47 -0
  465. package/src/permissions-generator.js +400 -0
  466. package/src/preflight-engine.js +654 -443
  467. package/src/prompt-tool.js +20 -20
  468. package/src/qa-html-report.js +472 -472
  469. package/src/recovery-context-session.js +154 -154
  470. package/src/runner/cascade.js +97 -97
  471. package/src/runner/cli-launcher.js +109 -109
  472. package/src/runner/plan-importer.js +63 -63
  473. package/src/runner/queue-store.js +159 -159
  474. package/src/runtime-store.js +2720 -2676
  475. package/src/sandbox.js +194 -177
  476. package/src/self-healing.js +142 -0
  477. package/src/session-handoff.js +295 -77
  478. package/src/squad/agent-teams-adapter.js +270 -264
  479. package/src/squad/brief-validator.js +350 -350
  480. package/src/squad/bus-bridge.js +140 -140
  481. package/src/squad/context-compactor.js +265 -265
  482. package/src/squad/cross-ai-synthesizer.js +250 -250
  483. package/src/squad/external-session.js +180 -180
  484. package/src/squad/hooks-generator.js +196 -196
  485. package/src/squad/inter-squad-events.js +175 -175
  486. package/src/squad/inter-squad.js +74 -74
  487. package/src/squad/intra-bus.js +345 -345
  488. package/src/squad/learning-extractor.js +213 -213
  489. package/src/squad/pattern-detector.js +365 -365
  490. package/src/squad/preflight-context.js +296 -296
  491. package/src/squad/recovery-context.js +372 -372
  492. package/src/squad/reflection.js +365 -365
  493. package/src/squad/squad-scaffold.js +341 -177
  494. package/src/squad/state-manager.js +310 -310
  495. package/src/squad/task-decomposer.js +652 -652
  496. package/src/squad/verify-gate.js +303 -303
  497. package/src/squad/worktree-manager.js +114 -114
  498. package/src/squad-daemon.js +490 -490
  499. package/src/squad-dashboard/api.js +223 -223
  500. package/src/squad-dashboard/attachment-handler.js +93 -93
  501. package/src/squad-dashboard/context-monitor.js +157 -157
  502. package/src/squad-dashboard/execution-logs.js +115 -115
  503. package/src/squad-dashboard/hunk-review.js +209 -209
  504. package/src/squad-dashboard/metrics.js +133 -133
  505. package/src/squad-dashboard/process-monitor.js +125 -125
  506. package/src/squad-dashboard/renderer.js +858 -858
  507. package/src/squad-dashboard/server.js +232 -232
  508. package/src/squad-dashboard/styles.js +525 -525
  509. package/src/squad-dashboard/token-tracker.js +99 -99
  510. package/src/squads/apply-genome.js +21 -21
  511. package/src/squads/genome-binding-service.js +154 -154
  512. package/src/sub-task-engine.js +415 -0
  513. package/src/sub-task-schemas.js +150 -0
  514. package/src/sub-task-state.js +152 -0
  515. package/src/sub-task-telemetry.js +69 -0
  516. package/src/test-briefing.js +226 -0
  517. package/src/tool-executor.js +94 -94
  518. package/src/updater.js +39 -39
  519. package/src/utils.js +49 -46
  520. package/src/version.js +50 -50
  521. package/src/web.js +284 -284
  522. package/src/worker-runner.js +541 -524
  523. package/src/workflow-gates.js +185 -0
  524. package/template/.aioson/advisors/.gitkeep +1 -1
  525. package/template/.aioson/agents/analyst.md +333 -372
  526. package/template/.aioson/agents/architect.md +325 -338
  527. package/template/.aioson/agents/briefing.md +264 -0
  528. package/template/.aioson/agents/committer.md +161 -0
  529. package/template/.aioson/agents/copywriter.md +937 -463
  530. package/template/.aioson/agents/design-hybrid-forge.md +141 -141
  531. package/template/.aioson/agents/dev.md +295 -779
  532. package/template/.aioson/agents/deyvin.md +198 -290
  533. package/template/.aioson/agents/discover.md +235 -0
  534. package/template/.aioson/agents/discovery-design-doc.md +56 -264
  535. package/template/.aioson/agents/genome.md +1904 -314
  536. package/template/.aioson/agents/manifests/analyst.manifest.json +26 -0
  537. package/template/.aioson/agents/manifests/architect.manifest.json +23 -0
  538. package/template/.aioson/agents/manifests/committer.manifest.json +23 -0
  539. package/template/.aioson/agents/manifests/dev.manifest.json +54 -0
  540. package/template/.aioson/agents/manifests/deyvin.manifest.json +41 -0
  541. package/template/.aioson/agents/manifests/orchestrator.manifest.json +30 -0
  542. package/template/.aioson/agents/manifests/pentester.manifest.json +39 -0
  543. package/template/.aioson/agents/manifests/pm.manifest.json +26 -0
  544. package/template/.aioson/agents/manifests/product.manifest.json +23 -0
  545. package/template/.aioson/agents/manifests/qa.manifest.json +41 -0
  546. package/template/.aioson/agents/manifests/setup.manifest.json +20 -0
  547. package/template/.aioson/agents/manifests/ux-ui.manifest.json +24 -0
  548. package/template/.aioson/agents/neo.md +341 -233
  549. package/template/.aioson/agents/orache.md +430 -434
  550. package/template/.aioson/agents/orchestrator.md +274 -364
  551. package/template/.aioson/agents/pair.md +5 -5
  552. package/template/.aioson/agents/pentester.md +289 -0
  553. package/template/.aioson/agents/pm.md +141 -194
  554. package/template/.aioson/agents/product.md +351 -518
  555. package/template/.aioson/agents/profiler-enricher.md +331 -280
  556. package/template/.aioson/agents/profiler-forge.md +212 -202
  557. package/template/.aioson/agents/profiler-researcher.md +282 -259
  558. package/template/.aioson/agents/qa.md +432 -688
  559. package/template/.aioson/agents/setup.md +423 -649
  560. package/template/.aioson/agents/sheldon.md +259 -829
  561. package/template/.aioson/agents/site-forge.md +281 -1753
  562. package/template/.aioson/agents/squad.md +160 -2027
  563. package/template/.aioson/agents/tester.md +536 -463
  564. package/template/.aioson/agents/ux-ui.md +195 -870
  565. package/template/.aioson/agents/validator.md +101 -0
  566. package/template/.aioson/brains/README.md +132 -128
  567. package/template/.aioson/brains/_archived/.gitkeep +0 -0
  568. package/template/.aioson/brains/_index.json +34 -16
  569. package/template/.aioson/brains/dev/patterns.brain.json +79 -0
  570. package/template/.aioson/brains/scripts/query.js +107 -103
  571. package/template/.aioson/brains/sheldon/architecture-decisions.brain.json +79 -0
  572. package/template/.aioson/brains/site-forge/visual-patterns.brain.json +205 -205
  573. package/template/.aioson/config/autonomy-protocol.json +125 -0
  574. package/template/.aioson/config/learning-loop.json +10 -0
  575. package/template/.aioson/config/scout-engine.json +1 -0
  576. package/template/.aioson/config.md +410 -382
  577. package/template/.aioson/constitution.md +36 -33
  578. package/template/.aioson/context/_archived/.gitkeep +0 -0
  579. package/template/.aioson/context/design-doc.md +136 -0
  580. package/template/.aioson/context/project-map.md +57 -0
  581. package/template/.aioson/context/project-pulse.md +34 -34
  582. package/template/.aioson/context/seeds/seed-example.md +27 -27
  583. package/template/.aioson/context/spec.md.template +54 -54
  584. package/template/.aioson/context/user-profile.md +42 -42
  585. package/template/.aioson/design-docs/code-reuse.md +48 -0
  586. package/template/.aioson/design-docs/componentization.md +47 -0
  587. package/template/.aioson/design-docs/file-size.md +52 -0
  588. package/template/.aioson/design-docs/folder-structure.md +51 -0
  589. package/template/.aioson/design-docs/naming.md +54 -0
  590. package/template/.aioson/docs/LAYERS.md +89 -79
  591. package/template/.aioson/docs/README.md +76 -76
  592. package/template/.aioson/docs/autonomy-protocol.md +80 -0
  593. package/template/.aioson/docs/briefing/briefing-craft.md +237 -0
  594. package/template/.aioson/docs/dev/execution-discipline.md +106 -0
  595. package/template/.aioson/docs/dev/stack-conventions.md +83 -0
  596. package/template/.aioson/docs/deyvin/continuity-recovery.md +57 -0
  597. package/template/.aioson/docs/deyvin/debugging-escalation.md +30 -0
  598. package/template/.aioson/docs/deyvin/pair-execution.md +44 -0
  599. package/template/.aioson/docs/deyvin/runtime-handoffs.md +36 -0
  600. package/template/.aioson/docs/example-external-api-context.md +72 -72
  601. package/template/.aioson/docs/pentester/app-playbooks.md +206 -0
  602. package/template/.aioson/docs/pentester/llm-supplychain.md +165 -0
  603. package/template/.aioson/docs/product/conversation-playbook.md +116 -0
  604. package/template/.aioson/docs/product/prd-contract.md +107 -0
  605. package/template/.aioson/docs/product/quality-lens.md +57 -0
  606. package/template/.aioson/docs/product/research-loop.md +65 -0
  607. package/template/.aioson/docs/sheldon/enrichment-paths.md +134 -0
  608. package/template/.aioson/docs/sheldon/harness-contract.md +118 -0
  609. package/template/.aioson/docs/sheldon/quality-lens.md +57 -0
  610. package/template/.aioson/docs/sheldon/research-loop.md +56 -0
  611. package/template/.aioson/docs/sheldon/web-intelligence.md +75 -0
  612. package/template/.aioson/docs/site-forge-build.md +195 -0
  613. package/template/.aioson/docs/site-forge-extraction.md +135 -0
  614. package/template/.aioson/docs/site-forge-qa.md +155 -0
  615. package/template/.aioson/docs/site-forge-recon.md +434 -0
  616. package/template/.aioson/docs/site-forge-transform.md +249 -0
  617. package/template/.aioson/docs/squad/content-output.md +91 -0
  618. package/template/.aioson/docs/squad/creation-flow.md +149 -0
  619. package/template/.aioson/docs/squad/domain-breadth.md +322 -0
  620. package/template/.aioson/docs/squad/domain-classification.md +117 -0
  621. package/template/.aioson/docs/squad/genome-bindings.md +47 -0
  622. package/template/.aioson/docs/squad/package-contract.md +260 -0
  623. package/template/.aioson/docs/squad/quality-lens.md +60 -0
  624. package/template/.aioson/docs/squad/research-loop.md +59 -0
  625. package/template/.aioson/docs/squad/session-operations.md +117 -0
  626. package/template/.aioson/docs/squad/workflow-quality.md +165 -0
  627. package/template/.aioson/docs/tester/coverage-quality.md +351 -0
  628. package/template/.aioson/docs/ux-ui/accessibility-audit.md +55 -0
  629. package/template/.aioson/docs/ux-ui/audit-mode.md +86 -0
  630. package/template/.aioson/docs/ux-ui/component-map.md +35 -0
  631. package/template/.aioson/docs/ux-ui/design-execution.md +111 -0
  632. package/template/.aioson/docs/ux-ui/design-gate.md +27 -0
  633. package/template/.aioson/docs/ux-ui/research-mode.md +39 -0
  634. package/template/.aioson/docs/ux-ui/site-delivery.md +156 -0
  635. package/template/.aioson/docs/ux-ui/token-contract.md +57 -0
  636. package/template/.aioson/genomes/INDEX.md +195 -0
  637. package/template/.aioson/genomes/copywriting/SKILL.md +137 -0
  638. package/template/.aioson/genomes/copywriting/manifest.json +140 -0
  639. package/template/.aioson/genomes/copywriting/references/application-notes.md +145 -0
  640. package/template/.aioson/genomes/copywriting/references/decision-weights.md +45 -0
  641. package/template/.aioson/genomes/copywriting/references/frameworks/5-act-narrative.md +184 -0
  642. package/template/.aioson/genomes/copywriting/references/frameworks/classical-formulas.md +164 -0
  643. package/template/.aioson/genomes/copywriting/references/frameworks/offer-stack.md +195 -0
  644. package/template/.aioson/genomes/copywriting/references/frameworks/one-belief.md +135 -0
  645. package/template/.aioson/genomes/copywriting/references/frameworks/pms-research.md +211 -0
  646. package/template/.aioson/genomes/copywriting/references/frameworks/two-paths-close.md +190 -0
  647. package/template/.aioson/genomes/copywriting/references/heuristics.md +114 -0
  648. package/template/.aioson/genomes/copywriting/references/meta-axioms.md +68 -0
  649. package/template/.aioson/genomes/copywriting/references/methodology.md +115 -0
  650. package/template/.aioson/genomes/copywriting-brunson/SKILL.md +133 -0
  651. package/template/.aioson/genomes/copywriting-brunson/manifest.json +152 -0
  652. package/template/.aioson/genomes/copywriting-brunson/references/application-notes.md +113 -0
  653. package/template/.aioson/genomes/copywriting-brunson/references/decision-weights.md +33 -0
  654. package/template/.aioson/genomes/copywriting-brunson/references/evidence-and-attribution.md +81 -0
  655. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/6-part-structure.md +136 -0
  656. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/origin-story.md +121 -0
  657. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/perfect-webinar-script.md +139 -0
  658. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/persuasive-storytelling-5-structures.md +164 -0
  659. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/value-stack.md +136 -0
  660. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/who-what-why-how.md +110 -0
  661. package/template/.aioson/genomes/copywriting-brunson/references/meta-axioms.md +36 -0
  662. package/template/.aioson/genomes/copywriting-brunson/references/methodology.md +112 -0
  663. package/template/.aioson/git-guard.json +12 -0
  664. package/template/.aioson/mcp/servers.md +23 -24
  665. package/template/.aioson/profiler-reports/.gitkeep +1 -1
  666. package/template/.aioson/rules/README.md +69 -69
  667. package/template/.aioson/rules/_archived/.gitkeep +0 -0
  668. package/template/.aioson/rules/agent-language-policy.md +93 -0
  669. package/template/.aioson/rules/aioson-context-boundary.md +63 -0
  670. package/template/.aioson/rules/canonical-path-contract.md +47 -0
  671. package/template/.aioson/rules/data-format-convention.md +74 -136
  672. package/template/.aioson/rules/disk-first-artifacts.md +44 -0
  673. package/template/.aioson/rules/example-monetary-values.md +30 -30
  674. package/template/.aioson/rules/output-brevity.md +44 -0
  675. package/template/.aioson/rules/prd-section-ownership.md +49 -0
  676. package/template/.aioson/rules/security-baseline.md +139 -0
  677. package/template/.aioson/rules/spec-level-ownership.md +61 -0
  678. package/template/.aioson/rules/squad/README.md +50 -50
  679. package/template/.aioson/rules/squad-driver-pattern.md +81 -0
  680. package/template/.aioson/schemas/content-blueprint.schema.json +30 -30
  681. package/template/.aioson/schemas/genome-meta.schema.json +150 -150
  682. package/template/.aioson/schemas/genome.schema.json +115 -115
  683. package/template/.aioson/schemas/readiness.schema.json +27 -27
  684. package/template/.aioson/schemas/squad-blueprint.schema.json +228 -204
  685. package/template/.aioson/schemas/squad-manifest.schema.json +874 -830
  686. package/template/.aioson/skills/design/aurora-command-ui/SKILL.md +243 -243
  687. package/template/.aioson/skills/design/aurora-command-ui/references/art-direction.md +293 -293
  688. package/template/.aioson/skills/design/aurora-command-ui/references/components.md +827 -827
  689. package/template/.aioson/skills/design/aurora-command-ui/references/dashboards.md +250 -250
  690. package/template/.aioson/skills/design/aurora-command-ui/references/design-tokens.md +585 -585
  691. package/template/.aioson/skills/design/aurora-command-ui/references/motion.md +365 -365
  692. package/template/.aioson/skills/design/aurora-command-ui/references/patterns.md +482 -482
  693. package/template/.aioson/skills/design/aurora-command-ui/references/websites.md +387 -387
  694. package/template/.aioson/skills/design/bold-editorial-ui/SKILL.md +205 -205
  695. package/template/.aioson/skills/design/bold-editorial-ui/references/art-direction.md +338 -338
  696. package/template/.aioson/skills/design/bold-editorial-ui/references/components.md +977 -977
  697. package/template/.aioson/skills/design/bold-editorial-ui/references/dashboards.md +218 -218
  698. package/template/.aioson/skills/design/bold-editorial-ui/references/design-tokens.md +326 -326
  699. package/template/.aioson/skills/design/bold-editorial-ui/references/motion.md +461 -461
  700. package/template/.aioson/skills/design/bold-editorial-ui/references/patterns.md +293 -293
  701. package/template/.aioson/skills/design/bold-editorial-ui/references/websites.md +352 -352
  702. package/template/.aioson/skills/design/clean-saas-ui/SKILL.md +210 -210
  703. package/template/.aioson/skills/design/clean-saas-ui/references/art-direction.md +319 -319
  704. package/template/.aioson/skills/design/clean-saas-ui/references/components.md +365 -365
  705. package/template/.aioson/skills/design/clean-saas-ui/references/dashboards.md +196 -196
  706. package/template/.aioson/skills/design/clean-saas-ui/references/design-tokens.md +244 -244
  707. package/template/.aioson/skills/design/clean-saas-ui/references/motion.md +235 -235
  708. package/template/.aioson/skills/design/clean-saas-ui/references/patterns.md +215 -215
  709. package/template/.aioson/skills/design/clean-saas-ui/references/websites.md +295 -295
  710. package/template/.aioson/skills/design/cognitive-core-ui/SKILL.md +203 -203
  711. package/template/.aioson/skills/design/cognitive-core-ui/references/art-direction.md +339 -339
  712. package/template/.aioson/skills/design/cognitive-core-ui/references/components.md +407 -407
  713. package/template/.aioson/skills/design/cognitive-core-ui/references/dashboards.md +272 -272
  714. package/template/.aioson/skills/design/cognitive-core-ui/references/design-tokens.md +524 -524
  715. package/template/.aioson/skills/design/cognitive-core-ui/references/motion.md +279 -279
  716. package/template/.aioson/skills/design/cognitive-core-ui/references/patterns.md +289 -289
  717. package/template/.aioson/skills/design/cognitive-core-ui/references/websites.md +437 -437
  718. package/template/.aioson/skills/design/glassmorphism-ui/SKILL.md +222 -222
  719. package/template/.aioson/skills/design/glassmorphism-ui/references/art-direction.md +159 -159
  720. package/template/.aioson/skills/design/glassmorphism-ui/references/components.md +498 -498
  721. package/template/.aioson/skills/design/glassmorphism-ui/references/dashboards.md +236 -236
  722. package/template/.aioson/skills/design/glassmorphism-ui/references/design-tokens.md +274 -274
  723. package/template/.aioson/skills/design/glassmorphism-ui/references/motion.md +355 -355
  724. package/template/.aioson/skills/design/glassmorphism-ui/references/patterns.md +198 -198
  725. package/template/.aioson/skills/design/glassmorphism-ui/references/websites.md +307 -307
  726. package/template/.aioson/skills/design/interface-design/SKILL.md +47 -47
  727. package/template/.aioson/skills/design/interface-design/references/components-and-states.md +105 -105
  728. package/template/.aioson/skills/design/interface-design/references/design-directions.md +101 -101
  729. package/template/.aioson/skills/design/interface-design/references/handoff-and-quality.md +71 -71
  730. package/template/.aioson/skills/design/interface-design/references/intent-and-domain.md +74 -74
  731. package/template/.aioson/skills/design/interface-design/references/tokens-and-depth.md +173 -173
  732. package/template/.aioson/skills/design/neo-brutalist-ui/SKILL.md +213 -213
  733. package/template/.aioson/skills/design/neo-brutalist-ui/references/art-direction.md +228 -228
  734. package/template/.aioson/skills/design/neo-brutalist-ui/references/components.md +855 -855
  735. package/template/.aioson/skills/design/neo-brutalist-ui/references/dashboards.md +334 -334
  736. package/template/.aioson/skills/design/neo-brutalist-ui/references/design-tokens.md +342 -342
  737. package/template/.aioson/skills/design/neo-brutalist-ui/references/motion.md +286 -286
  738. package/template/.aioson/skills/design/neo-brutalist-ui/references/patterns.md +458 -458
  739. package/template/.aioson/skills/design/neo-brutalist-ui/references/websites.md +723 -723
  740. package/template/.aioson/skills/design/premium-command-center-ui/SKILL.md +62 -62
  741. package/template/.aioson/skills/design/premium-command-center-ui/references/operations.md +74 -74
  742. package/template/.aioson/skills/design/premium-command-center-ui/references/patterns.md +116 -116
  743. package/template/.aioson/skills/design/premium-command-center-ui/references/validation.md +47 -47
  744. package/template/.aioson/skills/design/premium-command-center-ui/references/visual-system.md +215 -215
  745. package/template/.aioson/skills/design/pt.squarespace.com/.skill-meta.json +31 -31
  746. package/template/.aioson/skills/design/pt.squarespace.com/SKILL.md +66 -66
  747. package/template/.aioson/skills/design/pt.squarespace.com/references/components.md +368 -368
  748. package/template/.aioson/skills/design/pt.squarespace.com/references/design-tokens.md +150 -150
  749. package/template/.aioson/skills/design/pt.squarespace.com/references/motion.md +270 -270
  750. package/template/.aioson/skills/design/pt.squarespace.com/references/patterns.md +189 -189
  751. package/template/.aioson/skills/design/pt.squarespace.com/references/websites.md +165 -165
  752. package/template/.aioson/skills/design/warm-craft-ui/SKILL.md +209 -209
  753. package/template/.aioson/skills/design/warm-craft-ui/references/art-direction.md +324 -324
  754. package/template/.aioson/skills/design/warm-craft-ui/references/components.md +508 -508
  755. package/template/.aioson/skills/design/warm-craft-ui/references/dashboards.md +223 -223
  756. package/template/.aioson/skills/design/warm-craft-ui/references/design-tokens.md +374 -374
  757. package/template/.aioson/skills/design/warm-craft-ui/references/motion.md +356 -356
  758. package/template/.aioson/skills/design/warm-craft-ui/references/patterns.md +288 -288
  759. package/template/.aioson/skills/design/warm-craft-ui/references/websites.md +289 -289
  760. package/template/.aioson/skills/design-system/SKILL.md +92 -92
  761. package/template/.aioson/skills/design-system/components/SKILL.md +274 -274
  762. package/template/.aioson/skills/design-system/dashboards/SKILL.md +184 -184
  763. package/template/.aioson/skills/design-system/foundations/SKILL.md +250 -250
  764. package/template/.aioson/skills/design-system/motion/SKILL.md +197 -197
  765. package/template/.aioson/skills/design-system/patterns/SKILL.md +231 -231
  766. package/template/.aioson/skills/dynamic/README.md +30 -30
  767. package/template/.aioson/skills/dynamic/cardano-docs.md +16 -16
  768. package/template/.aioson/skills/dynamic/ethereum-docs.md +17 -17
  769. package/template/.aioson/skills/dynamic/flux-ui-docs.md +13 -13
  770. package/template/.aioson/skills/dynamic/laravel-docs.md +41 -41
  771. package/template/.aioson/skills/dynamic/npm-packages.md +16 -16
  772. package/template/.aioson/skills/dynamic/solana-docs.md +16 -16
  773. package/template/.aioson/skills/marketing/references/anti-patterns.md +254 -254
  774. package/template/.aioson/skills/marketing/references/cta-matrix.md +361 -0
  775. package/template/.aioson/skills/marketing/references/fascinations.md +192 -192
  776. package/template/.aioson/skills/marketing/references/five-acts.md +248 -248
  777. package/template/.aioson/skills/marketing/references/headline-matrix.md +358 -0
  778. package/template/.aioson/skills/marketing/references/market-intelligence.md +198 -198
  779. package/template/.aioson/skills/marketing/references/offer-structure.md +203 -203
  780. package/template/.aioson/skills/marketing/references/one-belief.md +149 -149
  781. package/template/.aioson/skills/marketing/references/patterns.md +218 -218
  782. package/template/.aioson/skills/marketing/references/platform-constraints.md +337 -0
  783. package/template/.aioson/skills/marketing/references/pms-research.md +193 -193
  784. package/template/.aioson/skills/marketing/vsl-craft.md +385 -385
  785. package/template/.aioson/skills/premium-visual-design/SKILL.md +83 -83
  786. package/template/.aioson/skills/premium-visual-design/components/agent-badge.md +92 -92
  787. package/template/.aioson/skills/premium-visual-design/components/dependency-node.md +102 -102
  788. package/template/.aioson/skills/premium-visual-design/components/mention-autocomplete.md +136 -136
  789. package/template/.aioson/skills/premium-visual-design/components/notification-center.md +136 -136
  790. package/template/.aioson/skills/premium-visual-design/components/review-action-bar.md +188 -188
  791. package/template/.aioson/skills/premium-visual-design/components/team-switcher.md +131 -131
  792. package/template/.aioson/skills/premium-visual-design/patterns/agent-message-thread.md +198 -198
  793. package/template/.aioson/skills/premium-visual-design/patterns/notification-panel.md +275 -275
  794. package/template/.aioson/skills/premium-visual-design/patterns/review-workflow-ui.md +234 -234
  795. package/template/.aioson/skills/premium-visual-design/patterns/task-dependency-graph.md +147 -147
  796. package/template/.aioson/skills/premium-visual-design/tokens/status-extended.md +142 -142
  797. package/template/.aioson/skills/process/aioson-spec-driven/SKILL.md +46 -46
  798. package/template/.aioson/skills/process/aioson-spec-driven/references/analyst.md +30 -30
  799. package/template/.aioson/skills/process/aioson-spec-driven/references/approval-gates.md +109 -109
  800. package/template/.aioson/skills/process/aioson-spec-driven/references/architect.md +23 -23
  801. package/template/.aioson/skills/process/aioson-spec-driven/references/artifact-map.md +44 -44
  802. package/template/.aioson/skills/process/aioson-spec-driven/references/classification-map.md +37 -37
  803. package/template/.aioson/skills/process/aioson-spec-driven/references/dev.md +47 -47
  804. package/template/.aioson/skills/process/aioson-spec-driven/references/deyvin.md +27 -27
  805. package/template/.aioson/skills/process/aioson-spec-driven/references/hardening-lane.md +49 -49
  806. package/template/.aioson/skills/process/aioson-spec-driven/references/maintenance-and-state.md +101 -101
  807. package/template/.aioson/skills/process/aioson-spec-driven/references/pm.md +30 -0
  808. package/template/.aioson/skills/process/aioson-spec-driven/references/product.md +25 -25
  809. package/template/.aioson/skills/process/aioson-spec-driven/references/qa.md +30 -30
  810. package/template/.aioson/skills/process/aioson-spec-driven/references/sheldon.md +25 -25
  811. package/template/.aioson/skills/process/aioson-spec-driven/references/ui-language.md +75 -75
  812. package/template/.aioson/skills/process/design-hybrid-forge/SKILL.md +147 -147
  813. package/template/.aioson/skills/process/design-hybrid-forge/references/crossover-protocol.md +221 -221
  814. package/template/.aioson/skills/process/design-hybrid-forge/references/naming-registry.md +88 -88
  815. package/template/.aioson/skills/process/design-hybrid-forge/references/output-contract.md +306 -306
  816. package/template/.aioson/skills/process/design-hybrid-forge/references/pair-compatibility.md +149 -149
  817. package/template/.aioson/skills/process/design-hybrid-forge/references/quality-gates.md +208 -208
  818. package/template/.aioson/skills/process/design-hybrid-forge/references/variation-library.md +125 -125
  819. package/template/.aioson/skills/process/secure-tdd/SKILL.md +97 -0
  820. package/template/.aioson/skills/process/simplify/SKILL.md +173 -173
  821. package/template/.aioson/skills/references/premium-command-center-ui/master-application-prompt.md +79 -79
  822. package/template/.aioson/skills/references/premium-command-center-ui/operational-ux-playbook.md +253 -253
  823. package/template/.aioson/skills/references/premium-command-center-ui/quality-validation-checklist.md +82 -82
  824. package/template/.aioson/skills/references/premium-command-center-ui/visual-system-and-component-patterns.md +270 -270
  825. package/template/.aioson/skills/squad/SKILL.md +58 -58
  826. package/template/.aioson/skills/squad/formats/catalog.json +15 -15
  827. package/template/.aioson/skills/squad/formats/content/blog-post.md +47 -47
  828. package/template/.aioson/skills/squad/formats/content/newsletter.md +47 -47
  829. package/template/.aioson/skills/squad/formats/creative/podcast-script.md +43 -43
  830. package/template/.aioson/skills/squad/formats/creative/video-script.md +41 -41
  831. package/template/.aioson/skills/squad/formats/social/instagram-feed.md +42 -42
  832. package/template/.aioson/skills/squad/formats/social/linkedin-post.md +42 -42
  833. package/template/.aioson/skills/squad/formats/social/tiktok.md +39 -39
  834. package/template/.aioson/skills/squad/formats/social/twitter-thread.md +39 -39
  835. package/template/.aioson/skills/squad/formats/social/youtube-long.md +47 -47
  836. package/template/.aioson/skills/squad/formats/social/youtube-shorts.md +39 -39
  837. package/template/.aioson/skills/squad/patterns/multi-platform-pattern.md +108 -108
  838. package/template/.aioson/skills/squad/patterns/persona-based-pattern.md +98 -98
  839. package/template/.aioson/skills/squad/patterns/pipeline-pattern.md +106 -106
  840. package/template/.aioson/skills/squad/patterns/review-loop-pattern.md +81 -81
  841. package/template/.aioson/skills/squad/references/checklist-templates.md +122 -122
  842. package/template/.aioson/skills/squad/references/executor-archetypes.md +123 -123
  843. package/template/.aioson/skills/squad/references/workflow-templates.md +169 -169
  844. package/template/.aioson/skills/static/context-budget-guide.md +46 -46
  845. package/template/.aioson/skills/static/debugging-protocol.md +42 -42
  846. package/template/.aioson/skills/static/django-patterns.md +342 -342
  847. package/template/.aioson/skills/static/fastapi-patterns.md +344 -344
  848. package/template/.aioson/skills/static/filament-patterns.md +267 -267
  849. package/template/.aioson/skills/static/flux-ui-components.md +262 -262
  850. package/template/.aioson/skills/static/git-conventions.md +227 -227
  851. package/template/.aioson/skills/static/git-worktrees.md +36 -36
  852. package/template/.aioson/skills/static/harness-sensors.md +74 -74
  853. package/template/.aioson/skills/static/harness-validate/SKILL.md +46 -0
  854. package/template/.aioson/skills/static/jetstream-setup.md +200 -200
  855. package/template/.aioson/skills/static/landing-page-deploy.md +192 -192
  856. package/template/.aioson/skills/static/landing-page-forge.md +730 -730
  857. package/template/.aioson/skills/static/laravel-conventions.md +491 -491
  858. package/template/.aioson/skills/static/multi-agent-patterns.md +43 -43
  859. package/template/.aioson/skills/static/nextjs-patterns.md +321 -321
  860. package/template/.aioson/skills/static/node-express-patterns.md +317 -317
  861. package/template/.aioson/skills/static/node-typescript-patterns.md +282 -282
  862. package/template/.aioson/skills/static/rails-conventions.md +307 -307
  863. package/template/.aioson/skills/static/react-motion-patterns.md +599 -599
  864. package/template/.aioson/skills/static/static-html-patterns/checklists.md +43 -43
  865. package/template/.aioson/skills/static/static-html-patterns/css-tokens.md +609 -609
  866. package/template/.aioson/skills/static/static-html-patterns/motion.md +193 -193
  867. package/template/.aioson/skills/static/static-html-patterns/premium.md +711 -711
  868. package/template/.aioson/skills/static/static-html-patterns/structure.md +209 -209
  869. package/template/.aioson/skills/static/static-html-patterns/utilities.md +190 -190
  870. package/template/.aioson/skills/static/static-html-patterns.md +80 -80
  871. package/template/.aioson/skills/static/tall-stack-patterns.md +286 -286
  872. package/template/.aioson/skills/static/threejs-patterns.md +929 -929
  873. package/template/.aioson/skills/static/ui-ux-modern.md +76 -76
  874. package/template/.aioson/skills/static/web-research-cache.md +115 -112
  875. package/template/.aioson/skills/static/web3-cardano-patterns.md +337 -337
  876. package/template/.aioson/skills/static/web3-ethereum-patterns.md +310 -310
  877. package/template/.aioson/skills/static/web3-security-checklist.md +284 -284
  878. package/template/.aioson/skills/static/web3-solana-patterns.md +324 -324
  879. package/template/.aioson/squads/memory.md +5 -5
  880. package/template/.aioson/tasks/implementation-plan.md +327 -327
  881. package/template/.aioson/tasks/squad-analyze.md +83 -83
  882. package/template/.aioson/tasks/squad-create.md +148 -121
  883. package/template/.aioson/tasks/squad-design.md +206 -158
  884. package/template/.aioson/tasks/squad-execution-plan.md +279 -279
  885. package/template/.aioson/tasks/squad-export.md +20 -20
  886. package/template/.aioson/tasks/squad-extend.md +68 -68
  887. package/template/.aioson/tasks/squad-investigate.md +57 -44
  888. package/template/.aioson/tasks/squad-learning-review.md +44 -44
  889. package/template/.aioson/tasks/squad-output-config.md +177 -177
  890. package/template/.aioson/tasks/squad-pipeline.md +122 -122
  891. package/template/.aioson/tasks/squad-profile.md +48 -48
  892. package/template/.aioson/tasks/squad-refresh.md +236 -0
  893. package/template/.aioson/tasks/squad-repair.md +85 -85
  894. package/template/.aioson/tasks/squad-review.md +61 -61
  895. package/template/.aioson/tasks/squad-task-decompose.md +66 -66
  896. package/template/.aioson/tasks/squad-validate.md +58 -58
  897. package/template/.aioson/templates/reflect-prompts/current-state.md +36 -0
  898. package/template/.aioson/templates/reflect-prompts/how-it-works.md +23 -0
  899. package/template/.aioson/templates/reflect-prompts/what-it-does.md +21 -0
  900. package/template/.aioson/templates/squads/content-basic/template.json +21 -21
  901. package/template/.aioson/templates/squads/digital-marketing-agency/template.json +96 -96
  902. package/template/.aioson/templates/squads/media-channel/template.json +24 -24
  903. package/template/.aioson/templates/squads/research-analysis/template.json +22 -22
  904. package/template/.aioson/templates/squads/software-delivery/template.json +21 -21
  905. package/template/.claude/commands/aioson/agent/analyst.md +5 -5
  906. package/template/.claude/commands/aioson/agent/architect.md +5 -5
  907. package/template/.claude/commands/aioson/agent/briefing.md +5 -0
  908. package/template/.claude/commands/aioson/agent/committer.md +5 -0
  909. package/template/.claude/commands/aioson/agent/copywriter.md +5 -0
  910. package/template/.claude/commands/aioson/agent/design-hybrid-forge.md +5 -5
  911. package/template/.claude/commands/aioson/agent/dev.md +5 -5
  912. package/template/.claude/commands/aioson/agent/deyvin.md +5 -5
  913. package/template/.claude/commands/aioson/agent/discover.md +5 -0
  914. package/template/.claude/commands/aioson/agent/discovery-design-doc.md +5 -5
  915. package/template/.claude/commands/aioson/agent/genome.md +5 -5
  916. package/template/.claude/commands/aioson/agent/neo.md +5 -5
  917. package/template/.claude/commands/aioson/agent/orache.md +5 -5
  918. package/template/.claude/commands/aioson/agent/orchestrator.md +5 -5
  919. package/template/.claude/commands/aioson/agent/pair.md +5 -0
  920. package/template/.claude/commands/aioson/agent/pentester.md +5 -0
  921. package/template/.claude/commands/aioson/agent/pm.md +5 -5
  922. package/template/.claude/commands/aioson/agent/product.md +5 -5
  923. package/template/.claude/commands/aioson/agent/profiler-enricher.md +5 -5
  924. package/template/.claude/commands/aioson/agent/profiler-forge.md +5 -5
  925. package/template/.claude/commands/aioson/agent/profiler-researcher.md +5 -5
  926. package/template/.claude/commands/aioson/agent/qa.md +5 -5
  927. package/template/.claude/commands/aioson/agent/setup.md +5 -5
  928. package/template/.claude/commands/aioson/agent/sheldon.md +5 -5
  929. package/template/.claude/commands/aioson/agent/site-forge.md +5 -5
  930. package/template/.claude/commands/aioson/agent/squad.md +5 -5
  931. package/template/.claude/commands/aioson/agent/tester.md +5 -5
  932. package/template/.claude/commands/aioson/agent/ux-ui.md +5 -5
  933. package/template/.claude/commands/aioson/agent/validator.md +5 -0
  934. package/template/.gemini/GEMINI.md +13 -13
  935. package/template/.gemini/commands/aios-analyst.toml +7 -4
  936. package/template/.gemini/commands/aios-architect.toml +8 -7
  937. package/template/.gemini/commands/aios-committer.toml +7 -0
  938. package/template/.gemini/commands/aios-copywriter.toml +7 -0
  939. package/template/.gemini/commands/aios-cypher.toml +7 -0
  940. package/template/.gemini/commands/aios-dev.toml +9 -8
  941. package/template/.gemini/commands/aios-deyvin.toml +7 -6
  942. package/template/.gemini/commands/aios-discover.toml +6 -0
  943. package/template/.gemini/commands/aios-discovery-design-doc.toml +7 -4
  944. package/template/.gemini/commands/aios-genome.toml +7 -0
  945. package/template/.gemini/commands/aios-neo.toml +6 -4
  946. package/template/.gemini/commands/aios-orache.toml +7 -0
  947. package/template/.gemini/commands/aios-orchestrator.toml +9 -8
  948. package/template/.gemini/commands/aios-pair.toml +7 -6
  949. package/template/.gemini/commands/aios-pm.toml +9 -8
  950. package/template/.gemini/commands/aios-product.toml +6 -4
  951. package/template/.gemini/commands/aios-qa.toml +7 -6
  952. package/template/.gemini/commands/aios-setup.toml +6 -3
  953. package/template/.gemini/commands/aios-sheldon.toml +7 -0
  954. package/template/.gemini/commands/aios-site-forge.toml +7 -0
  955. package/template/.gemini/commands/aios-squad.toml +7 -0
  956. package/template/.gemini/commands/aios-tester.toml +7 -6
  957. package/template/.gemini/commands/aios-ux-ui.toml +9 -8
  958. package/template/.gemini/commands/aios-validator.toml +7 -0
  959. package/template/AGENTS.md +184 -172
  960. package/template/CLAUDE.md +98 -93
  961. package/template/OPENCODE.md +35 -34
  962. package/template/aioson-models.json +40 -40
  963. package/docs/en/i18n.md +0 -52
  964. package/docs/en/schemas/parallel-status.schema.json +0 -94
  965. package/template/.aioson/genomes/copywriting.md +0 -204
  966. package/template/.aioson/locales/en/agents/analyst.md +0 -244
  967. package/template/.aioson/locales/en/agents/architect.md +0 -245
  968. package/template/.aioson/locales/en/agents/dev.md +0 -397
  969. package/template/.aioson/locales/en/agents/deyvin.md +0 -137
  970. package/template/.aioson/locales/en/agents/discovery-design-doc.md +0 -27
  971. package/template/.aioson/locales/en/agents/genome.md +0 -212
  972. package/template/.aioson/locales/en/agents/neo.md +0 -8
  973. package/template/.aioson/locales/en/agents/orache.md +0 -6
  974. package/template/.aioson/locales/en/agents/orchestrator.md +0 -189
  975. package/template/.aioson/locales/en/agents/pair.md +0 -5
  976. package/template/.aioson/locales/en/agents/pm.md +0 -84
  977. package/template/.aioson/locales/en/agents/product.md +0 -378
  978. package/template/.aioson/locales/en/agents/profiler-enricher.md +0 -5
  979. package/template/.aioson/locales/en/agents/profiler-forge.md +0 -5
  980. package/template/.aioson/locales/en/agents/profiler-researcher.md +0 -5
  981. package/template/.aioson/locales/en/agents/qa.md +0 -270
  982. package/template/.aioson/locales/en/agents/setup.md +0 -421
  983. package/template/.aioson/locales/en/agents/sheldon.md +0 -455
  984. package/template/.aioson/locales/en/agents/squad.md +0 -449
  985. package/template/.aioson/locales/en/agents/tester.md +0 -6
  986. package/template/.aioson/locales/en/agents/ux-ui.md +0 -668
  987. package/template/.aioson/locales/es/agents/analyst.md +0 -225
  988. package/template/.aioson/locales/es/agents/architect.md +0 -245
  989. package/template/.aioson/locales/es/agents/dev.md +0 -370
  990. package/template/.aioson/locales/es/agents/deyvin.md +0 -99
  991. package/template/.aioson/locales/es/agents/discovery-design-doc.md +0 -21
  992. package/template/.aioson/locales/es/agents/genome.md +0 -104
  993. package/template/.aioson/locales/es/agents/neo.md +0 -50
  994. package/template/.aioson/locales/es/agents/orache.md +0 -105
  995. package/template/.aioson/locales/es/agents/orchestrator.md +0 -194
  996. package/template/.aioson/locales/es/agents/pair.md +0 -7
  997. package/template/.aioson/locales/es/agents/pm.md +0 -90
  998. package/template/.aioson/locales/es/agents/product.md +0 -372
  999. package/template/.aioson/locales/es/agents/profiler-enricher.md +0 -7
  1000. package/template/.aioson/locales/es/agents/profiler-forge.md +0 -7
  1001. package/template/.aioson/locales/es/agents/profiler-researcher.md +0 -7
  1002. package/template/.aioson/locales/es/agents/qa.md +0 -198
  1003. package/template/.aioson/locales/es/agents/setup.md +0 -405
  1004. package/template/.aioson/locales/es/agents/sheldon.md +0 -309
  1005. package/template/.aioson/locales/es/agents/squad.md +0 -532
  1006. package/template/.aioson/locales/es/agents/tester.md +0 -9
  1007. package/template/.aioson/locales/es/agents/ux-ui.md +0 -212
  1008. package/template/.aioson/locales/fr/agents/analyst.md +0 -225
  1009. package/template/.aioson/locales/fr/agents/architect.md +0 -245
  1010. package/template/.aioson/locales/fr/agents/dev.md +0 -370
  1011. package/template/.aioson/locales/fr/agents/deyvin.md +0 -99
  1012. package/template/.aioson/locales/fr/agents/discovery-design-doc.md +0 -21
  1013. package/template/.aioson/locales/fr/agents/genome.md +0 -104
  1014. package/template/.aioson/locales/fr/agents/neo.md +0 -50
  1015. package/template/.aioson/locales/fr/agents/orache.md +0 -106
  1016. package/template/.aioson/locales/fr/agents/orchestrator.md +0 -194
  1017. package/template/.aioson/locales/fr/agents/pair.md +0 -7
  1018. package/template/.aioson/locales/fr/agents/pm.md +0 -90
  1019. package/template/.aioson/locales/fr/agents/product.md +0 -372
  1020. package/template/.aioson/locales/fr/agents/profiler-enricher.md +0 -7
  1021. package/template/.aioson/locales/fr/agents/profiler-forge.md +0 -7
  1022. package/template/.aioson/locales/fr/agents/profiler-researcher.md +0 -7
  1023. package/template/.aioson/locales/fr/agents/qa.md +0 -198
  1024. package/template/.aioson/locales/fr/agents/setup.md +0 -405
  1025. package/template/.aioson/locales/fr/agents/sheldon.md +0 -309
  1026. package/template/.aioson/locales/fr/agents/squad.md +0 -532
  1027. package/template/.aioson/locales/fr/agents/tester.md +0 -9
  1028. package/template/.aioson/locales/fr/agents/ux-ui.md +0 -212
  1029. package/template/.aioson/locales/pt-BR/agents/analyst.md +0 -319
  1030. package/template/.aioson/locales/pt-BR/agents/architect.md +0 -284
  1031. package/template/.aioson/locales/pt-BR/agents/dev.md +0 -483
  1032. package/template/.aioson/locales/pt-BR/agents/deyvin.md +0 -184
  1033. package/template/.aioson/locales/pt-BR/agents/discovery-design-doc.md +0 -198
  1034. package/template/.aioson/locales/pt-BR/agents/genome.md +0 -297
  1035. package/template/.aioson/locales/pt-BR/agents/neo.md +0 -208
  1036. package/template/.aioson/locales/pt-BR/agents/orache.md +0 -137
  1037. package/template/.aioson/locales/pt-BR/agents/orchestrator.md +0 -324
  1038. package/template/.aioson/locales/pt-BR/agents/pair.md +0 -5
  1039. package/template/.aioson/locales/pt-BR/agents/pm.md +0 -182
  1040. package/template/.aioson/locales/pt-BR/agents/product.md +0 -466
  1041. package/template/.aioson/locales/pt-BR/agents/profiler-enricher.md +0 -5
  1042. package/template/.aioson/locales/pt-BR/agents/profiler-forge.md +0 -5
  1043. package/template/.aioson/locales/pt-BR/agents/profiler-researcher.md +0 -5
  1044. package/template/.aioson/locales/pt-BR/agents/qa.md +0 -300
  1045. package/template/.aioson/locales/pt-BR/agents/setup.md +0 -533
  1046. package/template/.aioson/locales/pt-BR/agents/sheldon.md +0 -323
  1047. package/template/.aioson/locales/pt-BR/agents/squad.md +0 -1330
  1048. package/template/.aioson/locales/pt-BR/agents/tester.md +0 -449
  1049. package/template/.aioson/locales/pt-BR/agents/ux-ui.md +0 -669
@@ -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 };