@contractspec/bundle.workspace 1.44.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/LICENSE +21 -0
- package/README.md +79 -0
- package/dist/_virtual/rolldown_runtime.js +36 -0
- package/dist/adapters/ai.d.ts +12 -0
- package/dist/adapters/ai.d.ts.map +1 -0
- package/dist/adapters/ai.js +83 -0
- package/dist/adapters/ai.js.map +1 -0
- package/dist/adapters/factory.d.ts +29 -0
- package/dist/adapters/factory.d.ts.map +1 -0
- package/dist/adapters/factory.js +37 -0
- package/dist/adapters/factory.js.map +1 -0
- package/dist/adapters/fs.d.ts +11 -0
- package/dist/adapters/fs.d.ts.map +1 -0
- package/dist/adapters/fs.js +131 -0
- package/dist/adapters/fs.js.map +1 -0
- package/dist/adapters/git.d.ts +11 -0
- package/dist/adapters/git.d.ts.map +1 -0
- package/dist/adapters/git.js +55 -0
- package/dist/adapters/git.js.map +1 -0
- package/dist/adapters/index.d.ts +7 -0
- package/dist/adapters/index.js +7 -0
- package/dist/adapters/logger.d.ts +18 -0
- package/dist/adapters/logger.d.ts.map +1 -0
- package/dist/adapters/logger.js +81 -0
- package/dist/adapters/logger.js.map +1 -0
- package/dist/adapters/watcher.d.ts +11 -0
- package/dist/adapters/watcher.d.ts.map +1 -0
- package/dist/adapters/watcher.js +74 -0
- package/dist/adapters/watcher.js.map +1 -0
- package/dist/adapters/workspace.d.ts +148 -0
- package/dist/adapters/workspace.d.ts.map +1 -0
- package/dist/adapters/workspace.js +275 -0
- package/dist/adapters/workspace.js.map +1 -0
- package/dist/ai/agents/claude-code-agent.d.ts +22 -0
- package/dist/ai/agents/claude-code-agent.d.ts.map +1 -0
- package/dist/ai/agents/claude-code-agent.js +182 -0
- package/dist/ai/agents/claude-code-agent.js.map +1 -0
- package/dist/ai/agents/cursor-agent.d.ts +68 -0
- package/dist/ai/agents/cursor-agent.d.ts.map +1 -0
- package/dist/ai/agents/cursor-agent.js +436 -0
- package/dist/ai/agents/cursor-agent.js.map +1 -0
- package/dist/ai/agents/index.js +5 -0
- package/dist/ai/agents/openai-codex-agent.d.ts +22 -0
- package/dist/ai/agents/openai-codex-agent.d.ts.map +1 -0
- package/dist/ai/agents/openai-codex-agent.js +167 -0
- package/dist/ai/agents/openai-codex-agent.js.map +1 -0
- package/dist/ai/agents/orchestrator.d.ts +50 -0
- package/dist/ai/agents/orchestrator.d.ts.map +1 -0
- package/dist/ai/agents/orchestrator.js +143 -0
- package/dist/ai/agents/orchestrator.js.map +1 -0
- package/dist/ai/agents/simple-agent.d.ts +17 -0
- package/dist/ai/agents/simple-agent.d.ts.map +1 -0
- package/dist/ai/agents/simple-agent.js +92 -0
- package/dist/ai/agents/simple-agent.js.map +1 -0
- package/dist/ai/agents/types.d.ts +36 -0
- package/dist/ai/agents/types.d.ts.map +1 -0
- package/dist/ai/client.d.ts +83 -0
- package/dist/ai/client.d.ts.map +1 -0
- package/dist/ai/client.js +163 -0
- package/dist/ai/client.js.map +1 -0
- package/dist/ai/index.d.ts +17 -0
- package/dist/ai/index.d.ts.map +1 -0
- package/dist/ai/index.js +28 -0
- package/dist/ai/index.js.map +1 -0
- package/dist/ai/prompts/code-generation.d.ts +26 -0
- package/dist/ai/prompts/code-generation.d.ts.map +1 -0
- package/dist/ai/prompts/code-generation.js +143 -0
- package/dist/ai/prompts/code-generation.js.map +1 -0
- package/dist/ai/prompts/index.d.ts +10 -0
- package/dist/ai/prompts/index.d.ts.map +1 -0
- package/dist/ai/prompts/index.js +13 -0
- package/dist/ai/prompts/index.js.map +1 -0
- package/dist/ai/prompts/spec-creation.d.ts +29 -0
- package/dist/ai/prompts/spec-creation.d.ts.map +1 -0
- package/dist/ai/prompts/spec-creation.js +111 -0
- package/dist/ai/prompts/spec-creation.js.map +1 -0
- package/dist/ai/providers.d.ts +29 -0
- package/dist/ai/providers.d.ts.map +1 -0
- package/dist/ai/providers.js +39 -0
- package/dist/ai/providers.js.map +1 -0
- package/dist/formatters/index.d.ts +11 -0
- package/dist/formatters/index.d.ts.map +1 -0
- package/dist/formatters/index.js +19 -0
- package/dist/formatters/index.js.map +1 -0
- package/dist/formatters/json.d.ts +89 -0
- package/dist/formatters/json.d.ts.map +1 -0
- package/dist/formatters/json.js +72 -0
- package/dist/formatters/json.js.map +1 -0
- package/dist/formatters/sarif.d.ts +101 -0
- package/dist/formatters/sarif.d.ts.map +1 -0
- package/dist/formatters/sarif.js +163 -0
- package/dist/formatters/sarif.js.map +1 -0
- package/dist/formatters/text.d.ts +35 -0
- package/dist/formatters/text.d.ts.map +1 -0
- package/dist/formatters/text.js +209 -0
- package/dist/formatters/text.js.map +1 -0
- package/dist/index.d.ts +82 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +204 -0
- package/dist/index.js.map +1 -0
- package/dist/ports/ai.d.ts +59 -0
- package/dist/ports/ai.d.ts.map +1 -0
- package/dist/ports/fs.d.ts +81 -0
- package/dist/ports/fs.d.ts.map +1 -0
- package/dist/ports/git.d.ts +33 -0
- package/dist/ports/git.d.ts.map +1 -0
- package/dist/ports/index.d.ts +5 -0
- package/dist/ports/logger.d.ts +88 -0
- package/dist/ports/logger.d.ts.map +1 -0
- package/dist/ports/watcher.d.ts +52 -0
- package/dist/ports/watcher.d.ts.map +1 -0
- package/dist/services/agent-guide/adapters/claude-code.d.ts +35 -0
- package/dist/services/agent-guide/adapters/claude-code.d.ts.map +1 -0
- package/dist/services/agent-guide/adapters/claude-code.js +144 -0
- package/dist/services/agent-guide/adapters/claude-code.js.map +1 -0
- package/dist/services/agent-guide/adapters/cursor-cli.d.ts +39 -0
- package/dist/services/agent-guide/adapters/cursor-cli.d.ts.map +1 -0
- package/dist/services/agent-guide/adapters/cursor-cli.js +135 -0
- package/dist/services/agent-guide/adapters/cursor-cli.js.map +1 -0
- package/dist/services/agent-guide/adapters/generic-mcp.d.ts +53 -0
- package/dist/services/agent-guide/adapters/generic-mcp.d.ts.map +1 -0
- package/dist/services/agent-guide/adapters/generic-mcp.js +159 -0
- package/dist/services/agent-guide/adapters/generic-mcp.js.map +1 -0
- package/dist/services/agent-guide/adapters/index.d.ts +23 -0
- package/dist/services/agent-guide/adapters/index.d.ts.map +1 -0
- package/dist/services/agent-guide/adapters/index.js +31 -0
- package/dist/services/agent-guide/adapters/index.js.map +1 -0
- package/dist/services/agent-guide/agent-guide-service.d.ts +56 -0
- package/dist/services/agent-guide/agent-guide-service.d.ts.map +1 -0
- package/dist/services/agent-guide/agent-guide-service.js +147 -0
- package/dist/services/agent-guide/agent-guide-service.js.map +1 -0
- package/dist/services/agent-guide/index.d.ts +6 -0
- package/dist/services/agent-guide/index.js +5 -0
- package/dist/services/agent-guide/types.d.ts +58 -0
- package/dist/services/agent-guide/types.d.ts.map +1 -0
- package/dist/services/build.d.ts +59 -0
- package/dist/services/build.d.ts.map +1 -0
- package/dist/services/build.js +140 -0
- package/dist/services/build.js.map +1 -0
- package/dist/services/ci-check/ci-check-service.d.ts +16 -0
- package/dist/services/ci-check/ci-check-service.d.ts.map +1 -0
- package/dist/services/ci-check/ci-check-service.js +392 -0
- package/dist/services/ci-check/ci-check-service.js.map +1 -0
- package/dist/services/ci-check/index.d.ts +2 -0
- package/dist/services/ci-check/index.js +2 -0
- package/dist/services/ci-check/types.d.ts +143 -0
- package/dist/services/ci-check/types.d.ts.map +1 -0
- package/dist/services/ci-check/types.js +29 -0
- package/dist/services/ci-check/types.js.map +1 -0
- package/dist/services/clean.d.ts +41 -0
- package/dist/services/clean.d.ts.map +1 -0
- package/dist/services/clean.js +72 -0
- package/dist/services/clean.js.map +1 -0
- package/dist/services/config.d.ts +26 -0
- package/dist/services/config.d.ts.map +1 -0
- package/dist/services/config.js +77 -0
- package/dist/services/config.js.map +1 -0
- package/dist/services/deps.d.ts +53 -0
- package/dist/services/deps.d.ts.map +1 -0
- package/dist/services/deps.js +62 -0
- package/dist/services/deps.js.map +1 -0
- package/dist/services/diff.d.ts +34 -0
- package/dist/services/diff.d.ts.map +1 -0
- package/dist/services/diff.js +34 -0
- package/dist/services/diff.js.map +1 -0
- package/dist/services/doctor/checks/ai.js +119 -0
- package/dist/services/doctor/checks/ai.js.map +1 -0
- package/dist/services/doctor/checks/cli.js +147 -0
- package/dist/services/doctor/checks/cli.js.map +1 -0
- package/dist/services/doctor/checks/config.js +171 -0
- package/dist/services/doctor/checks/config.js.map +1 -0
- package/dist/services/doctor/checks/deps.js +247 -0
- package/dist/services/doctor/checks/deps.js.map +1 -0
- package/dist/services/doctor/checks/index.js +6 -0
- package/dist/services/doctor/checks/mcp.js +145 -0
- package/dist/services/doctor/checks/mcp.js.map +1 -0
- package/dist/services/doctor/checks/workspace.js +244 -0
- package/dist/services/doctor/checks/workspace.js.map +1 -0
- package/dist/services/doctor/doctor-service.d.ts +24 -0
- package/dist/services/doctor/doctor-service.d.ts.map +1 -0
- package/dist/services/doctor/doctor-service.js +116 -0
- package/dist/services/doctor/doctor-service.js.map +1 -0
- package/dist/services/doctor/index.d.ts +2 -0
- package/dist/services/doctor/index.js +2 -0
- package/dist/services/doctor/types.d.ts +118 -0
- package/dist/services/doctor/types.d.ts.map +1 -0
- package/dist/services/doctor/types.js +27 -0
- package/dist/services/doctor/types.js.map +1 -0
- package/dist/services/impact/formatters.d.ts +27 -0
- package/dist/services/impact/formatters.d.ts.map +1 -0
- package/dist/services/impact/formatters.js +111 -0
- package/dist/services/impact/formatters.js.map +1 -0
- package/dist/services/impact/impact-detection-service.d.ts +22 -0
- package/dist/services/impact/impact-detection-service.d.ts.map +1 -0
- package/dist/services/impact/impact-detection-service.js +96 -0
- package/dist/services/impact/impact-detection-service.js.map +1 -0
- package/dist/services/impact/index.d.ts +11 -0
- package/dist/services/impact/index.d.ts.map +1 -0
- package/dist/services/impact/index.js +16 -0
- package/dist/services/impact/index.js.map +1 -0
- package/dist/services/impact/types.d.ts +58 -0
- package/dist/services/impact/types.d.ts.map +1 -0
- package/dist/services/implementation/discovery.d.ts +30 -0
- package/dist/services/implementation/discovery.d.ts.map +1 -0
- package/dist/services/implementation/discovery.js +144 -0
- package/dist/services/implementation/discovery.js.map +1 -0
- package/dist/services/implementation/index.d.ts +3 -0
- package/dist/services/implementation/index.js +2 -0
- package/dist/services/implementation/resolver.d.ts +44 -0
- package/dist/services/implementation/resolver.d.ts.map +1 -0
- package/dist/services/implementation/resolver.js +224 -0
- package/dist/services/implementation/resolver.js.map +1 -0
- package/dist/services/implementation/types.d.ts +79 -0
- package/dist/services/implementation/types.d.ts.map +1 -0
- package/dist/services/index.d.ts +60 -0
- package/dist/services/index.js +57 -0
- package/dist/services/integrity-diagram.d.ts +36 -0
- package/dist/services/integrity-diagram.d.ts.map +1 -0
- package/dist/services/integrity-diagram.js +275 -0
- package/dist/services/integrity-diagram.js.map +1 -0
- package/dist/services/integrity.d.ts +134 -0
- package/dist/services/integrity.d.ts.map +1 -0
- package/dist/services/integrity.js +272 -0
- package/dist/services/integrity.js.map +1 -0
- package/dist/services/list.d.ts +31 -0
- package/dist/services/list.d.ts.map +1 -0
- package/dist/services/list.js +36 -0
- package/dist/services/list.js.map +1 -0
- package/dist/services/openapi/export-service.d.ts +53 -0
- package/dist/services/openapi/export-service.d.ts.map +1 -0
- package/dist/services/openapi/export-service.js +50 -0
- package/dist/services/openapi/export-service.js.map +1 -0
- package/dist/services/openapi/import-service.d.ts +17 -0
- package/dist/services/openapi/import-service.d.ts.map +1 -0
- package/dist/services/openapi/import-service.js +74 -0
- package/dist/services/openapi/import-service.js.map +1 -0
- package/dist/services/openapi/index.d.ts +5 -0
- package/dist/services/openapi/index.js +4 -0
- package/dist/services/openapi/sync-service.d.ts +17 -0
- package/dist/services/openapi/sync-service.d.ts.map +1 -0
- package/dist/services/openapi/sync-service.js +120 -0
- package/dist/services/openapi/sync-service.js.map +1 -0
- package/dist/services/openapi/types.d.ts +162 -0
- package/dist/services/openapi/types.d.ts.map +1 -0
- package/dist/services/openapi/validate-service.d.ts +16 -0
- package/dist/services/openapi/validate-service.d.ts.map +1 -0
- package/dist/services/openapi/validate-service.js +130 -0
- package/dist/services/openapi/validate-service.js.map +1 -0
- package/dist/services/quickstart/dependencies.d.ts +31 -0
- package/dist/services/quickstart/dependencies.d.ts.map +1 -0
- package/dist/services/quickstart/dependencies.js +57 -0
- package/dist/services/quickstart/dependencies.js.map +1 -0
- package/dist/services/quickstart/index.js +2 -0
- package/dist/services/quickstart/quickstart-service.d.ts +20 -0
- package/dist/services/quickstart/quickstart-service.d.ts.map +1 -0
- package/dist/services/quickstart/quickstart-service.js +196 -0
- package/dist/services/quickstart/quickstart-service.js.map +1 -0
- package/dist/services/quickstart/types.d.ts +81 -0
- package/dist/services/quickstart/types.d.ts.map +1 -0
- package/dist/services/regenerator.d.ts +18 -0
- package/dist/services/regenerator.d.ts.map +1 -0
- package/dist/services/regenerator.js +23 -0
- package/dist/services/regenerator.js.map +1 -0
- package/dist/services/registry.d.ts +53 -0
- package/dist/services/registry.d.ts.map +1 -0
- package/dist/services/registry.js +74 -0
- package/dist/services/registry.js.map +1 -0
- package/dist/services/setup/config-generators.d.ts +42 -0
- package/dist/services/setup/config-generators.d.ts.map +1 -0
- package/dist/services/setup/config-generators.js +238 -0
- package/dist/services/setup/config-generators.js.map +1 -0
- package/dist/services/setup/file-merger.d.ts +27 -0
- package/dist/services/setup/file-merger.d.ts.map +1 -0
- package/dist/services/setup/file-merger.js +61 -0
- package/dist/services/setup/file-merger.js.map +1 -0
- package/dist/services/setup/index.js +4 -0
- package/dist/services/setup/setup-service.d.ts +12 -0
- package/dist/services/setup/setup-service.d.ts.map +1 -0
- package/dist/services/setup/setup-service.js +96 -0
- package/dist/services/setup/setup-service.js.map +1 -0
- package/dist/services/setup/targets/agents-md.js +47 -0
- package/dist/services/setup/targets/agents-md.js.map +1 -0
- package/dist/services/setup/targets/cli-config.js +60 -0
- package/dist/services/setup/targets/cli-config.js.map +1 -0
- package/dist/services/setup/targets/cursor-rules.js +48 -0
- package/dist/services/setup/targets/cursor-rules.js.map +1 -0
- package/dist/services/setup/targets/mcp-claude.js +60 -0
- package/dist/services/setup/targets/mcp-claude.js.map +1 -0
- package/dist/services/setup/targets/mcp-cursor.js +59 -0
- package/dist/services/setup/targets/mcp-cursor.js.map +1 -0
- package/dist/services/setup/targets/vscode-settings.js +63 -0
- package/dist/services/setup/targets/vscode-settings.js.map +1 -0
- package/dist/services/setup/types.d.ts +85 -0
- package/dist/services/setup/types.d.ts.map +1 -0
- package/dist/services/setup/types.js +27 -0
- package/dist/services/setup/types.js.map +1 -0
- package/dist/services/sync.d.ts +41 -0
- package/dist/services/sync.d.ts.map +1 -0
- package/dist/services/sync.js +63 -0
- package/dist/services/sync.js.map +1 -0
- package/dist/services/test.d.ts +15 -0
- package/dist/services/test.d.ts.map +1 -0
- package/dist/services/test.js +30 -0
- package/dist/services/test.js.map +1 -0
- package/dist/services/validate-implementation.d.ts +32 -0
- package/dist/services/validate-implementation.d.ts.map +1 -0
- package/dist/services/validate-implementation.js +64 -0
- package/dist/services/validate-implementation.js.map +1 -0
- package/dist/services/validate.d.ts +41 -0
- package/dist/services/validate.d.ts.map +1 -0
- package/dist/services/validate.js +48 -0
- package/dist/services/validate.js.map +1 -0
- package/dist/services/verification-cache/adapters/filesystem.d.ts +46 -0
- package/dist/services/verification-cache/adapters/filesystem.d.ts.map +1 -0
- package/dist/services/verification-cache/adapters/filesystem.js +120 -0
- package/dist/services/verification-cache/adapters/filesystem.js.map +1 -0
- package/dist/services/verification-cache/adapters/in-memory.d.ts +27 -0
- package/dist/services/verification-cache/adapters/in-memory.d.ts.map +1 -0
- package/dist/services/verification-cache/adapters/in-memory.js +46 -0
- package/dist/services/verification-cache/adapters/in-memory.js.map +1 -0
- package/dist/services/verification-cache/adapters/index.d.ts +3 -0
- package/dist/services/verification-cache/adapters/index.js +3 -0
- package/dist/services/verification-cache/adapters/workspace-state.d.ts +49 -0
- package/dist/services/verification-cache/adapters/workspace-state.d.ts.map +1 -0
- package/dist/services/verification-cache/adapters/workspace-state.js +91 -0
- package/dist/services/verification-cache/adapters/workspace-state.js.map +1 -0
- package/dist/services/verification-cache/cache-service.d.ts +70 -0
- package/dist/services/verification-cache/cache-service.d.ts.map +1 -0
- package/dist/services/verification-cache/cache-service.js +256 -0
- package/dist/services/verification-cache/cache-service.js.map +1 -0
- package/dist/services/verification-cache/index.d.ts +6 -0
- package/dist/services/verification-cache/index.js +6 -0
- package/dist/services/verification-cache/types.d.ts +124 -0
- package/dist/services/verification-cache/types.d.ts.map +1 -0
- package/dist/services/verification-cache/types.js +16 -0
- package/dist/services/verification-cache/types.js.map +1 -0
- package/dist/services/verify/ai-verifier.d.ts +25 -0
- package/dist/services/verify/ai-verifier.d.ts.map +1 -0
- package/dist/services/verify/ai-verifier.js +403 -0
- package/dist/services/verify/ai-verifier.js.map +1 -0
- package/dist/services/verify/behavior-verifier.d.ts +12 -0
- package/dist/services/verify/behavior-verifier.d.ts.map +1 -0
- package/dist/services/verify/behavior-verifier.js +186 -0
- package/dist/services/verify/behavior-verifier.js.map +1 -0
- package/dist/services/verify/index.d.ts +5 -0
- package/dist/services/verify/index.js +4 -0
- package/dist/services/verify/structure-verifier.d.ts +12 -0
- package/dist/services/verify/structure-verifier.d.ts.map +1 -0
- package/dist/services/verify/structure-verifier.js +196 -0
- package/dist/services/verify/structure-verifier.js.map +1 -0
- package/dist/services/verify/types.d.ts +137 -0
- package/dist/services/verify/types.d.ts.map +1 -0
- package/dist/services/verify/verify-service.d.ts +60 -0
- package/dist/services/verify/verify-service.d.ts.map +1 -0
- package/dist/services/verify/verify-service.js +204 -0
- package/dist/services/verify/verify-service.js.map +1 -0
- package/dist/services/watch.d.ts +25 -0
- package/dist/services/watch.d.ts.map +1 -0
- package/dist/services/watch.js +32 -0
- package/dist/services/watch.js.map +1 -0
- package/dist/services/workspace-info.d.ts +62 -0
- package/dist/services/workspace-info.d.ts.map +1 -0
- package/dist/services/workspace-info.js +103 -0
- package/dist/services/workspace-info.js.map +1 -0
- package/dist/templates/app-config.template.d.ts +7 -0
- package/dist/templates/app-config.template.d.ts.map +1 -0
- package/dist/templates/app-config.template.js +107 -0
- package/dist/templates/app-config.template.js.map +1 -0
- package/dist/templates/data-view.template.d.ts +7 -0
- package/dist/templates/data-view.template.d.ts.map +1 -0
- package/dist/templates/data-view.template.js +70 -0
- package/dist/templates/data-view.template.js.map +1 -0
- package/dist/templates/event.template.d.ts +11 -0
- package/dist/templates/event.template.d.ts.map +1 -0
- package/dist/templates/event.template.js +40 -0
- package/dist/templates/event.template.js.map +1 -0
- package/dist/templates/experiment.template.d.ts +7 -0
- package/dist/templates/experiment.template.d.ts.map +1 -0
- package/dist/templates/experiment.template.js +89 -0
- package/dist/templates/experiment.template.js.map +1 -0
- package/dist/templates/handler.template.d.ts +16 -0
- package/dist/templates/handler.template.d.ts.map +1 -0
- package/dist/templates/handler.template.js +100 -0
- package/dist/templates/handler.template.js.map +1 -0
- package/dist/templates/index.d.ts +21 -0
- package/dist/templates/index.d.ts.map +1 -0
- package/dist/templates/index.js +37 -0
- package/dist/templates/index.js.map +1 -0
- package/dist/templates/integration.template.d.ts +7 -0
- package/dist/templates/integration.template.d.ts.map +1 -0
- package/dist/templates/integration.template.js +160 -0
- package/dist/templates/integration.template.js.map +1 -0
- package/dist/templates/knowledge.template.d.ts +7 -0
- package/dist/templates/knowledge.template.d.ts.map +1 -0
- package/dist/templates/knowledge.template.js +75 -0
- package/dist/templates/knowledge.template.js.map +1 -0
- package/dist/templates/migration.template.d.ts +7 -0
- package/dist/templates/migration.template.d.ts.map +1 -0
- package/dist/templates/migration.template.js +62 -0
- package/dist/templates/migration.template.js.map +1 -0
- package/dist/templates/operation.template.d.ts +11 -0
- package/dist/templates/operation.template.d.ts.map +1 -0
- package/dist/templates/operation.template.js +105 -0
- package/dist/templates/operation.template.js.map +1 -0
- package/dist/templates/presentation.template.d.ts +11 -0
- package/dist/templates/presentation.template.d.ts.map +1 -0
- package/dist/templates/presentation.template.js +80 -0
- package/dist/templates/presentation.template.js.map +1 -0
- package/dist/templates/telemetry.template.d.ts +7 -0
- package/dist/templates/telemetry.template.d.ts.map +1 -0
- package/dist/templates/telemetry.template.js +91 -0
- package/dist/templates/telemetry.template.js.map +1 -0
- package/dist/templates/workflow-runner.template.d.ts +16 -0
- package/dist/templates/workflow-runner.template.d.ts.map +1 -0
- package/dist/templates/workflow-runner.template.js +50 -0
- package/dist/templates/workflow-runner.template.js.map +1 -0
- package/dist/templates/workflow.template.d.ts +7 -0
- package/dist/templates/workflow.template.d.ts.map +1 -0
- package/dist/templates/workflow.template.js +69 -0
- package/dist/templates/workflow.template.js.map +1 -0
- package/dist/types/config.d.ts +34 -0
- package/dist/types/config.d.ts.map +1 -0
- package/dist/types.d.ts +324 -0
- package/dist/types.d.ts.map +1 -0
- package/package.json +68 -0
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import { FsAdapter } from "../ports/fs.js";
|
|
2
|
+
import { GitAdapter } from "../ports/git.js";
|
|
3
|
+
import { SemanticDiffItem, SemanticDiffOptions } from "@contractspec/module.workspace";
|
|
4
|
+
|
|
5
|
+
//#region src/services/diff.d.ts
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Options for comparing specs.
|
|
9
|
+
*/
|
|
10
|
+
interface CompareSpecsOptions extends SemanticDiffOptions {
|
|
11
|
+
/**
|
|
12
|
+
* Git ref to compare against (branch, tag, commit).
|
|
13
|
+
* If provided, spec2 is loaded from this ref instead of filesystem.
|
|
14
|
+
*/
|
|
15
|
+
baseline?: string;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Result of spec comparison.
|
|
19
|
+
*/
|
|
20
|
+
interface CompareSpecsResult {
|
|
21
|
+
spec1: string;
|
|
22
|
+
spec2: string;
|
|
23
|
+
differences: SemanticDiffItem[];
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Compare two spec files semantically.
|
|
27
|
+
*/
|
|
28
|
+
declare function compareSpecs(spec1Path: string, spec2Path: string, adapters: {
|
|
29
|
+
fs: FsAdapter;
|
|
30
|
+
git: GitAdapter;
|
|
31
|
+
}, options?: CompareSpecsOptions): Promise<CompareSpecsResult>;
|
|
32
|
+
//#endregion
|
|
33
|
+
export { CompareSpecsOptions, CompareSpecsResult, compareSpecs };
|
|
34
|
+
//# sourceMappingURL=diff.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"diff.d.ts","names":[],"sources":["../../src/services/diff.ts"],"sourcesContent":[],"mappings":";;;;;;AA0BA;AASA;;AAGkC,UAvBjB,mBAAA,SAA4B,mBAuBX,CAAA;EACvB;;;;;;;;;UAbM,kBAAA;;;eAGF;;;;;iBAMO,YAAA;MAGJ;OAAgB;aACvB,sBACR,QAAQ"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import { computeSemanticDiff } from "@contractspec/module.workspace";
|
|
2
|
+
|
|
3
|
+
//#region src/services/diff.ts
|
|
4
|
+
/**
|
|
5
|
+
* Diff service.
|
|
6
|
+
*/
|
|
7
|
+
/**
|
|
8
|
+
* Compare two spec files semantically.
|
|
9
|
+
*/
|
|
10
|
+
async function compareSpecs(spec1Path, spec2Path, adapters, options = {}) {
|
|
11
|
+
const { fs, git } = adapters;
|
|
12
|
+
if (!await fs.exists(spec1Path)) throw new Error(`Spec file not found: ${spec1Path}`);
|
|
13
|
+
const spec1Code = await fs.readFile(spec1Path);
|
|
14
|
+
let spec2Code;
|
|
15
|
+
let spec2Label;
|
|
16
|
+
if (options.baseline) {
|
|
17
|
+
spec2Code = await git.showFile(options.baseline, spec1Path);
|
|
18
|
+
spec2Label = `${options.baseline}:${spec1Path}`;
|
|
19
|
+
} else {
|
|
20
|
+
if (!await fs.exists(spec2Path)) throw new Error(`Spec file not found: ${spec2Path}`);
|
|
21
|
+
spec2Code = await fs.readFile(spec2Path);
|
|
22
|
+
spec2Label = spec2Path;
|
|
23
|
+
}
|
|
24
|
+
const differences = computeSemanticDiff(spec1Code, spec1Path, spec2Code, spec2Label, { breakingOnly: options.breakingOnly });
|
|
25
|
+
return {
|
|
26
|
+
spec1: spec1Path,
|
|
27
|
+
spec2: spec2Label,
|
|
28
|
+
differences
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
//#endregion
|
|
33
|
+
export { compareSpecs };
|
|
34
|
+
//# sourceMappingURL=diff.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"diff.js","names":["spec2Code: string","spec2Label: string"],"sources":["../../src/services/diff.ts"],"sourcesContent":["/**\n * Diff service.\n */\n\nimport {\n computeSemanticDiff,\n type SemanticDiffItem,\n type SemanticDiffOptions,\n} from '@contractspec/module.workspace';\nimport type { FsAdapter } from '../ports/fs';\nimport type { GitAdapter } from '../ports/git';\n\n/**\n * Options for comparing specs.\n */\nexport interface CompareSpecsOptions extends SemanticDiffOptions {\n /**\n * Git ref to compare against (branch, tag, commit).\n * If provided, spec2 is loaded from this ref instead of filesystem.\n */\n baseline?: string;\n}\n\n/**\n * Result of spec comparison.\n */\nexport interface CompareSpecsResult {\n spec1: string;\n spec2: string;\n differences: SemanticDiffItem[];\n}\n\n/**\n * Compare two spec files semantically.\n */\nexport async function compareSpecs(\n spec1Path: string,\n spec2Path: string,\n adapters: { fs: FsAdapter; git: GitAdapter },\n options: CompareSpecsOptions = {}\n): Promise<CompareSpecsResult> {\n const { fs, git } = adapters;\n\n // Load first spec from filesystem\n const exists1 = await fs.exists(spec1Path);\n if (!exists1) {\n throw new Error(`Spec file not found: ${spec1Path}`);\n }\n const spec1Code = await fs.readFile(spec1Path);\n\n // Load second spec either from git baseline or filesystem\n let spec2Code: string;\n let spec2Label: string;\n\n if (options.baseline) {\n spec2Code = await git.showFile(options.baseline, spec1Path);\n spec2Label = `${options.baseline}:${spec1Path}`;\n } else {\n const exists2 = await fs.exists(spec2Path);\n if (!exists2) {\n throw new Error(`Spec file not found: ${spec2Path}`);\n }\n spec2Code = await fs.readFile(spec2Path);\n spec2Label = spec2Path;\n }\n\n // Compute semantic diff\n const differences = computeSemanticDiff(\n spec1Code,\n spec1Path,\n spec2Code,\n spec2Label,\n { breakingOnly: options.breakingOnly }\n );\n\n return {\n spec1: spec1Path,\n spec2: spec2Label,\n differences,\n };\n}\n"],"mappings":";;;;;;;;;AAmCA,eAAsB,aACpB,WACA,WACA,UACA,UAA+B,EAAE,EACJ;CAC7B,MAAM,EAAE,IAAI,QAAQ;AAIpB,KAAI,CADY,MAAM,GAAG,OAAO,UAAU,CAExC,OAAM,IAAI,MAAM,wBAAwB,YAAY;CAEtD,MAAM,YAAY,MAAM,GAAG,SAAS,UAAU;CAG9C,IAAIA;CACJ,IAAIC;AAEJ,KAAI,QAAQ,UAAU;AACpB,cAAY,MAAM,IAAI,SAAS,QAAQ,UAAU,UAAU;AAC3D,eAAa,GAAG,QAAQ,SAAS,GAAG;QAC/B;AAEL,MAAI,CADY,MAAM,GAAG,OAAO,UAAU,CAExC,OAAM,IAAI,MAAM,wBAAwB,YAAY;AAEtD,cAAY,MAAM,GAAG,SAAS,UAAU;AACxC,eAAa;;CAIf,MAAM,cAAc,oBAClB,WACA,WACA,WACA,YACA,EAAE,cAAc,QAAQ,cAAc,CACvC;AAED,QAAO;EACL,OAAO;EACP,OAAO;EACP;EACD"}
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
//#region src/services/doctor/checks/ai.ts
|
|
2
|
+
/**
|
|
3
|
+
* Run AI provider-related health checks.
|
|
4
|
+
*/
|
|
5
|
+
async function runAiChecks(fs, ctx, prompts) {
|
|
6
|
+
const results = [];
|
|
7
|
+
const providerResult = await checkAiProvider(fs, ctx);
|
|
8
|
+
results.push(providerResult);
|
|
9
|
+
if (providerResult.status === "pass") results.push(await checkApiKey(fs, ctx, prompts));
|
|
10
|
+
return results;
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* Check which AI provider is configured.
|
|
14
|
+
*/
|
|
15
|
+
async function checkAiProvider(fs, ctx) {
|
|
16
|
+
const configPath = fs.join(ctx.workspaceRoot, ".contractsrc.json");
|
|
17
|
+
try {
|
|
18
|
+
if (!await fs.exists(configPath)) return {
|
|
19
|
+
category: "ai",
|
|
20
|
+
name: "AI Provider",
|
|
21
|
+
status: "skip",
|
|
22
|
+
message: "No config file found"
|
|
23
|
+
};
|
|
24
|
+
const content = await fs.readFile(configPath);
|
|
25
|
+
const config = JSON.parse(content);
|
|
26
|
+
return {
|
|
27
|
+
category: "ai",
|
|
28
|
+
name: "AI Provider",
|
|
29
|
+
status: "pass",
|
|
30
|
+
message: `Provider: ${config.aiProvider ?? "claude"}, Model: ${config.aiModel ?? "default"}`
|
|
31
|
+
};
|
|
32
|
+
} catch {
|
|
33
|
+
return {
|
|
34
|
+
category: "ai",
|
|
35
|
+
name: "AI Provider",
|
|
36
|
+
status: "skip",
|
|
37
|
+
message: "Could not read AI config"
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Check if API key is set for the configured provider.
|
|
43
|
+
*/
|
|
44
|
+
async function checkApiKey(fs, ctx, prompts) {
|
|
45
|
+
const configPath = fs.join(ctx.workspaceRoot, ".contractsrc.json");
|
|
46
|
+
try {
|
|
47
|
+
const content = await fs.readFile(configPath);
|
|
48
|
+
const provider = JSON.parse(content).aiProvider ?? "claude";
|
|
49
|
+
const envVar = {
|
|
50
|
+
claude: "ANTHROPIC_API_KEY",
|
|
51
|
+
openai: "OPENAI_API_KEY",
|
|
52
|
+
ollama: "",
|
|
53
|
+
custom: "CONTRACTSPEC_LLM_API_KEY"
|
|
54
|
+
}[provider];
|
|
55
|
+
if (!envVar) return {
|
|
56
|
+
category: "ai",
|
|
57
|
+
name: "API Key",
|
|
58
|
+
status: "pass",
|
|
59
|
+
message: `${provider} does not require an API key`
|
|
60
|
+
};
|
|
61
|
+
if (!!process.env[envVar]) return {
|
|
62
|
+
category: "ai",
|
|
63
|
+
name: "API Key",
|
|
64
|
+
status: "pass",
|
|
65
|
+
message: `${envVar} is set`
|
|
66
|
+
};
|
|
67
|
+
return {
|
|
68
|
+
category: "ai",
|
|
69
|
+
name: "API Key",
|
|
70
|
+
status: "warn",
|
|
71
|
+
message: `${envVar} not set`,
|
|
72
|
+
details: `Set ${envVar} in your environment to use AI features`,
|
|
73
|
+
fix: prompts ? {
|
|
74
|
+
description: `Set ${envVar} environment variable`,
|
|
75
|
+
apply: async () => {
|
|
76
|
+
const key = await prompts.input(`Enter your ${provider} API key:`, { password: true });
|
|
77
|
+
if (!key) return {
|
|
78
|
+
success: false,
|
|
79
|
+
message: "No API key provided"
|
|
80
|
+
};
|
|
81
|
+
const envPath = fs.join(ctx.workspaceRoot, ".env");
|
|
82
|
+
try {
|
|
83
|
+
let envContent = "";
|
|
84
|
+
if (await fs.exists(envPath)) {
|
|
85
|
+
envContent = await fs.readFile(envPath);
|
|
86
|
+
if (envContent.includes(`${envVar}=`)) return {
|
|
87
|
+
success: false,
|
|
88
|
+
message: `${envVar} already in .env, update manually`
|
|
89
|
+
};
|
|
90
|
+
envContent += "\n";
|
|
91
|
+
}
|
|
92
|
+
envContent += `${envVar}=${key}\n`;
|
|
93
|
+
await fs.writeFile(envPath, envContent);
|
|
94
|
+
return {
|
|
95
|
+
success: true,
|
|
96
|
+
message: `Added ${envVar} to .env (restart required)`
|
|
97
|
+
};
|
|
98
|
+
} catch (error) {
|
|
99
|
+
return {
|
|
100
|
+
success: false,
|
|
101
|
+
message: `Failed: ${error instanceof Error ? error.message : String(error)}`
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
} : void 0
|
|
106
|
+
};
|
|
107
|
+
} catch {
|
|
108
|
+
return {
|
|
109
|
+
category: "ai",
|
|
110
|
+
name: "API Key",
|
|
111
|
+
status: "skip",
|
|
112
|
+
message: "Could not check API key"
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
//#endregion
|
|
118
|
+
export { runAiChecks };
|
|
119
|
+
//# sourceMappingURL=ai.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ai.js","names":["results: CheckResult[]"],"sources":["../../../../src/services/doctor/checks/ai.ts"],"sourcesContent":["/**\n * AI provider health checks.\n */\n\nimport type { FsAdapter } from '../../../ports/fs';\nimport type {\n CheckResult,\n CheckContext,\n DoctorPromptCallbacks,\n FixResult,\n} from '../types';\n\n/**\n * Run AI provider-related health checks.\n */\nexport async function runAiChecks(\n fs: FsAdapter,\n ctx: CheckContext,\n prompts?: DoctorPromptCallbacks\n): Promise<CheckResult[]> {\n const results: CheckResult[] = [];\n\n // Check configured AI provider\n const providerResult = await checkAiProvider(fs, ctx);\n results.push(providerResult);\n\n // Check API key based on provider\n if (providerResult.status === 'pass') {\n results.push(await checkApiKey(fs, ctx, prompts));\n }\n\n return results;\n}\n\n/**\n * Check which AI provider is configured.\n */\nasync function checkAiProvider(\n fs: FsAdapter,\n ctx: CheckContext\n): Promise<CheckResult> {\n const configPath = fs.join(ctx.workspaceRoot, '.contractsrc.json');\n\n try {\n const exists = await fs.exists(configPath);\n if (!exists) {\n return {\n category: 'ai',\n name: 'AI Provider',\n status: 'skip',\n message: 'No config file found',\n };\n }\n\n const content = await fs.readFile(configPath);\n const config = JSON.parse(content) as {\n aiProvider?: string;\n aiModel?: string;\n };\n\n const provider = config.aiProvider ?? 'claude';\n const model = config.aiModel ?? 'default';\n\n return {\n category: 'ai',\n name: 'AI Provider',\n status: 'pass',\n message: `Provider: ${provider}, Model: ${model}`,\n };\n } catch {\n return {\n category: 'ai',\n name: 'AI Provider',\n status: 'skip',\n message: 'Could not read AI config',\n };\n }\n}\n\n/**\n * Check if API key is set for the configured provider.\n */\nasync function checkApiKey(\n fs: FsAdapter,\n ctx: CheckContext,\n prompts?: DoctorPromptCallbacks\n): Promise<CheckResult> {\n const configPath = fs.join(ctx.workspaceRoot, '.contractsrc.json');\n\n try {\n const content = await fs.readFile(configPath);\n const config = JSON.parse(content) as { aiProvider?: string };\n\n const provider = config.aiProvider ?? 'claude';\n\n // Map provider to env var\n const envVarMap: Record<string, string> = {\n claude: 'ANTHROPIC_API_KEY',\n openai: 'OPENAI_API_KEY',\n ollama: '', // Ollama doesn't need API key\n custom: 'CONTRACTSPEC_LLM_API_KEY',\n };\n\n const envVar = envVarMap[provider];\n\n // Ollama doesn't need an API key\n if (!envVar) {\n return {\n category: 'ai',\n name: 'API Key',\n status: 'pass',\n message: `${provider} does not require an API key`,\n };\n }\n\n const hasKey = !!process.env[envVar];\n\n if (hasKey) {\n return {\n category: 'ai',\n name: 'API Key',\n status: 'pass',\n message: `${envVar} is set`,\n };\n }\n\n return {\n category: 'ai',\n name: 'API Key',\n status: 'warn',\n message: `${envVar} not set`,\n details: `Set ${envVar} in your environment to use AI features`,\n fix: prompts\n ? {\n description: `Set ${envVar} environment variable`,\n apply: async (): Promise<FixResult> => {\n const key = await prompts.input(\n `Enter your ${provider} API key:`,\n { password: true }\n );\n\n if (!key) {\n return { success: false, message: 'No API key provided' };\n }\n\n // We can't actually set env vars permanently, but we can create a .env file\n const envPath = fs.join(ctx.workspaceRoot, '.env');\n try {\n let envContent = '';\n if (await fs.exists(envPath)) {\n envContent = await fs.readFile(envPath);\n // Check if var already exists\n if (envContent.includes(`${envVar}=`)) {\n return {\n success: false,\n message: `${envVar} already in .env, update manually`,\n };\n }\n envContent += '\\n';\n }\n\n envContent += `${envVar}=${key}\\n`;\n await fs.writeFile(envPath, envContent);\n\n return {\n success: true,\n message: `Added ${envVar} to .env (restart required)`,\n };\n } catch (error) {\n const msg =\n error instanceof Error ? error.message : String(error);\n return { success: false, message: `Failed: ${msg}` };\n }\n },\n }\n : undefined,\n };\n } catch {\n return {\n category: 'ai',\n name: 'API Key',\n status: 'skip',\n message: 'Could not check API key',\n };\n }\n}\n"],"mappings":";;;;AAeA,eAAsB,YACpB,IACA,KACA,SACwB;CACxB,MAAMA,UAAyB,EAAE;CAGjC,MAAM,iBAAiB,MAAM,gBAAgB,IAAI,IAAI;AACrD,SAAQ,KAAK,eAAe;AAG5B,KAAI,eAAe,WAAW,OAC5B,SAAQ,KAAK,MAAM,YAAY,IAAI,KAAK,QAAQ,CAAC;AAGnD,QAAO;;;;;AAMT,eAAe,gBACb,IACA,KACsB;CACtB,MAAM,aAAa,GAAG,KAAK,IAAI,eAAe,oBAAoB;AAElE,KAAI;AAEF,MAAI,CADW,MAAM,GAAG,OAAO,WAAW,CAExC,QAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV;EAGH,MAAM,UAAU,MAAM,GAAG,SAAS,WAAW;EAC7C,MAAM,SAAS,KAAK,MAAM,QAAQ;AAQlC,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS,aAPM,OAAO,cAAc,SAOL,WANnB,OAAO,WAAW;GAO/B;SACK;AACN,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV;;;;;;AAOL,eAAe,YACb,IACA,KACA,SACsB;CACtB,MAAM,aAAa,GAAG,KAAK,IAAI,eAAe,oBAAoB;AAElE,KAAI;EACF,MAAM,UAAU,MAAM,GAAG,SAAS,WAAW;EAG7C,MAAM,WAFS,KAAK,MAAM,QAAQ,CAEV,cAAc;EAUtC,MAAM,SAPoC;GACxC,QAAQ;GACR,QAAQ;GACR,QAAQ;GACR,QAAQ;GACT,CAEwB;AAGzB,MAAI,CAAC,OACH,QAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS,GAAG,SAAS;GACtB;AAKH,MAFe,CAAC,CAAC,QAAQ,IAAI,QAG3B,QAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS,GAAG,OAAO;GACpB;AAGH,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS,GAAG,OAAO;GACnB,SAAS,OAAO,OAAO;GACvB,KAAK,UACD;IACE,aAAa,OAAO,OAAO;IAC3B,OAAO,YAAgC;KACrC,MAAM,MAAM,MAAM,QAAQ,MACxB,cAAc,SAAS,YACvB,EAAE,UAAU,MAAM,CACnB;AAED,SAAI,CAAC,IACH,QAAO;MAAE,SAAS;MAAO,SAAS;MAAuB;KAI3D,MAAM,UAAU,GAAG,KAAK,IAAI,eAAe,OAAO;AAClD,SAAI;MACF,IAAI,aAAa;AACjB,UAAI,MAAM,GAAG,OAAO,QAAQ,EAAE;AAC5B,oBAAa,MAAM,GAAG,SAAS,QAAQ;AAEvC,WAAI,WAAW,SAAS,GAAG,OAAO,GAAG,CACnC,QAAO;QACL,SAAS;QACT,SAAS,GAAG,OAAO;QACpB;AAEH,qBAAc;;AAGhB,oBAAc,GAAG,OAAO,GAAG,IAAI;AAC/B,YAAM,GAAG,UAAU,SAAS,WAAW;AAEvC,aAAO;OACL,SAAS;OACT,SAAS,SAAS,OAAO;OAC1B;cACM,OAAO;AAGd,aAAO;OAAE,SAAS;OAAO,SAAS,WADhC,iBAAiB,QAAQ,MAAM,UAAU,OAAO,MAAM;OACJ;;;IAGzD,GACD;GACL;SACK;AACN,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV"}
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
import { exec } from "node:child_process";
|
|
2
|
+
import { promisify } from "node:util";
|
|
3
|
+
|
|
4
|
+
//#region src/services/doctor/checks/cli.ts
|
|
5
|
+
/**
|
|
6
|
+
* CLI installation health checks.
|
|
7
|
+
*/
|
|
8
|
+
const execAsync = promisify(exec);
|
|
9
|
+
/**
|
|
10
|
+
* Run CLI-related health checks.
|
|
11
|
+
*/
|
|
12
|
+
async function runCliChecks(fs, ctx) {
|
|
13
|
+
const results = [];
|
|
14
|
+
results.push(await checkCliAccessible(ctx));
|
|
15
|
+
results.push(await checkCliVersion(ctx));
|
|
16
|
+
results.push(await checkCliInstallLocation(fs, ctx));
|
|
17
|
+
return results;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Check if the CLI is accessible.
|
|
21
|
+
*/
|
|
22
|
+
async function checkCliAccessible(ctx) {
|
|
23
|
+
try {
|
|
24
|
+
await execAsync("npx contractspec --version", {
|
|
25
|
+
cwd: ctx.workspaceRoot,
|
|
26
|
+
timeout: 1e4
|
|
27
|
+
});
|
|
28
|
+
return {
|
|
29
|
+
category: "cli",
|
|
30
|
+
name: "CLI Accessible",
|
|
31
|
+
status: "pass",
|
|
32
|
+
message: "ContractSpec CLI is accessible"
|
|
33
|
+
};
|
|
34
|
+
} catch {
|
|
35
|
+
return {
|
|
36
|
+
category: "cli",
|
|
37
|
+
name: "CLI Accessible",
|
|
38
|
+
status: "fail",
|
|
39
|
+
message: "ContractSpec CLI is not accessible",
|
|
40
|
+
details: "Could not run \"npx contractspec --version\"",
|
|
41
|
+
fix: {
|
|
42
|
+
description: "Install ContractSpec CLI globally",
|
|
43
|
+
apply: async () => {
|
|
44
|
+
try {
|
|
45
|
+
await execAsync("npm install -g @contractspec/app.cli-contractspec", {
|
|
46
|
+
cwd: ctx.workspaceRoot,
|
|
47
|
+
timeout: 6e4
|
|
48
|
+
});
|
|
49
|
+
return {
|
|
50
|
+
success: true,
|
|
51
|
+
message: "CLI installed globally"
|
|
52
|
+
};
|
|
53
|
+
} catch (error) {
|
|
54
|
+
return {
|
|
55
|
+
success: false,
|
|
56
|
+
message: `Failed to install: ${error instanceof Error ? error.message : String(error)}`
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Check CLI version.
|
|
66
|
+
*/
|
|
67
|
+
async function checkCliVersion(ctx) {
|
|
68
|
+
try {
|
|
69
|
+
const { stdout } = await execAsync("npx contractspec --version", {
|
|
70
|
+
cwd: ctx.workspaceRoot,
|
|
71
|
+
timeout: 1e4
|
|
72
|
+
});
|
|
73
|
+
return {
|
|
74
|
+
category: "cli",
|
|
75
|
+
name: "CLI Version",
|
|
76
|
+
status: "pass",
|
|
77
|
+
message: `CLI version: ${stdout.trim()}`
|
|
78
|
+
};
|
|
79
|
+
} catch {
|
|
80
|
+
return {
|
|
81
|
+
category: "cli",
|
|
82
|
+
name: "CLI Version",
|
|
83
|
+
status: "skip",
|
|
84
|
+
message: "Could not determine CLI version"
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Check if CLI is installed locally in the project.
|
|
90
|
+
*/
|
|
91
|
+
async function checkCliInstallLocation(fs, ctx) {
|
|
92
|
+
const packageJsonPath = fs.join(ctx.workspaceRoot, "package.json");
|
|
93
|
+
try {
|
|
94
|
+
if (!await fs.exists(packageJsonPath)) return {
|
|
95
|
+
category: "cli",
|
|
96
|
+
name: "Local Installation",
|
|
97
|
+
status: "skip",
|
|
98
|
+
message: "No package.json found"
|
|
99
|
+
};
|
|
100
|
+
const content = await fs.readFile(packageJsonPath);
|
|
101
|
+
const pkg = JSON.parse(content);
|
|
102
|
+
if (pkg.dependencies?.["@contractspec/app.cli-contractspec"] !== void 0 || pkg.devDependencies?.["@contractspec/app.cli-contractspec"] !== void 0) return {
|
|
103
|
+
category: "cli",
|
|
104
|
+
name: "Local Installation",
|
|
105
|
+
status: "pass",
|
|
106
|
+
message: "CLI is installed as a project dependency"
|
|
107
|
+
};
|
|
108
|
+
return {
|
|
109
|
+
category: "cli",
|
|
110
|
+
name: "Local Installation",
|
|
111
|
+
status: "warn",
|
|
112
|
+
message: "CLI is not installed as a project dependency",
|
|
113
|
+
details: "Consider adding @contractspec/app.cli-contractspec to devDependencies",
|
|
114
|
+
fix: {
|
|
115
|
+
description: "Add CLI as a dev dependency",
|
|
116
|
+
apply: async () => {
|
|
117
|
+
try {
|
|
118
|
+
await execAsync("npm install -D @contractspec/app.cli-contractspec", {
|
|
119
|
+
cwd: ctx.workspaceRoot,
|
|
120
|
+
timeout: 6e4
|
|
121
|
+
});
|
|
122
|
+
return {
|
|
123
|
+
success: true,
|
|
124
|
+
message: "CLI added as dev dependency"
|
|
125
|
+
};
|
|
126
|
+
} catch (error) {
|
|
127
|
+
return {
|
|
128
|
+
success: false,
|
|
129
|
+
message: `Failed to install: ${error instanceof Error ? error.message : String(error)}`
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
};
|
|
135
|
+
} catch {
|
|
136
|
+
return {
|
|
137
|
+
category: "cli",
|
|
138
|
+
name: "Local Installation",
|
|
139
|
+
status: "skip",
|
|
140
|
+
message: "Could not check local installation"
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
//#endregion
|
|
146
|
+
export { runCliChecks };
|
|
147
|
+
//# sourceMappingURL=cli.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cli.js","names":["results: CheckResult[]"],"sources":["../../../../src/services/doctor/checks/cli.ts"],"sourcesContent":["/**\n * CLI installation health checks.\n */\n\nimport { exec } from 'node:child_process';\nimport { promisify } from 'node:util';\nimport type { FsAdapter } from '../../../ports/fs';\nimport type { CheckResult, CheckContext, FixResult } from '../types';\n\nconst execAsync = promisify(exec);\n\n/**\n * Run CLI-related health checks.\n */\nexport async function runCliChecks(\n fs: FsAdapter,\n ctx: CheckContext\n): Promise<CheckResult[]> {\n const results: CheckResult[] = [];\n\n // Check if CLI is accessible\n results.push(await checkCliAccessible(ctx));\n\n // Check CLI version\n results.push(await checkCliVersion(ctx));\n\n // Check if CLI is installed globally or locally\n results.push(await checkCliInstallLocation(fs, ctx));\n\n return results;\n}\n\n/**\n * Check if the CLI is accessible.\n */\nasync function checkCliAccessible(ctx: CheckContext): Promise<CheckResult> {\n try {\n await execAsync('npx contractspec --version', {\n cwd: ctx.workspaceRoot,\n timeout: 10000,\n });\n\n return {\n category: 'cli',\n name: 'CLI Accessible',\n status: 'pass',\n message: 'ContractSpec CLI is accessible',\n };\n } catch {\n return {\n category: 'cli',\n name: 'CLI Accessible',\n status: 'fail',\n message: 'ContractSpec CLI is not accessible',\n details: 'Could not run \"npx contractspec --version\"',\n fix: {\n description: 'Install ContractSpec CLI globally',\n apply: async (): Promise<FixResult> => {\n try {\n await execAsync(\n 'npm install -g @contractspec/app.cli-contractspec',\n {\n cwd: ctx.workspaceRoot,\n timeout: 60000,\n }\n );\n return { success: true, message: 'CLI installed globally' };\n } catch (error) {\n const msg = error instanceof Error ? error.message : String(error);\n return { success: false, message: `Failed to install: ${msg}` };\n }\n },\n },\n };\n }\n}\n\n/**\n * Check CLI version.\n */\nasync function checkCliVersion(ctx: CheckContext): Promise<CheckResult> {\n try {\n const { stdout } = await execAsync('npx contractspec --version', {\n cwd: ctx.workspaceRoot,\n timeout: 10000,\n });\n\n const version = stdout.trim();\n\n return {\n category: 'cli',\n name: 'CLI Version',\n status: 'pass',\n message: `CLI version: ${version}`,\n };\n } catch {\n return {\n category: 'cli',\n name: 'CLI Version',\n status: 'skip',\n message: 'Could not determine CLI version',\n };\n }\n}\n\n/**\n * Check if CLI is installed locally in the project.\n */\nasync function checkCliInstallLocation(\n fs: FsAdapter,\n ctx: CheckContext\n): Promise<CheckResult> {\n const packageJsonPath = fs.join(ctx.workspaceRoot, 'package.json');\n\n try {\n const exists = await fs.exists(packageJsonPath);\n if (!exists) {\n return {\n category: 'cli',\n name: 'Local Installation',\n status: 'skip',\n message: 'No package.json found',\n };\n }\n\n const content = await fs.readFile(packageJsonPath);\n const pkg = JSON.parse(content) as {\n dependencies?: Record<string, string>;\n devDependencies?: Record<string, string>;\n };\n\n const hasDep =\n pkg.dependencies?.['@contractspec/app.cli-contractspec'] !== undefined ||\n pkg.devDependencies?.['@contractspec/app.cli-contractspec'] !== undefined;\n\n if (hasDep) {\n return {\n category: 'cli',\n name: 'Local Installation',\n status: 'pass',\n message: 'CLI is installed as a project dependency',\n };\n }\n\n return {\n category: 'cli',\n name: 'Local Installation',\n status: 'warn',\n message: 'CLI is not installed as a project dependency',\n details:\n 'Consider adding @contractspec/app.cli-contractspec to devDependencies',\n fix: {\n description: 'Add CLI as a dev dependency',\n apply: async (): Promise<FixResult> => {\n try {\n await execAsync(\n 'npm install -D @contractspec/app.cli-contractspec',\n {\n cwd: ctx.workspaceRoot,\n timeout: 60000,\n }\n );\n return { success: true, message: 'CLI added as dev dependency' };\n } catch (error) {\n const msg = error instanceof Error ? error.message : String(error);\n return { success: false, message: `Failed to install: ${msg}` };\n }\n },\n },\n };\n } catch {\n return {\n category: 'cli',\n name: 'Local Installation',\n status: 'skip',\n message: 'Could not check local installation',\n };\n }\n}\n"],"mappings":";;;;;;;AASA,MAAM,YAAY,UAAU,KAAK;;;;AAKjC,eAAsB,aACpB,IACA,KACwB;CACxB,MAAMA,UAAyB,EAAE;AAGjC,SAAQ,KAAK,MAAM,mBAAmB,IAAI,CAAC;AAG3C,SAAQ,KAAK,MAAM,gBAAgB,IAAI,CAAC;AAGxC,SAAQ,KAAK,MAAM,wBAAwB,IAAI,IAAI,CAAC;AAEpD,QAAO;;;;;AAMT,eAAe,mBAAmB,KAAyC;AACzE,KAAI;AACF,QAAM,UAAU,8BAA8B;GAC5C,KAAK,IAAI;GACT,SAAS;GACV,CAAC;AAEF,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV;SACK;AACN,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACT,SAAS;GACT,KAAK;IACH,aAAa;IACb,OAAO,YAAgC;AACrC,SAAI;AACF,YAAM,UACJ,qDACA;OACE,KAAK,IAAI;OACT,SAAS;OACV,CACF;AACD,aAAO;OAAE,SAAS;OAAM,SAAS;OAA0B;cACpD,OAAO;AAEd,aAAO;OAAE,SAAS;OAAO,SAAS,sBADtB,iBAAiB,QAAQ,MAAM,UAAU,OAAO,MAAM;OACH;;;IAGpE;GACF;;;;;;AAOL,eAAe,gBAAgB,KAAyC;AACtE,KAAI;EACF,MAAM,EAAE,WAAW,MAAM,UAAU,8BAA8B;GAC/D,KAAK,IAAI;GACT,SAAS;GACV,CAAC;AAIF,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS,gBANK,OAAO,MAAM;GAO5B;SACK;AACN,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV;;;;;;AAOL,eAAe,wBACb,IACA,KACsB;CACtB,MAAM,kBAAkB,GAAG,KAAK,IAAI,eAAe,eAAe;AAElE,KAAI;AAEF,MAAI,CADW,MAAM,GAAG,OAAO,gBAAgB,CAE7C,QAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV;EAGH,MAAM,UAAU,MAAM,GAAG,SAAS,gBAAgB;EAClD,MAAM,MAAM,KAAK,MAAM,QAAQ;AAS/B,MAHE,IAAI,eAAe,0CAA0C,UAC7D,IAAI,kBAAkB,0CAA0C,OAGhE,QAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV;AAGH,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACT,SACE;GACF,KAAK;IACH,aAAa;IACb,OAAO,YAAgC;AACrC,SAAI;AACF,YAAM,UACJ,qDACA;OACE,KAAK,IAAI;OACT,SAAS;OACV,CACF;AACD,aAAO;OAAE,SAAS;OAAM,SAAS;OAA+B;cACzD,OAAO;AAEd,aAAO;OAAE,SAAS;OAAO,SAAS,sBADtB,iBAAiB,QAAQ,MAAM,UAAU,OAAO,MAAM;OACH;;;IAGpE;GACF;SACK;AACN,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV"}
|
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
import { generateContractsrcConfig } from "../../setup/config-generators.js";
|
|
2
|
+
import { formatJson } from "../../setup/file-merger.js";
|
|
3
|
+
|
|
4
|
+
//#region src/services/doctor/checks/config.ts
|
|
5
|
+
/**
|
|
6
|
+
* Run configuration-related health checks.
|
|
7
|
+
*/
|
|
8
|
+
async function runConfigChecks(fs, ctx) {
|
|
9
|
+
const results = [];
|
|
10
|
+
results.push(await checkContractsrcExists(fs, ctx));
|
|
11
|
+
results.push(await checkContractsrcValid(fs, ctx));
|
|
12
|
+
results.push(await checkContractsrcFields(fs, ctx));
|
|
13
|
+
return results;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Check if .contractsrc.json exists.
|
|
17
|
+
*/
|
|
18
|
+
async function checkContractsrcExists(fs, ctx) {
|
|
19
|
+
const configPath = fs.join(ctx.workspaceRoot, ".contractsrc.json");
|
|
20
|
+
if (await fs.exists(configPath)) return {
|
|
21
|
+
category: "config",
|
|
22
|
+
name: "Config File Exists",
|
|
23
|
+
status: "pass",
|
|
24
|
+
message: ".contractsrc.json found"
|
|
25
|
+
};
|
|
26
|
+
return {
|
|
27
|
+
category: "config",
|
|
28
|
+
name: "Config File Exists",
|
|
29
|
+
status: "fail",
|
|
30
|
+
message: ".contractsrc.json not found",
|
|
31
|
+
fix: {
|
|
32
|
+
description: "Create .contractsrc.json with defaults",
|
|
33
|
+
apply: async () => {
|
|
34
|
+
try {
|
|
35
|
+
const defaults = generateContractsrcConfig({
|
|
36
|
+
workspaceRoot: ctx.workspaceRoot,
|
|
37
|
+
interactive: false,
|
|
38
|
+
targets: []
|
|
39
|
+
});
|
|
40
|
+
await fs.writeFile(configPath, formatJson(defaults));
|
|
41
|
+
return {
|
|
42
|
+
success: true,
|
|
43
|
+
message: "Created .contractsrc.json"
|
|
44
|
+
};
|
|
45
|
+
} catch (error) {
|
|
46
|
+
return {
|
|
47
|
+
success: false,
|
|
48
|
+
message: `Failed to create: ${error instanceof Error ? error.message : String(error)}`
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Check if .contractsrc.json is valid JSON.
|
|
57
|
+
*/
|
|
58
|
+
async function checkContractsrcValid(fs, ctx) {
|
|
59
|
+
const configPath = fs.join(ctx.workspaceRoot, ".contractsrc.json");
|
|
60
|
+
if (!await fs.exists(configPath)) return {
|
|
61
|
+
category: "config",
|
|
62
|
+
name: "Config Valid JSON",
|
|
63
|
+
status: "skip",
|
|
64
|
+
message: "Config file does not exist"
|
|
65
|
+
};
|
|
66
|
+
try {
|
|
67
|
+
const content = await fs.readFile(configPath);
|
|
68
|
+
JSON.parse(content);
|
|
69
|
+
return {
|
|
70
|
+
category: "config",
|
|
71
|
+
name: "Config Valid JSON",
|
|
72
|
+
status: "pass",
|
|
73
|
+
message: ".contractsrc.json is valid JSON"
|
|
74
|
+
};
|
|
75
|
+
} catch (error) {
|
|
76
|
+
return {
|
|
77
|
+
category: "config",
|
|
78
|
+
name: "Config Valid JSON",
|
|
79
|
+
status: "fail",
|
|
80
|
+
message: ".contractsrc.json is not valid JSON",
|
|
81
|
+
details: error instanceof Error ? error.message : String(error),
|
|
82
|
+
fix: {
|
|
83
|
+
description: "Replace with valid default config",
|
|
84
|
+
apply: async () => {
|
|
85
|
+
try {
|
|
86
|
+
const defaults = generateContractsrcConfig({
|
|
87
|
+
workspaceRoot: ctx.workspaceRoot,
|
|
88
|
+
interactive: false,
|
|
89
|
+
targets: []
|
|
90
|
+
});
|
|
91
|
+
await fs.writeFile(configPath, formatJson(defaults));
|
|
92
|
+
return {
|
|
93
|
+
success: true,
|
|
94
|
+
message: "Replaced with valid config"
|
|
95
|
+
};
|
|
96
|
+
} catch (err) {
|
|
97
|
+
return {
|
|
98
|
+
success: false,
|
|
99
|
+
message: `Failed: ${err instanceof Error ? err.message : String(err)}`
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Check required fields in .contractsrc.json.
|
|
109
|
+
*/
|
|
110
|
+
async function checkContractsrcFields(fs, ctx) {
|
|
111
|
+
const configPath = fs.join(ctx.workspaceRoot, ".contractsrc.json");
|
|
112
|
+
if (!await fs.exists(configPath)) return {
|
|
113
|
+
category: "config",
|
|
114
|
+
name: "Config Fields",
|
|
115
|
+
status: "skip",
|
|
116
|
+
message: "Config file does not exist"
|
|
117
|
+
};
|
|
118
|
+
try {
|
|
119
|
+
const content = await fs.readFile(configPath);
|
|
120
|
+
const config = JSON.parse(content);
|
|
121
|
+
const missingFields = [];
|
|
122
|
+
if (!config["outputDir"]) missingFields.push("outputDir");
|
|
123
|
+
if (!config["conventions"]) missingFields.push("conventions");
|
|
124
|
+
if (missingFields.length === 0) return {
|
|
125
|
+
category: "config",
|
|
126
|
+
name: "Config Fields",
|
|
127
|
+
status: "pass",
|
|
128
|
+
message: "All recommended fields present"
|
|
129
|
+
};
|
|
130
|
+
return {
|
|
131
|
+
category: "config",
|
|
132
|
+
name: "Config Fields",
|
|
133
|
+
status: "warn",
|
|
134
|
+
message: `Missing recommended fields: ${missingFields.join(", ")}`,
|
|
135
|
+
fix: {
|
|
136
|
+
description: "Add missing fields with defaults",
|
|
137
|
+
apply: async () => {
|
|
138
|
+
try {
|
|
139
|
+
const defaults = generateContractsrcConfig({
|
|
140
|
+
workspaceRoot: ctx.workspaceRoot,
|
|
141
|
+
interactive: false,
|
|
142
|
+
targets: []
|
|
143
|
+
});
|
|
144
|
+
for (const field of missingFields) if (defaults[field] !== void 0) config[field] = defaults[field];
|
|
145
|
+
await fs.writeFile(configPath, formatJson(config));
|
|
146
|
+
return {
|
|
147
|
+
success: true,
|
|
148
|
+
message: "Added missing fields"
|
|
149
|
+
};
|
|
150
|
+
} catch (err) {
|
|
151
|
+
return {
|
|
152
|
+
success: false,
|
|
153
|
+
message: `Failed: ${err instanceof Error ? err.message : String(err)}`
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
};
|
|
159
|
+
} catch {
|
|
160
|
+
return {
|
|
161
|
+
category: "config",
|
|
162
|
+
name: "Config Fields",
|
|
163
|
+
status: "skip",
|
|
164
|
+
message: "Could not parse config"
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
//#endregion
|
|
170
|
+
export { runConfigChecks };
|
|
171
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","names":["results: CheckResult[]","missingFields: string[]"],"sources":["../../../../src/services/doctor/checks/config.ts"],"sourcesContent":["/**\n * Configuration file health checks.\n */\n\nimport type { FsAdapter } from '../../../ports/fs';\nimport type { CheckResult, CheckContext, FixResult } from '../types';\nimport { generateContractsrcConfig } from '../../setup/config-generators';\nimport { formatJson } from '../../setup/file-merger';\n\n/**\n * Run configuration-related health checks.\n */\nexport async function runConfigChecks(\n fs: FsAdapter,\n ctx: CheckContext\n): Promise<CheckResult[]> {\n const results: CheckResult[] = [];\n\n // Check if .contractsrc.json exists\n results.push(await checkContractsrcExists(fs, ctx));\n\n // Check if .contractsrc.json is valid\n results.push(await checkContractsrcValid(fs, ctx));\n\n // Check required fields in config\n results.push(await checkContractsrcFields(fs, ctx));\n\n return results;\n}\n\n/**\n * Check if .contractsrc.json exists.\n */\nasync function checkContractsrcExists(\n fs: FsAdapter,\n ctx: CheckContext\n): Promise<CheckResult> {\n const configPath = fs.join(ctx.workspaceRoot, '.contractsrc.json');\n\n const exists = await fs.exists(configPath);\n if (exists) {\n return {\n category: 'config',\n name: 'Config File Exists',\n status: 'pass',\n message: '.contractsrc.json found',\n };\n }\n\n return {\n category: 'config',\n name: 'Config File Exists',\n status: 'fail',\n message: '.contractsrc.json not found',\n fix: {\n description: 'Create .contractsrc.json with defaults',\n apply: async (): Promise<FixResult> => {\n try {\n const defaults = generateContractsrcConfig({\n workspaceRoot: ctx.workspaceRoot,\n interactive: false,\n targets: [],\n });\n await fs.writeFile(configPath, formatJson(defaults));\n return { success: true, message: 'Created .contractsrc.json' };\n } catch (error) {\n const msg = error instanceof Error ? error.message : String(error);\n return { success: false, message: `Failed to create: ${msg}` };\n }\n },\n },\n };\n}\n\n/**\n * Check if .contractsrc.json is valid JSON.\n */\nasync function checkContractsrcValid(\n fs: FsAdapter,\n ctx: CheckContext\n): Promise<CheckResult> {\n const configPath = fs.join(ctx.workspaceRoot, '.contractsrc.json');\n\n const exists = await fs.exists(configPath);\n if (!exists) {\n return {\n category: 'config',\n name: 'Config Valid JSON',\n status: 'skip',\n message: 'Config file does not exist',\n };\n }\n\n try {\n const content = await fs.readFile(configPath);\n JSON.parse(content);\n\n return {\n category: 'config',\n name: 'Config Valid JSON',\n status: 'pass',\n message: '.contractsrc.json is valid JSON',\n };\n } catch (error) {\n const msg = error instanceof Error ? error.message : String(error);\n\n return {\n category: 'config',\n name: 'Config Valid JSON',\n status: 'fail',\n message: '.contractsrc.json is not valid JSON',\n details: msg,\n fix: {\n description: 'Replace with valid default config',\n apply: async (): Promise<FixResult> => {\n try {\n const defaults = generateContractsrcConfig({\n workspaceRoot: ctx.workspaceRoot,\n interactive: false,\n targets: [],\n });\n await fs.writeFile(configPath, formatJson(defaults));\n return { success: true, message: 'Replaced with valid config' };\n } catch (err) {\n const m = err instanceof Error ? err.message : String(err);\n return { success: false, message: `Failed: ${m}` };\n }\n },\n },\n };\n }\n}\n\n/**\n * Check required fields in .contractsrc.json.\n */\nasync function checkContractsrcFields(\n fs: FsAdapter,\n ctx: CheckContext\n): Promise<CheckResult> {\n const configPath = fs.join(ctx.workspaceRoot, '.contractsrc.json');\n\n const exists = await fs.exists(configPath);\n if (!exists) {\n return {\n category: 'config',\n name: 'Config Fields',\n status: 'skip',\n message: 'Config file does not exist',\n };\n }\n\n try {\n const content = await fs.readFile(configPath);\n const config = JSON.parse(content) as Record<string, unknown>;\n\n const missingFields: string[] = [];\n\n // Check for recommended fields\n if (!config['outputDir']) {\n missingFields.push('outputDir');\n }\n if (!config['conventions']) {\n missingFields.push('conventions');\n }\n\n if (missingFields.length === 0) {\n return {\n category: 'config',\n name: 'Config Fields',\n status: 'pass',\n message: 'All recommended fields present',\n };\n }\n\n return {\n category: 'config',\n name: 'Config Fields',\n status: 'warn',\n message: `Missing recommended fields: ${missingFields.join(', ')}`,\n fix: {\n description: 'Add missing fields with defaults',\n apply: async (): Promise<FixResult> => {\n try {\n const defaults = generateContractsrcConfig({\n workspaceRoot: ctx.workspaceRoot,\n interactive: false,\n targets: [],\n }) as Record<string, unknown>;\n\n // Merge missing fields\n for (const field of missingFields) {\n if (defaults[field] !== undefined) {\n config[field] = defaults[field];\n }\n }\n\n await fs.writeFile(configPath, formatJson(config));\n return { success: true, message: 'Added missing fields' };\n } catch (err) {\n const m = err instanceof Error ? err.message : String(err);\n return { success: false, message: `Failed: ${m}` };\n }\n },\n },\n };\n } catch {\n return {\n category: 'config',\n name: 'Config Fields',\n status: 'skip',\n message: 'Could not parse config',\n };\n }\n}\n"],"mappings":";;;;;;;AAYA,eAAsB,gBACpB,IACA,KACwB;CACxB,MAAMA,UAAyB,EAAE;AAGjC,SAAQ,KAAK,MAAM,uBAAuB,IAAI,IAAI,CAAC;AAGnD,SAAQ,KAAK,MAAM,sBAAsB,IAAI,IAAI,CAAC;AAGlD,SAAQ,KAAK,MAAM,uBAAuB,IAAI,IAAI,CAAC;AAEnD,QAAO;;;;;AAMT,eAAe,uBACb,IACA,KACsB;CACtB,MAAM,aAAa,GAAG,KAAK,IAAI,eAAe,oBAAoB;AAGlE,KADe,MAAM,GAAG,OAAO,WAAW,CAExC,QAAO;EACL,UAAU;EACV,MAAM;EACN,QAAQ;EACR,SAAS;EACV;AAGH,QAAO;EACL,UAAU;EACV,MAAM;EACN,QAAQ;EACR,SAAS;EACT,KAAK;GACH,aAAa;GACb,OAAO,YAAgC;AACrC,QAAI;KACF,MAAM,WAAW,0BAA0B;MACzC,eAAe,IAAI;MACnB,aAAa;MACb,SAAS,EAAE;MACZ,CAAC;AACF,WAAM,GAAG,UAAU,YAAY,WAAW,SAAS,CAAC;AACpD,YAAO;MAAE,SAAS;MAAM,SAAS;MAA6B;aACvD,OAAO;AAEd,YAAO;MAAE,SAAS;MAAO,SAAS,qBADtB,iBAAiB,QAAQ,MAAM,UAAU,OAAO,MAAM;MACJ;;;GAGnE;EACF;;;;;AAMH,eAAe,sBACb,IACA,KACsB;CACtB,MAAM,aAAa,GAAG,KAAK,IAAI,eAAe,oBAAoB;AAGlE,KAAI,CADW,MAAM,GAAG,OAAO,WAAW,CAExC,QAAO;EACL,UAAU;EACV,MAAM;EACN,QAAQ;EACR,SAAS;EACV;AAGH,KAAI;EACF,MAAM,UAAU,MAAM,GAAG,SAAS,WAAW;AAC7C,OAAK,MAAM,QAAQ;AAEnB,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV;UACM,OAAO;AAGd,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACT,SAPU,iBAAiB,QAAQ,MAAM,UAAU,OAAO,MAAM;GAQhE,KAAK;IACH,aAAa;IACb,OAAO,YAAgC;AACrC,SAAI;MACF,MAAM,WAAW,0BAA0B;OACzC,eAAe,IAAI;OACnB,aAAa;OACb,SAAS,EAAE;OACZ,CAAC;AACF,YAAM,GAAG,UAAU,YAAY,WAAW,SAAS,CAAC;AACpD,aAAO;OAAE,SAAS;OAAM,SAAS;OAA8B;cACxD,KAAK;AAEZ,aAAO;OAAE,SAAS;OAAO,SAAS,WADxB,eAAe,QAAQ,IAAI,UAAU,OAAO,IAAI;OACR;;;IAGvD;GACF;;;;;;AAOL,eAAe,uBACb,IACA,KACsB;CACtB,MAAM,aAAa,GAAG,KAAK,IAAI,eAAe,oBAAoB;AAGlE,KAAI,CADW,MAAM,GAAG,OAAO,WAAW,CAExC,QAAO;EACL,UAAU;EACV,MAAM;EACN,QAAQ;EACR,SAAS;EACV;AAGH,KAAI;EACF,MAAM,UAAU,MAAM,GAAG,SAAS,WAAW;EAC7C,MAAM,SAAS,KAAK,MAAM,QAAQ;EAElC,MAAMC,gBAA0B,EAAE;AAGlC,MAAI,CAAC,OAAO,aACV,eAAc,KAAK,YAAY;AAEjC,MAAI,CAAC,OAAO,eACV,eAAc,KAAK,cAAc;AAGnC,MAAI,cAAc,WAAW,EAC3B,QAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV;AAGH,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS,+BAA+B,cAAc,KAAK,KAAK;GAChE,KAAK;IACH,aAAa;IACb,OAAO,YAAgC;AACrC,SAAI;MACF,MAAM,WAAW,0BAA0B;OACzC,eAAe,IAAI;OACnB,aAAa;OACb,SAAS,EAAE;OACZ,CAAC;AAGF,WAAK,MAAM,SAAS,cAClB,KAAI,SAAS,WAAW,OACtB,QAAO,SAAS,SAAS;AAI7B,YAAM,GAAG,UAAU,YAAY,WAAW,OAAO,CAAC;AAClD,aAAO;OAAE,SAAS;OAAM,SAAS;OAAwB;cAClD,KAAK;AAEZ,aAAO;OAAE,SAAS;OAAO,SAAS,WADxB,eAAe,QAAQ,IAAI,UAAU,OAAO,IAAI;OACR;;;IAGvD;GACF;SACK;AACN,SAAO;GACL,UAAU;GACV,MAAM;GACN,QAAQ;GACR,SAAS;GACV"}
|