voidforge-build 23.9.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (772) hide show
  1. package/dist/.claude/agents/adolin-brand.md +40 -0
  2. package/dist/.claude/agents/ahsoka-access-control.md +56 -0
  3. package/dist/.claude/agents/alfred-dependencies.md +43 -0
  4. package/dist/.claude/agents/alia-threat-detect.md +40 -0
  5. package/dist/.claude/agents/anakin-dark-side.md +41 -0
  6. package/dist/.claude/agents/aquaman-deep-dive.md +43 -0
  7. package/dist/.claude/agents/aragorn-orchestration.md +39 -0
  8. package/dist/.claude/agents/archer-greenfield.md +48 -0
  9. package/dist/.claude/agents/armin-clever.md +39 -0
  10. package/dist/.claude/agents/arwen-ui-polish.md +42 -0
  11. package/dist/.claude/agents/ashitaka-tech-debt.md +39 -0
  12. package/dist/.claude/agents/asuka-performance.md +39 -0
  13. package/dist/.claude/agents/bail-organa-governance.md +37 -0
  14. package/dist/.claude/agents/banner-database.md +44 -0
  15. package/dist/.claude/agents/barton-smoke-test.md +59 -0
  16. package/dist/.claude/agents/bashir-field-medic.md +63 -0
  17. package/dist/.claude/agents/batgirl-detail.md +43 -0
  18. package/dist/.claude/agents/batman-qa.md +73 -0
  19. package/dist/.claude/agents/bayta-evals.md +41 -0
  20. package/dist/.claude/agents/beast-boy-cross-env.md +43 -0
  21. package/dist/.claude/agents/beerus-destroyer.md +39 -0
  22. package/dist/.claude/agents/bel-riose-orchestration.md +40 -0
  23. package/dist/.claude/agents/beru-subprocess.md +37 -0
  24. package/dist/.claude/agents/bilbo-microcopy.md +43 -0
  25. package/dist/.claude/agents/black-canary-monitoring.md +43 -0
  26. package/dist/.claude/agents/bliss-ai-safety.md +40 -0
  27. package/dist/.claude/agents/bo-katan-perimeter.md +40 -0
  28. package/dist/.claude/agents/bombadil-forge-sync.md +62 -0
  29. package/dist/.claude/agents/boromir-hubris.md +40 -0
  30. package/dist/.claude/agents/breeze-platform-relations.md +40 -0
  31. package/dist/.claude/agents/bucky-legacy.md +43 -0
  32. package/dist/.claude/agents/bulma-engineering.md +40 -0
  33. package/dist/.claude/agents/calcifer-daemon.md +39 -0
  34. package/dist/.claude/agents/cara-dune-enforcement.md +37 -0
  35. package/dist/.claude/agents/cassian-recon.md +37 -0
  36. package/dist/.claude/agents/cc-persistent-process.md +39 -0
  37. package/dist/.claude/agents/celeborn-design-system.md +40 -0
  38. package/dist/.claude/agents/celebrimbor-forge-artist.md +62 -0
  39. package/dist/.claude/agents/chakotay-bridge.md +47 -0
  40. package/dist/.claude/agents/chani-worm-rider.md +61 -0
  41. package/dist/.claude/agents/chewie-dependency-audit.md +41 -0
  42. package/dist/.claude/agents/chrome-discovery.md +37 -0
  43. package/dist/.claude/agents/constantine-cursed-code.md +59 -0
  44. package/dist/.claude/agents/coulson-release.md +64 -0
  45. package/dist/.claude/agents/crusher-diagnostics.md +48 -0
  46. package/dist/.claude/agents/cyborg-system-integration.md +43 -0
  47. package/dist/.claude/agents/dalinar-positioning.md +40 -0
  48. package/dist/.claude/agents/daneel-model-migration.md +40 -0
  49. package/dist/.claude/agents/data-tech-debt.md +48 -0
  50. package/dist/.claude/agents/dax-legacy-wisdom.md +48 -0
  51. package/dist/.claude/agents/deathstroke-adversarial.md +59 -0
  52. package/dist/.claude/agents/denji-determination.md +39 -0
  53. package/dist/.claude/agents/din-djarin-bounty.md +42 -0
  54. package/dist/.claude/agents/dockson-treasury.md +67 -0
  55. package/dist/.claude/agents/dori-integration-check.md +37 -0
  56. package/dist/.claude/agents/dors-observability.md +40 -0
  57. package/dist/.claude/agents/drax-exact-match.md +42 -0
  58. package/dist/.claude/agents/ducem-token-economics.md +41 -0
  59. package/dist/.claude/agents/duncan-relay.md +40 -0
  60. package/dist/.claude/agents/duo-teardown.md +38 -0
  61. package/dist/.claude/agents/ed-network-scan.md +38 -0
  62. package/dist/.claude/agents/elrond-ux-strategy.md +39 -0
  63. package/dist/.claude/agents/eowyn-delight.md +56 -0
  64. package/dist/.claude/agents/erwin-strategy.md +39 -0
  65. package/dist/.claude/agents/ezra-catches-missed.md +40 -0
  66. package/dist/.claude/agents/ezri-session-analyst.md +45 -0
  67. package/dist/.claude/agents/falcon-migration.md +43 -0
  68. package/dist/.claude/agents/faramir-judgment.md +40 -0
  69. package/dist/.claude/agents/faye-resourceful.md +39 -0
  70. package/dist/.claude/agents/fenring-passive-monitor.md +37 -0
  71. package/dist/.claude/agents/fern-protocol.md +37 -0
  72. package/dist/.claude/agents/feyd-adversarial.md +41 -0
  73. package/dist/.claude/agents/flash-rapid-test.md +43 -0
  74. package/dist/.claude/agents/friday-automation.md +42 -0
  75. package/dist/.claude/agents/frieren-long-term.md +39 -0
  76. package/dist/.claude/agents/frodo-critical-path.md +40 -0
  77. package/dist/.claude/agents/fury-initiative.md +65 -0
  78. package/dist/.claude/agents/gaal-prompt-arch.md +41 -0
  79. package/dist/.claude/agents/galadriel-frontend.md +69 -0
  80. package/dist/.claude/agents/gamora-perf-assassin.md +43 -0
  81. package/dist/.claude/agents/gandalf-setup-wizard.md +63 -0
  82. package/dist/.claude/agents/gen-docs.md +37 -0
  83. package/dist/.claude/agents/ghanima-paired-monitor.md +37 -0
  84. package/dist/.claude/agents/gimli-performance.md +41 -0
  85. package/dist/.claude/agents/giyu-silent-guard.md +39 -0
  86. package/dist/.claude/agents/glorfindel-rendering.md +40 -0
  87. package/dist/.claude/agents/gohan-hidden-power.md +39 -0
  88. package/dist/.claude/agents/gojo-infinite-scale.md +39 -0
  89. package/dist/.claude/agents/goku-scaling.md +39 -0
  90. package/dist/.claude/agents/goldberry-change-detect.md +37 -0
  91. package/dist/.claude/agents/gordon-escalation.md +42 -0
  92. package/dist/.claude/agents/green-arrow-precision.md +43 -0
  93. package/dist/.claude/agents/green-lantern-scenarios.md +43 -0
  94. package/dist/.claude/agents/grogu-tiny-vulns.md +38 -0
  95. package/dist/.claude/agents/groot-caching.md +43 -0
  96. package/dist/.claude/agents/gurney-delivery.md +40 -0
  97. package/dist/.claude/agents/haku-deploy-wizard.md +65 -0
  98. package/dist/.claude/agents/haldir-boundaries.md +40 -0
  99. package/dist/.claude/agents/han-vuln-hunter.md +40 -0
  100. package/dist/.claude/agents/hange-experimentation.md +39 -0
  101. package/dist/.claude/agents/harah-protocol.md +38 -0
  102. package/dist/.claude/agents/hawkgirl-regression-sweep.md +43 -0
  103. package/dist/.claude/agents/heero-mission-deploy.md +39 -0
  104. package/dist/.claude/agents/hera-navigation.md +41 -0
  105. package/dist/.claude/agents/hill-mission-control.md +43 -0
  106. package/dist/.claude/agents/himmel-legacy.md +37 -0
  107. package/dist/.claude/agents/hober-tool-schema.md +40 -0
  108. package/dist/.claude/agents/hoid-copywriting.md +40 -0
  109. package/dist/.claude/agents/howl-migration.md +39 -0
  110. package/dist/.claude/agents/hughes-observability.md +37 -0
  111. package/dist/.claude/agents/huntress-flaky-bugs.md +42 -0
  112. package/dist/.claude/agents/irulan-historian.md +37 -0
  113. package/dist/.claude/agents/jake-reporter.md +45 -0
  114. package/dist/.claude/agents/janeway-novel-arch.md +48 -0
  115. package/dist/.claude/agents/janov-context-eng.md +40 -0
  116. package/dist/.claude/agents/jarvis-status.md +42 -0
  117. package/dist/.claude/agents/jean-pragmatic.md +39 -0
  118. package/dist/.claude/agents/jessica-voice.md +40 -0
  119. package/dist/.claude/agents/jet-maintenance.md +39 -0
  120. package/dist/.claude/agents/jin-disciplined-adv.md +39 -0
  121. package/dist/.claude/agents/kaji-intelligence.md +39 -0
  122. package/dist/.claude/agents/kaladin-organic-growth.md +40 -0
  123. package/dist/.claude/agents/kallen-hard-deploy.md +39 -0
  124. package/dist/.claude/agents/kanan-intuitive.md +41 -0
  125. package/dist/.claude/agents/kaoru-harmony.md +37 -0
  126. package/dist/.claude/agents/kaworu-solver.md +39 -0
  127. package/dist/.claude/agents/kelsier-growth.md +64 -0
  128. package/dist/.claude/agents/kenobi-security.md +70 -0
  129. package/dist/.claude/agents/kim-api-design.md +49 -0
  130. package/dist/.claude/agents/kira-pragmatic.md +48 -0
  131. package/dist/.claude/agents/kishibe-hardening.md +39 -0
  132. package/dist/.claude/agents/kohaku-rapid-response.md +36 -0
  133. package/dist/.claude/agents/krillin-support.md +36 -0
  134. package/dist/.claude/agents/kusanagi-devops.md +70 -0
  135. package/dist/.claude/agents/la-forge-reliability.md +63 -0
  136. package/dist/.claude/agents/lang-micro-changes.md +43 -0
  137. package/dist/.claude/agents/legolas-precision.md +42 -0
  138. package/dist/.claude/agents/leia-secrets.md +41 -0
  139. package/dist/.claude/agents/lelouch-orchestration.md +39 -0
  140. package/dist/.claude/agents/leto-ii-persistence.md +40 -0
  141. package/dist/.claude/agents/leto-protection.md +40 -0
  142. package/dist/.claude/agents/levi-deploy.md +40 -0
  143. package/dist/.claude/agents/liet-kynes-deep-system.md +40 -0
  144. package/dist/.claude/agents/lift-social-media.md +40 -0
  145. package/dist/.claude/agents/loki-chaos.md +58 -0
  146. package/dist/.claude/agents/lucius-config.md +43 -0
  147. package/dist/.claude/agents/luke-audit-journey.md +41 -0
  148. package/dist/.claude/agents/manhunter-shapeshifting.md +43 -0
  149. package/dist/.claude/agents/marsh-competitive-intel.md +41 -0
  150. package/dist/.claude/agents/maul-red-team.md +57 -0
  151. package/dist/.claude/agents/merry-pair-review.md +40 -0
  152. package/dist/.claude/agents/mikasa-protection.md +39 -0
  153. package/dist/.claude/agents/miles-teg-perf.md +40 -0
  154. package/dist/.claude/agents/milim-load-test.md +39 -0
  155. package/dist/.claude/agents/misato-operations.md +39 -0
  156. package/dist/.claude/agents/mob-capacity.md +39 -0
  157. package/dist/.claude/agents/mohiam-authentication.md +40 -0
  158. package/dist/.claude/agents/mon-mothma-security-mgmt.md +41 -0
  159. package/dist/.claude/agents/mugen-chaos.md +39 -0
  160. package/dist/.claude/agents/mule-adversarial-ai.md +41 -0
  161. package/dist/.claude/agents/mustang-cleanup.md +39 -0
  162. package/dist/.claude/agents/nanami-structured-ops.md +39 -0
  163. package/dist/.claude/agents/nausicaa-resources.md +39 -0
  164. package/dist/.claude/agents/navani-technical-seo.md +40 -0
  165. package/dist/.claude/agents/nebula-optimization.md +43 -0
  166. package/dist/.claude/agents/nightwing-regression.md +59 -0
  167. package/dist/.claude/agents/nobara-direct-fix.md +39 -0
  168. package/dist/.claude/agents/nog-solutions.md +48 -0
  169. package/dist/.claude/agents/nori-asset-scanner.md +37 -0
  170. package/dist/.claude/agents/obrien-root-cause.md +48 -0
  171. package/dist/.claude/agents/odo-structural-anomaly.md +48 -0
  172. package/dist/.claude/agents/okoye-data-integrity.md +43 -0
  173. package/dist/.claude/agents/olivier-hardening.md +39 -0
  174. package/dist/.claude/agents/oracle-static-analysis.md +59 -0
  175. package/dist/.claude/agents/ori-prompt-crafter.md +37 -0
  176. package/dist/.claude/agents/padme-data-protection.md +42 -0
  177. package/dist/.claude/agents/paris-route-planner.md +47 -0
  178. package/dist/.claude/agents/parker-connections.md +43 -0
  179. package/dist/.claude/agents/paul-orchestration.md +40 -0
  180. package/dist/.claude/agents/picard-architecture.md +64 -0
  181. package/dist/.claude/agents/piccolo-tactics.md +39 -0
  182. package/dist/.claude/agents/pike-bold-decisions.md +48 -0
  183. package/dist/.claude/agents/pippin-discovery.md +40 -0
  184. package/dist/.claude/agents/plo-koon-edge-cases.md +37 -0
  185. package/dist/.claude/agents/power-chaotic.md +39 -0
  186. package/dist/.claude/agents/qui-gon-subtle-vulns.md +40 -0
  187. package/dist/.claude/agents/radagast-edge-cases.md +41 -0
  188. package/dist/.claude/agents/raoden-conversion.md +41 -0
  189. package/dist/.claude/agents/raven-deep-analysis.md +43 -0
  190. package/dist/.claude/agents/red-hood-aggressive.md +47 -0
  191. package/dist/.claude/agents/rei-dangerous-tasks.md +39 -0
  192. package/dist/.claude/agents/reigen-debugger.md +39 -0
  193. package/dist/.claude/agents/rengoku-intense-monitor.md +39 -0
  194. package/dist/.claude/agents/rex-infrastructure.md +41 -0
  195. package/dist/.claude/agents/rhodes-production.md +43 -0
  196. package/dist/.claude/agents/riker-review.md +62 -0
  197. package/dist/.claude/agents/rimuru-adapter.md +39 -0
  198. package/dist/.claude/agents/riza-backup.md +39 -0
  199. package/dist/.claude/agents/robin-apprentice.md +42 -0
  200. package/dist/.claude/agents/rocket-scrappy.md +43 -0
  201. package/dist/.claude/agents/rogers-api-design.md +43 -0
  202. package/dist/.claude/agents/romanoff-integrations.md +44 -0
  203. package/dist/.claude/agents/sabine-unconventional.md +40 -0
  204. package/dist/.claude/agents/salvor-model-selection.md +42 -0
  205. package/dist/.claude/agents/samwise-accessibility.md +43 -0
  206. package/dist/.claude/agents/sarene-outreach.md +40 -0
  207. package/dist/.claude/agents/sasha-resources.md +37 -0
  208. package/dist/.claude/agents/scotty-infrastructure.md +48 -0
  209. package/dist/.claude/agents/seldon-ai.md +67 -0
  210. package/dist/.claude/agents/senku-provisioning.md +40 -0
  211. package/dist/.claude/agents/sentaro-scheduling.md +37 -0
  212. package/dist/.claude/agents/seven-optimization.md +48 -0
  213. package/dist/.claude/agents/shallan-creative.md +40 -0
  214. package/dist/.claude/agents/sheeana-transport.md +40 -0
  215. package/dist/.claude/agents/shuri-innovation.md +43 -0
  216. package/dist/.claude/agents/silver-surfer-herald.md +90 -0
  217. package/dist/.claude/agents/siona-evasion.md +41 -0
  218. package/dist/.claude/agents/sisko-campaign.md +68 -0
  219. package/dist/.claude/agents/spike-routing.md +40 -0
  220. package/dist/.claude/agents/spock-schema.md +62 -0
  221. package/dist/.claude/agents/starfire-brute-force.md +43 -0
  222. package/dist/.claude/agents/stark-backend.md +71 -0
  223. package/dist/.claude/agents/steris-budget.md +41 -0
  224. package/dist/.claude/agents/stilgar-channel-security.md +40 -0
  225. package/dist/.claude/agents/strange-service-arch.md +44 -0
  226. package/dist/.claude/agents/sung-workers.md +39 -0
  227. package/dist/.claude/agents/superman-strength-test.md +43 -0
  228. package/dist/.claude/agents/suzaku-execution.md +39 -0
  229. package/dist/.claude/agents/szeth-compliance.md +40 -0
  230. package/dist/.claude/agents/tanjiro-persistent.md +39 -0
  231. package/dist/.claude/agents/tchalla-quality.md +43 -0
  232. package/dist/.claude/agents/thanos-gauntlet.md +68 -0
  233. package/dist/.claude/agents/theoden-rally.md +40 -0
  234. package/dist/.claude/agents/thor-queues.md +44 -0
  235. package/dist/.claude/agents/thufir-protocol-parsing.md +40 -0
  236. package/dist/.claude/agents/todo-brute-force.md +39 -0
  237. package/dist/.claude/agents/torres-site-scanner.md +47 -0
  238. package/dist/.claude/agents/totoro-guardian.md +39 -0
  239. package/dist/.claude/agents/tpol-disciplined.md +48 -0
  240. package/dist/.claude/agents/treebeard-deliberation.md +41 -0
  241. package/dist/.claude/agents/troi-prd-compliance.md +64 -0
  242. package/dist/.claude/agents/trunks-rollback.md +39 -0
  243. package/dist/.claude/agents/tuvok-deep-current.md +63 -0
  244. package/dist/.claude/agents/uhura-integration.md +47 -0
  245. package/dist/.claude/agents/valkyrie-recovery.md +43 -0
  246. package/dist/.claude/agents/vegeta-monitoring.md +39 -0
  247. package/dist/.claude/agents/veldora-dormant.md +37 -0
  248. package/dist/.claude/agents/vin-analytics.md +41 -0
  249. package/dist/.claude/agents/vision-data-analysis.md +43 -0
  250. package/dist/.claude/agents/wanda-seldon-validation.md +38 -0
  251. package/dist/.claude/agents/wanda-state.md +43 -0
  252. package/dist/.claude/agents/wax-paid-ads.md +40 -0
  253. package/dist/.claude/agents/wayne-ab-testing.md +40 -0
  254. package/dist/.claude/agents/whis-precision.md +39 -0
  255. package/dist/.claude/agents/windu-input-validation.md +41 -0
  256. package/dist/.claude/agents/winry-maintenance.md +39 -0
  257. package/dist/.claude/agents/wonder-woman-truth.md +43 -0
  258. package/dist/.claude/agents/wong-documentation.md +58 -0
  259. package/dist/.claude/agents/worf-security-arch.md +49 -0
  260. package/dist/.claude/agents/yoda-auth.md +57 -0
  261. package/dist/.claude/agents/yueh-trust-verify.md +40 -0
  262. package/dist/.claude/agents/zatanna-impossible.md +43 -0
  263. package/dist/.claude/agents/zechs-rival.md +39 -0
  264. package/dist/.claude/agents/zenitsu-alerts.md +37 -0
  265. package/dist/.claude/commands/ai.md +84 -0
  266. package/dist/.claude/commands/architect.md +107 -0
  267. package/dist/.claude/commands/assemble.md +223 -0
  268. package/dist/.claude/commands/assess.md +86 -0
  269. package/dist/.claude/commands/blueprint.md +135 -0
  270. package/dist/.claude/commands/build.md +138 -0
  271. package/dist/.claude/commands/campaign.md +224 -0
  272. package/dist/.claude/commands/cultivation.md +184 -0
  273. package/dist/.claude/commands/current.md +128 -0
  274. package/dist/.claude/commands/dangerroom.md +74 -0
  275. package/dist/.claude/commands/debrief.md +180 -0
  276. package/dist/.claude/commands/deploy.md +108 -0
  277. package/dist/.claude/commands/devops.md +160 -0
  278. package/dist/.claude/commands/engage.md +135 -0
  279. package/dist/.claude/commands/gauntlet.md +179 -0
  280. package/dist/.claude/commands/git.md +104 -0
  281. package/dist/.claude/commands/grow.md +160 -0
  282. package/dist/.claude/commands/imagine.md +126 -0
  283. package/dist/.claude/commands/portfolio.md +51 -0
  284. package/dist/.claude/commands/prd.md +113 -0
  285. package/dist/.claude/commands/qa.md +130 -0
  286. package/dist/.claude/commands/review.md +9 -0
  287. package/dist/.claude/commands/security.md +9 -0
  288. package/dist/.claude/commands/sentinel.md +90 -0
  289. package/dist/.claude/commands/test.md +114 -0
  290. package/dist/.claude/commands/thumper.md +116 -0
  291. package/dist/.claude/commands/treasury.md +117 -0
  292. package/dist/.claude/commands/ux.md +132 -0
  293. package/dist/.claude/commands/vault.md +198 -0
  294. package/dist/.claude/commands/void.md +148 -0
  295. package/dist/CHANGELOG.md +2621 -0
  296. package/dist/CLAUDE.md +292 -0
  297. package/dist/HOLOCRON.md +859 -0
  298. package/dist/VERSION.md +149 -0
  299. package/dist/docs/NAMING_REGISTRY.md +479 -0
  300. package/dist/docs/methods/AI_INTELLIGENCE.md +276 -0
  301. package/dist/docs/methods/ASSEMBLER.md +142 -0
  302. package/dist/docs/methods/BACKEND_ENGINEER.md +165 -0
  303. package/dist/docs/methods/BUILD_JOURNAL.md +214 -0
  304. package/dist/docs/methods/BUILD_PROTOCOL.md +436 -0
  305. package/dist/docs/methods/CAMPAIGN.md +569 -0
  306. package/dist/docs/methods/CONTEXT_MANAGEMENT.md +189 -0
  307. package/dist/docs/methods/DEEP_CURRENT.md +184 -0
  308. package/dist/docs/methods/DEVOPS_ENGINEER.md +297 -0
  309. package/dist/docs/methods/FIELD_MEDIC.md +265 -0
  310. package/dist/docs/methods/FORGE_ARTIST.md +108 -0
  311. package/dist/docs/methods/FORGE_KEEPER.md +270 -0
  312. package/dist/docs/methods/GAUNTLET.md +364 -0
  313. package/dist/docs/methods/GROWTH_STRATEGIST.md +466 -0
  314. package/dist/docs/methods/HEARTBEAT.md +168 -0
  315. package/dist/docs/methods/MCP_INTEGRATION.md +139 -0
  316. package/dist/docs/methods/MUSTER.md +152 -0
  317. package/dist/docs/methods/PRD_GENERATOR.md +186 -0
  318. package/dist/docs/methods/PRODUCT_DESIGN_FRONTEND.md +252 -0
  319. package/dist/docs/methods/QA_ENGINEER.md +360 -0
  320. package/dist/docs/methods/RELEASE_MANAGER.md +145 -0
  321. package/dist/docs/methods/SECURITY_AUDITOR.md +328 -0
  322. package/dist/docs/methods/SUB_AGENTS.md +375 -0
  323. package/dist/docs/methods/SYSTEMS_ARCHITECT.md +180 -0
  324. package/dist/docs/methods/TESTING.md +359 -0
  325. package/dist/docs/methods/THUMPER.md +175 -0
  326. package/dist/docs/methods/TIME_VAULT.md +120 -0
  327. package/dist/docs/methods/TREASURY.md +184 -0
  328. package/dist/docs/methods/TROUBLESHOOTING.md +265 -0
  329. package/dist/docs/patterns/README.md +52 -0
  330. package/dist/docs/patterns/ad-billing-adapter.ts +537 -0
  331. package/dist/docs/patterns/ad-platform-adapter.ts +421 -0
  332. package/dist/docs/patterns/ai-classifier.ts +195 -0
  333. package/dist/docs/patterns/ai-eval.ts +272 -0
  334. package/dist/docs/patterns/ai-orchestrator.ts +341 -0
  335. package/dist/docs/patterns/ai-router.ts +194 -0
  336. package/dist/docs/patterns/ai-tool-schema.ts +237 -0
  337. package/dist/docs/patterns/api-route.ts +241 -0
  338. package/dist/docs/patterns/backtest-engine.ts +499 -0
  339. package/dist/docs/patterns/browser-review.ts +292 -0
  340. package/dist/docs/patterns/combobox.tsx +300 -0
  341. package/dist/docs/patterns/component.tsx +262 -0
  342. package/dist/docs/patterns/daemon-process.ts +338 -0
  343. package/dist/docs/patterns/data-pipeline.ts +297 -0
  344. package/dist/docs/patterns/database-migration.ts +466 -0
  345. package/dist/docs/patterns/e2e-test.ts +629 -0
  346. package/dist/docs/patterns/error-handling.ts +312 -0
  347. package/dist/docs/patterns/execution-safety.ts +601 -0
  348. package/dist/docs/patterns/financial-transaction.ts +366 -0
  349. package/dist/docs/patterns/funding-plan.ts +462 -0
  350. package/dist/docs/patterns/game-entity.ts +137 -0
  351. package/dist/docs/patterns/game-loop.ts +113 -0
  352. package/dist/docs/patterns/game-state.ts +143 -0
  353. package/dist/docs/patterns/job-queue.ts +225 -0
  354. package/dist/docs/patterns/kongo-integration.ts +164 -0
  355. package/dist/docs/patterns/middleware.ts +363 -0
  356. package/dist/docs/patterns/mobile-screen.tsx +139 -0
  357. package/dist/docs/patterns/mobile-service.ts +167 -0
  358. package/dist/docs/patterns/multi-tenant.ts +382 -0
  359. package/dist/docs/patterns/oauth-token-lifecycle.ts +223 -0
  360. package/dist/docs/patterns/outbound-rate-limiter.ts +260 -0
  361. package/dist/docs/patterns/prompt-template.ts +195 -0
  362. package/dist/docs/patterns/revenue-source-adapter.ts +311 -0
  363. package/dist/docs/patterns/service.ts +224 -0
  364. package/dist/docs/patterns/sse-endpoint.ts +118 -0
  365. package/dist/docs/patterns/stablecoin-adapter.ts +511 -0
  366. package/dist/docs/patterns/third-party-script.ts +68 -0
  367. package/dist/scripts/thumper/gom-jabbar.sh +241 -0
  368. package/dist/scripts/thumper/relay.sh +610 -0
  369. package/dist/scripts/thumper/scan.sh +359 -0
  370. package/dist/scripts/thumper/thumper.sh +190 -0
  371. package/dist/scripts/thumper/water-rings.sh +76 -0
  372. package/dist/scripts/vault-read.d.ts +11 -0
  373. package/dist/scripts/vault-read.js +89 -0
  374. package/dist/scripts/voidforge.d.ts +21 -0
  375. package/dist/scripts/voidforge.js +614 -0
  376. package/dist/wizard/api/auth.d.ts +5 -0
  377. package/dist/wizard/api/auth.js +139 -0
  378. package/dist/wizard/api/blueprint.d.ts +34 -0
  379. package/dist/wizard/api/blueprint.js +161 -0
  380. package/dist/wizard/api/cloud-providers.d.ts +16 -0
  381. package/dist/wizard/api/cloud-providers.js +363 -0
  382. package/dist/wizard/api/credentials.d.ts +1 -0
  383. package/dist/wizard/api/credentials.js +265 -0
  384. package/dist/wizard/api/danger-room.d.ts +24 -0
  385. package/dist/wizard/api/danger-room.js +274 -0
  386. package/dist/wizard/api/deploy.d.ts +4 -0
  387. package/dist/wizard/api/deploy.js +164 -0
  388. package/dist/wizard/api/prd.d.ts +1 -0
  389. package/dist/wizard/api/prd.js +363 -0
  390. package/dist/wizard/api/project.d.ts +1 -0
  391. package/dist/wizard/api/project.js +241 -0
  392. package/dist/wizard/api/projects-data.d.ts +5 -0
  393. package/dist/wizard/api/projects-data.js +234 -0
  394. package/dist/wizard/api/projects-list.d.ts +5 -0
  395. package/dist/wizard/api/projects-list.js +227 -0
  396. package/dist/wizard/api/projects.d.ts +7 -0
  397. package/dist/wizard/api/projects.js +273 -0
  398. package/dist/wizard/api/provision-status.d.ts +5 -0
  399. package/dist/wizard/api/provision-status.js +47 -0
  400. package/dist/wizard/api/provision-steps.d.ts +21 -0
  401. package/dist/wizard/api/provision-steps.js +44 -0
  402. package/dist/wizard/api/provision-validate.d.ts +22 -0
  403. package/dist/wizard/api/provision-validate.js +164 -0
  404. package/dist/wizard/api/provision.d.ts +2 -0
  405. package/dist/wizard/api/provision.js +239 -0
  406. package/dist/wizard/api/terminal.d.ts +25 -0
  407. package/dist/wizard/api/terminal.js +246 -0
  408. package/dist/wizard/api/users.d.ts +6 -0
  409. package/dist/wizard/api/users.js +244 -0
  410. package/dist/wizard/api/war-room.d.ts +16 -0
  411. package/dist/wizard/api/war-room.js +70 -0
  412. package/dist/wizard/danger-room.config.json +5 -0
  413. package/dist/wizard/lib/ad-platform-core.d.ts +6 -0
  414. package/dist/wizard/lib/ad-platform-core.js +1 -0
  415. package/dist/wizard/lib/adapters/index.d.ts +52 -0
  416. package/dist/wizard/lib/adapters/index.js +38 -0
  417. package/dist/wizard/lib/adapters/sandbox-bank.d.ts +17 -0
  418. package/dist/wizard/lib/adapters/sandbox-bank.js +77 -0
  419. package/dist/wizard/lib/adapters/sandbox.d.ts +39 -0
  420. package/dist/wizard/lib/adapters/sandbox.js +174 -0
  421. package/dist/wizard/lib/adapters/stripe.d.ts +19 -0
  422. package/dist/wizard/lib/adapters/stripe.js +143 -0
  423. package/dist/wizard/lib/adapters/types.d.ts +9 -0
  424. package/dist/wizard/lib/adapters/types.js +10 -0
  425. package/dist/wizard/lib/agent-memory.d.ts +36 -0
  426. package/dist/wizard/lib/agent-memory.js +114 -0
  427. package/dist/wizard/lib/agent-registry.d.ts +21 -0
  428. package/dist/wizard/lib/agent-registry.js +105 -0
  429. package/dist/wizard/lib/anomaly-detection.d.ts +59 -0
  430. package/dist/wizard/lib/anomaly-detection.js +122 -0
  431. package/dist/wizard/lib/anthropic.d.ts +21 -0
  432. package/dist/wizard/lib/anthropic.js +105 -0
  433. package/dist/wizard/lib/asset-scanner.d.ts +23 -0
  434. package/dist/wizard/lib/asset-scanner.js +107 -0
  435. package/dist/wizard/lib/audit-log.d.ts +23 -0
  436. package/dist/wizard/lib/audit-log.js +70 -0
  437. package/dist/wizard/lib/autonomy-controller.d.ts +76 -0
  438. package/dist/wizard/lib/autonomy-controller.js +184 -0
  439. package/dist/wizard/lib/body-parser.d.ts +2 -0
  440. package/dist/wizard/lib/body-parser.js +36 -0
  441. package/dist/wizard/lib/build-analytics.d.ts +39 -0
  442. package/dist/wizard/lib/build-analytics.js +91 -0
  443. package/dist/wizard/lib/build-step.d.ts +21 -0
  444. package/dist/wizard/lib/build-step.js +104 -0
  445. package/dist/wizard/lib/campaign-proposer.d.ts +39 -0
  446. package/dist/wizard/lib/campaign-proposer.js +181 -0
  447. package/dist/wizard/lib/campaign-state-machine.d.ts +63 -0
  448. package/dist/wizard/lib/campaign-state-machine.js +114 -0
  449. package/dist/wizard/lib/ci-generator.d.ts +14 -0
  450. package/dist/wizard/lib/ci-generator.js +187 -0
  451. package/dist/wizard/lib/claude-merge.d.ts +38 -0
  452. package/dist/wizard/lib/claude-merge.js +115 -0
  453. package/dist/wizard/lib/codegen/erd-gen.d.ts +16 -0
  454. package/dist/wizard/lib/codegen/erd-gen.js +98 -0
  455. package/dist/wizard/lib/codegen/integrations.d.ts +18 -0
  456. package/dist/wizard/lib/codegen/integrations.js +189 -0
  457. package/dist/wizard/lib/codegen/openapi-gen.d.ts +15 -0
  458. package/dist/wizard/lib/codegen/openapi-gen.js +79 -0
  459. package/dist/wizard/lib/codegen/prisma-types.d.ts +15 -0
  460. package/dist/wizard/lib/codegen/prisma-types.js +44 -0
  461. package/dist/wizard/lib/codegen/seed-gen.d.ts +16 -0
  462. package/dist/wizard/lib/codegen/seed-gen.js +128 -0
  463. package/dist/wizard/lib/compliance.d.ts +51 -0
  464. package/dist/wizard/lib/compliance.js +112 -0
  465. package/dist/wizard/lib/correlation-engine.d.ts +59 -0
  466. package/dist/wizard/lib/correlation-engine.js +152 -0
  467. package/dist/wizard/lib/cost-estimator.d.ts +22 -0
  468. package/dist/wizard/lib/cost-estimator.js +72 -0
  469. package/dist/wizard/lib/cost-tracker.d.ts +27 -0
  470. package/dist/wizard/lib/cost-tracker.js +37 -0
  471. package/dist/wizard/lib/daemon-aggregator.d.ts +76 -0
  472. package/dist/wizard/lib/daemon-aggregator.js +241 -0
  473. package/dist/wizard/lib/daemon-core.d.ts +16 -0
  474. package/dist/wizard/lib/daemon-core.js +39 -0
  475. package/dist/wizard/lib/dashboard-data.d.ts +123 -0
  476. package/dist/wizard/lib/dashboard-data.js +314 -0
  477. package/dist/wizard/lib/dashboard-ws.d.ts +28 -0
  478. package/dist/wizard/lib/dashboard-ws.js +117 -0
  479. package/dist/wizard/lib/deep-current.d.ts +77 -0
  480. package/dist/wizard/lib/deep-current.js +247 -0
  481. package/dist/wizard/lib/deploy-coordinator.d.ts +40 -0
  482. package/dist/wizard/lib/deploy-coordinator.js +86 -0
  483. package/dist/wizard/lib/deploy-log.d.ts +28 -0
  484. package/dist/wizard/lib/deploy-log.js +52 -0
  485. package/dist/wizard/lib/desktop-notify.d.ts +27 -0
  486. package/dist/wizard/lib/desktop-notify.js +98 -0
  487. package/dist/wizard/lib/dns/cloudflare-dns.d.ts +35 -0
  488. package/dist/wizard/lib/dns/cloudflare-dns.js +216 -0
  489. package/dist/wizard/lib/dns/cloudflare-registrar.d.ts +31 -0
  490. package/dist/wizard/lib/dns/cloudflare-registrar.js +148 -0
  491. package/dist/wizard/lib/dns/types.d.ts +22 -0
  492. package/dist/wizard/lib/dns/types.js +4 -0
  493. package/dist/wizard/lib/document-discovery.d.ts +33 -0
  494. package/dist/wizard/lib/document-discovery.js +145 -0
  495. package/dist/wizard/lib/env-validator.d.ts +14 -0
  496. package/dist/wizard/lib/env-validator.js +205 -0
  497. package/dist/wizard/lib/env-writer.d.ts +13 -0
  498. package/dist/wizard/lib/env-writer.js +26 -0
  499. package/dist/wizard/lib/exec.d.ts +30 -0
  500. package/dist/wizard/lib/exec.js +52 -0
  501. package/dist/wizard/lib/experiment.d.ts +70 -0
  502. package/dist/wizard/lib/experiment.js +169 -0
  503. package/dist/wizard/lib/extensions.d.ts +20 -0
  504. package/dist/wizard/lib/extensions.js +183 -0
  505. package/dist/wizard/lib/financial/adapter-factory.d.ts +47 -0
  506. package/dist/wizard/lib/financial/adapter-factory.js +225 -0
  507. package/dist/wizard/lib/financial/billing/base.d.ts +6 -0
  508. package/dist/wizard/lib/financial/billing/base.js +1 -0
  509. package/dist/wizard/lib/financial/billing/google-billing.d.ts +56 -0
  510. package/dist/wizard/lib/financial/billing/google-billing.js +298 -0
  511. package/dist/wizard/lib/financial/billing/meta-billing.d.ts +54 -0
  512. package/dist/wizard/lib/financial/billing/meta-billing.js +243 -0
  513. package/dist/wizard/lib/financial/billing/tiktok-billing.d.ts +54 -0
  514. package/dist/wizard/lib/financial/billing/tiktok-billing.js +260 -0
  515. package/dist/wizard/lib/financial/campaign/base.d.ts +13 -0
  516. package/dist/wizard/lib/financial/campaign/base.js +1 -0
  517. package/dist/wizard/lib/financial/campaign/campaign-common.d.ts +21 -0
  518. package/dist/wizard/lib/financial/campaign/campaign-common.js +58 -0
  519. package/dist/wizard/lib/financial/campaign/google-api.d.ts +35 -0
  520. package/dist/wizard/lib/financial/campaign/google-api.js +118 -0
  521. package/dist/wizard/lib/financial/campaign/google-campaign.d.ts +38 -0
  522. package/dist/wizard/lib/financial/campaign/google-campaign.js +186 -0
  523. package/dist/wizard/lib/financial/campaign/meta-api.d.ts +28 -0
  524. package/dist/wizard/lib/financial/campaign/meta-api.js +93 -0
  525. package/dist/wizard/lib/financial/campaign/meta-campaign.d.ts +32 -0
  526. package/dist/wizard/lib/financial/campaign/meta-campaign.js +189 -0
  527. package/dist/wizard/lib/financial/campaign/sandbox-campaign.d.ts +45 -0
  528. package/dist/wizard/lib/financial/campaign/sandbox-campaign.js +261 -0
  529. package/dist/wizard/lib/financial/campaign/tiktok-api.d.ts +25 -0
  530. package/dist/wizard/lib/financial/campaign/tiktok-api.js +81 -0
  531. package/dist/wizard/lib/financial/campaign/tiktok-campaign.d.ts +37 -0
  532. package/dist/wizard/lib/financial/campaign/tiktok-campaign.js +155 -0
  533. package/dist/wizard/lib/financial/funding-auto.d.ts +44 -0
  534. package/dist/wizard/lib/financial/funding-auto.js +52 -0
  535. package/dist/wizard/lib/financial/funding-policy.d.ts +60 -0
  536. package/dist/wizard/lib/financial/funding-policy.js +179 -0
  537. package/dist/wizard/lib/financial/platform-planner.d.ts +47 -0
  538. package/dist/wizard/lib/financial/platform-planner.js +134 -0
  539. package/dist/wizard/lib/financial/reconciliation-engine.d.ts +78 -0
  540. package/dist/wizard/lib/financial/reconciliation-engine.js +193 -0
  541. package/dist/wizard/lib/financial/registry.d.ts +22 -0
  542. package/dist/wizard/lib/financial/registry.js +26 -0
  543. package/dist/wizard/lib/financial/reporting.d.ts +96 -0
  544. package/dist/wizard/lib/financial/reporting.js +198 -0
  545. package/dist/wizard/lib/financial/stablecoin/base.d.ts +6 -0
  546. package/dist/wizard/lib/financial/stablecoin/base.js +1 -0
  547. package/dist/wizard/lib/financial/stablecoin/circle.d.ts +54 -0
  548. package/dist/wizard/lib/financial/stablecoin/circle.js +367 -0
  549. package/dist/wizard/lib/financial/stablecoin/mercury.d.ts +24 -0
  550. package/dist/wizard/lib/financial/stablecoin/mercury.js +171 -0
  551. package/dist/wizard/lib/financial/stablecoin/sandbox-stablecoin.d.ts +47 -0
  552. package/dist/wizard/lib/financial/stablecoin/sandbox-stablecoin.js +202 -0
  553. package/dist/wizard/lib/financial/treasury-planner.d.ts +52 -0
  554. package/dist/wizard/lib/financial/treasury-planner.js +128 -0
  555. package/dist/wizard/lib/financial-core.d.ts +6 -0
  556. package/dist/wizard/lib/financial-core.js +5 -0
  557. package/dist/wizard/lib/financial-vault.d.ts +34 -0
  558. package/dist/wizard/lib/financial-vault.js +200 -0
  559. package/dist/wizard/lib/frontmatter.d.ts +30 -0
  560. package/dist/wizard/lib/frontmatter.js +99 -0
  561. package/dist/wizard/lib/gap-analysis.d.ts +37 -0
  562. package/dist/wizard/lib/gap-analysis.js +218 -0
  563. package/dist/wizard/lib/github.d.ts +22 -0
  564. package/dist/wizard/lib/github.js +261 -0
  565. package/dist/wizard/lib/headless-deploy.d.ts +14 -0
  566. package/dist/wizard/lib/headless-deploy.js +452 -0
  567. package/dist/wizard/lib/health-monitor.d.ts +15 -0
  568. package/dist/wizard/lib/health-monitor.js +91 -0
  569. package/dist/wizard/lib/health-poller.d.ts +9 -0
  570. package/dist/wizard/lib/health-poller.js +123 -0
  571. package/dist/wizard/lib/heartbeat-lifecycle.d.ts +71 -0
  572. package/dist/wizard/lib/heartbeat-lifecycle.js +107 -0
  573. package/dist/wizard/lib/heartbeat-scheduler.d.ts +26 -0
  574. package/dist/wizard/lib/heartbeat-scheduler.js +155 -0
  575. package/dist/wizard/lib/heartbeat.d.ts +22 -0
  576. package/dist/wizard/lib/heartbeat.js +538 -0
  577. package/dist/wizard/lib/herald.d.ts +28 -0
  578. package/dist/wizard/lib/herald.js +167 -0
  579. package/dist/wizard/lib/http-helpers.d.ts +9 -0
  580. package/dist/wizard/lib/http-helpers.js +24 -0
  581. package/dist/wizard/lib/image-gen.d.ts +56 -0
  582. package/dist/wizard/lib/image-gen.js +159 -0
  583. package/dist/wizard/lib/instance-sizing.d.ts +26 -0
  584. package/dist/wizard/lib/instance-sizing.js +51 -0
  585. package/dist/wizard/lib/kongo/analytics.d.ts +29 -0
  586. package/dist/wizard/lib/kongo/analytics.js +179 -0
  587. package/dist/wizard/lib/kongo/campaigns.d.ts +52 -0
  588. package/dist/wizard/lib/kongo/campaigns.js +91 -0
  589. package/dist/wizard/lib/kongo/client.d.ts +58 -0
  590. package/dist/wizard/lib/kongo/client.js +221 -0
  591. package/dist/wizard/lib/kongo/jobs.d.ts +57 -0
  592. package/dist/wizard/lib/kongo/jobs.js +122 -0
  593. package/dist/wizard/lib/kongo/pages.d.ts +60 -0
  594. package/dist/wizard/lib/kongo/pages.js +150 -0
  595. package/dist/wizard/lib/kongo/provisioner.d.ts +64 -0
  596. package/dist/wizard/lib/kongo/provisioner.js +116 -0
  597. package/dist/wizard/lib/kongo/seed.d.ts +49 -0
  598. package/dist/wizard/lib/kongo/seed.js +237 -0
  599. package/dist/wizard/lib/kongo/types.d.ts +323 -0
  600. package/dist/wizard/lib/kongo/types.js +11 -0
  601. package/dist/wizard/lib/kongo/variants.d.ts +57 -0
  602. package/dist/wizard/lib/kongo/variants.js +88 -0
  603. package/dist/wizard/lib/kongo/webhooks.d.ts +41 -0
  604. package/dist/wizard/lib/kongo/webhooks.js +112 -0
  605. package/dist/wizard/lib/marker.d.ts +28 -0
  606. package/dist/wizard/lib/marker.js +79 -0
  607. package/dist/wizard/lib/migrator.d.ts +35 -0
  608. package/dist/wizard/lib/migrator.js +190 -0
  609. package/dist/wizard/lib/natural-language-deploy.d.ts +30 -0
  610. package/dist/wizard/lib/natural-language-deploy.js +186 -0
  611. package/dist/wizard/lib/network.d.ts +22 -0
  612. package/dist/wizard/lib/network.js +72 -0
  613. package/dist/wizard/lib/oauth-core.d.ts +6 -0
  614. package/dist/wizard/lib/oauth-core.js +5 -0
  615. package/dist/wizard/lib/open-browser.d.ts +1 -0
  616. package/dist/wizard/lib/open-browser.js +26 -0
  617. package/dist/wizard/lib/patterns/ad-billing-adapter.d.ts +209 -0
  618. package/dist/wizard/lib/patterns/ad-billing-adapter.js +269 -0
  619. package/dist/wizard/lib/patterns/ad-platform-adapter.d.ts +200 -0
  620. package/dist/wizard/lib/patterns/ad-platform-adapter.js +212 -0
  621. package/dist/wizard/lib/patterns/daemon-process.d.ts +88 -0
  622. package/dist/wizard/lib/patterns/daemon-process.js +271 -0
  623. package/dist/wizard/lib/patterns/financial-transaction.d.ts +171 -0
  624. package/dist/wizard/lib/patterns/financial-transaction.js +154 -0
  625. package/dist/wizard/lib/patterns/funding-plan.d.ts +136 -0
  626. package/dist/wizard/lib/patterns/funding-plan.js +200 -0
  627. package/dist/wizard/lib/patterns/oauth-token-lifecycle.d.ts +94 -0
  628. package/dist/wizard/lib/patterns/oauth-token-lifecycle.js +139 -0
  629. package/dist/wizard/lib/patterns/outbound-rate-limiter.d.ts +67 -0
  630. package/dist/wizard/lib/patterns/outbound-rate-limiter.js +216 -0
  631. package/dist/wizard/lib/patterns/revenue-source-adapter.d.ts +96 -0
  632. package/dist/wizard/lib/patterns/revenue-source-adapter.js +182 -0
  633. package/dist/wizard/lib/patterns/stablecoin-adapter.d.ts +218 -0
  634. package/dist/wizard/lib/patterns/stablecoin-adapter.js +264 -0
  635. package/dist/wizard/lib/prd-validator.d.ts +39 -0
  636. package/dist/wizard/lib/prd-validator.js +137 -0
  637. package/dist/wizard/lib/project-init.d.ts +24 -0
  638. package/dist/wizard/lib/project-init.js +228 -0
  639. package/dist/wizard/lib/project-registry.d.ts +86 -0
  640. package/dist/wizard/lib/project-registry.js +359 -0
  641. package/dist/wizard/lib/project-scope.d.ts +64 -0
  642. package/dist/wizard/lib/project-scope.js +96 -0
  643. package/dist/wizard/lib/project-vault.d.ts +47 -0
  644. package/dist/wizard/lib/project-vault.js +221 -0
  645. package/dist/wizard/lib/provision-manifest.d.ts +44 -0
  646. package/dist/wizard/lib/provision-manifest.js +164 -0
  647. package/dist/wizard/lib/provisioner-registry.d.ts +15 -0
  648. package/dist/wizard/lib/provisioner-registry.js +34 -0
  649. package/dist/wizard/lib/provisioners/aws-config.d.ts +36 -0
  650. package/dist/wizard/lib/provisioners/aws-config.js +56 -0
  651. package/dist/wizard/lib/provisioners/aws-ec2.d.ts +19 -0
  652. package/dist/wizard/lib/provisioners/aws-ec2.js +241 -0
  653. package/dist/wizard/lib/provisioners/aws-rds.d.ts +10 -0
  654. package/dist/wizard/lib/provisioners/aws-rds.js +199 -0
  655. package/dist/wizard/lib/provisioners/aws-vps.d.ts +6 -0
  656. package/dist/wizard/lib/provisioners/aws-vps.js +231 -0
  657. package/dist/wizard/lib/provisioners/cloudflare.d.ts +6 -0
  658. package/dist/wizard/lib/provisioners/cloudflare.js +300 -0
  659. package/dist/wizard/lib/provisioners/docker.d.ts +6 -0
  660. package/dist/wizard/lib/provisioners/docker.js +75 -0
  661. package/dist/wizard/lib/provisioners/http-client.d.ts +20 -0
  662. package/dist/wizard/lib/provisioners/http-client.js +79 -0
  663. package/dist/wizard/lib/provisioners/railway-config.d.ts +24 -0
  664. package/dist/wizard/lib/provisioners/railway-config.js +220 -0
  665. package/dist/wizard/lib/provisioners/railway-deploy.d.ts +19 -0
  666. package/dist/wizard/lib/provisioners/railway-deploy.js +205 -0
  667. package/dist/wizard/lib/provisioners/railway.d.ts +6 -0
  668. package/dist/wizard/lib/provisioners/railway.js +45 -0
  669. package/dist/wizard/lib/provisioners/scripts/caddyfile.d.ts +10 -0
  670. package/dist/wizard/lib/provisioners/scripts/caddyfile.js +54 -0
  671. package/dist/wizard/lib/provisioners/scripts/deploy-vps.d.ts +10 -0
  672. package/dist/wizard/lib/provisioners/scripts/deploy-vps.js +112 -0
  673. package/dist/wizard/lib/provisioners/scripts/docker-compose.d.ts +11 -0
  674. package/dist/wizard/lib/provisioners/scripts/docker-compose.js +91 -0
  675. package/dist/wizard/lib/provisioners/scripts/dockerfile.d.ts +5 -0
  676. package/dist/wizard/lib/provisioners/scripts/dockerfile.js +185 -0
  677. package/dist/wizard/lib/provisioners/scripts/ecosystem-config.d.ts +10 -0
  678. package/dist/wizard/lib/provisioners/scripts/ecosystem-config.js +36 -0
  679. package/dist/wizard/lib/provisioners/scripts/provision-vps.d.ts +14 -0
  680. package/dist/wizard/lib/provisioners/scripts/provision-vps.js +202 -0
  681. package/dist/wizard/lib/provisioners/scripts/rollback-vps.d.ts +10 -0
  682. package/dist/wizard/lib/provisioners/scripts/rollback-vps.js +67 -0
  683. package/dist/wizard/lib/provisioners/self-deploy.d.ts +41 -0
  684. package/dist/wizard/lib/provisioners/self-deploy.js +185 -0
  685. package/dist/wizard/lib/provisioners/static-s3.d.ts +6 -0
  686. package/dist/wizard/lib/provisioners/static-s3.js +235 -0
  687. package/dist/wizard/lib/provisioners/types.d.ts +40 -0
  688. package/dist/wizard/lib/provisioners/types.js +4 -0
  689. package/dist/wizard/lib/provisioners/vercel.d.ts +6 -0
  690. package/dist/wizard/lib/provisioners/vercel.js +287 -0
  691. package/dist/wizard/lib/pty-manager.d.ts +42 -0
  692. package/dist/wizard/lib/pty-manager.js +244 -0
  693. package/dist/wizard/lib/rate-limiter-core.d.ts +5 -0
  694. package/dist/wizard/lib/rate-limiter-core.js +5 -0
  695. package/dist/wizard/lib/reconciliation.d.ts +43 -0
  696. package/dist/wizard/lib/reconciliation.js +173 -0
  697. package/dist/wizard/lib/revenue-types.d.ts +5 -0
  698. package/dist/wizard/lib/revenue-types.js +1 -0
  699. package/dist/wizard/lib/route-optimizer.d.ts +28 -0
  700. package/dist/wizard/lib/route-optimizer.js +93 -0
  701. package/dist/wizard/lib/s3-deploy.d.ts +19 -0
  702. package/dist/wizard/lib/s3-deploy.js +156 -0
  703. package/dist/wizard/lib/safety-tiers.d.ts +76 -0
  704. package/dist/wizard/lib/safety-tiers.js +134 -0
  705. package/dist/wizard/lib/sentry-generator.d.ts +15 -0
  706. package/dist/wizard/lib/sentry-generator.js +116 -0
  707. package/dist/wizard/lib/server-config.d.ts +13 -0
  708. package/dist/wizard/lib/server-config.js +23 -0
  709. package/dist/wizard/lib/service-install.d.ts +18 -0
  710. package/dist/wizard/lib/service-install.js +182 -0
  711. package/dist/wizard/lib/site-scanner.d.ts +80 -0
  712. package/dist/wizard/lib/site-scanner.js +262 -0
  713. package/dist/wizard/lib/ssh-deploy.d.ts +25 -0
  714. package/dist/wizard/lib/ssh-deploy.js +225 -0
  715. package/dist/wizard/lib/templates.d.ts +24 -0
  716. package/dist/wizard/lib/templates.js +219 -0
  717. package/dist/wizard/lib/totp.d.ts +35 -0
  718. package/dist/wizard/lib/totp.js +277 -0
  719. package/dist/wizard/lib/tower-auth.d.ts +43 -0
  720. package/dist/wizard/lib/tower-auth.js +352 -0
  721. package/dist/wizard/lib/tower-rate-limit.d.ts +14 -0
  722. package/dist/wizard/lib/tower-rate-limit.js +61 -0
  723. package/dist/wizard/lib/tower-session.d.ts +28 -0
  724. package/dist/wizard/lib/tower-session.js +119 -0
  725. package/dist/wizard/lib/treasury-backup.d.ts +23 -0
  726. package/dist/wizard/lib/treasury-backup.js +127 -0
  727. package/dist/wizard/lib/treasury-circuit-breakers.d.ts +28 -0
  728. package/dist/wizard/lib/treasury-circuit-breakers.js +74 -0
  729. package/dist/wizard/lib/treasury-handlers.d.ts +21 -0
  730. package/dist/wizard/lib/treasury-handlers.js +281 -0
  731. package/dist/wizard/lib/treasury-heartbeat.d.ts +18 -0
  732. package/dist/wizard/lib/treasury-heartbeat.js +20 -0
  733. package/dist/wizard/lib/treasury-io.d.ts +107 -0
  734. package/dist/wizard/lib/treasury-io.js +254 -0
  735. package/dist/wizard/lib/treasury-jobs.d.ts +14 -0
  736. package/dist/wizard/lib/treasury-jobs.js +589 -0
  737. package/dist/wizard/lib/treasury-migrator.d.ts +59 -0
  738. package/dist/wizard/lib/treasury-migrator.js +227 -0
  739. package/dist/wizard/lib/treasury-reader.d.ts +52 -0
  740. package/dist/wizard/lib/treasury-reader.js +235 -0
  741. package/dist/wizard/lib/updater.d.ts +29 -0
  742. package/dist/wizard/lib/updater.js +203 -0
  743. package/dist/wizard/lib/user-manager.d.ts +39 -0
  744. package/dist/wizard/lib/user-manager.js +182 -0
  745. package/dist/wizard/lib/vault.d.ts +26 -0
  746. package/dist/wizard/lib/vault.js +161 -0
  747. package/dist/wizard/router.d.ts +12 -0
  748. package/dist/wizard/router.js +58 -0
  749. package/dist/wizard/server.d.ts +18 -0
  750. package/dist/wizard/server.js +427 -0
  751. package/dist/wizard/ui/app.js +1357 -0
  752. package/dist/wizard/ui/danger-room-prophecy.js +217 -0
  753. package/dist/wizard/ui/danger-room.html +27 -0
  754. package/dist/wizard/ui/danger-room.js +29 -0
  755. package/dist/wizard/ui/deploy.html +181 -0
  756. package/dist/wizard/ui/deploy.js +616 -0
  757. package/dist/wizard/ui/favicon.svg +11 -0
  758. package/dist/wizard/ui/index.html +407 -0
  759. package/dist/wizard/ui/lobby.html +235 -0
  760. package/dist/wizard/ui/lobby.js +843 -0
  761. package/dist/wizard/ui/login.html +111 -0
  762. package/dist/wizard/ui/login.js +199 -0
  763. package/dist/wizard/ui/project.html +285 -0
  764. package/dist/wizard/ui/project.js +324 -0
  765. package/dist/wizard/ui/rollback.js +107 -0
  766. package/dist/wizard/ui/styles.css +1040 -0
  767. package/dist/wizard/ui/tower.html +177 -0
  768. package/dist/wizard/ui/tower.js +445 -0
  769. package/dist/wizard/ui/war-room-prophecy.js +217 -0
  770. package/dist/wizard/ui/war-room.html +27 -0
  771. package/dist/wizard/ui/war-room.js +29 -0
  772. package/package.json +60 -0
@@ -0,0 +1,1357 @@
1
+ /**
2
+ * VoidForge Wizard — Vanilla JS Step Machine
3
+ * Gandalf — Setup Wizard (Three-Act Flow, v7.1)
4
+ * Act 1: 1=Vault, 2=API Key
5
+ * Act 2: 3=Project, 4=PRD, 4b=Credentials
6
+ * Act 3: 5=Operations Menu, 6=Review, 7=Create/Done
7
+ */
8
+
9
+ (function () {
10
+ 'use strict';
11
+
12
+ const TOTAL_STEPS = 7;
13
+ let currentStep = 1;
14
+
15
+ // State
16
+ const state = {
17
+ anthropicKeyStored: false,
18
+ cloudProviders: {}, // { aws: true, vercel: false, ... }
19
+ projectName: '',
20
+ projectDir: '',
21
+ projectDesc: '',
22
+ projectDomain: '',
23
+ projectHostname: '',
24
+ prdMode: 'generate', // 'generate' | 'paste' | 'skip'
25
+ prdContent: '',
26
+ generatedPrd: '',
27
+ deployTarget: '',
28
+ createdDir: '',
29
+ envGroups: [], // PRD-driven env credential groups
30
+ envCredentials: {}, // { VAR_NAME: 'value', ... }
31
+ };
32
+
33
+ // DOM refs
34
+ const $ = (sel) => document.querySelector(sel);
35
+ const $$ = (sel) => document.querySelectorAll(sel);
36
+
37
+ const progressBar = $('#progress-bar');
38
+ const stepLabel = $('#step-label');
39
+ const btnBack = $('#btn-back');
40
+ const btnNext = $('#btn-next');
41
+
42
+ // --- Navigation ---
43
+
44
+ function showStep(step) {
45
+ // ENCHANT-R2-012 + CROSS-R4-016: Determine direction for animation
46
+ const goingBack = (typeof step === 'number' && typeof currentStep === 'number' && step < currentStep) ||
47
+ (step === 4 && currentStep === '4b') ||
48
+ (step === '4b' && currentStep === 5);
49
+ $$('.step').forEach((el) => {
50
+ el.classList.add('hidden');
51
+ el.classList.remove('entering-backward');
52
+ });
53
+ const target = $(`#step-${step}`);
54
+ if (target) {
55
+ if (goingBack) target.classList.add('entering-backward');
56
+ target.classList.remove('hidden');
57
+ }
58
+
59
+ currentStep = step;
60
+
61
+ // Act-based progress: Act 1 (steps 1-2), Act 2 (steps 3-4b), Act 3 (steps 5-7)
62
+ const hasEnvStep = state.envGroups.length > 0;
63
+ const visibleSteps = hasEnvStep ? [1, 2, 3, 4, '4b', 5, 6, 7] : [1, 2, 3, 4, 5, 6, 7];
64
+ const currentIdx = visibleSteps.indexOf(step);
65
+ const totalVisible = visibleSteps.length;
66
+ const displayNum = currentIdx >= 0 ? currentIdx + 1 : (typeof step === 'number' ? step : 5);
67
+
68
+ const pct = Math.round((displayNum / totalVisible) * 100);
69
+ progressBar.style.width = `${pct}%`;
70
+ progressBar.setAttribute('aria-valuenow', String(pct));
71
+
72
+ // Show act labels instead of step numbers
73
+ let actLabel = 'Act 1 — Secure Your Forge';
74
+ if (step >= 3 && step !== 5 && step !== 6 && step !== 7) actLabel = 'Act 2 — Describe Your Vision';
75
+ if (step === '4b') actLabel = 'Act 2 — Describe Your Vision';
76
+ if (step >= 5) actLabel = 'Act 3 — Equip Your Project';
77
+ if (step === 6) actLabel = 'Review';
78
+ if (step === 7) actLabel = 'Creating';
79
+ stepLabel.textContent = actLabel;
80
+
81
+ btnBack.disabled = step === 1;
82
+
83
+ if (step === 6) {
84
+ btnNext.textContent = 'Create Project';
85
+ } else if (step === 7) {
86
+ btnNext.style.display = 'none';
87
+ btnBack.style.display = 'none';
88
+ } else if (step === '4b') {
89
+ // Step 4b has its own Store/Skip buttons, hide main nav Next
90
+ btnNext.style.display = 'none';
91
+ btnBack.style.display = '';
92
+ } else {
93
+ btnNext.textContent = 'Next';
94
+ btnNext.style.display = '';
95
+ btnBack.style.display = '';
96
+ }
97
+
98
+ // Focus first input
99
+ const firstInput = target?.querySelector('input, textarea, select');
100
+ if (firstInput) setTimeout(() => firstInput.focus(), 100);
101
+ }
102
+
103
+ function syncState() {
104
+ if (currentStep === 3) {
105
+ state.projectName = $('#project-name').value.trim();
106
+ state.projectDir = $('#project-dir').value.trim();
107
+ state.projectDesc = $('#project-desc').value.trim();
108
+ }
109
+ if (currentStep === 5) {
110
+ // Domain/hostname are now in the operations menu
111
+ state.projectDomain = $('#project-domain')?.value.trim() || '';
112
+ state.projectHostname = $('#project-hostname')?.value.trim() || '';
113
+ }
114
+ if (currentStep === 4) {
115
+ if (state.prdMode === 'paste') {
116
+ state.prdContent = $('#prd-paste').value.trim();
117
+ } else if (state.prdMode === 'generate' && state.generatedPrd) {
118
+ state.prdContent = state.generatedPrd;
119
+ } else {
120
+ state.prdContent = '';
121
+ }
122
+ }
123
+ // Deploy target is set directly by card clicks — no sync needed
124
+ }
125
+
126
+ function canAdvance() {
127
+ switch (currentStep) {
128
+ case 1: return true; // Vault unlock handled by button, not Next
129
+ case 2: return true; // API key has skip option
130
+ case 3: return state.projectName.trim() !== '' && state.projectDir.trim() !== '';
131
+ case 4: return true; // PRD optional
132
+ case 5: return true; // Operations menu — all optional
133
+ case 6: return true;
134
+ default: return false;
135
+ }
136
+ }
137
+
138
+ async function nextStep() {
139
+ syncState();
140
+ if (!canAdvance()) {
141
+ showValidationErrors();
142
+ return;
143
+ }
144
+ clearValidationErrors();
145
+
146
+ // After PRD step (4), check for env requirements then go to operations menu
147
+ if (currentStep === 4) {
148
+ const prdText = state.prdContent || state.generatedPrd || '';
149
+ if (prdText) {
150
+ const envGroups = await loadEnvRequirements(prdText);
151
+ state.envGroups = envGroups;
152
+ if (envGroups.length > 0) {
153
+ renderEnvCredentials(envGroups);
154
+ showStep('4b');
155
+ return;
156
+ }
157
+ }
158
+ // No env requirements — go to operations menu
159
+ await loadDeployTargets();
160
+ loadCloudProviders();
161
+ showStep(5);
162
+ return;
163
+ }
164
+
165
+ // After 4b, proceed to operations menu
166
+ if (currentStep === '4b') {
167
+ await loadDeployTargets();
168
+ loadCloudProviders();
169
+ showStep(5);
170
+ return;
171
+ }
172
+
173
+ if (currentStep === 6) {
174
+ createProject();
175
+ showStep(7);
176
+ return;
177
+ }
178
+
179
+ // Blueprint auto-detection: when moving from Step 3 → Step 4,
180
+ // check if docs/PRD.md already exists in the TARGET project directory (not CWD).
181
+ // During init, the project dir doesn't exist yet — detection is skipped.
182
+ if (currentStep === 3 && state.projectDir) {
183
+ try {
184
+ const detectRes = await fetch('/api/blueprint/detect?dir=' + encodeURIComponent(state.projectDir));
185
+ const detectData = await detectRes.json();
186
+ if (detectData.detected) {
187
+ const blueprintBanner = $('#blueprint-detection');
188
+ if (blueprintBanner) {
189
+ $('#blueprint-project-name').textContent = detectData.name || 'your project';
190
+ blueprintBanner.classList.remove('hidden');
191
+ // User clicks "Use my blueprint" → redirect to /blueprint flow
192
+ // User clicks "Start fresh" → continue to Step 4 normally
193
+ return; // Pause navigation until user decides
194
+ }
195
+ }
196
+ } catch { /* detection failed — proceed normally */ }
197
+ }
198
+
199
+ if (currentStep < TOTAL_STEPS) {
200
+ const nextStepNum = currentStep + 1;
201
+ if (nextStepNum === 5) await loadDeployTargets();
202
+ if (nextStepNum === 6) populateReview();
203
+ showStep(nextStepNum);
204
+ }
205
+ }
206
+
207
+ function prevStep() {
208
+ if (currentStep === 1) return;
209
+ if (currentStep === '4b') { showStep(4); return; }
210
+ if (currentStep === 5 && state.envGroups.length > 0) { showStep('4b'); return; }
211
+ if (currentStep === 5) { showStep(4); return; }
212
+ if (currentStep === 6) { showStep(5); return; }
213
+ showStep(currentStep - 1);
214
+ }
215
+
216
+ btnNext.addEventListener('click', nextStep);
217
+ btnBack.addEventListener('click', prevStep);
218
+
219
+ // Keyboard: Enter triggers contextual action
220
+ document.addEventListener('keydown', (e) => {
221
+ if (e.key !== 'Enter') return;
222
+ if (e.target.tagName === 'TEXTAREA') return;
223
+ // Don't intercept Enter on buttons, links, or selects — let native behavior handle them
224
+ if (e.target.tagName === 'BUTTON' || e.target.tagName === 'A' || e.target.tagName === 'SELECT') return;
225
+ e.preventDefault();
226
+
227
+ if (currentStep === 1) {
228
+ if (vaultPasswordInput.value) unlockVaultBtn.click();
229
+ return;
230
+ }
231
+ if (currentStep === 2) {
232
+ if (keyInput.value.trim()) validateKeyBtn.click();
233
+ else nextStep(); // skip
234
+ return;
235
+ }
236
+
237
+ if (currentStep === '4b') return; // Step 4b has its own Store/Skip buttons
238
+
239
+ if (currentStep === 4) {
240
+ const generateTab = $('#tab-generate');
241
+ const ideaField = $('#prd-idea');
242
+ if (generateTab.classList.contains('active') && ideaField.value.trim() && !state.generatedPrd) {
243
+ generatePrdBtn.click();
244
+ return;
245
+ }
246
+ }
247
+
248
+ nextStep();
249
+ });
250
+
251
+ // =============================================
252
+ // Step 1: Vault + API Key
253
+ // =============================================
254
+
255
+ const vaultPasswordInput = $('#vault-password');
256
+ const vaultStatus = $('#vault-status');
257
+ const unlockVaultBtn = $('#unlock-vault');
258
+ const toggleVaultBtn = $('#toggle-vault-visibility');
259
+ const vaultCard = $('#vault-card');
260
+ const apikeyCard = $('#apikey-card');
261
+
262
+ const keyInput = $('#anthropic-key');
263
+ const keyStatus = $('#key-status');
264
+ const validateKeyBtn = $('#validate-key');
265
+ const toggleKeyBtn = $('#toggle-key-visibility');
266
+
267
+ toggleVaultBtn.addEventListener('click', () => {
268
+ const isPassword = vaultPasswordInput.type === 'password';
269
+ vaultPasswordInput.type = isPassword ? 'text' : 'password';
270
+ toggleVaultBtn.textContent = isPassword ? 'Hide' : 'Show';
271
+ });
272
+
273
+ toggleKeyBtn.addEventListener('click', () => {
274
+ const isPassword = keyInput.type === 'password';
275
+ keyInput.type = isPassword ? 'text' : 'password';
276
+ toggleKeyBtn.textContent = isPassword ? 'Hide' : 'Show';
277
+ });
278
+
279
+ // Check vault state on load
280
+ fetch('/api/credentials/status')
281
+ .then((r) => r.json())
282
+ .then((data) => {
283
+ if (data.vaultPath) $('#vault-path').textContent = data.vaultPath;
284
+ if (data.vaultExists) {
285
+ $('#vault-password-label').textContent = 'Vault Password';
286
+ vaultPasswordInput.placeholder = 'Enter your vault password';
287
+ $('#vault-hint').textContent = 'Enter the password you used to create this vault.';
288
+ }
289
+ if (data.unlocked && data.anthropic) {
290
+ state.anthropicKeyStored = true;
291
+ vaultCard.classList.add('hidden');
292
+ apikeyCard.classList.remove('hidden');
293
+ showStatus(keyStatus, 'success', 'API key already stored in vault');
294
+ keyInput.placeholder = 'Key already stored — enter a new one to replace';
295
+ }
296
+ })
297
+ .catch(() => {});
298
+
299
+ unlockVaultBtn.addEventListener('click', async () => {
300
+ const password = vaultPasswordInput.value;
301
+ if (!password) { showStatus(vaultStatus, 'error', 'Please enter a password'); return; }
302
+ if (password.length < 8) { showStatus(vaultStatus, 'error', 'Password must be at least 8 characters'); return; }
303
+
304
+ showStatus(vaultStatus, 'loading', 'Unlocking...');
305
+ unlockVaultBtn.disabled = true;
306
+
307
+ try {
308
+ const res = await fetch('/api/credentials/unlock', {
309
+ method: 'POST',
310
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
311
+ body: JSON.stringify({ password }),
312
+ });
313
+ const data = await res.json();
314
+
315
+ if (res.ok && data.unlocked) {
316
+ // ENCHANT-R2-003 + CROSS-R4-017: Forge-lit pulse on vault unlock (force replay)
317
+ const vc = document.getElementById('vault-card');
318
+ if (vc) { vc.classList.remove('forge-lit'); void vc.offsetWidth; vc.classList.add('forge-lit'); }
319
+ if (data.anthropic) {
320
+ state.anthropicKeyStored = true;
321
+ showStatus(vaultStatus, 'success', 'Vault unlocked — API key found');
322
+ setTimeout(() => showStep(3), 900);
323
+ } else {
324
+ showStatus(vaultStatus, 'success', 'Vault unlocked');
325
+ setTimeout(() => showStep(2), 900);
326
+ }
327
+ } else {
328
+ showStatus(vaultStatus, 'error', data.error || 'Failed to unlock');
329
+ }
330
+ } catch (err) {
331
+ showStatus(vaultStatus, 'error', 'Connection error: ' + err.message);
332
+ } finally {
333
+ unlockVaultBtn.disabled = false;
334
+ }
335
+ });
336
+
337
+ validateKeyBtn.addEventListener('click', async () => {
338
+ const apiKey = keyInput.value.trim();
339
+ if (!apiKey) { showStatus(keyStatus, 'error', 'Please enter your API key'); return; }
340
+
341
+ showStatus(keyStatus, 'loading', 'Validating...');
342
+ validateKeyBtn.disabled = true;
343
+
344
+ try {
345
+ const res = await fetch('/api/credentials/anthropic', {
346
+ method: 'POST',
347
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
348
+ body: JSON.stringify({ apiKey }),
349
+ });
350
+ const data = await res.json();
351
+
352
+ if (res.ok && data.stored) {
353
+ showStatus(keyStatus, 'success', 'Key validated and stored in vault');
354
+ state.anthropicKeyStored = true;
355
+ setTimeout(() => showStep(3), 600);
356
+ } else {
357
+ showStatus(keyStatus, 'error', data.error || 'Validation failed');
358
+ }
359
+ } catch (err) {
360
+ showStatus(keyStatus, 'error', 'Connection error: ' + err.message);
361
+ } finally {
362
+ validateKeyBtn.disabled = false;
363
+ }
364
+ });
365
+
366
+ // =============================================
367
+ // Step 2: API Key + Skip
368
+ // =============================================
369
+
370
+ let providersLoaded = false;
371
+
372
+ /** Make a div act as a keyboard-accessible button */
373
+ function activateOnKeyboard(el, handler) {
374
+ el.addEventListener('click', handler);
375
+ el.addEventListener('keydown', (e) => {
376
+ if (e.key === 'Enter' || e.key === ' ') {
377
+ e.preventDefault();
378
+ handler(e);
379
+ }
380
+ });
381
+ }
382
+
383
+ // Skip API key button
384
+ $('#skip-key')?.addEventListener('click', () => {
385
+ showStep(3);
386
+ });
387
+
388
+ async function loadCloudProviders() {
389
+ if (providersLoaded) return;
390
+ providersLoaded = true;
391
+
392
+ try {
393
+ const [provRes, statusRes] = await Promise.all([
394
+ fetch('/api/cloud/providers').then((r) => r.json()),
395
+ fetch('/api/cloud/status').then((r) => r.json()),
396
+ ]);
397
+
398
+ const container = $('#cloud-providers-list');
399
+ container.innerHTML = '';
400
+
401
+ for (const provider of provRes.providers) {
402
+ const configured = statusRes.status[provider.id] || false;
403
+ state.cloudProviders[provider.id] = configured;
404
+
405
+ const card = document.createElement('div');
406
+ card.className = 'provider-card' + (configured ? ' configured' : '');
407
+ card.dataset.provider = provider.id;
408
+
409
+ const badge = configured
410
+ ? '<span class="provider-badge connected">Connected</span>'
411
+ : '<span class="provider-badge not-connected">Not connected</span>';
412
+
413
+ let fieldsHtml = '';
414
+ for (const field of provider.fields) {
415
+ fieldsHtml += `
416
+ <div class="field">
417
+ <label for="cloud-${field.key}">${field.label}</label>
418
+ <input type="${field.secret ? 'password' : 'text'}" id="cloud-${field.key}"
419
+ data-field-key="${field.key}" placeholder="${field.placeholder}" autocomplete="off">
420
+ </div>`;
421
+ }
422
+
423
+ card.innerHTML = `
424
+ <div class="provider-header" data-toggle="${provider.id}" role="button" tabindex="0" aria-expanded="false">
425
+ <div class="provider-header-left">
426
+ <span class="provider-chevron">&#9654;</span>
427
+ <div>
428
+ <div class="provider-name">${provider.name} <button class="provider-help-btn" data-help="${provider.id}" type="button" title="How to get credentials">?</button></div>
429
+ <div class="provider-desc">${provider.description}</div>
430
+ </div>
431
+ </div>
432
+ ${badge}
433
+ </div>
434
+ <div class="provider-body hidden" id="body-${provider.id}">
435
+ <div class="provider-help hidden" id="help-${provider.id}">
436
+ <button class="provider-help-close" data-close-help="${provider.id}" type="button" title="Close">&times;</button>
437
+ ${provider.help}
438
+ <a class="help-link" href="${provider.credentialUrl}" target="_blank" rel="noopener">Open ${provider.name} Credentials Page &rarr;</a>
439
+ </div>
440
+ <div class="provider-fields" id="fields-${provider.id}">
441
+ ${fieldsHtml}
442
+ <div class="btn-row">
443
+ <button class="btn btn-primary btn-small" data-validate="${provider.id}" type="button">
444
+ ${configured ? 'Update' : 'Connect'}
445
+ </button>
446
+ ${configured ? `<button class="btn btn-secondary btn-small" data-remove="${provider.id}" type="button">Remove</button>` : ''}
447
+ </div>
448
+ <div class="status-row" id="cloud-status-${provider.id}"></div>
449
+ </div>
450
+ </div>`;
451
+
452
+ container.appendChild(card);
453
+ }
454
+
455
+ // Toggle accordion on header click or keyboard (Enter/Space)
456
+ function toggleAccordion(toggle) {
457
+ const id = toggle.dataset.toggle;
458
+ const card = toggle.closest('.provider-card');
459
+ const body = $(`#body-${id}`);
460
+ const isOpen = !body.classList.contains('hidden');
461
+ body.classList.toggle('hidden');
462
+ card.classList.toggle('open', !isOpen);
463
+ toggle.setAttribute('aria-expanded', isOpen ? 'false' : 'true');
464
+ }
465
+
466
+ container.addEventListener('click', (e) => {
467
+ if (e.target.closest('[data-help]')) return;
468
+ if (e.target.closest('[data-close-help]')) return;
469
+ const toggle = e.target.closest('[data-toggle]');
470
+ if (toggle) toggleAccordion(toggle);
471
+ });
472
+
473
+ container.addEventListener('keydown', (e) => {
474
+ if (e.key !== 'Enter' && e.key !== ' ') return;
475
+ if (e.target.closest('[data-help]')) return;
476
+ if (e.target.closest('[data-close-help]')) return;
477
+ const toggle = e.target.closest('[data-toggle]');
478
+ if (toggle) {
479
+ e.preventDefault();
480
+ toggleAccordion(toggle);
481
+ }
482
+ });
483
+
484
+ // Help button toggles
485
+ container.addEventListener('click', (e) => {
486
+ const helpBtn = e.target.closest('[data-help]');
487
+ if (helpBtn) {
488
+ e.stopPropagation();
489
+ const providerId = helpBtn.dataset.help;
490
+ // Expand the accordion if it's collapsed so the help panel is visible
491
+ const body = $(`#body-${providerId}`);
492
+ const card = helpBtn.closest('.provider-card');
493
+ if (body.classList.contains('hidden')) {
494
+ body.classList.remove('hidden');
495
+ card.classList.add('open');
496
+ }
497
+ $(`#help-${providerId}`).classList.toggle('hidden');
498
+ return;
499
+ }
500
+ const closeBtn = e.target.closest('[data-close-help]');
501
+ if (closeBtn) {
502
+ e.stopPropagation();
503
+ $(`#help-${closeBtn.dataset.closeHelp}`).classList.add('hidden');
504
+ return;
505
+ }
506
+ });
507
+
508
+ // Validate buttons
509
+ container.addEventListener('click', async (e) => {
510
+ const validateBtn = e.target.closest('[data-validate]');
511
+ if (!validateBtn) return;
512
+
513
+ const providerId = validateBtn.dataset.validate;
514
+ const statusEl = $(`#cloud-status-${providerId}`);
515
+ const card = validateBtn.closest('.provider-card');
516
+
517
+ const credentials = {};
518
+ card.querySelectorAll('[data-field-key]').forEach((input) => {
519
+ if (input.value.trim()) credentials[input.dataset.fieldKey] = input.value.trim();
520
+ });
521
+
522
+ const provider = provRes.providers.find((p) => p.id === providerId);
523
+ const missing = provider.fields.filter((f) => !f.optional && !credentials[f.key]);
524
+ if (missing.length > 0) {
525
+ showStatus(statusEl, 'error', `Missing: ${missing.map((f) => f.label).join(', ')}`);
526
+ return;
527
+ }
528
+
529
+ showStatus(statusEl, 'loading', 'Validating...');
530
+ validateBtn.disabled = true;
531
+
532
+ try {
533
+ const res = await fetch('/api/cloud/validate', {
534
+ method: 'POST',
535
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
536
+ body: JSON.stringify({ provider: providerId, credentials }),
537
+ });
538
+ const data = await res.json();
539
+
540
+ if (res.ok && data.stored) {
541
+ const identity = data.identity ? ` (${data.identity})` : '';
542
+ showStatus(statusEl, 'success', `Connected${identity}`);
543
+ state.cloudProviders[providerId] = true;
544
+ card.classList.add('configured');
545
+ card.querySelector('.provider-badge').className = 'provider-badge connected';
546
+ card.querySelector('.provider-badge').textContent = 'Connected';
547
+ } else {
548
+ showStatus(statusEl, 'error', data.error || 'Validation failed');
549
+ }
550
+ } catch (err) {
551
+ showStatus(statusEl, 'error', 'Connection error: ' + err.message);
552
+ } finally {
553
+ validateBtn.disabled = false;
554
+ }
555
+ });
556
+
557
+ // Remove buttons
558
+ container.addEventListener('click', async (e) => {
559
+ const removeBtn = e.target.closest('[data-remove]');
560
+ if (!removeBtn) return;
561
+
562
+ const providerId = removeBtn.dataset.remove;
563
+ const statusEl = $(`#cloud-status-${providerId}`);
564
+ const card = removeBtn.closest('.provider-card');
565
+
566
+ try {
567
+ await fetch('/api/cloud/remove', {
568
+ method: 'POST',
569
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
570
+ body: JSON.stringify({ provider: providerId }),
571
+ });
572
+
573
+ state.cloudProviders[providerId] = false;
574
+ card.classList.remove('configured');
575
+ card.querySelector('.provider-badge').className = 'provider-badge not-connected';
576
+ card.querySelector('.provider-badge').textContent = 'Not connected';
577
+ showStatus(statusEl, 'info', 'Credentials removed');
578
+ removeBtn.remove();
579
+ } catch (err) {
580
+ showStatus(statusEl, 'error', 'Failed to remove: ' + err.message);
581
+ }
582
+ });
583
+
584
+ } catch (err) {
585
+ $('#cloud-providers-list').innerHTML = `<div class="status-row error">Failed to load providers: ${escapeHtml(err.message)}</div>`;
586
+ }
587
+ }
588
+
589
+ // Step 4b needs to re-show nav buttons properly
590
+ const origShowStep = showStep;
591
+ showStep = function (step) {
592
+ if (step === '4b') {
593
+ btnNext.style.display = 'none';
594
+ btnBack.style.display = '';
595
+ btnBack.disabled = false;
596
+ }
597
+ origShowStep(step);
598
+ };
599
+
600
+ // =============================================
601
+ // Step 3: Project Setup
602
+ // =============================================
603
+
604
+ const projectNameInput = $('#project-name');
605
+ const projectDirInput = $('#project-dir');
606
+
607
+ projectNameInput.addEventListener('input', () => {
608
+ const name = projectNameInput.value.trim();
609
+ if (name && !projectDirInput.dataset.manual) {
610
+ const slug = name.toLowerCase().replace(/[^a-z0-9\-_\s]/g, '').replace(/\s+/g, '-');
611
+ fetch('/api/project/defaults')
612
+ .then((r) => r.json())
613
+ .then((data) => {
614
+ if (!projectDirInput.dataset.manual) {
615
+ projectDirInput.value = data.baseDir + '/' + slug;
616
+ }
617
+ })
618
+ .catch(() => {});
619
+ }
620
+ state.projectName = name;
621
+ });
622
+
623
+ projectDirInput.addEventListener('input', () => {
624
+ projectDirInput.dataset.manual = 'true';
625
+ state.projectDir = projectDirInput.value.trim();
626
+ });
627
+
628
+ // Clear individual validation errors on input
629
+ projectNameInput.addEventListener('input', () => {
630
+ const err = $('#project-name-error');
631
+ if (err && !err.classList.contains('hidden')) {
632
+ projectNameInput.style.borderColor = '';
633
+ projectNameInput.removeAttribute('aria-invalid');
634
+ err.textContent = '';
635
+ err.classList.add('hidden');
636
+ }
637
+ });
638
+ projectDirInput.addEventListener('input', () => {
639
+ const err = $('#project-dir-error');
640
+ if (err && !err.classList.contains('hidden')) {
641
+ projectDirInput.style.borderColor = '';
642
+ projectDirInput.removeAttribute('aria-invalid');
643
+ err.textContent = '';
644
+ err.classList.add('hidden');
645
+ }
646
+ });
647
+
648
+ // =============================================
649
+ // Step 4: PRD
650
+ // =============================================
651
+
652
+ function activatePrdTab(tab) {
653
+ const allTabs = Array.from($$('.tab'));
654
+ allTabs.forEach((t) => {
655
+ t.classList.remove('active');
656
+ t.setAttribute('aria-selected', 'false');
657
+ t.setAttribute('tabindex', '-1');
658
+ });
659
+ $$('.tab-panel').forEach((p) => p.classList.remove('active'));
660
+ tab.classList.add('active');
661
+ tab.setAttribute('aria-selected', 'true');
662
+ tab.setAttribute('tabindex', '0');
663
+ tab.focus();
664
+ const panel = $(`#tab-${tab.dataset.tab}`);
665
+ if (panel) panel.classList.add('active');
666
+ state.prdMode = tab.dataset.tab;
667
+ }
668
+
669
+ $$('.tab').forEach((tab, i) => {
670
+ // Set initial tabindex: only active tab is tabbable
671
+ tab.setAttribute('tabindex', i === 0 ? '0' : '-1');
672
+ tab.addEventListener('click', () => { activatePrdTab(tab); });
673
+ });
674
+
675
+ // Keyboard navigation for PRD tabs (WAI-ARIA tab pattern)
676
+ const prdTablist = $('[role="tablist"]');
677
+ if (prdTablist) {
678
+ prdTablist.addEventListener('keydown', (e) => {
679
+ const tabArray = Array.from($$('.tab'));
680
+ const idx = tabArray.indexOf(document.activeElement);
681
+ if (idx === -1) return;
682
+
683
+ let newIdx = -1;
684
+ if (e.key === 'ArrowRight') {
685
+ newIdx = (idx + 1) % tabArray.length;
686
+ } else if (e.key === 'ArrowLeft') {
687
+ newIdx = (idx - 1 + tabArray.length) % tabArray.length;
688
+ } else if (e.key === 'Home') {
689
+ newIdx = 0;
690
+ } else if (e.key === 'End') {
691
+ newIdx = tabArray.length - 1;
692
+ }
693
+
694
+ if (newIdx >= 0) {
695
+ e.preventDefault();
696
+ activatePrdTab(tabArray[newIdx]);
697
+ }
698
+ });
699
+ }
700
+
701
+ const validatePrdBtn = $('#validate-prd');
702
+ const prdStatus = $('#prd-status');
703
+
704
+ validatePrdBtn.addEventListener('click', async () => {
705
+ const content = $('#prd-paste').value.trim();
706
+ if (!content) { showStatus(prdStatus, 'error', 'Paste your PRD content first'); return; }
707
+
708
+ try {
709
+ const res = await fetch('/api/prd/validate', {
710
+ method: 'POST',
711
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
712
+ body: JSON.stringify({ content }),
713
+ });
714
+ const data = await res.json();
715
+
716
+ if (data.valid) {
717
+ showStatus(prdStatus, 'success', `Valid frontmatter: ${data.frontmatter.name || 'unnamed'} (${data.frontmatter.type || 'no type'})`);
718
+ } else {
719
+ showStatus(prdStatus, 'error', data.errors.join(', '));
720
+ }
721
+ } catch (err) {
722
+ showStatus(prdStatus, 'error', 'Validation error: ' + err.message);
723
+ }
724
+ });
725
+
726
+ // PRD file upload + drag-and-drop
727
+ const dropzone = $('#prd-dropzone');
728
+ const fileInput = $('#prd-file-input');
729
+ const prdTextarea = $('#prd-paste');
730
+
731
+ if (dropzone && fileInput) {
732
+ function handlePrdFile(file) {
733
+ const reader = new FileReader();
734
+ reader.onload = function (e) {
735
+ prdTextarea.value = e.target.result;
736
+ showStatus(prdStatus, 'success', 'Loaded: ' + file.name + ' (' + file.size + ' bytes)');
737
+ dropzone.style.borderColor = 'var(--accent, #5b5bf7)';
738
+ setTimeout(function () { dropzone.style.borderColor = ''; }, 2000);
739
+ };
740
+ reader.onerror = function () {
741
+ showStatus(prdStatus, 'error', 'Failed to read file');
742
+ };
743
+ reader.readAsText(file);
744
+ }
745
+
746
+ dropzone.addEventListener('click', function () { fileInput.click(); });
747
+ dropzone.addEventListener('keydown', function (e) { if (e.key === 'Enter' || e.key === ' ') fileInput.click(); });
748
+
749
+ fileInput.addEventListener('change', function () {
750
+ if (fileInput.files && fileInput.files[0]) handlePrdFile(fileInput.files[0]);
751
+ });
752
+
753
+ dropzone.addEventListener('dragover', function (e) {
754
+ e.preventDefault();
755
+ dropzone.style.borderColor = 'var(--accent, #5b5bf7)';
756
+ dropzone.style.background = 'rgba(91, 91, 247, 0.05)';
757
+ });
758
+ dropzone.addEventListener('dragleave', function () {
759
+ dropzone.style.borderColor = '';
760
+ dropzone.style.background = '';
761
+ });
762
+ dropzone.addEventListener('drop', function (e) {
763
+ e.preventDefault();
764
+ dropzone.style.borderColor = '';
765
+ dropzone.style.background = '';
766
+ if (e.dataTransfer.files && e.dataTransfer.files[0]) handlePrdFile(e.dataTransfer.files[0]);
767
+ });
768
+ }
769
+
770
+ let cachedPrompt = null;
771
+ $('#copy-prd-prompt').addEventListener('click', async () => {
772
+ const promptCopyStatus = $('#prompt-copy-status');
773
+ try {
774
+ if (!cachedPrompt) {
775
+ showStatus(promptCopyStatus, 'loading', 'Loading prompt...');
776
+ const res = await fetch('/api/prd/prompt');
777
+ const data = await res.json();
778
+ if (!res.ok) throw new Error(data.error || 'Failed to load prompt');
779
+ cachedPrompt = data.prompt;
780
+ }
781
+ await copyToClipboard(cachedPrompt);
782
+ showStatus(promptCopyStatus, 'success', 'Prompt copied — paste it into your AI of choice, add your idea, then paste the result above');
783
+ } catch (err) {
784
+ showStatus(promptCopyStatus, 'error', 'Failed to copy: ' + err.message);
785
+ }
786
+ });
787
+
788
+ const generatePrdBtn = $('#generate-prd');
789
+ const generationOutput = $('#generation-output');
790
+ const generatedContent = $('#generated-prd-content');
791
+
792
+ generatePrdBtn.addEventListener('click', async () => {
793
+ const idea = $('#prd-idea').value.trim();
794
+ if (!idea) {
795
+ showStatus($('#generate-status'), 'error', 'Describe your idea first');
796
+ return;
797
+ }
798
+ $('#generate-status').className = 'status-row';
799
+
800
+ generatePrdBtn.disabled = true;
801
+ generatePrdBtn.textContent = 'Generating...';
802
+ generationOutput.classList.remove('hidden');
803
+ generatedContent.textContent = '';
804
+ // ENCHANT-R2-006: Add streaming cursor
805
+ generatedContent.classList.add('streaming');
806
+ state.generatedPrd = '';
807
+
808
+ try {
809
+ const res = await fetch('/api/prd/generate', {
810
+ method: 'POST',
811
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
812
+ body: JSON.stringify({
813
+ idea,
814
+ name: state.projectName,
815
+ framework: $('#pref-framework').value,
816
+ database: $('#pref-database').value,
817
+ deploy: $('#pref-deploy').value,
818
+ }),
819
+ });
820
+
821
+ const reader = res.body.getReader();
822
+ const decoder = new TextDecoder();
823
+ let buffer = '';
824
+ let wasTruncated = false;
825
+
826
+ while (true) {
827
+ const { done, value } = await reader.read();
828
+ if (done) break;
829
+
830
+ buffer += decoder.decode(value, { stream: true });
831
+ const lines = buffer.split('\n');
832
+ buffer = lines.pop() || '';
833
+
834
+ for (const line of lines) {
835
+ if (!line.startsWith('data: ')) continue;
836
+ const data = line.slice(6);
837
+ if (data === '[DONE]') continue;
838
+
839
+ try {
840
+ const parsed = JSON.parse(data);
841
+ if (parsed.text) {
842
+ state.generatedPrd += parsed.text;
843
+ generatedContent.textContent = state.generatedPrd;
844
+ generatedContent.scrollTop = generatedContent.scrollHeight;
845
+ }
846
+ if (parsed.truncated) {
847
+ wasTruncated = true;
848
+ }
849
+ if (parsed.error) {
850
+ generatedContent.textContent += '\n\nError: ' + parsed.error;
851
+ }
852
+ } catch { /* skip */ }
853
+ }
854
+ }
855
+
856
+ if (wasTruncated) {
857
+ showStatus($('#generate-status'), 'error',
858
+ 'PRD was truncated — the output hit the model token limit. Try a shorter idea description or generate again with fewer details.');
859
+ }
860
+ } catch (err) {
861
+ generatedContent.textContent += '\n\nConnection error: ' + err.message;
862
+ } finally {
863
+ // ENCHANT-R2-006: Remove streaming cursor when done
864
+ generatedContent.classList.remove('streaming');
865
+ generatePrdBtn.disabled = false;
866
+ generatePrdBtn.textContent = 'Generate PRD with Claude';
867
+ }
868
+ });
869
+
870
+ $('#copy-generated').addEventListener('click', () => {
871
+ if (!state.generatedPrd) return;
872
+ copyToClipboard(state.generatedPrd).then(() => {
873
+ $('#copy-generated').textContent = 'Copied!';
874
+ setTimeout(() => { $('#copy-generated').textContent = 'Copy'; }, 2000);
875
+ });
876
+ });
877
+
878
+ // =============================================
879
+ // Step 4b: PRD-Driven Credentials
880
+ // =============================================
881
+
882
+ async function loadEnvRequirements(prdText) {
883
+ try {
884
+ const res = await fetch('/api/prd/env-requirements', {
885
+ method: 'POST',
886
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
887
+ body: JSON.stringify({ content: prdText }),
888
+ });
889
+ const data = await res.json();
890
+ return data.groups || [];
891
+ } catch {
892
+ return [];
893
+ }
894
+ }
895
+
896
+ function renderEnvCredentials(groups) {
897
+ const container = $('#env-credentials-list');
898
+ const emptyEl = $('#env-credentials-empty');
899
+ container.innerHTML = '';
900
+
901
+ if (groups.length === 0) {
902
+ emptyEl.classList.remove('hidden');
903
+ return;
904
+ }
905
+ emptyEl.classList.add('hidden');
906
+
907
+ for (const group of groups) {
908
+ const section = document.createElement('div');
909
+ section.className = 'card';
910
+ section.style.marginBottom = '12px';
911
+
912
+ let fieldsHtml = '';
913
+ for (const field of group.fields) {
914
+ fieldsHtml += `
915
+ <div class="field">
916
+ <label for="env-${field.key}">${escapeHtml(field.label)}</label>
917
+ <input type="${field.secret ? 'password' : 'text'}" id="env-${field.key}"
918
+ data-env-key="${field.key}" placeholder="${escapeHtml(field.placeholder)}" autocomplete="off">
919
+ </div>`;
920
+ }
921
+
922
+ section.innerHTML = `
923
+ <h3>${escapeHtml(group.name)}</h3>
924
+ ${fieldsHtml}`;
925
+ container.appendChild(section);
926
+ }
927
+ }
928
+
929
+ // Store All button
930
+ $('#store-env-credentials')?.addEventListener('click', async () => {
931
+ const statusEl = $('#env-store-status');
932
+ const inputs = $$('[data-env-key]');
933
+ const credentials = {};
934
+ let count = 0;
935
+
936
+ inputs.forEach((input) => {
937
+ if (input.value.trim()) {
938
+ credentials[input.dataset.envKey] = input.value.trim();
939
+ count++;
940
+ }
941
+ });
942
+
943
+ if (count === 0) {
944
+ showStatus(statusEl, 'info', 'No credentials entered — skipping.');
945
+ proceedFromEnvStep();
946
+ return;
947
+ }
948
+
949
+ showStatus(statusEl, 'loading', `Storing ${count} credentials in vault...`);
950
+ try {
951
+ const res = await fetch('/api/credentials/env-batch', {
952
+ method: 'POST',
953
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
954
+ body: JSON.stringify({ credentials }),
955
+ });
956
+ const data = await res.json();
957
+
958
+ if (res.ok && data.stored) {
959
+ state.envCredentials = credentials;
960
+ showStatus(statusEl, 'success', `${count} credentials stored in vault`);
961
+ setTimeout(proceedFromEnvStep, 600);
962
+ } else {
963
+ showStatus(statusEl, 'error', data.error || 'Failed to store credentials');
964
+ }
965
+ } catch (err) {
966
+ showStatus(statusEl, 'error', 'Connection error: ' + err.message);
967
+ }
968
+ });
969
+
970
+ // Skip button
971
+ $('#skip-env-credentials')?.addEventListener('click', () => {
972
+ proceedFromEnvStep();
973
+ });
974
+
975
+ function proceedFromEnvStep() {
976
+ Promise.all([loadDeployTargets(), loadCloudProviders()]).then(() => showStep(5));
977
+ }
978
+
979
+ // =============================================
980
+ // Step 5: Deploy Target
981
+ // =============================================
982
+
983
+ async function loadDeployTargets() {
984
+ try {
985
+ const res = await fetch('/api/cloud/deploy-targets');
986
+ const data = await res.json();
987
+
988
+ const container = $('#deploy-targets-list');
989
+ const noteEl = $('#deploy-note');
990
+ container.innerHTML = '';
991
+
992
+ if (!state.deployTarget) {
993
+ state.deployTarget = 'docker';
994
+ }
995
+
996
+ const availableTargets = data.targets.filter((t) => t.available);
997
+ const onlyDocker = availableTargets.length === 1 && availableTargets[0].id === 'docker';
998
+
999
+ if (onlyDocker) {
1000
+ state.deployTarget = 'docker';
1001
+ noteEl.innerHTML = '<div class="status-row info" style="margin-top: 16px;">No cloud providers configured, so <strong>Docker (local)</strong> is preselected. You can add cloud credentials later by re-running the wizard.</div>';
1002
+ } else {
1003
+ noteEl.innerHTML = '';
1004
+ }
1005
+
1006
+ for (const target of data.targets) {
1007
+ const card = document.createElement('div');
1008
+ card.className = 'deploy-card' + (target.available ? '' : ' unavailable');
1009
+ if (state.deployTarget === target.id) card.classList.add('selected');
1010
+ card.dataset.target = target.id;
1011
+ card.setAttribute('role', 'radio');
1012
+ card.setAttribute('aria-checked', state.deployTarget === target.id ? 'true' : 'false');
1013
+ if (target.available) {
1014
+ card.tabIndex = 0;
1015
+ } else {
1016
+ card.setAttribute('aria-disabled', 'true');
1017
+ }
1018
+
1019
+ const badge = target.available
1020
+ ? '<span class="provider-badge connected">Ready</span>'
1021
+ : `<span class="provider-badge not-connected">Needs ${target.provider || 'setup'}</span>`;
1022
+
1023
+ card.innerHTML = `
1024
+ <div class="deploy-card-header">
1025
+ <span class="deploy-card-name">${target.name}</span>
1026
+ ${badge}
1027
+ </div>
1028
+ <div class="deploy-card-desc">${target.description}</div>`;
1029
+
1030
+ const selectTarget = () => {
1031
+ if (!target.available) return;
1032
+ $$('.deploy-card').forEach((c) => {
1033
+ c.classList.remove('selected');
1034
+ c.setAttribute('aria-checked', 'false');
1035
+ });
1036
+ card.classList.add('selected');
1037
+ card.setAttribute('aria-checked', 'true');
1038
+ state.deployTarget = target.id;
1039
+ };
1040
+ activateOnKeyboard(card, selectTarget);
1041
+
1042
+ container.appendChild(card);
1043
+ }
1044
+ } catch (err) {
1045
+ $('#deploy-targets-list').innerHTML = `<div class="status-row error">Failed to load targets: ${escapeHtml(err.message)}</div>`;
1046
+ }
1047
+ }
1048
+
1049
+ // =============================================
1050
+ // Step 6: Review
1051
+ // =============================================
1052
+
1053
+ function populateReview() {
1054
+ $('#review-name').textContent = state.projectName;
1055
+ $('#review-dir').textContent = state.projectDir;
1056
+ $('#review-desc').textContent = state.projectDesc || '(not set)';
1057
+ $('#review-domain').textContent = state.projectDomain || '(not set)';
1058
+ $('#review-hostname').textContent = state.projectHostname || '(not set)';
1059
+ const deployNames = { vps: 'VPS (AWS EC2)', vercel: 'Vercel', railway: 'Railway', cloudflare: 'Cloudflare Workers/Pages', static: 'Static (S3 + CloudFront)', docker: 'Docker (local)' };
1060
+ $('#review-deploy').textContent = deployNames[state.deployTarget] || state.deployTarget || 'Docker (local)';
1061
+
1062
+ if (state.prdMode === 'paste' && state.prdContent) {
1063
+ $('#review-prd').textContent = 'Custom PRD (pasted)';
1064
+ } else if (state.prdMode === 'generate' && state.generatedPrd) {
1065
+ $('#review-prd').textContent = 'Generated by Claude';
1066
+ } else {
1067
+ $('#review-prd').textContent = 'Default template (edit later)';
1068
+ }
1069
+
1070
+ // Show env credentials count if any were stored
1071
+ const envCount = Object.keys(state.envCredentials).length;
1072
+ const envRow = $('#review-env-credentials');
1073
+ if (envRow) {
1074
+ envRow.textContent = envCount > 0 ? `${envCount} keys stored in vault` : 'None (add to .env later)';
1075
+ }
1076
+ }
1077
+
1078
+ // =============================================
1079
+ // Step 7: Create
1080
+ // =============================================
1081
+
1082
+ async function createProject() {
1083
+ const creatingState = $('#creating-state');
1084
+ const statusText = $('#create-status-text');
1085
+
1086
+ creatingState.classList.remove('hidden');
1087
+
1088
+ try {
1089
+ statusText.textContent = 'Creating project files...';
1090
+ let prd = state.prdContent || undefined;
1091
+
1092
+ const res = await fetch('/api/project/create', {
1093
+ method: 'POST',
1094
+ headers: { 'Content-Type': 'application/json', 'X-VoidForge-Request': '1' },
1095
+ body: JSON.stringify({
1096
+ name: state.projectName,
1097
+ directory: state.projectDir,
1098
+ description: state.projectDesc || undefined,
1099
+ domain: state.projectDomain || undefined,
1100
+ hostname: state.projectHostname || undefined,
1101
+ deploy: state.deployTarget || undefined,
1102
+ prd,
1103
+ }),
1104
+ });
1105
+
1106
+ const data = await res.json();
1107
+
1108
+ if (res.ok && data.created) {
1109
+ state.createdDir = data.directory;
1110
+ // Show done state
1111
+ creatingState.classList.add('hidden');
1112
+ $('#done-state').classList.remove('hidden');
1113
+ // Switch section label to done heading
1114
+ const step7Section = $('#step-7');
1115
+ if (step7Section) step7Section.setAttribute('aria-labelledby', 'step-7-heading');
1116
+ $('#done-details').innerHTML = `
1117
+ <p><strong>${escapeHtml(state.projectName)}</strong></p>
1118
+ <p style="color: var(--text-dim); font-family: var(--mono); font-size: 13px;">${escapeHtml(data.directory)}</p>
1119
+ <p style="color: var(--text-dim); margin-top: 8px;">${data.files.length} files created</p>
1120
+ `;
1121
+ setTimeout(() => { const h = $('#step-7-heading'); if (h) h.focus(); }, 100);
1122
+ } else {
1123
+ showCreateError('Error: ' + (data.error || 'Unknown error'));
1124
+ }
1125
+ } catch (err) {
1126
+ showCreateError('Error: ' + err.message);
1127
+ }
1128
+ }
1129
+
1130
+ function showCreateError(message) {
1131
+ const creatingState = $('#creating-state');
1132
+ const statusText = $('#create-status-text');
1133
+ creatingState.querySelector('.spinner')?.classList.add('hidden');
1134
+ statusText.textContent = message;
1135
+ statusText.style.color = 'var(--error)';
1136
+ // Show back + retry buttons so the user isn't trapped
1137
+ btnBack.style.display = '';
1138
+ btnBack.disabled = false;
1139
+ btnNext.style.display = '';
1140
+ btnNext.textContent = 'Retry';
1141
+ }
1142
+
1143
+ $('#open-tower')?.addEventListener('click', () => {
1144
+ if (state.createdDir) {
1145
+ const name = encodeURIComponent(state.projectName);
1146
+ const dir = encodeURIComponent(state.createdDir);
1147
+ window.location.href = `/tower.html?name=${name}&dir=${dir}`;
1148
+ }
1149
+ });
1150
+
1151
+ $('#open-terminal')?.addEventListener('click', () => {
1152
+ if (state.createdDir) {
1153
+ const cmd = `cd "${state.createdDir}" && claude`;
1154
+ copyToClipboard(cmd).then(() => {
1155
+ alert(`Copied to clipboard:\n\n${cmd}\n\nPaste this in your terminal.`);
1156
+ }).catch(() => {
1157
+ alert(`Run this in your terminal:\n\n${cmd}`);
1158
+ });
1159
+ }
1160
+ });
1161
+
1162
+ $('#open-finder')?.addEventListener('click', () => {
1163
+ if (state.createdDir) {
1164
+ copyToClipboard(state.createdDir).then(() => {
1165
+ alert(`Path copied. Open Finder and press Cmd+Shift+G, then paste:\n\n${state.createdDir}`);
1166
+ });
1167
+ }
1168
+ });
1169
+
1170
+ // Provisioning moved to Haku (deploy wizard) — launch with `npm run deploy`
1171
+
1172
+ // =============================================
1173
+ // Step 5: Operations Menu — Card Expand/Collapse
1174
+ // =============================================
1175
+
1176
+ $$('.ops-card-header').forEach((header) => {
1177
+ function toggle() {
1178
+ const card = header.closest('.ops-card');
1179
+ const body = card.querySelector('.ops-card-body');
1180
+ const isOpen = !body.classList.contains('hidden');
1181
+ body.classList.toggle('hidden');
1182
+ card.classList.toggle('expanded', !isOpen);
1183
+ header.setAttribute('aria-expanded', isOpen ? 'false' : 'true');
1184
+ if (!isOpen) {
1185
+ const firstInput = body.querySelector('input, select');
1186
+ if (firstInput) setTimeout(() => firstInput.focus(), 100);
1187
+ }
1188
+ }
1189
+ header.addEventListener('click', toggle);
1190
+ header.addEventListener('keydown', (e) => {
1191
+ if (e.key === 'Enter' || e.key === ' ') { e.preventDefault(); toggle(); }
1192
+ });
1193
+ });
1194
+
1195
+ // Skip All button
1196
+ $('#ops-skip-all')?.addEventListener('click', () => {
1197
+ populateReview();
1198
+ showStep(6);
1199
+ });
1200
+
1201
+ // Update header when project name is typed (Éowyn's enchantment)
1202
+ projectNameInput.addEventListener('input', () => {
1203
+ const name = projectNameInput.value.trim();
1204
+ const logo = $('.logo');
1205
+ if (name) {
1206
+ logo.textContent = 'Gandalf — ' + name;
1207
+ } else {
1208
+ logo.textContent = 'Gandalf — VoidForge Setup';
1209
+ }
1210
+ });
1211
+
1212
+ // =============================================
1213
+ // Utilities
1214
+ // =============================================
1215
+
1216
+ function showValidationErrors() {
1217
+ if (currentStep === 3) {
1218
+ const nameInput = $('#project-name');
1219
+ const dirInput = $('#project-dir');
1220
+ const nameError = $('#project-name-error');
1221
+ const dirError = $('#project-dir-error');
1222
+ if (!state.projectName) {
1223
+ nameInput.style.borderColor = 'var(--error)';
1224
+ nameInput.setAttribute('aria-invalid', 'true');
1225
+ if (nameError) { nameError.textContent = 'Project name is required'; nameError.classList.remove('hidden'); }
1226
+ }
1227
+ if (!state.projectDir) {
1228
+ dirInput.style.borderColor = 'var(--error)';
1229
+ dirInput.setAttribute('aria-invalid', 'true');
1230
+ if (dirError) { dirError.textContent = 'Project directory is required'; dirError.classList.remove('hidden'); }
1231
+ }
1232
+ }
1233
+ }
1234
+
1235
+ function clearValidationErrors() {
1236
+ const nameInput = $('#project-name');
1237
+ const dirInput = $('#project-dir');
1238
+ const nameError = $('#project-name-error');
1239
+ const dirError = $('#project-dir-error');
1240
+ if (nameInput) { nameInput.style.borderColor = ''; nameInput.removeAttribute('aria-invalid'); }
1241
+ if (dirInput) { dirInput.style.borderColor = ''; dirInput.removeAttribute('aria-invalid'); }
1242
+ if (nameError) { nameError.textContent = ''; nameError.classList.add('hidden'); }
1243
+ if (dirError) { dirError.textContent = ''; dirError.classList.add('hidden'); }
1244
+ }
1245
+
1246
+ function escapeHtml(str) {
1247
+ const div = document.createElement('div');
1248
+ div.appendChild(document.createTextNode(str));
1249
+ return div.innerHTML;
1250
+ }
1251
+
1252
+ function showStatus(el, type, message) {
1253
+ el.className = 'status-row ' + type;
1254
+ el.textContent = message;
1255
+ }
1256
+
1257
+ function copyToClipboard(text) {
1258
+ if (navigator.clipboard && navigator.clipboard.writeText) {
1259
+ return navigator.clipboard.writeText(text);
1260
+ }
1261
+ return new Promise((resolve, reject) => {
1262
+ const ta = document.createElement('textarea');
1263
+ ta.value = text;
1264
+ ta.style.position = 'fixed';
1265
+ ta.style.opacity = '0';
1266
+ document.body.appendChild(ta);
1267
+ ta.select();
1268
+ try {
1269
+ document.execCommand('copy');
1270
+ resolve();
1271
+ } catch (e) {
1272
+ reject(e);
1273
+ } finally {
1274
+ document.body.removeChild(ta);
1275
+ }
1276
+ });
1277
+ }
1278
+
1279
+ // Blueprint detection handlers
1280
+ const btnUseBlueprint = $('#btn-use-blueprint');
1281
+ const btnStartFresh = $('#btn-start-fresh');
1282
+
1283
+ if (btnUseBlueprint) {
1284
+ btnUseBlueprint.addEventListener('click', async () => {
1285
+ // User chose blueprint path — validate and show results
1286
+ const blueprintBanner = $('#blueprint-detection');
1287
+
1288
+ try {
1289
+ const res = await fetch('/api/blueprint/validate');
1290
+ const data = await res.json();
1291
+
1292
+ if (data.valid) {
1293
+ // Show validation success in the banner itself
1294
+ if (blueprintBanner) {
1295
+ blueprintBanner.innerHTML = `
1296
+ <h2 style="margin: 0 0 0.5rem 0; color: var(--success, #4ade80);">Blueprint Validated</h2>
1297
+ <p style="margin: 0 0 0.5rem 0;">${escapeHtml(data.summary || 'PRD is valid and ready to build.')}</p>
1298
+ <p style="margin: 0 0 1rem 0; opacity: 0.8;">Run <code>/blueprint</code> in Claude Code to provision and start building.</p>
1299
+ <button type="button" onclick="this.closest('#blueprint-detection').classList.add('hidden'); document.querySelector('#step-4').classList.remove('hidden');"
1300
+ style="padding: 0.5rem 1rem; background: transparent; color: var(--text, #ccc); border: 1px solid var(--border, #333); border-radius: 6px; cursor: pointer;">
1301
+ Continue to wizard instead
1302
+ </button>
1303
+ `;
1304
+ }
1305
+ } else {
1306
+ // Show errors but don't dead-end — let user fix or continue
1307
+ if (blueprintBanner) {
1308
+ const errors = (data.frontmatterErrors || ['Unknown validation error']).map(e => escapeHtml(e)).join('<br>');
1309
+ blueprintBanner.innerHTML = `
1310
+ <h2 style="margin: 0 0 0.5rem 0; color: var(--warning, #fbbf24);">Blueprint Has Issues</h2>
1311
+ <p style="margin: 0 0 0.5rem 0; font-size: 0.9rem;">${errors}</p>
1312
+ <p style="margin: 0 0 1rem 0; opacity: 0.8;">Fix these in docs/PRD.md, or continue with the wizard interview.</p>
1313
+ <button type="button" onclick="this.closest('#blueprint-detection').classList.add('hidden'); document.querySelector('#step-4').classList.remove('hidden');"
1314
+ style="padding: 0.5rem 1rem; background: transparent; color: var(--text, #ccc); border: 1px solid var(--border, #333); border-radius: 6px; cursor: pointer;">
1315
+ Continue to wizard
1316
+ </button>
1317
+ `;
1318
+ }
1319
+ }
1320
+ } catch {
1321
+ // API unavailable — still show useful guidance, never dead-end
1322
+ if (blueprintBanner) {
1323
+ blueprintBanner.innerHTML = `
1324
+ <h2 style="margin: 0 0 0.5rem 0; color: var(--accent, #5b5bf7);">Blueprint Path Available</h2>
1325
+ <p style="margin: 0 0 1rem 0;">Run <code>/blueprint</code> in Claude Code to validate your PRD and provision infrastructure.</p>
1326
+ <button type="button" onclick="this.closest('#blueprint-detection').classList.add('hidden'); document.querySelector('#step-4').classList.remove('hidden');"
1327
+ style="padding: 0.5rem 1rem; background: transparent; color: var(--text, #ccc); border: 1px solid var(--border, #333); border-radius: 6px; cursor: pointer;">
1328
+ Continue to wizard
1329
+ </button>
1330
+ `;
1331
+ }
1332
+ }
1333
+ });
1334
+ }
1335
+
1336
+ if (btnStartFresh) {
1337
+ btnStartFresh.addEventListener('click', () => {
1338
+ const blueprintBanner = $('#blueprint-detection');
1339
+ if (blueprintBanner) blueprintBanner.classList.add('hidden');
1340
+ // Continue to Step 4 (PRD interview) normally
1341
+ showStep(4);
1342
+ });
1343
+ }
1344
+
1345
+ // Dismiss button — same behavior as "Start fresh"
1346
+ const btnDismissBlueprint = $('#btn-dismiss-blueprint');
1347
+ if (btnDismissBlueprint) {
1348
+ btnDismissBlueprint.addEventListener('click', () => {
1349
+ const blueprintBanner = $('#blueprint-detection');
1350
+ if (blueprintBanner) blueprintBanner.classList.add('hidden');
1351
+ showStep(4);
1352
+ });
1353
+ }
1354
+
1355
+ // Init
1356
+ showStep(1);
1357
+ })();