groundwork-method 0.0.1 → 0.11.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +823 -0
- package/LICENSE +21 -0
- package/README.md +44 -29
- package/bin/groundwork.js +1723 -0
- package/dist/src/generators/add-capability/generator.d.ts +8 -0
- package/dist/src/generators/add-capability/generator.js +60 -0
- package/dist/src/generators/add-capability/generator.js.map +1 -0
- package/dist/src/generators/cli-app/generator.d.ts +9 -0
- package/dist/src/generators/cli-app/generator.js +140 -0
- package/dist/src/generators/cli-app/generator.js.map +1 -0
- package/dist/src/generators/docs-site/generator.d.ts +5 -0
- package/dist/src/generators/docs-site/generator.js +441 -0
- package/dist/src/generators/docs-site/generator.js.map +1 -0
- package/dist/src/generators/electron-app/generator.d.ts +6 -0
- package/dist/src/generators/electron-app/generator.js +261 -0
- package/dist/src/generators/electron-app/generator.js.map +1 -0
- package/dist/src/generators/flutter-app/generator.d.ts +6 -0
- package/dist/src/generators/flutter-app/generator.js +314 -0
- package/dist/src/generators/flutter-app/generator.js.map +1 -0
- package/dist/src/generators/go-microservice/generator.d.ts +8 -0
- package/dist/src/generators/go-microservice/generator.js +232 -0
- package/dist/src/generators/go-microservice/generator.js.map +1 -0
- package/dist/src/generators/nextjs-app/generator.d.ts +8 -0
- package/dist/src/generators/nextjs-app/generator.js +294 -0
- package/dist/src/generators/nextjs-app/generator.js.map +1 -0
- package/dist/src/generators/python-microservice/generator.d.ts +13 -0
- package/dist/src/generators/python-microservice/generator.js +265 -0
- package/dist/src/generators/python-microservice/generator.js.map +1 -0
- package/dist/src/generators/shared/brand-tokens.d.ts +89 -0
- package/dist/src/generators/shared/brand-tokens.js +308 -0
- package/dist/src/generators/shared/brand-tokens.js.map +1 -0
- package/dist/src/generators/shared/capabilities.d.ts +101 -0
- package/dist/src/generators/shared/capabilities.js +279 -0
- package/dist/src/generators/shared/capabilities.js.map +1 -0
- package/dist/src/generators/shared/provenance.d.ts +2 -0
- package/dist/src/generators/shared/provenance.js +85 -0
- package/dist/src/generators/shared/provenance.js.map +1 -0
- package/dist/src/generators/shared/scaffold-helpers.d.ts +72 -0
- package/dist/src/generators/shared/scaffold-helpers.js +309 -0
- package/dist/src/generators/shared/scaffold-helpers.js.map +1 -0
- package/dist/src/generators/system-test-runner/generator.d.ts +23 -0
- package/dist/src/generators/system-test-runner/generator.js +173 -0
- package/dist/src/generators/system-test-runner/generator.js.map +1 -0
- package/dist/src/generators/workspace-dev-cli/generator.d.ts +7 -0
- package/dist/src/generators/workspace-dev-cli/generator.js +138 -0
- package/dist/src/generators/workspace-dev-cli/generator.js.map +1 -0
- package/generators.json +57 -0
- package/lib/repo-map/grammars/tree-sitter-c.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-cpp.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-csharp.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-dart.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-go.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-java.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-javascript.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-kotlin.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-lua.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-php.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-python.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-ruby.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-rust.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-scala.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-swift.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-tsx.wasm +0 -0
- package/lib/repo-map/grammars/tree-sitter-typescript.wasm +0 -0
- package/lib/repo-map/index.js +386 -0
- package/lib/repo-map/languages.js +514 -0
- package/lib/repo-map/pagerank.js +59 -0
- package/migrations/README.md +60 -0
- package/migrations/_template/cli-migration.js +27 -0
- package/migrations/gw-bet-prose-redesign.js +105 -0
- package/migrations/gw-drop-test-manifest.js +37 -0
- package/migrations/gw-register-serena-mcp.js +42 -0
- package/migrations/gw-relocate-hidden-skills.js +40 -0
- package/migrations/gw-seed-config-toml.js +24 -0
- package/migrations/index.json +40 -0
- package/package.json +70 -6
- package/src/AGENTS.md +36 -0
- package/src/config/config.toml +30 -0
- package/src/config/groundwork-state.json +5 -0
- package/src/docs/llms.txt +72 -0
- package/src/docs/principles/ai-native/agent-native-systems.md +90 -0
- package/src/docs/principles/ai-native/agentic-systems.md +78 -0
- package/src/docs/principles/ai-native/ai-engineering.md +100 -0
- package/src/docs/principles/ai-native/ai-native-product.md +76 -0
- package/src/docs/principles/delivery/cost-engineering.md +89 -0
- package/src/docs/principles/delivery/day-2-operational-baseline.md +57 -0
- package/src/docs/principles/delivery/devex.md +88 -0
- package/src/docs/principles/delivery/platform.md +101 -0
- package/src/docs/principles/delivery/progressive-delivery.md +92 -0
- package/src/docs/principles/design/ai-native-design.md +73 -0
- package/src/docs/principles/design/design-foundations.md +80 -0
- package/src/docs/principles/design/design-systems-and-tokens.md +72 -0
- package/src/docs/principles/design/interaction-and-motion.md +69 -0
- package/src/docs/principles/design/layout-and-space.md +72 -0
- package/src/docs/principles/design/usability-and-ux.md +79 -0
- package/src/docs/principles/design/visual-design.md +84 -0
- package/src/docs/principles/foundations/code-craft.md +86 -0
- package/src/docs/principles/foundations/continuous-discovery.md +75 -0
- package/src/docs/principles/foundations/documentation.md +102 -0
- package/src/docs/principles/foundations/prioritization-and-appetite.md +78 -0
- package/src/docs/principles/foundations/product-engineering.md +90 -0
- package/src/docs/principles/foundations/product-risks.md +89 -0
- package/src/docs/principles/foundations/requirements-and-specs.md +80 -0
- package/src/docs/principles/foundations/success-metrics.md +66 -0
- package/src/docs/principles/foundations/testing.md +108 -0
- package/src/docs/principles/index.md +24 -0
- package/src/docs/principles/quality/accessibility.md +88 -0
- package/src/docs/principles/quality/observability.md +84 -0
- package/src/docs/principles/quality/performance.md +84 -0
- package/src/docs/principles/quality/privacy.md +92 -0
- package/src/docs/principles/quality/reliability.md +89 -0
- package/src/docs/principles/quality/security.md +78 -0
- package/src/docs/principles/stack/postgres.md +100 -0
- package/src/docs/principles/system-design/api-design.md +86 -0
- package/src/docs/principles/system-design/architecture-decisions.md +81 -0
- package/src/docs/principles/system-design/code-structure.md +104 -0
- package/src/docs/principles/system-design/data-engineering.md +87 -0
- package/src/docs/principles/system-design/durable-execution.md +89 -0
- package/src/docs/principles/system-design/evolutionary-architecture.md +81 -0
- package/src/docs/principles/system-design/identity-and-access.md +76 -0
- package/src/docs/principles/system-design/integration-patterns.md +84 -0
- package/src/docs/principles/system-design/real-time.md +83 -0
- package/src/docs/principles/system-design/surface-architecture.md +74 -0
- package/src/docs/ways-of-working/documentation.md +69 -0
- package/src/docs/ways-of-working/how-we-work.md +76 -0
- package/src/docs/ways-of-working/units-of-work.md +40 -0
- package/src/engineer-skills/groundwork-electron-engineer/SKILL.md +123 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/documentation.md +126 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/ipc-contracts.md +138 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/observability.md +37 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/packaging-and-updates.md +82 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/performance-and-reliability.md +80 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/process-model.md +94 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/security.md +107 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/testing-and-smoke.md +129 -0
- package/src/engineer-skills/groundwork-electron-engineer/references/theming-and-tokens.md +74 -0
- package/src/engineer-skills/groundwork-electron-engineer/sync-anchor.md +22 -0
- package/src/engineer-skills/groundwork-flutter-engineer/SKILL.md +114 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/accessibility.md +92 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/architecture.md +189 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/data-and-contracts.md +136 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/documentation.md +122 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/navigation.md +122 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/observability.md +37 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/performance-and-reliability.md +100 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/platform-channels.md +93 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/releases-and-distribution.md +84 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/security.md +96 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/state-management.md +166 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/testing.md +160 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/theming-and-design-tokens.md +109 -0
- package/src/engineer-skills/groundwork-flutter-engineer/references/widgets-and-composition.md +123 -0
- package/src/engineer-skills/groundwork-flutter-engineer/sync-anchor.md +24 -0
- package/src/engineer-skills/groundwork-go-engineer/SKILL.md +174 -0
- package/src/engineer-skills/groundwork-go-engineer/references/api-design.md +82 -0
- package/src/engineer-skills/groundwork-go-engineer/references/architecture.md +42 -0
- package/src/engineer-skills/groundwork-go-engineer/references/capability-ports.md +50 -0
- package/src/engineer-skills/groundwork-go-engineer/references/code-craft-security.md +34 -0
- package/src/engineer-skills/groundwork-go-engineer/references/concurrency.md +108 -0
- package/src/engineer-skills/groundwork-go-engineer/references/documentation.md +130 -0
- package/src/engineer-skills/groundwork-go-engineer/references/go-services.md +77 -0
- package/src/engineer-skills/groundwork-go-engineer/references/http-handlers.md +172 -0
- package/src/engineer-skills/groundwork-go-engineer/references/implementation-patterns.md +156 -0
- package/src/engineer-skills/groundwork-go-engineer/references/integration-realtime-data.md +57 -0
- package/src/engineer-skills/groundwork-go-engineer/references/observability.md +49 -0
- package/src/engineer-skills/groundwork-go-engineer/references/postgres.md +41 -0
- package/src/engineer-skills/groundwork-go-engineer/references/reliability-performance.md +105 -0
- package/src/engineer-skills/groundwork-go-engineer/references/testing.md +201 -0
- package/src/engineer-skills/groundwork-go-engineer/sync-anchor.md +20 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/SKILL.md +112 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/accessibility.md +111 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/architecture.md +323 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/data-fetching.md +458 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/documentation.md +324 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/error-boundaries.md +383 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/mutations-and-forms.md +396 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/observability.md +48 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/performance-and-deployment.md +947 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/routing-and-navigation.md +405 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/security.md +131 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/server-components.md +394 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/tailwind-and-styling.md +134 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/testing.md +491 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/type-system.md +368 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/ux-principles.md +230 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/references/visual-language.md +69 -0
- package/src/engineer-skills/groundwork-nextjs-engineer/sync-anchor.md +16 -0
- package/src/engineer-skills/groundwork-python-engineer/SKILL.md +199 -0
- package/src/engineer-skills/groundwork-python-engineer/references/api-standards.md +88 -0
- package/src/engineer-skills/groundwork-python-engineer/references/architecture.md +57 -0
- package/src/engineer-skills/groundwork-python-engineer/references/async-patterns.md +103 -0
- package/src/engineer-skills/groundwork-python-engineer/references/capability-ports.md +44 -0
- package/src/engineer-skills/groundwork-python-engineer/references/database.md +88 -0
- package/src/engineer-skills/groundwork-python-engineer/references/documentation-mcp.md +167 -0
- package/src/engineer-skills/groundwork-python-engineer/references/implementation-patterns.md +166 -0
- package/src/engineer-skills/groundwork-python-engineer/references/ml-pipelines.md +119 -0
- package/src/engineer-skills/groundwork-python-engineer/references/ml-systems-ai-engineering.md +74 -0
- package/src/engineer-skills/groundwork-python-engineer/references/observability.md +57 -0
- package/src/engineer-skills/groundwork-python-engineer/references/resilience.md +126 -0
- package/src/engineer-skills/groundwork-python-engineer/references/security.md +148 -0
- package/src/engineer-skills/groundwork-python-engineer/references/testing.md +216 -0
- package/src/engineer-skills/groundwork-python-engineer/sync-anchor.md +20 -0
- package/src/generators/add-capability/generator.ts +70 -0
- package/src/generators/add-capability/schema.json +30 -0
- package/src/generators/capabilities/llm/capability.json +28 -0
- package/src/generators/capabilities/llm/providers/anthropic/footprint.json +13 -0
- package/src/generators/capabilities/llm/providers/anthropic/stacks/go/internal/llm/llm.go.template +102 -0
- package/src/generators/capabilities/llm/providers/anthropic/stacks/python/src/__packageName__/adapters/llm.py.template +61 -0
- package/src/generators/capabilities/llm/providers/local/footprint.json +13 -0
- package/src/generators/capabilities/llm/providers/local/stacks/go/internal/llm/llm.go.template +102 -0
- package/src/generators/capabilities/llm/providers/local/stacks/python/src/__packageName__/adapters/llm.py.template +53 -0
- package/src/generators/capabilities/llm/providers/localai/footprint.json +29 -0
- package/src/generators/capabilities/llm/providers/localai/stacks/go/internal/llm/llm.go.template +102 -0
- package/src/generators/capabilities/llm/providers/localai/stacks/python/src/__packageName__/adapters/llm.py.template +53 -0
- package/src/generators/capabilities/llm/providers/none/footprint.json +9 -0
- package/src/generators/capabilities/llm/providers/none/stacks/go/internal/llm/llm.go.template +35 -0
- package/src/generators/capabilities/llm/providers/none/stacks/python/src/__packageName__/adapters/llm.py.template +25 -0
- package/src/generators/capabilities/llm/providers/ollama/footprint.json +20 -0
- package/src/generators/capabilities/llm/providers/ollama/stacks/go/internal/llm/llm.go.template +102 -0
- package/src/generators/capabilities/llm/providers/ollama/stacks/python/src/__packageName__/adapters/llm.py.template +53 -0
- package/src/generators/capabilities/llm/providers/openai/footprint.json +13 -0
- package/src/generators/capabilities/llm/providers/openai/stacks/go/internal/llm/llm.go.template +98 -0
- package/src/generators/capabilities/llm/providers/openai/stacks/python/src/__packageName__/adapters/llm.py.template +60 -0
- package/src/generators/capabilities/llm/stacks/go/internal/core/service/llm.go.template +12 -0
- package/src/generators/capabilities/llm/stacks/go/internal/llm/llm_test.go.template +33 -0
- package/src/generators/capabilities/llm/stacks/python/src/__packageName__/core/llm.py.template +15 -0
- package/src/generators/capabilities/llm/stacks/python/tests/contracts/test_llm.py.template +37 -0
- package/src/generators/cli-app/files/README.md.template +76 -0
- package/src/generators/cli-app/files/build.mjs.template +15 -0
- package/src/generators/cli-app/files/package.json.template +21 -0
- package/src/generators/cli-app/files/src/cli.ts.template +67 -0
- package/src/generators/cli-app/files/src/commands/hello.ts.template +17 -0
- package/src/generators/cli-app/files/src/commands/status.ts.template +23 -0
- package/src/generators/cli-app/files/src/core/client.test.ts.template +80 -0
- package/src/generators/cli-app/files/src/core/client.ts.template +64 -0
- package/src/generators/cli-app/files/src/registry.test.ts.template +35 -0
- package/src/generators/cli-app/files/src/registry.ts.template +31 -0
- package/src/generators/cli-app/files/tsconfig.json.template +16 -0
- package/src/generators/cli-app/files/tsconfig.test.json.template +11 -0
- package/src/generators/cli-app/generator.ts +138 -0
- package/src/generators/cli-app/schema.json +24 -0
- package/src/generators/docs-site/files/.gitignore.ejs +40 -0
- package/src/generators/docs-site/files/app/docs/__slug__/page.tsx +101 -0
- package/src/generators/docs-site/files/app/docs/layout.tsx +14 -0
- package/src/generators/docs-site/files/app/docs.css +43 -0
- package/src/generators/docs-site/files/app/layout.tsx +24 -0
- package/src/generators/docs-site/files/app/page.tsx +135 -0
- package/src/generators/docs-site/files/app/source.ts +8 -0
- package/src/generators/docs-site/files/components/mermaid.tsx +67 -0
- package/src/generators/docs-site/files/next.config.mjs +10 -0
- package/src/generators/docs-site/files/package.json +32 -0
- package/src/generators/docs-site/files/pnpm-workspace.yaml +7 -0
- package/src/generators/docs-site/files/postcss.config.mjs +6 -0
- package/src/generators/docs-site/files/source.config.ts +77 -0
- package/src/generators/docs-site/files/tailwind.config.js +10 -0
- package/src/generators/docs-site/files/tsconfig.json +27 -0
- package/src/generators/docs-site/generator.ts +476 -0
- package/src/generators/docs-site/schema.json +17 -0
- package/src/generators/electron-app/docs/principles/stack/electron/index.md +49 -0
- package/src/generators/electron-app/docs/principles/stack/electron/ipc-contracts.md +71 -0
- package/src/generators/electron-app/docs/principles/stack/electron/packaging-and-updates.md +59 -0
- package/src/generators/electron-app/docs/principles/stack/electron/process-model.md +53 -0
- package/src/generators/electron-app/docs/principles/stack/electron/security.md +70 -0
- package/src/generators/electron-app/docs/principles/stack/typescript/frontend.md +65 -0
- package/src/generators/electron-app/files/.gitignore.template +20 -0
- package/src/generators/electron-app/files/README.md.template +125 -0
- package/src/generators/electron-app/files/electron.vite.config.ts +31 -0
- package/src/generators/electron-app/files/eslint.config.mjs +92 -0
- package/src/generators/electron-app/files/forge.config.ts.template +44 -0
- package/src/generators/electron-app/files/package.json.template +54 -0
- package/src/generators/electron-app/files/playwright.config.ts +18 -0
- package/src/generators/electron-app/files/project.json.template +65 -0
- package/src/generators/electron-app/files/src/main/core-client.test.ts +81 -0
- package/src/generators/electron-app/files/src/main/core-client.ts +55 -0
- package/src/generators/electron-app/files/src/main/index.ts +157 -0
- package/src/generators/electron-app/files/src/main/ipc.ts +52 -0
- package/src/generators/electron-app/files/src/main/policy.test.ts +71 -0
- package/src/generators/electron-app/files/src/main/policy.ts +73 -0
- package/src/generators/electron-app/files/src/preload/index.ts +23 -0
- package/src/generators/electron-app/files/src/renderer/index.html.template +20 -0
- package/src/generators/electron-app/files/src/renderer/src/App.test.tsx +61 -0
- package/src/generators/electron-app/files/src/renderer/src/App.tsx.template +43 -0
- package/src/generators/electron-app/files/src/renderer/src/assets/main.css +40 -0
- package/src/generators/electron-app/files/src/renderer/src/env.d.ts +14 -0
- package/src/generators/electron-app/files/src/renderer/src/main.tsx +25 -0
- package/src/generators/electron-app/files/src/shared/ipc.ts +54 -0
- package/src/generators/electron-app/files/tests/smoke/app.spec.ts.template +133 -0
- package/src/generators/electron-app/files/tool/electron_exec.sh.template +83 -0
- package/src/generators/electron-app/files/tsconfig.json +7 -0
- package/src/generators/electron-app/files/tsconfig.node.json +27 -0
- package/src/generators/electron-app/files/tsconfig.web.json +22 -0
- package/src/generators/electron-app/files/vitest.config.ts +32 -0
- package/src/generators/electron-app/files/vitest.setup.ts +1 -0
- package/src/generators/electron-app/generator.ts +288 -0
- package/src/generators/electron-app/schema.json +23 -0
- package/src/generators/flutter-app/docs/principles/stack/flutter/architecture.md +78 -0
- package/src/generators/flutter-app/docs/principles/stack/flutter/index.md +38 -0
- package/src/generators/flutter-app/docs/principles/stack/flutter/platform-channels.md +51 -0
- package/src/generators/flutter-app/docs/principles/stack/flutter/releases-and-distribution.md +59 -0
- package/src/generators/flutter-app/docs/principles/stack/flutter/state-management.md +85 -0
- package/src/generators/flutter-app/docs/principles/stack/flutter/testing.md +86 -0
- package/src/generators/flutter-app/docs/principles/stack/flutter/widgets-and-composition.md +69 -0
- package/src/generators/flutter-app/files/.gitignore.template +30 -0
- package/src/generators/flutter-app/files/README.md.template +100 -0
- package/src/generators/flutter-app/files/analysis_options.yaml.template +18 -0
- package/src/generators/flutter-app/files/integration_test/app_test.dart.template +64 -0
- package/src/generators/flutter-app/files/lib/app.dart.template +24 -0
- package/src/generators/flutter-app/files/lib/config/app_config.dart +15 -0
- package/src/generators/flutter-app/files/lib/data/repositories/status_repository.dart +36 -0
- package/src/generators/flutter-app/files/lib/data/services/api_client.dart +71 -0
- package/src/generators/flutter-app/files/lib/domain/models/health_status.dart +23 -0
- package/src/generators/flutter-app/files/lib/main.dart +11 -0
- package/src/generators/flutter-app/files/lib/router.dart +23 -0
- package/src/generators/flutter-app/files/lib/ui/core/theme/app_theme.dart +110 -0
- package/src/generators/flutter-app/files/lib/ui/home/home_view.dart +89 -0
- package/src/generators/flutter-app/files/lib/ui/home/home_view_model.dart.template +38 -0
- package/src/generators/flutter-app/files/project.json.template +51 -0
- package/src/generators/flutter-app/files/pubspec.yaml.template +47 -0
- package/src/generators/flutter-app/files/test/api_client_test.dart.template +63 -0
- package/src/generators/flutter-app/files/test/fakes/fake_status_repository.dart.template +19 -0
- package/src/generators/flutter-app/files/test/home_view_test.dart.template +58 -0
- package/src/generators/flutter-app/files/tool/flutter_exec.sh.template +60 -0
- package/src/generators/flutter-app/generator.ts +362 -0
- package/src/generators/flutter-app/schema.json +23 -0
- package/src/generators/go-microservice/docs/principles/stack/go/concurrency.md +123 -0
- package/src/generators/go-microservice/docs/principles/stack/go/index.md +70 -0
- package/src/generators/go-microservice/docs/principles/stack/go/testing.md +168 -0
- package/src/generators/go-microservice/files/.air.toml.template +38 -0
- package/src/generators/go-microservice/files/.env.template +4 -0
- package/src/generators/go-microservice/files/.golangci.yml.template +82 -0
- package/src/generators/go-microservice/files/Dockerfile.dev.template +12 -0
- package/src/generators/go-microservice/files/asyncapi-pubsub.yaml.template +33 -0
- package/src/generators/go-microservice/files/asyncapi-ws.yaml.template +34 -0
- package/src/generators/go-microservice/files/cmd/api/main.go.template +149 -0
- package/src/generators/go-microservice/files/cmd/api/main_test.go.template +99 -0
- package/src/generators/go-microservice/files/cmd/worker/cleanup/main.go.template +39 -0
- package/src/generators/go-microservice/files/db/schema.sql.template +24 -0
- package/src/generators/go-microservice/files/go.mod.template +39 -0
- package/src/generators/go-microservice/files/internal/config/config.go.template +52 -0
- package/src/generators/go-microservice/files/internal/config/otel.go.template +93 -0
- package/src/generators/go-microservice/files/internal/core/domain/errors.go.template +16 -0
- package/src/generators/go-microservice/files/internal/core/domain/model.go.template +28 -0
- package/src/generators/go-microservice/files/internal/core/domain/user.go.template +13 -0
- package/src/generators/go-microservice/files/internal/core/pagination.go.template +16 -0
- package/src/generators/go-microservice/files/internal/core/service/app_service.go.template +79 -0
- package/src/generators/go-microservice/files/internal/core/service/event_hub.go.template +9 -0
- package/src/generators/go-microservice/files/internal/core/service/message_queue.go.template +10 -0
- package/src/generators/go-microservice/files/internal/core/service/outbox_repository.go.template +31 -0
- package/src/generators/go-microservice/files/internal/core/service/repository.go.template +23 -0
- package/src/generators/go-microservice/files/internal/core/service/user_repository.go.template +15 -0
- package/src/generators/go-microservice/files/internal/core/service/user_service.go.template +43 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/app_handler.go.template +108 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/auth_middleware_test.go.template +52 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/clerk_webhook.go.template +202 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/clerk_webhook_test.go.template +82 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/health_handler.go.template +80 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/idempotency/middleware.go.template +87 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/idempotency/middleware_test.go.template +76 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/idempotency/repository.go.template +37 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/middleware_auth.go.template +40 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/middleware_loadshed.go.template +38 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/middleware_logging.go.template +40 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/middleware_ratelimit.go.template +48 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/middleware_test.go.template +81 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/router.go.template +105 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/types.go.template +70 -0
- package/src/generators/go-microservice/files/internal/entrypoints/api/websocket_handler.go.template +39 -0
- package/src/generators/go-microservice/files/internal/httpclient/http_client.go.template +87 -0
- package/src/generators/go-microservice/files/internal/kafka/kafka.go.template +34 -0
- package/src/generators/go-microservice/files/internal/postgres/postgres.go.template +195 -0
- package/src/generators/go-microservice/files/internal/postgres/postgres_test.go.template +156 -0
- package/src/generators/go-microservice/files/internal/postgres/user_repository.go.template +56 -0
- package/src/generators/go-microservice/files/internal/pubsub/gcp_pubsub.go.template +35 -0
- package/src/generators/go-microservice/files/internal/websocket/client.go.template +151 -0
- package/src/generators/go-microservice/files/internal/websocket/hub.go.template +261 -0
- package/src/generators/go-microservice/files/scripts/apply-schema.sh.template +21 -0
- package/src/generators/go-microservice/files/tools/tools.go.template +10 -0
- package/src/generators/go-microservice/generator.ts +240 -0
- package/src/generators/go-microservice/schema.json +63 -0
- package/src/generators/nextjs-app/docs/principles/stack/typescript/frontend.md +65 -0
- package/src/generators/nextjs-app/files/.dockerignore.template +7 -0
- package/src/generators/nextjs-app/files/.env.example.template +24 -0
- package/src/generators/nextjs-app/files/.gitignore.template +5 -0
- package/src/generators/nextjs-app/files/Dockerfile +53 -0
- package/src/generators/nextjs-app/files/app/(auth)/sign-in/__sign-in__/page.tsx.template +9 -0
- package/src/generators/nextjs-app/files/app/(auth)/sign-up/__sign-up__/page.tsx.template +9 -0
- package/src/generators/nextjs-app/files/app/api/config/route.ts.template +39 -0
- package/src/generators/nextjs-app/files/app/api/healthz/route.test.ts +15 -0
- package/src/generators/nextjs-app/files/app/api/healthz/route.ts +5 -0
- package/src/generators/nextjs-app/files/app/api/proxy/__path__/route.test.ts.template +55 -0
- package/src/generators/nextjs-app/files/app/api/proxy/__path__/route.ts.template +126 -0
- package/src/generators/nextjs-app/files/app/error.tsx +39 -0
- package/src/generators/nextjs-app/files/app/global-error.tsx +68 -0
- package/src/generators/nextjs-app/files/app/globals.css +105 -0
- package/src/generators/nextjs-app/files/app/layout.tsx +59 -0
- package/src/generators/nextjs-app/files/app/loading.tsx +13 -0
- package/src/generators/nextjs-app/files/app/not-found.tsx +30 -0
- package/src/generators/nextjs-app/files/app/page.tsx +20 -0
- package/src/generators/nextjs-app/files/components/providers/default.tsx +19 -0
- package/src/generators/nextjs-app/files/components/providers/production.tsx +32 -0
- package/src/generators/nextjs-app/files/components/providers/telemetry.tsx +76 -0
- package/src/generators/nextjs-app/files/components/render-smoke.test.tsx +29 -0
- package/src/generators/nextjs-app/files/components/theme-provider.tsx +11 -0
- package/src/generators/nextjs-app/files/components.json +21 -0
- package/src/generators/nextjs-app/files/eslint.config.mjs +120 -0
- package/src/generators/nextjs-app/files/hooks/use-toast.ts +7 -0
- package/src/generators/nextjs-app/files/instrumentation.ts +90 -0
- package/src/generators/nextjs-app/files/lib/api/fetcher.ts.template +130 -0
- package/src/generators/nextjs-app/files/lib/config.ts +21 -0
- package/src/generators/nextjs-app/files/lib/logger.ts +29 -0
- package/src/generators/nextjs-app/files/lib/schemas/index.ts +19 -0
- package/src/generators/nextjs-app/files/lib/utils.ts +6 -0
- package/src/generators/nextjs-app/files/next.config.mjs +9 -0
- package/src/generators/nextjs-app/files/package.json +70 -0
- package/src/generators/nextjs-app/files/postcss.config.mjs +8 -0
- package/src/generators/nextjs-app/files/proxy.test.ts.template +30 -0
- package/src/generators/nextjs-app/files/proxy.ts +31 -0
- package/src/generators/nextjs-app/files/public/.gitkeep +1 -0
- package/src/generators/nextjs-app/files/tsconfig.json +42 -0
- package/src/generators/nextjs-app/files/vitest.config.mts +15 -0
- package/src/generators/nextjs-app/files/vitest.setup.ts +7 -0
- package/src/generators/nextjs-app/generator.ts +307 -0
- package/src/generators/nextjs-app/schema.json +44 -0
- package/src/generators/python-microservice/docs/principles/stack/python/async.md +168 -0
- package/src/generators/python-microservice/docs/principles/stack/python/documentation.md +240 -0
- package/src/generators/python-microservice/docs/principles/stack/python/mcp.md +147 -0
- package/src/generators/python-microservice/docs/principles/stack/python/resilience.md +193 -0
- package/src/generators/python-microservice/docs/principles/stack/python/testing.md +322 -0
- package/src/generators/python-microservice/files/.env.example.template +30 -0
- package/src/generators/python-microservice/files/Dockerfile.template +36 -0
- package/src/generators/python-microservice/files/db/schema.sql.template +19 -0
- package/src/generators/python-microservice/files/pyproject.toml.template +76 -0
- package/src/generators/python-microservice/files/scripts/apply-schema.sh.template +25 -0
- package/src/generators/python-microservice/files/src/__packageName__/adapters/comfyui.py.template +87 -0
- package/src/generators/python-microservice/files/src/__packageName__/adapters/config.py.template +48 -0
- package/src/generators/python-microservice/files/src/__packageName__/adapters/database.py.template +21 -0
- package/src/generators/python-microservice/files/src/__packageName__/adapters/message_queue.py.template +29 -0
- package/src/generators/python-microservice/files/src/__packageName__/adapters/repository.py.template +130 -0
- package/src/generators/python-microservice/files/src/__packageName__/adapters/telemetry.py.template +68 -0
- package/src/generators/python-microservice/files/src/__packageName__/adapters/websocket_hub.py.template +36 -0
- package/src/generators/python-microservice/files/src/__packageName__/core/domain/entities.py.template +22 -0
- package/src/generators/python-microservice/files/src/__packageName__/core/domain/exceptions.py.template +43 -0
- package/src/generators/python-microservice/files/src/__packageName__/core/ports.py.template +42 -0
- package/src/generators/python-microservice/files/src/__packageName__/core/service/example_service.py.template +68 -0
- package/src/generators/python-microservice/files/src/__packageName__/entrypoints/api/dependencies.py.template +50 -0
- package/src/generators/python-microservice/files/src/__packageName__/entrypoints/api/middleware.py.template +131 -0
- package/src/generators/python-microservice/files/src/__packageName__/entrypoints/api/router.py.template +37 -0
- package/src/generators/python-microservice/files/src/__packageName__/entrypoints/api/websocket_handler.py.template +20 -0
- package/src/generators/python-microservice/files/src/__packageName__/entrypoints/worker/cleanup.py.template +35 -0
- package/src/generators/python-microservice/files/src/__packageName__/entrypoints/worker/worker.py.template +28 -0
- package/src/generators/python-microservice/files/src/__packageName__/main.py.template +108 -0
- package/src/generators/python-microservice/files/tests/test_main.py.template +74 -0
- package/src/generators/python-microservice/files/tests/test_middleware.py.template +109 -0
- package/src/generators/python-microservice/files/tests/test_worker.py.template +16 -0
- package/src/generators/python-microservice/generator.ts +286 -0
- package/src/generators/python-microservice/schema.json +86 -0
- package/src/generators/shared/brand-tokens.ts +301 -0
- package/src/generators/shared/capabilities.ts +349 -0
- package/src/generators/shared/provenance.ts +61 -0
- package/src/generators/shared/scaffold-helpers.ts +309 -0
- package/src/generators/system-test-runner/NATIVE-CHECK-CONTRACT.md +20 -0
- package/src/generators/system-test-runner/files/tests/bets/.gitkeep +0 -0
- package/src/generators/system-test-runner/files/tests/bets/_archive/.gitkeep +0 -0
- package/src/generators/system-test-runner/files/tests/conftest.py.template +503 -0
- package/src/generators/system-test-runner/files/tests/pyproject.toml.template +20 -0
- package/src/generators/system-test-runner/files/tests/system/pages/__init__.py.template +9 -0
- package/src/generators/system-test-runner/files/tests/system/pages/base_page.py.template +36 -0
- package/src/generators/system-test-runner/files/tests/system/test_a11y_smoke.py.template +132 -0
- package/src/generators/system-test-runner/files/tests/system/test_contract_conformance.py.template +140 -0
- package/src/generators/system-test-runner/files/tests/system/test_layout_geometry.py.template +109 -0
- package/src/generators/system-test-runner/files/tests/system/test_render_smoke.py.template +257 -0
- package/src/generators/system-test-runner/files/tests/system/test_system.py.template +158 -0
- package/src/generators/system-test-runner/files/tests/system/test_token_conformance.py.template +206 -0
- package/src/generators/system-test-runner/files/tests/system/test_visual_regression.py.template +104 -0
- package/src/generators/system-test-runner/generator.ts +196 -0
- package/src/generators/system-test-runner/schema.json +24 -0
- package/src/generators/workspace-dev-cli/cli-src/build.mjs +42 -0
- package/src/generators/workspace-dev-cli/cli-src/dist/dev-bundle.js +2168 -0
- package/src/generators/workspace-dev-cli/cli-src/src/commands/bet.ts +442 -0
- package/src/generators/workspace-dev-cli/cli-src/src/commands/completion.ts +87 -0
- package/src/generators/workspace-dev-cli/cli-src/src/commands/doctor.ts +139 -0
- package/src/generators/workspace-dev-cli/cli-src/src/commands/lifecycle.ts +548 -0
- package/src/generators/workspace-dev-cli/cli-src/src/commands/quality.ts +127 -0
- package/src/generators/workspace-dev-cli/cli-src/src/commands/surface.ts +214 -0
- package/src/generators/workspace-dev-cli/cli-src/src/index.ts +127 -0
- package/src/generators/workspace-dev-cli/cli-src/src/registry.ts +194 -0
- package/src/generators/workspace-dev-cli/cli-src/src/theme/color.ts +130 -0
- package/src/generators/workspace-dev-cli/cli-src/src/theme/render.ts +158 -0
- package/src/generators/workspace-dev-cli/cli-src/src/theme/tokens.ts +122 -0
- package/src/generators/workspace-dev-cli/cli-src/src/util/context.ts +43 -0
- package/src/generators/workspace-dev-cli/cli-src/src/util/extensions.ts +99 -0
- package/src/generators/workspace-dev-cli/cli-src/src/util/paths.ts +46 -0
- package/src/generators/workspace-dev-cli/cli-src/src/util/proc.ts +106 -0
- package/src/generators/workspace-dev-cli/cli-src/src/util/prompt.ts +108 -0
- package/src/generators/workspace-dev-cli/cli-src/src/util/runners.ts +70 -0
- package/src/generators/workspace-dev-cli/cli-src/src/util/services.ts +221 -0
- package/src/generators/workspace-dev-cli/cli-src/src/util/version.ts +21 -0
- package/src/generators/workspace-dev-cli/cli-src/tsconfig.json +16 -0
- package/src/generators/workspace-dev-cli/files/.agents/skills/workspace-cli/SKILL.md.template +74 -0
- package/src/generators/workspace-dev-cli/files/dev.template +16 -0
- package/src/generators/workspace-dev-cli/files/docker-compose.yml.template +20 -0
- package/src/generators/workspace-dev-cli/files/scripts/cli/templates/milestone-test.pytmpl.template +46 -0
- package/src/generators/workspace-dev-cli/files/scripts/cli/templates/slice-test.pytmpl.template +38 -0
- package/src/generators/workspace-dev-cli/generator.ts +136 -0
- package/src/generators/workspace-dev-cli/schema.json +22 -0
- package/src/hidden-skills/code-intelligence.md +135 -0
- package/src/hidden-skills/groundwork-architect/SKILL.md +114 -0
- package/src/hidden-skills/groundwork-architect/references/agentic-systems.md +44 -0
- package/src/hidden-skills/groundwork-architect/references/ai-native-architecture.md +37 -0
- package/src/hidden-skills/groundwork-architect/references/api-and-contracts.md +45 -0
- package/src/hidden-skills/groundwork-architect/references/core-and-boundaries.md +45 -0
- package/src/hidden-skills/groundwork-architect/references/data-architecture.md +33 -0
- package/src/hidden-skills/groundwork-architect/references/decision-records.md +34 -0
- package/src/hidden-skills/groundwork-architect/references/durable-execution.md +45 -0
- package/src/hidden-skills/groundwork-architect/references/evolutionary-architecture.md +37 -0
- package/src/hidden-skills/groundwork-architect/references/identity-and-access.md +41 -0
- package/src/hidden-skills/groundwork-architect/references/integration-patterns.md +39 -0
- package/src/hidden-skills/groundwork-architect/references/observability.md +36 -0
- package/src/hidden-skills/groundwork-architect/references/performance-and-scale.md +41 -0
- package/src/hidden-skills/groundwork-architect/references/platform-and-delivery.md +47 -0
- package/src/hidden-skills/groundwork-architect/references/realtime-and-async.md +28 -0
- package/src/hidden-skills/groundwork-architect/references/reliability.md +31 -0
- package/src/hidden-skills/groundwork-architect/references/security-and-trust.md +47 -0
- package/src/hidden-skills/groundwork-architect/references/surface-architecture.md +40 -0
- package/src/hidden-skills/groundwork-architect/sync-anchor.md +34 -0
- package/src/hidden-skills/groundwork-architecture/architecture-template.md +50 -0
- package/src/hidden-skills/groundwork-architecture/instructions.md +139 -0
- package/src/hidden-skills/groundwork-architecture/phases/01-context-ingestion.md +18 -0
- package/src/hidden-skills/groundwork-architecture/phases/02-technical-constraints.md +27 -0
- package/src/hidden-skills/groundwork-architecture/phases/03-service-design.md +19 -0
- package/src/hidden-skills/groundwork-architecture/phases/04-data-flow-communication.md +23 -0
- package/src/hidden-skills/groundwork-architecture/phases/05-component-boundaries-contracts.md +17 -0
- package/src/hidden-skills/groundwork-architecture/phases/06-draft-review-present.md +38 -0
- package/src/hidden-skills/groundwork-architecture/phases/07-commit.md +33 -0
- package/src/hidden-skills/groundwork-architecture/templates/architecture-cache.md +43 -0
- package/src/hidden-skills/groundwork-architecture-extract/instructions.md +163 -0
- package/src/hidden-skills/groundwork-architecture-extract/templates/architecture-extract-cache.md +21 -0
- package/src/hidden-skills/groundwork-bet/briefs/acceptance-auditor.md +68 -0
- package/src/hidden-skills/groundwork-bet/briefs/blind-reviewer.md +56 -0
- package/src/hidden-skills/groundwork-bet/briefs/coverage-auditor.md +95 -0
- package/src/hidden-skills/groundwork-bet/briefs/edge-case-tracer.md +64 -0
- package/src/hidden-skills/groundwork-bet/briefs/experience-auditor.md +83 -0
- package/src/hidden-skills/groundwork-bet/briefs/slice-worker.md +257 -0
- package/src/hidden-skills/groundwork-bet/instructions.md +88 -0
- package/src/hidden-skills/groundwork-bet/templates/bet-progress-test.md +115 -0
- package/src/hidden-skills/groundwork-bet/templates/change-proposal.md +38 -0
- package/src/hidden-skills/groundwork-bet/templates/decomposition/meta.json +4 -0
- package/src/hidden-skills/groundwork-bet/templates/decomposition/milestone-index.md +31 -0
- package/src/hidden-skills/groundwork-bet/templates/decomposition/slice.md +31 -0
- package/src/hidden-skills/groundwork-bet/templates/pitch.md +45 -0
- package/src/hidden-skills/groundwork-bet/templates/technical-design/01-ui-design.md +51 -0
- package/src/hidden-skills/groundwork-bet/templates/technical-design/02-data-flows.md +36 -0
- package/src/hidden-skills/groundwork-bet/templates/technical-design/03-api-design.md +90 -0
- package/src/hidden-skills/groundwork-bet/templates/technical-design/04-data-design.md +29 -0
- package/src/hidden-skills/groundwork-bet/workflows/01-discovery.md +200 -0
- package/src/hidden-skills/groundwork-bet/workflows/02-design.md +178 -0
- package/src/hidden-skills/groundwork-bet/workflows/03-decomposition.md +242 -0
- package/src/hidden-skills/groundwork-bet/workflows/04-delivery.md +226 -0
- package/src/hidden-skills/groundwork-bet/workflows/05-validation.md +210 -0
- package/src/hidden-skills/groundwork-design-system/instructions.md +125 -0
- package/src/hidden-skills/groundwork-design-system/templates/brand-tokens.md +182 -0
- package/src/hidden-skills/groundwork-design-system/templates/design-system-cache.md +64 -0
- package/src/hidden-skills/groundwork-design-system/tracks/_foundation.md +136 -0
- package/src/hidden-skills/groundwork-design-system/tracks/agentic-protocol.md +269 -0
- package/src/hidden-skills/groundwork-design-system/tracks/cli.md +355 -0
- package/src/hidden-skills/groundwork-design-system/tracks/graphical-ui.md +330 -0
- package/src/hidden-skills/groundwork-design-system-extract/instructions.md +124 -0
- package/src/hidden-skills/groundwork-design-system-extract/templates/design-system-extract-cache.md +19 -0
- package/src/hidden-skills/groundwork-designer/SKILL.md +108 -0
- package/src/hidden-skills/groundwork-designer/references/accessibility.md +33 -0
- package/src/hidden-skills/groundwork-designer/references/ai-native-design.md +37 -0
- package/src/hidden-skills/groundwork-designer/references/design-review.md +29 -0
- package/src/hidden-skills/groundwork-designer/references/design-systems-and-tokens.md +33 -0
- package/src/hidden-skills/groundwork-designer/references/interaction-and-motion.md +37 -0
- package/src/hidden-skills/groundwork-designer/references/layout-and-space.md +33 -0
- package/src/hidden-skills/groundwork-designer/references/usability-and-ux.md +33 -0
- package/src/hidden-skills/groundwork-designer/references/visual-craft.md +49 -0
- package/src/hidden-skills/groundwork-designer/sync-anchor.md +20 -0
- package/src/hidden-skills/groundwork-doc-sync/instructions.md +100 -0
- package/src/hidden-skills/groundwork-elicit/instructions.md +66 -0
- package/src/hidden-skills/groundwork-elicit/methods.md +65 -0
- package/src/hidden-skills/groundwork-infra-adopt/instructions.md +168 -0
- package/src/hidden-skills/groundwork-infra-adopt/templates/infra-adopt-cache.md +21 -0
- package/src/hidden-skills/groundwork-mvp/instructions.md +223 -0
- package/src/hidden-skills/groundwork-mvp/templates/mvp-cache.md +9 -0
- package/src/hidden-skills/groundwork-patch/instructions.md +40 -0
- package/src/hidden-skills/groundwork-persona/instructions.md +65 -0
- package/src/hidden-skills/groundwork-product/SKILL.md +102 -0
- package/src/hidden-skills/groundwork-product/references/ai-native-product.md +45 -0
- package/src/hidden-skills/groundwork-product/references/discovery-and-opportunity.md +38 -0
- package/src/hidden-skills/groundwork-product/references/product-risks.md +52 -0
- package/src/hidden-skills/groundwork-product/references/requirements-and-specs.md +39 -0
- package/src/hidden-skills/groundwork-product/references/scope-and-sequencing.md +35 -0
- package/src/hidden-skills/groundwork-product/references/shaping-and-appetite.md +48 -0
- package/src/hidden-skills/groundwork-product/references/success-metrics-and-signals.md +37 -0
- package/src/hidden-skills/groundwork-product/sync-anchor.md +19 -0
- package/src/hidden-skills/groundwork-product-brief/instructions.md +231 -0
- package/src/hidden-skills/groundwork-product-brief-extract/instructions.md +139 -0
- package/src/hidden-skills/groundwork-product-brief-extract/templates/product-brief-extract-cache.md +17 -0
- package/src/hidden-skills/groundwork-review/checklists/architecture.md +93 -0
- package/src/hidden-skills/groundwork-review/checklists/bet-pitch.md +94 -0
- package/src/hidden-skills/groundwork-review/checklists/decomposition.md +135 -0
- package/src/hidden-skills/groundwork-review/checklists/design-system.md +85 -0
- package/src/hidden-skills/groundwork-review/checklists/domain-entity.md +66 -0
- package/src/hidden-skills/groundwork-review/checklists/implementation-readiness.md +47 -0
- package/src/hidden-skills/groundwork-review/checklists/infrastructure.md +68 -0
- package/src/hidden-skills/groundwork-review/checklists/maturity.md +71 -0
- package/src/hidden-skills/groundwork-review/checklists/product-brief.md +69 -0
- package/src/hidden-skills/groundwork-review/checklists/technical-design.md +112 -0
- package/src/hidden-skills/groundwork-review/instructions.md +181 -0
- package/src/hidden-skills/groundwork-scaffold/instructions.md +254 -0
- package/src/hidden-skills/groundwork-scaffold/phases/01-ingestion-service-mapping.md +87 -0
- package/src/hidden-skills/groundwork-scaffold/phases/02-scaffolding-execution.md +15 -0
- package/src/hidden-skills/groundwork-scaffold/phases/03-service-documentation-api-stubs.md +100 -0
- package/src/hidden-skills/groundwork-scaffold/phases/04-infrastructure-verification.md +17 -0
- package/src/hidden-skills/groundwork-scaffold/phases/05-draft-review.md +19 -0
- package/src/hidden-skills/groundwork-scaffold/phases/06-commit.md +19 -0
- package/src/hidden-skills/groundwork-scaffold/templates/scaffold-cache.md +23 -0
- package/src/hidden-skills/groundwork-scan/instructions.md +164 -0
- package/src/hidden-skills/groundwork-scan/references/digest-schema.md +66 -0
- package/src/hidden-skills/groundwork-scan/references/exclusions.md +44 -0
- package/src/hidden-skills/groundwork-scan/templates/architecture-findings.md +42 -0
- package/src/hidden-skills/groundwork-scan/templates/design-findings.md +23 -0
- package/src/hidden-skills/groundwork-scan/templates/overview.md +26 -0
- package/src/hidden-skills/groundwork-scan/templates/product-findings.md +23 -0
- package/src/hidden-skills/groundwork-scan/templates/scan-state.json +19 -0
- package/src/hidden-skills/groundwork-stack-forge/instructions.md +150 -0
- package/src/hidden-skills/groundwork-stack-forge/references/authoring-engineer-skills.md +107 -0
- package/src/hidden-skills/groundwork-surface-activation/instructions.md +138 -0
- package/src/hidden-skills/groundwork-update/briefs/reconcile-worker.md +196 -0
- package/src/hidden-skills/groundwork-update/instructions.md +200 -0
- package/src/hidden-skills/groundwork-writer/SKILL.md +278 -0
- package/src/hidden-skills/maturity-model.md +125 -0
- package/src/hidden-skills/operating-contract.md +400 -0
- package/src/hidden-skills/repo-map-schema.md +90 -0
- package/src/hidden-skills/templates/adr.md +57 -0
- package/src/hidden-skills/templates/capability-ports.md +71 -0
- package/src/hidden-skills/templates/discovery-notes.md +33 -0
- package/src/hidden-skills/templates/domain-entity.md +80 -0
- package/src/hidden-skills/templates/gap-ledger.md +21 -0
- package/src/hidden-skills/templates/handoff.md +37 -0
- package/src/hidden-skills/templates/maturity.md +39 -0
- package/src/hidden-skills/templates/surfaces.md +207 -0
- package/src/skills/groundwork-check/SKILL.md +56 -0
- package/src/skills/groundwork-check/instructions.md +70 -0
- package/src/skills/groundwork-orchestrator/SKILL.md +176 -0
- package/src/skills/groundwork-orchestrator/workflow-index.md +50 -0
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* The render layer — the typographic and structural vocabulary of the CLI.
|
|
3
|
+
*
|
|
4
|
+
* This is the Node port of the bash `_ui.sh` helpers, rebuilt on the brand-token
|
|
5
|
+
* painter so output is themed from the design system rather than hard-coded. It is
|
|
6
|
+
* the shared surface both `./dev` and the `cli-app` starter render through.
|
|
7
|
+
*
|
|
8
|
+
* All human chrome is written to stderr so stdout stays clean for `--json` and pipes.
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { Painter, detectCaps } from './color';
|
|
12
|
+
import { BrandTokens, mergeTokens, SymbolName } from './tokens';
|
|
13
|
+
|
|
14
|
+
const PAD = ' ';
|
|
15
|
+
|
|
16
|
+
export class Renderer {
|
|
17
|
+
readonly painter: Painter;
|
|
18
|
+
private readonly tokens: BrandTokens;
|
|
19
|
+
private spinnerTimer: NodeJS.Timeout | null = null;
|
|
20
|
+
private spinnerText = '';
|
|
21
|
+
|
|
22
|
+
constructor(tokens: BrandTokens, stream: NodeJS.WriteStream = process.stderr) {
|
|
23
|
+
this.tokens = tokens;
|
|
24
|
+
this.out = stream;
|
|
25
|
+
this.painter = new Painter(tokens, detectCaps(stream));
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
private out: NodeJS.WriteStream;
|
|
29
|
+
|
|
30
|
+
/** A twin renderer bound to a different stream (e.g. stdout for command *results*,
|
|
31
|
+
* while progress and spinners stay on stderr). */
|
|
32
|
+
asStream(stream: NodeJS.WriteStream): Renderer {
|
|
33
|
+
return new Renderer(this.tokens, stream);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
private sym(name: SymbolName): string {
|
|
37
|
+
const t = this.tokens.terminal?.symbols?.[name];
|
|
38
|
+
if (!t) return '';
|
|
39
|
+
return this.painter.caps.unicode ? t.unicode : t.ascii;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
private write(line: string): void {
|
|
43
|
+
this.out.write(line + '\n');
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
logo(subtitle?: string): void {
|
|
47
|
+
const { wordmark, appName } = this.tokens.identity;
|
|
48
|
+
const mark = this.painter.primary(`${wordmark} ${appName}`.trim());
|
|
49
|
+
this.write('');
|
|
50
|
+
this.write(subtitle ? `${PAD}${this.painter.bold(mark)} ${this.painter.dim('— ' + subtitle)}` : `${PAD}${this.painter.bold(mark)}`);
|
|
51
|
+
this.write('');
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
step(text: string): void {
|
|
55
|
+
this.write(`\n${PAD}${this.painter.primary(this.sym('step'))} ${this.painter.bold(text)}`);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
substep(text: string): void {
|
|
59
|
+
this.write(`${PAD}${PAD}${this.painter.dim(`${this.sym('substep')} ${text}`)}`);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
info(text: string): void {
|
|
63
|
+
this.write(`${PAD}${this.painter.dim(this.sym('info'))} ${text}`);
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
success(text: string): void {
|
|
67
|
+
this.write(`${PAD}${this.painter.paint('success', this.sym('success'))} ${text}`);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
error(text: string): void {
|
|
71
|
+
this.write(`${PAD}${this.painter.paint('error', this.sym('error'))} ${text}`);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
warn(text: string): void {
|
|
75
|
+
this.write(`${PAD}${this.painter.paint('warning', this.sym('warning'))} ${text}`);
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
category(text: string): void {
|
|
79
|
+
this.write(`\n${PAD}${this.painter.dim('■')} ${this.painter.paint('header', text)}`);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
cmd(name: string, desc: string): void {
|
|
83
|
+
this.write(` ${this.painter.paint('accent', name.padEnd(15))} ${desc}`);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
/** A boxed error card with an optional action line. */
|
|
87
|
+
errorCard(msg: string, action?: string): void {
|
|
88
|
+
const bar = this.painter.paint('error', '│');
|
|
89
|
+
this.write('');
|
|
90
|
+
this.write(`${PAD}${this.painter.paint('error', '╭' + '─'.repeat(58) + '╮')}`);
|
|
91
|
+
this.write(`${PAD}${bar} ${this.painter.paint('error', this.sym('error'))} ${this.painter.bold('ERROR:')} ${msg}`);
|
|
92
|
+
if (action) {
|
|
93
|
+
this.write(`${PAD}${bar}`);
|
|
94
|
+
this.write(`${PAD}${bar} ${this.painter.dim('Action required:')}`);
|
|
95
|
+
this.write(`${PAD}${bar} ${this.painter.paint('accent', this.sym('active'))} ${action}`);
|
|
96
|
+
}
|
|
97
|
+
this.write(`${PAD}${this.painter.paint('error', '╰' + '─'.repeat(58) + '╯')}`);
|
|
98
|
+
this.write('');
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
/** A simple three-column table for status output. */
|
|
102
|
+
table(title: string, rows: Array<[string, string, string]>): void {
|
|
103
|
+
this.write(`${PAD}${this.painter.dim('╭─')} ${this.painter.bold(title)}`);
|
|
104
|
+
if (rows.length === 0) {
|
|
105
|
+
this.write(`${PAD}${this.painter.dim('│')} ${this.painter.dim('(none)')}`);
|
|
106
|
+
}
|
|
107
|
+
for (const [a, b, c] of rows) {
|
|
108
|
+
this.write(`${PAD}${this.painter.dim('│')} ${a.padEnd(28)} ${b.padEnd(16)} ${this.painter.dim(c)}`);
|
|
109
|
+
}
|
|
110
|
+
this.write(`${PAD}${this.painter.dim('╰' + '─'.repeat(40))}`);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// --- Spinner (TTY only; degrades to a static line) -------------------------
|
|
114
|
+
|
|
115
|
+
startSpinner(text: string): void {
|
|
116
|
+
this.spinnerText = text;
|
|
117
|
+
if (!this.painter.caps.isTTY) {
|
|
118
|
+
this.write(`${PAD}${this.painter.dim(this.sym('info'))} ${text}...`);
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
const frames = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];
|
|
122
|
+
const asciiFrames = ['|', '/', '-', '\\'];
|
|
123
|
+
const set = this.painter.caps.unicode ? frames : asciiFrames;
|
|
124
|
+
let i = 0;
|
|
125
|
+
this.out.write('\x1b[?25l'); // hide cursor
|
|
126
|
+
this.spinnerTimer = setInterval(() => {
|
|
127
|
+
const frame = this.painter.primary(set[i % set.length]);
|
|
128
|
+
this.out.write(`\r${PAD}${frame} ${this.spinnerText}`);
|
|
129
|
+
i += 1;
|
|
130
|
+
}, 90);
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
stopSpinner(successMsg: string, elapsed?: string): void {
|
|
134
|
+
if (this.spinnerTimer) {
|
|
135
|
+
clearInterval(this.spinnerTimer);
|
|
136
|
+
this.spinnerTimer = null;
|
|
137
|
+
this.out.write('\r\x1b[K'); // clear line
|
|
138
|
+
this.out.write('\x1b[?25h'); // restore cursor
|
|
139
|
+
}
|
|
140
|
+
const time = elapsed ? ` ${this.painter.dim(`(${elapsed})`)}` : '';
|
|
141
|
+
this.success(`${successMsg}${time}`);
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
failSpinner(failMsg: string): void {
|
|
145
|
+
if (this.spinnerTimer) {
|
|
146
|
+
clearInterval(this.spinnerTimer);
|
|
147
|
+
this.spinnerTimer = null;
|
|
148
|
+
this.out.write('\r\x1b[K');
|
|
149
|
+
this.out.write('\x1b[?25h');
|
|
150
|
+
}
|
|
151
|
+
this.error(failMsg);
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
/** Build a renderer from a (possibly partial) tokens object loaded from disk. */
|
|
156
|
+
export function makeRenderer(partialTokens: unknown, stream?: NodeJS.WriteStream): Renderer {
|
|
157
|
+
return new Renderer(mergeTokens(partialTokens), stream);
|
|
158
|
+
}
|
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Brand tokens — the machine-readable projection of the design system's branding.
|
|
3
|
+
*
|
|
4
|
+
* This module is the shared theme contract. The `./dev` CLI is its first consumer;
|
|
5
|
+
* the `cli-app` product starter is the second. Both read a `dev.config.json` (itself
|
|
6
|
+
* projected from `.groundwork/config/brand-tokens.json`) and render from it.
|
|
7
|
+
*
|
|
8
|
+
* Keep this module free of `./dev`-specific logic so it can be extracted to a shared
|
|
9
|
+
* package when the second consumer lands. See the contract at
|
|
10
|
+
* groundwork-design-system/templates/brand-tokens.md.
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
export type NoColorTreatment =
|
|
14
|
+
| 'bold'
|
|
15
|
+
| 'dim'
|
|
16
|
+
| 'underline'
|
|
17
|
+
| 'bold+upper'
|
|
18
|
+
| 'plain';
|
|
19
|
+
|
|
20
|
+
export interface ColorRole {
|
|
21
|
+
truecolor: string | null; // "#rrggbb" or null
|
|
22
|
+
ansi256: number | null; // 0-255 or null
|
|
23
|
+
noColor: NoColorTreatment;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export type RoleName =
|
|
27
|
+
| 'success'
|
|
28
|
+
| 'error'
|
|
29
|
+
| 'warning'
|
|
30
|
+
| 'info'
|
|
31
|
+
| 'muted'
|
|
32
|
+
| 'accent'
|
|
33
|
+
| 'header'
|
|
34
|
+
| 'key'
|
|
35
|
+
| 'value';
|
|
36
|
+
|
|
37
|
+
export interface SymbolToken {
|
|
38
|
+
unicode: string;
|
|
39
|
+
ascii: string;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export type SymbolName =
|
|
43
|
+
| 'success'
|
|
44
|
+
| 'error'
|
|
45
|
+
| 'warning'
|
|
46
|
+
| 'info'
|
|
47
|
+
| 'step'
|
|
48
|
+
| 'substep'
|
|
49
|
+
| 'active';
|
|
50
|
+
|
|
51
|
+
export interface BrandTokens {
|
|
52
|
+
identity: {
|
|
53
|
+
appName: string;
|
|
54
|
+
wordmark: string;
|
|
55
|
+
primary: string; // "#rrggbb"
|
|
56
|
+
accent: string; // "#rrggbb"
|
|
57
|
+
voice: string;
|
|
58
|
+
};
|
|
59
|
+
terminal?: {
|
|
60
|
+
colorRoles: Record<RoleName, ColorRole>;
|
|
61
|
+
symbols: Record<SymbolName, SymbolToken>;
|
|
62
|
+
splash: { style: 'wordmark-line' | 'banner' | 'none'; tagline: string };
|
|
63
|
+
typography: Record<'header' | 'title' | 'body' | 'muted', string>;
|
|
64
|
+
};
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/** The fallback theme used when no brand tokens are present. Mirrors the values in
|
|
68
|
+
* the design-system colour exemplar so a themed and an un-themed CLI look like kin. */
|
|
69
|
+
export const DEFAULT_TOKENS: BrandTokens = {
|
|
70
|
+
identity: {
|
|
71
|
+
appName: 'Workspace',
|
|
72
|
+
wordmark: '◢◤',
|
|
73
|
+
primary: '#5fafff',
|
|
74
|
+
accent: '#d7afff',
|
|
75
|
+
voice: 'clear, modern',
|
|
76
|
+
},
|
|
77
|
+
terminal: {
|
|
78
|
+
colorRoles: {
|
|
79
|
+
success: { truecolor: '#5faf87', ansi256: 72, noColor: 'bold' },
|
|
80
|
+
error: { truecolor: '#d75f5f', ansi256: 167, noColor: 'bold' },
|
|
81
|
+
warning: { truecolor: '#d7af5f', ansi256: 179, noColor: 'bold' },
|
|
82
|
+
info: { truecolor: '#5fafff', ansi256: 75, noColor: 'dim' },
|
|
83
|
+
muted: { truecolor: '#8a8a8a', ansi256: 245, noColor: 'dim' },
|
|
84
|
+
accent: { truecolor: '#d7afff', ansi256: 183, noColor: 'underline' },
|
|
85
|
+
header: { truecolor: null, ansi256: null, noColor: 'bold+upper' },
|
|
86
|
+
key: { truecolor: '#5fafff', ansi256: 75, noColor: 'plain' },
|
|
87
|
+
value: { truecolor: '#d0d0d0', ansi256: 252, noColor: 'plain' },
|
|
88
|
+
},
|
|
89
|
+
symbols: {
|
|
90
|
+
success: { unicode: '✔', ascii: 'OK' },
|
|
91
|
+
error: { unicode: '✖', ascii: 'x' },
|
|
92
|
+
warning: { unicode: '⚠', ascii: '!' },
|
|
93
|
+
info: { unicode: '●', ascii: '*' },
|
|
94
|
+
step: { unicode: '▶', ascii: '>' },
|
|
95
|
+
substep: { unicode: '↳', ascii: '-' },
|
|
96
|
+
active: { unicode: '❯', ascii: '>' },
|
|
97
|
+
},
|
|
98
|
+
splash: { style: 'wordmark-line', tagline: '' },
|
|
99
|
+
typography: {
|
|
100
|
+
header: 'bold + UPPERCASE',
|
|
101
|
+
title: 'bold + primary',
|
|
102
|
+
body: 'plain',
|
|
103
|
+
muted: 'dim',
|
|
104
|
+
},
|
|
105
|
+
},
|
|
106
|
+
};
|
|
107
|
+
|
|
108
|
+
/** Deep-merge a partial tokens object from dev.config.json over the defaults so a
|
|
109
|
+
* Tier-1 file (identity only) still yields a complete, renderable theme. */
|
|
110
|
+
export function mergeTokens(partial: unknown): BrandTokens {
|
|
111
|
+
const p = (partial ?? {}) as Partial<BrandTokens>;
|
|
112
|
+
const identity = { ...DEFAULT_TOKENS.identity, ...(p.identity ?? {}) };
|
|
113
|
+
const terminal = p.terminal
|
|
114
|
+
? {
|
|
115
|
+
colorRoles: { ...DEFAULT_TOKENS.terminal!.colorRoles, ...p.terminal.colorRoles },
|
|
116
|
+
symbols: { ...DEFAULT_TOKENS.terminal!.symbols, ...p.terminal.symbols },
|
|
117
|
+
splash: { ...DEFAULT_TOKENS.terminal!.splash, ...p.terminal.splash },
|
|
118
|
+
typography: { ...DEFAULT_TOKENS.terminal!.typography, ...p.terminal.typography },
|
|
119
|
+
}
|
|
120
|
+
: DEFAULT_TOKENS.terminal;
|
|
121
|
+
return { identity, terminal };
|
|
122
|
+
}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { Renderer } from '../theme/render';
|
|
2
|
+
import { Runner } from './runners';
|
|
3
|
+
import type { CommandDef } from '../registry';
|
|
4
|
+
|
|
5
|
+
/** Raised by a command to abort with an error card and exit code 1, mirroring the
|
|
6
|
+
* bash `fail()` helper. */
|
|
7
|
+
export class CliError extends Error {
|
|
8
|
+
constructor(
|
|
9
|
+
message: string,
|
|
10
|
+
public readonly action?: string,
|
|
11
|
+
) {
|
|
12
|
+
super(message);
|
|
13
|
+
this.name = 'CliError';
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
/** Raised for usage errors (bad/missing args, unknown flags) → exit code 2. */
|
|
18
|
+
export class UsageError extends Error {
|
|
19
|
+
constructor(message: string) {
|
|
20
|
+
super(message);
|
|
21
|
+
this.name = 'UsageError';
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export interface Ctx {
|
|
26
|
+
/** Human-facing renderer (writes to stderr). */
|
|
27
|
+
r: Renderer;
|
|
28
|
+
/** Global --json mode: machine output to stdout, no spinners/chrome. */
|
|
29
|
+
json: boolean;
|
|
30
|
+
/** Positional args after the command verb. */
|
|
31
|
+
args: string[];
|
|
32
|
+
/** Project prefix (for container names etc.), from dev.config.json. */
|
|
33
|
+
projectPrefix: string;
|
|
34
|
+
/** Declared native runners (surfaces, sidecars) from dev.config.json. */
|
|
35
|
+
runners: Runner[];
|
|
36
|
+
/** The merged command registry (built-ins + project commands) — the single source
|
|
37
|
+
* of truth shared by dispatch, `--help`, and completion. */
|
|
38
|
+
commands: CommandDef[];
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export function elapsedSince(startMs: number): string {
|
|
42
|
+
return `${Math.round((Date.now() - startMs) / 1000)}s`;
|
|
43
|
+
}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import * as fs from 'fs';
|
|
2
|
+
import * as path from 'path';
|
|
3
|
+
import { CONFIG_PATH, DEV_DIR } from './paths';
|
|
4
|
+
|
|
5
|
+
/** A project-owned command — a verb the project adds to `./dev` without touching the
|
|
6
|
+
* framework bundle. This is the composability seam: the shipped commands are a
|
|
7
|
+
* starting point, and a project grows the toolkit with every convenience it needs.
|
|
8
|
+
*
|
|
9
|
+
* Declared in either place (both carry the same shape):
|
|
10
|
+
* - inline in `.dev/dev.config.json` under a `commands` array, or
|
|
11
|
+
* - one-per-file under `.dev/commands/*.json` (easier to manage and diff).
|
|
12
|
+
*
|
|
13
|
+
* The CLI runs `run` as a subprocess (`bash -c`) with any extra args appended, so a
|
|
14
|
+
* project command is a shell one-liner or a call into a project script — no rebuild,
|
|
15
|
+
* no dependency added to the zero-dependency bundle. A project command whose `name`
|
|
16
|
+
* matches a built-in **shadows** it, so a project can redefine `start` for a stack the
|
|
17
|
+
* default lifecycle does not fit. These declarations are project-owned: `groundwork
|
|
18
|
+
* update` never overwrites them. */
|
|
19
|
+
export interface ProjectCommand {
|
|
20
|
+
name: string;
|
|
21
|
+
summary: string;
|
|
22
|
+
/** Display group in `--help`. Defaults to `PROJECT`. */
|
|
23
|
+
group?: string;
|
|
24
|
+
/** Shell command, run via `bash -c` from `cwd`; extra args are appended. */
|
|
25
|
+
run: string;
|
|
26
|
+
/** Working directory, relative to the repo root. */
|
|
27
|
+
cwd?: string;
|
|
28
|
+
/** Extra environment merged over the parent process env. */
|
|
29
|
+
env?: Record<string, string>;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
/** Command names are lowercase verbs (optionally namespaced with `:` or `-`), matching
|
|
33
|
+
* the built-in vocabulary so completion and help stay legible. */
|
|
34
|
+
const NAME_RX = /^[a-z][a-z0-9:-]*$/;
|
|
35
|
+
|
|
36
|
+
/** Coerce one untrusted entry into a validated ProjectCommand, or null if malformed.
|
|
37
|
+
* Dropping bad entries (rather than throwing) means one broken command never breaks
|
|
38
|
+
* the whole CLI. */
|
|
39
|
+
function coerce(item: unknown): ProjectCommand | null {
|
|
40
|
+
if (!item || typeof item !== 'object') return null;
|
|
41
|
+
const c = item as Record<string, unknown>;
|
|
42
|
+
if (typeof c.name !== 'string' || !NAME_RX.test(c.name)) return null;
|
|
43
|
+
if (typeof c.run !== 'string' || !c.run.trim()) return null;
|
|
44
|
+
return {
|
|
45
|
+
name: c.name,
|
|
46
|
+
summary: typeof c.summary === 'string' && c.summary.trim() ? c.summary : '(project command)',
|
|
47
|
+
group:
|
|
48
|
+
typeof c.group === 'string' && c.group.trim() ? c.group.trim().toUpperCase() : 'PROJECT',
|
|
49
|
+
run: c.run,
|
|
50
|
+
cwd: typeof c.cwd === 'string' ? c.cwd : undefined,
|
|
51
|
+
env:
|
|
52
|
+
c.env && typeof c.env === 'object' && !Array.isArray(c.env)
|
|
53
|
+
? (c.env as Record<string, string>)
|
|
54
|
+
: undefined,
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/** Discover project commands from both sources. A `.dev/commands/*.json` file wins over
|
|
59
|
+
* an inline `commands` entry of the same name, so a project can promote an inline
|
|
60
|
+
* command to its own file without a duplicate. Any read/parse failure degrades to
|
|
61
|
+
* fewer commands rather than breaking the CLI. */
|
|
62
|
+
export function loadProjectCommands(): ProjectCommand[] {
|
|
63
|
+
const byName = new Map<string, ProjectCommand>();
|
|
64
|
+
|
|
65
|
+
// Inline: .dev/dev.config.json → commands: [ ... ]
|
|
66
|
+
try {
|
|
67
|
+
if (fs.existsSync(CONFIG_PATH)) {
|
|
68
|
+
const raw = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf8')) as { commands?: unknown };
|
|
69
|
+
if (Array.isArray(raw.commands)) {
|
|
70
|
+
for (const item of raw.commands) {
|
|
71
|
+
const c = coerce(item);
|
|
72
|
+
if (c) byName.set(c.name, c);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
} catch {
|
|
77
|
+
/* no inline commands */
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Per-file: .dev/commands/*.json
|
|
81
|
+
try {
|
|
82
|
+
const dir = path.join(DEV_DIR, 'commands');
|
|
83
|
+
if (fs.existsSync(dir)) {
|
|
84
|
+
for (const f of fs.readdirSync(dir).sort()) {
|
|
85
|
+
if (!f.endsWith('.json')) continue;
|
|
86
|
+
try {
|
|
87
|
+
const c = coerce(JSON.parse(fs.readFileSync(path.join(dir, f), 'utf8')));
|
|
88
|
+
if (c) byName.set(c.name, c);
|
|
89
|
+
} catch {
|
|
90
|
+
/* skip a malformed file */
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
} catch {
|
|
95
|
+
/* no commands dir */
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
return [...byName.values()];
|
|
99
|
+
}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import * as path from 'path';
|
|
2
|
+
import * as fs from 'fs';
|
|
3
|
+
|
|
4
|
+
/** The project root. The `dev` launcher sets DEV_ROOT to its own directory; we fall
|
|
5
|
+
* back to walking up from cwd to find a project marker so the CLI also works when
|
|
6
|
+
* invoked directly. */
|
|
7
|
+
export const ROOT: string = (() => {
|
|
8
|
+
if (process.env.DEV_ROOT) return process.env.DEV_ROOT;
|
|
9
|
+
let dir = process.cwd();
|
|
10
|
+
for (let i = 0; i < 12; i += 1) {
|
|
11
|
+
if (
|
|
12
|
+
fs.existsSync(path.join(dir, 'dev')) ||
|
|
13
|
+
fs.existsSync(path.join(dir, 'docker-compose.yml')) ||
|
|
14
|
+
fs.existsSync(path.join(dir, '.groundwork'))
|
|
15
|
+
) {
|
|
16
|
+
return dir;
|
|
17
|
+
}
|
|
18
|
+
const parent = path.dirname(dir);
|
|
19
|
+
if (parent === dir) break;
|
|
20
|
+
dir = parent;
|
|
21
|
+
}
|
|
22
|
+
return process.cwd();
|
|
23
|
+
})();
|
|
24
|
+
|
|
25
|
+
export const DEV_DIR = path.join(ROOT, '.dev');
|
|
26
|
+
export const PID_DIR = path.join(DEV_DIR, 'pids');
|
|
27
|
+
export const LOG_DIR = path.join(DEV_DIR, 'logs');
|
|
28
|
+
export const SERVICES_DIR = path.join(ROOT, 'services');
|
|
29
|
+
export const TESTS_DIR = path.join(ROOT, 'tests');
|
|
30
|
+
export const DOCS_DIR = path.join(ROOT, 'docs');
|
|
31
|
+
export const CONFIG_PATH = path.join(DEV_DIR, 'dev.config.json');
|
|
32
|
+
/** Machine-readable twin of docs/surfaces.md (surface registry + capability ledger). */
|
|
33
|
+
export const GROUNDWORK_SURFACES_FILE = path.join(ROOT, '.groundwork', 'surfaces.json');
|
|
34
|
+
|
|
35
|
+
export function ensureDirs(): void {
|
|
36
|
+
fs.mkdirSync(PID_DIR, { recursive: true });
|
|
37
|
+
fs.mkdirSync(LOG_DIR, { recursive: true });
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export function pidFile(svc: string): string {
|
|
41
|
+
return path.join(PID_DIR, `${svc}.pid`);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export function logFile(svc: string): string {
|
|
45
|
+
return path.join(LOG_DIR, `${svc}.log`);
|
|
46
|
+
}
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import { spawn, spawnSync, SpawnSyncOptions } from 'child_process';
|
|
2
|
+
import * as http from 'http';
|
|
3
|
+
|
|
4
|
+
export interface RunResult {
|
|
5
|
+
status: number;
|
|
6
|
+
stdout: string;
|
|
7
|
+
stderr: string;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
/** Run a command synchronously, inheriting stdio (for user-visible subprocesses
|
|
11
|
+
* like `docker compose up`). Returns the exit status. */
|
|
12
|
+
export function run(cmd: string, args: string[], opts: SpawnSyncOptions = {}): number {
|
|
13
|
+
const r = spawnSync(cmd, args, { stdio: 'inherit', ...opts });
|
|
14
|
+
return r.status ?? 1;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
/** Run a command synchronously and capture its output (for parsing, e.g.
|
|
18
|
+
* `docker compose ps`). */
|
|
19
|
+
export function capture(cmd: string, args: string[], opts: SpawnSyncOptions = {}): RunResult {
|
|
20
|
+
const r = spawnSync(cmd, args, { encoding: 'utf8', ...opts });
|
|
21
|
+
return {
|
|
22
|
+
status: r.status ?? (r.error ? 127 : 1),
|
|
23
|
+
stdout: (r.stdout as string) ?? '',
|
|
24
|
+
stderr: (r.stderr as string) ?? '',
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/** Run a shell command string (for piped/compound commands ported from bash). */
|
|
29
|
+
export function sh(command: string, opts: SpawnSyncOptions = {}): RunResult {
|
|
30
|
+
const r = spawnSync(command, { shell: true, encoding: 'utf8', ...opts });
|
|
31
|
+
return {
|
|
32
|
+
status: r.status ?? (r.error ? 127 : 1),
|
|
33
|
+
stdout: (r.stdout as string) ?? '',
|
|
34
|
+
stderr: (r.stderr as string) ?? '',
|
|
35
|
+
};
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/** True if a command is resolvable on PATH. Passes a single command string (never
|
|
39
|
+
* args with shell:true) to avoid Node's DEP0190 shell-injection warning. */
|
|
40
|
+
export function commandExists(cmd: string): boolean {
|
|
41
|
+
const safe = cmd.replace(/[^a-zA-Z0-9._-]/g, '');
|
|
42
|
+
const probe = process.platform === 'win32' ? `where ${safe}` : `command -v ${safe}`;
|
|
43
|
+
const r = spawnSync(probe, { shell: true, stdio: 'ignore' });
|
|
44
|
+
return (r.status ?? 1) === 0;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/** `docker compose` (v2) is the only supported form. */
|
|
48
|
+
export const COMPOSE = ['compose'];
|
|
49
|
+
|
|
50
|
+
export function dockerComposeCapture(args: string[]): RunResult {
|
|
51
|
+
return capture('docker', [...COMPOSE, ...args]);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
export function dockerComposeRun(args: string[]): number {
|
|
55
|
+
return run('docker', [...COMPOSE, ...args]);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/** Spawn a detached background process writing to a log file; returns its PID.
|
|
59
|
+
* Optional cwd/env let a declared runner launch from its own directory with a
|
|
60
|
+
* merged environment (native runners, sidecars). */
|
|
61
|
+
export function spawnBackground(
|
|
62
|
+
command: string,
|
|
63
|
+
logStream: number,
|
|
64
|
+
opts: { cwd?: string; env?: NodeJS.ProcessEnv } = {},
|
|
65
|
+
): number {
|
|
66
|
+
const child = spawn('bash', ['-c', command], {
|
|
67
|
+
stdio: ['ignore', logStream, logStream],
|
|
68
|
+
detached: true,
|
|
69
|
+
cwd: opts.cwd,
|
|
70
|
+
env: opts.env,
|
|
71
|
+
});
|
|
72
|
+
child.unref();
|
|
73
|
+
return child.pid ?? -1;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
export function sleep(ms: number): Promise<void> {
|
|
77
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
/** Probe an HTTP endpoint and resolve with its status code, or 0 on a connection
|
|
81
|
+
* error/timeout. Never throws — used by health/doctor connectivity checks so a
|
|
82
|
+
* hung or absent port renders as a row, not a crash. Localhost-only (http). */
|
|
83
|
+
export function httpProbe(url: string, timeoutMs = 3000): Promise<number> {
|
|
84
|
+
return new Promise((resolve) => {
|
|
85
|
+
let settled = false;
|
|
86
|
+
const done = (code: number): void => {
|
|
87
|
+
if (settled) return;
|
|
88
|
+
settled = true;
|
|
89
|
+
resolve(code);
|
|
90
|
+
};
|
|
91
|
+
try {
|
|
92
|
+
const req = http.get(url, (res) => {
|
|
93
|
+
const code = res.statusCode ?? 0;
|
|
94
|
+
res.resume(); // drain so the socket can close
|
|
95
|
+
done(code);
|
|
96
|
+
});
|
|
97
|
+
req.setTimeout(timeoutMs, () => {
|
|
98
|
+
req.destroy();
|
|
99
|
+
done(0);
|
|
100
|
+
});
|
|
101
|
+
req.on('error', () => done(0));
|
|
102
|
+
} catch {
|
|
103
|
+
done(0);
|
|
104
|
+
}
|
|
105
|
+
});
|
|
106
|
+
}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Interactive prompts — arrow-key selection and validated text input — built on Node's
|
|
3
|
+
* raw-mode stdin and ANSI, so they add no dependencies and bundle to nothing.
|
|
4
|
+
*
|
|
5
|
+
* Every prompt is TTY-gated by `isInteractive()`. Callers must check it first and fall
|
|
6
|
+
* back to a non-interactive path (the agent contract: the CLI never blocks off a TTY).
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import * as readline from 'readline';
|
|
10
|
+
import { Painter } from '../theme/color';
|
|
11
|
+
|
|
12
|
+
export function isInteractive(): boolean {
|
|
13
|
+
return Boolean(process.stdin.isTTY && process.stdout.isTTY);
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
interface Key {
|
|
17
|
+
name?: string;
|
|
18
|
+
ctrl?: boolean;
|
|
19
|
+
sequence?: string;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/** Arrow-key single-select. Renders to stderr; returns the chosen value. */
|
|
23
|
+
export function selectPrompt<T>(
|
|
24
|
+
painter: Painter,
|
|
25
|
+
message: string,
|
|
26
|
+
choices: Array<{ label: string; value: T; hint?: string }>,
|
|
27
|
+
): Promise<T> {
|
|
28
|
+
return new Promise((resolve, reject) => {
|
|
29
|
+
let index = 0;
|
|
30
|
+
const out = process.stderr;
|
|
31
|
+
readline.emitKeypressEvents(process.stdin);
|
|
32
|
+
process.stdin.setRawMode(true);
|
|
33
|
+
process.stdin.resume();
|
|
34
|
+
out.write('\x1b[?25l'); // hide cursor
|
|
35
|
+
|
|
36
|
+
const render = (first: boolean) => {
|
|
37
|
+
if (!first) out.write(`\x1b[${choices.length + 1}A`); // move up to redraw
|
|
38
|
+
out.write(`\x1b[2K ${painter.bold(message)}\n`);
|
|
39
|
+
choices.forEach((c, i) => {
|
|
40
|
+
const active = i === index;
|
|
41
|
+
const marker = active ? painter.primary('❯') : ' ';
|
|
42
|
+
const label = active ? painter.primary(c.label) : c.label;
|
|
43
|
+
const hint = c.hint ? painter.dim(` ${c.hint}`) : '';
|
|
44
|
+
out.write(`\x1b[2K ${marker} ${label}${hint}\n`);
|
|
45
|
+
});
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
const cleanup = () => {
|
|
49
|
+
process.stdin.setRawMode(false);
|
|
50
|
+
process.stdin.pause();
|
|
51
|
+
process.stdin.removeListener('keypress', onKey);
|
|
52
|
+
out.write('\x1b[?25h'); // restore cursor
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
const onKey = (_str: string, key: Key) => {
|
|
56
|
+
if (key.ctrl && key.name === 'c') {
|
|
57
|
+
cleanup();
|
|
58
|
+
process.exit(130);
|
|
59
|
+
} else if (key.name === 'up' || key.name === 'k') {
|
|
60
|
+
index = (index - 1 + choices.length) % choices.length;
|
|
61
|
+
render(false);
|
|
62
|
+
} else if (key.name === 'down' || key.name === 'j') {
|
|
63
|
+
index = (index + 1) % choices.length;
|
|
64
|
+
render(false);
|
|
65
|
+
} else if (key.name === 'return' || key.name === 'enter') {
|
|
66
|
+
cleanup();
|
|
67
|
+
resolve(choices[index].value);
|
|
68
|
+
}
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
render(true);
|
|
72
|
+
process.stdin.on('keypress', onKey);
|
|
73
|
+
process.stdin.on('error', (e) => {
|
|
74
|
+
cleanup();
|
|
75
|
+
reject(e);
|
|
76
|
+
});
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
/** Validated single-line text input. Re-prompts until `validate` returns null. */
|
|
81
|
+
export function textPrompt(
|
|
82
|
+
painter: Painter,
|
|
83
|
+
message: string,
|
|
84
|
+
validate?: (value: string) => string | null,
|
|
85
|
+
): Promise<string> {
|
|
86
|
+
const rl = readline.createInterface({ input: process.stdin, output: process.stderr });
|
|
87
|
+
const ask = (): Promise<string> =>
|
|
88
|
+
new Promise((resolve) => {
|
|
89
|
+
rl.question(` ${painter.bold(message)} `, (answer) => resolve(answer.trim()));
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
return (async () => {
|
|
93
|
+
try {
|
|
94
|
+
for (;;) {
|
|
95
|
+
const value = await ask();
|
|
96
|
+
const err = validate ? validate(value) : null;
|
|
97
|
+
if (!err) {
|
|
98
|
+
rl.close();
|
|
99
|
+
return value;
|
|
100
|
+
}
|
|
101
|
+
process.stderr.write(` ${painter.paint('error', '✖')} ${err}\n`);
|
|
102
|
+
}
|
|
103
|
+
} catch (e) {
|
|
104
|
+
rl.close();
|
|
105
|
+
throw e;
|
|
106
|
+
}
|
|
107
|
+
})();
|
|
108
|
+
}
|