@cleocode/core 2026.4.9 → 2026.4.11
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/dist/adapters/adapter-registry.js +64 -0
- package/dist/adapters/adapter-registry.js.map +1 -0
- package/dist/adapters/discovery.js +83 -0
- package/dist/adapters/discovery.js.map +1 -0
- package/dist/adapters/index.js +9 -0
- package/dist/adapters/index.js.map +1 -0
- package/dist/adapters/manager.js +260 -0
- package/dist/adapters/manager.js.map +1 -0
- package/dist/admin/export-tasks.js +171 -0
- package/dist/admin/export-tasks.js.map +1 -0
- package/dist/admin/export.js +103 -0
- package/dist/admin/export.js.map +1 -0
- package/dist/admin/help.js +106 -0
- package/dist/admin/help.js.map +1 -0
- package/dist/admin/import-tasks.js +182 -0
- package/dist/admin/import-tasks.js.map +1 -0
- package/dist/admin/import.js +129 -0
- package/dist/admin/import.js.map +1 -0
- package/dist/admin/index.js +13 -0
- package/dist/admin/index.js.map +1 -0
- package/dist/adrs/find.js +134 -0
- package/dist/adrs/find.js.map +1 -0
- package/dist/adrs/index.js +15 -0
- package/dist/adrs/index.js.map +1 -0
- package/dist/adrs/link-pipeline.js +160 -0
- package/dist/adrs/link-pipeline.js.map +1 -0
- package/dist/adrs/list.js +43 -0
- package/dist/adrs/list.js.map +1 -0
- package/dist/adrs/parse.js +51 -0
- package/dist/adrs/parse.js.map +1 -0
- package/dist/adrs/show.js +22 -0
- package/dist/adrs/show.js.map +1 -0
- package/dist/adrs/sync.js +188 -0
- package/dist/adrs/sync.js.map +1 -0
- package/dist/adrs/types.js +9 -0
- package/dist/adrs/types.js.map +1 -0
- package/dist/adrs/validate.js +57 -0
- package/dist/adrs/validate.js.map +1 -0
- package/dist/agents/agent-registry.js +288 -0
- package/dist/agents/agent-registry.js.map +1 -0
- package/dist/agents/agent-schema.d.ts +2 -2
- package/dist/agents/agent-schema.js +82 -0
- package/dist/agents/agent-schema.js.map +1 -0
- package/dist/agents/capacity.js +116 -0
- package/dist/agents/capacity.js.map +1 -0
- package/dist/agents/execution-learning.js +474 -0
- package/dist/agents/execution-learning.js.map +1 -0
- package/dist/agents/health-monitor.js +217 -0
- package/dist/agents/health-monitor.js.map +1 -0
- package/dist/agents/index.js +29 -0
- package/dist/agents/index.js.map +1 -0
- package/dist/agents/registry.js +314 -0
- package/dist/agents/registry.js.map +1 -0
- package/dist/agents/retry.js +215 -0
- package/dist/agents/retry.js.map +1 -0
- package/dist/audit-prune.js +94 -0
- package/dist/audit-prune.js.map +1 -0
- package/dist/audit.js +68 -0
- package/dist/audit.js.map +1 -0
- package/dist/backfill/index.js +229 -0
- package/dist/backfill/index.js.map +1 -0
- package/dist/bootstrap.js +344 -0
- package/dist/bootstrap.js.map +1 -0
- package/dist/caamp/adapter.js +259 -0
- package/dist/caamp/adapter.js.map +1 -0
- package/dist/caamp/capability-check.js +38 -0
- package/dist/caamp/capability-check.js.map +1 -0
- package/dist/caamp/index.js +21 -0
- package/dist/caamp/index.js.map +1 -0
- package/dist/caamp-init.js +16 -0
- package/dist/caamp-init.js.map +1 -0
- package/dist/cleo.js +322 -0
- package/dist/cleo.js.map +1 -0
- package/dist/code/index.js +10 -0
- package/dist/code/index.js.map +1 -0
- package/dist/code/outline.js +165 -0
- package/dist/code/outline.js.map +1 -0
- package/dist/code/parser.js +295 -0
- package/dist/code/parser.js.map +1 -0
- package/dist/code/search.js +135 -0
- package/dist/code/search.js.map +1 -0
- package/dist/code/unfold.js +155 -0
- package/dist/code/unfold.js.map +1 -0
- package/dist/codebase-map/analyzers/architecture.js +130 -0
- package/dist/codebase-map/analyzers/architecture.js.map +1 -0
- package/dist/codebase-map/analyzers/concerns.js +122 -0
- package/dist/codebase-map/analyzers/concerns.js.map +1 -0
- package/dist/codebase-map/analyzers/conventions.js +149 -0
- package/dist/codebase-map/analyzers/conventions.js.map +1 -0
- package/dist/codebase-map/analyzers/integrations.js +108 -0
- package/dist/codebase-map/analyzers/integrations.js.map +1 -0
- package/dist/codebase-map/analyzers/stack.js +117 -0
- package/dist/codebase-map/analyzers/stack.js.map +1 -0
- package/dist/codebase-map/analyzers/structure.js +137 -0
- package/dist/codebase-map/analyzers/structure.js.map +1 -0
- package/dist/codebase-map/analyzers/testing.js +118 -0
- package/dist/codebase-map/analyzers/testing.js.map +1 -0
- package/dist/codebase-map/index.js +57 -0
- package/dist/codebase-map/index.js.map +1 -0
- package/dist/codebase-map/store.js +122 -0
- package/dist/codebase-map/store.js.map +1 -0
- package/dist/codebase-map/summary.js +152 -0
- package/dist/codebase-map/summary.js.map +1 -0
- package/dist/compliance/index.js +288 -0
- package/dist/compliance/index.js.map +1 -0
- package/dist/compliance/protocol-enforcement.js +332 -0
- package/dist/compliance/protocol-enforcement.js.map +1 -0
- package/dist/compliance/protocol-rules.js +786 -0
- package/dist/compliance/protocol-rules.js.map +1 -0
- package/dist/compliance/protocol-types.js +79 -0
- package/dist/compliance/protocol-types.js.map +1 -0
- package/dist/compliance/store.js +53 -0
- package/dist/compliance/store.js.map +1 -0
- package/dist/conduit/conduit-client.js +107 -0
- package/dist/conduit/conduit-client.js.map +1 -0
- package/dist/conduit/factory.js +52 -0
- package/dist/conduit/factory.js.map +1 -0
- package/dist/conduit/http-transport.js +155 -0
- package/dist/conduit/http-transport.js.map +1 -0
- package/dist/conduit/index.js +15 -0
- package/dist/conduit/index.js.map +1 -0
- package/dist/conduit/local-transport.js +245 -0
- package/dist/conduit/local-transport.js.map +1 -0
- package/dist/conduit/sse-transport.js +299 -0
- package/dist/conduit/sse-transport.js.map +1 -0
- package/dist/config/build-config.js +29 -0
- package/dist/config/build-config.js.map +1 -0
- package/dist/config.js +401 -0
- package/dist/config.js.map +1 -0
- package/dist/constants.js +18 -0
- package/dist/constants.js.map +1 -0
- package/dist/context/index.js +137 -0
- package/dist/context/index.js.map +1 -0
- package/dist/crypto/credentials.js +191 -0
- package/dist/crypto/credentials.js.map +1 -0
- package/dist/discovery.js +182 -0
- package/dist/discovery.js.map +1 -0
- package/dist/engine-result.js +12 -0
- package/dist/engine-result.js.map +1 -0
- package/dist/error-catalog.js +404 -0
- package/dist/error-catalog.js.map +1 -0
- package/dist/error-registry.js +393 -0
- package/dist/error-registry.js.map +1 -0
- package/dist/errors.js +167 -0
- package/dist/errors.js.map +1 -0
- package/dist/hooks/handlers/agent-hooks.js +106 -0
- package/dist/hooks/handlers/agent-hooks.js.map +1 -0
- package/dist/hooks/handlers/context-hooks.js +111 -0
- package/dist/hooks/handlers/context-hooks.js.map +1 -0
- package/dist/hooks/handlers/error-hooks.js +52 -0
- package/dist/hooks/handlers/error-hooks.js.map +1 -0
- package/dist/hooks/handlers/file-hooks.js +104 -0
- package/dist/hooks/handlers/file-hooks.js.map +1 -0
- package/dist/hooks/handlers/handler-helpers.js +61 -0
- package/dist/hooks/handlers/handler-helpers.js.map +1 -0
- package/dist/hooks/handlers/index.js +28 -0
- package/dist/hooks/handlers/index.js.map +1 -0
- package/dist/hooks/handlers/memory-bridge-refresh.js +42 -0
- package/dist/hooks/handlers/memory-bridge-refresh.js.map +1 -0
- package/dist/hooks/handlers/notification-hooks.js +62 -0
- package/dist/hooks/handlers/notification-hooks.js.map +1 -0
- package/dist/hooks/handlers/session-hooks.d.ts +21 -0
- package/dist/hooks/handlers/session-hooks.d.ts.map +1 -1
- package/dist/hooks/handlers/session-hooks.js +142 -0
- package/dist/hooks/handlers/session-hooks.js.map +1 -0
- package/dist/hooks/handlers/task-hooks.js +65 -0
- package/dist/hooks/handlers/task-hooks.js.map +1 -0
- package/dist/hooks/handlers/work-capture-hooks.js +165 -0
- package/dist/hooks/handlers/work-capture-hooks.js.map +1 -0
- package/dist/hooks/index.js +13 -0
- package/dist/hooks/index.js.map +1 -0
- package/dist/hooks/payload-schemas.d.ts +2 -2
- package/dist/hooks/payload-schemas.js +220 -0
- package/dist/hooks/payload-schemas.js.map +1 -0
- package/dist/hooks/provider-hooks.js +66 -0
- package/dist/hooks/provider-hooks.js.map +1 -0
- package/dist/hooks/registry.js +229 -0
- package/dist/hooks/registry.js.map +1 -0
- package/dist/hooks/types.js +66 -0
- package/dist/hooks/types.js.map +1 -0
- package/dist/hooks.js +136 -0
- package/dist/hooks.js.map +1 -0
- package/dist/index.js +3361 -3095
- package/dist/index.js.map +4 -4
- package/dist/init.js +852 -0
- package/dist/init.js.map +1 -0
- package/dist/inject/index.js +82 -0
- package/dist/inject/index.js.map +1 -0
- package/dist/injection.js +343 -0
- package/dist/injection.js.map +1 -0
- package/dist/intelligence/adaptive-validation.js +497 -0
- package/dist/intelligence/adaptive-validation.js.map +1 -0
- package/dist/intelligence/impact.js +675 -0
- package/dist/intelligence/impact.js.map +1 -0
- package/dist/intelligence/index.js +22 -0
- package/dist/intelligence/index.js.map +1 -0
- package/dist/intelligence/patterns.js +492 -0
- package/dist/intelligence/patterns.js.map +1 -0
- package/dist/intelligence/prediction.js +499 -0
- package/dist/intelligence/prediction.js.map +1 -0
- package/dist/intelligence/types.js +13 -0
- package/dist/intelligence/types.js.map +1 -0
- package/dist/internal.d.ts +7 -2
- package/dist/internal.d.ts.map +1 -1
- package/dist/internal.js +299 -0
- package/dist/internal.js.map +1 -0
- package/dist/issue/create.js +121 -0
- package/dist/issue/create.js.map +1 -0
- package/dist/issue/diagnostics.js +59 -0
- package/dist/issue/diagnostics.js.map +1 -0
- package/dist/issue/index.js +10 -0
- package/dist/issue/index.js.map +1 -0
- package/dist/issue/template-parser.js +267 -0
- package/dist/issue/template-parser.js.map +1 -0
- package/dist/json-schema-validator.js +76 -0
- package/dist/json-schema-validator.js.map +1 -0
- package/dist/lib/index.js +11 -0
- package/dist/lib/index.js.map +1 -0
- package/dist/lib/retry.js +152 -0
- package/dist/lib/retry.js.map +1 -0
- package/dist/lib/tree-sitter-languages.js +71 -0
- package/dist/lib/tree-sitter-languages.js.map +1 -0
- package/dist/lifecycle/chain-composition.js +152 -0
- package/dist/lifecycle/chain-composition.js.map +1 -0
- package/dist/lifecycle/chain-store.js +246 -0
- package/dist/lifecycle/chain-store.js.map +1 -0
- package/dist/lifecycle/consolidate-rcasd.js +352 -0
- package/dist/lifecycle/consolidate-rcasd.js.map +1 -0
- package/dist/lifecycle/default-chain.js +176 -0
- package/dist/lifecycle/default-chain.js.map +1 -0
- package/dist/lifecycle/evidence.js +180 -0
- package/dist/lifecycle/evidence.js.map +1 -0
- package/dist/lifecycle/frontmatter.js +363 -0
- package/dist/lifecycle/frontmatter.js.map +1 -0
- package/dist/lifecycle/index.js +756 -0
- package/dist/lifecycle/index.js.map +1 -0
- package/dist/lifecycle/pipeline.js +656 -0
- package/dist/lifecycle/pipeline.js.map +1 -0
- package/dist/lifecycle/rcasd-index.js +326 -0
- package/dist/lifecycle/rcasd-index.js.map +1 -0
- package/dist/lifecycle/rcasd-paths.js +220 -0
- package/dist/lifecycle/rcasd-paths.js.map +1 -0
- package/dist/lifecycle/resume.js +864 -0
- package/dist/lifecycle/resume.js.map +1 -0
- package/dist/lifecycle/stage-artifacts.js +94 -0
- package/dist/lifecycle/stage-artifacts.js.map +1 -0
- package/dist/lifecycle/stage-guidance.js +234 -0
- package/dist/lifecycle/stage-guidance.js.map +1 -0
- package/dist/lifecycle/stages.js +534 -0
- package/dist/lifecycle/stages.js.map +1 -0
- package/dist/lifecycle/state-machine.js +516 -0
- package/dist/lifecycle/state-machine.js.map +1 -0
- package/dist/lifecycle/tessera-engine.js +249 -0
- package/dist/lifecycle/tessera-engine.js.map +1 -0
- package/dist/logger.js +140 -0
- package/dist/logger.js.map +1 -0
- package/dist/memory/auto-extract.js +177 -0
- package/dist/memory/auto-extract.js.map +1 -0
- package/dist/memory/brain-embedding.js +66 -0
- package/dist/memory/brain-embedding.js.map +1 -0
- package/dist/memory/brain-lifecycle.js +298 -0
- package/dist/memory/brain-lifecycle.js.map +1 -0
- package/dist/memory/brain-links.js +161 -0
- package/dist/memory/brain-links.js.map +1 -0
- package/dist/memory/brain-maintenance.js +114 -0
- package/dist/memory/brain-maintenance.js.map +1 -0
- package/dist/memory/brain-migration.js +149 -0
- package/dist/memory/brain-migration.js.map +1 -0
- package/dist/memory/brain-reasoning.js +215 -0
- package/dist/memory/brain-reasoning.js.map +1 -0
- package/dist/memory/brain-retrieval.js +542 -0
- package/dist/memory/brain-retrieval.js.map +1 -0
- package/dist/memory/brain-row-types.js +10 -0
- package/dist/memory/brain-row-types.js.map +1 -0
- package/dist/memory/brain-search.js +519 -0
- package/dist/memory/brain-search.js.map +1 -0
- package/dist/memory/brain-similarity.js +145 -0
- package/dist/memory/brain-similarity.js.map +1 -0
- package/dist/memory/claude-mem-migration.js +277 -0
- package/dist/memory/claude-mem-migration.js.map +1 -0
- package/dist/memory/decisions.js +162 -0
- package/dist/memory/decisions.js.map +1 -0
- package/dist/memory/embedding-local.js +97 -0
- package/dist/memory/embedding-local.js.map +1 -0
- package/dist/memory/embedding-queue.js +271 -0
- package/dist/memory/embedding-queue.js.map +1 -0
- package/dist/memory/embedding-worker.js +58 -0
- package/dist/memory/embedding-worker.js.map +1 -0
- package/dist/memory/engine-compat.js +1397 -0
- package/dist/memory/engine-compat.js.map +1 -0
- package/dist/memory/index.js +1140 -0
- package/dist/memory/index.js.map +1 -0
- package/dist/memory/learnings.d.ts +4 -4
- package/dist/memory/learnings.js +121 -0
- package/dist/memory/learnings.js.map +1 -0
- package/dist/memory/memory-bridge.js +370 -0
- package/dist/memory/memory-bridge.js.map +1 -0
- package/dist/memory/patterns.d.ts +6 -6
- package/dist/memory/patterns.js +122 -0
- package/dist/memory/patterns.js.map +1 -0
- package/dist/memory/pipeline-manifest-sqlite.js +975 -0
- package/dist/memory/pipeline-manifest-sqlite.js.map +1 -0
- package/dist/memory/session-memory.js +331 -0
- package/dist/memory/session-memory.js.map +1 -0
- package/dist/metrics/ab-test.js +260 -0
- package/dist/metrics/ab-test.js.map +1 -0
- package/dist/metrics/aggregation.js +363 -0
- package/dist/metrics/aggregation.js.map +1 -0
- package/dist/metrics/common.js +64 -0
- package/dist/metrics/common.js.map +1 -0
- package/dist/metrics/enums.js +78 -0
- package/dist/metrics/enums.js.map +1 -0
- package/dist/metrics/index.js +19 -0
- package/dist/metrics/index.js.map +1 -0
- package/dist/metrics/model-provider-registry.js +88 -0
- package/dist/metrics/model-provider-registry.js.map +1 -0
- package/dist/metrics/otel-integration.js +263 -0
- package/dist/metrics/otel-integration.js.map +1 -0
- package/dist/metrics/provider-detection.js +103 -0
- package/dist/metrics/provider-detection.js.map +1 -0
- package/dist/metrics/token-estimation.js +253 -0
- package/dist/metrics/token-estimation.js.map +1 -0
- package/dist/metrics/token-service.js +450 -0
- package/dist/metrics/token-service.js.map +1 -0
- package/dist/migration/agent-outputs.js +316 -0
- package/dist/migration/agent-outputs.js.map +1 -0
- package/dist/migration/checksum.js +92 -0
- package/dist/migration/checksum.js.map +1 -0
- package/dist/migration/index.js +282 -0
- package/dist/migration/index.js.map +1 -0
- package/dist/migration/logger.js +360 -0
- package/dist/migration/logger.js.map +1 -0
- package/dist/migration/preflight.js +9 -0
- package/dist/migration/preflight.js.map +1 -0
- package/dist/migration/state.js +421 -0
- package/dist/migration/state.js.map +1 -0
- package/dist/migration/validate.js +241 -0
- package/dist/migration/validate.js.map +1 -0
- package/dist/mvi-helpers.js +74 -0
- package/dist/mvi-helpers.js.map +1 -0
- package/dist/nexus/deps.js +375 -0
- package/dist/nexus/deps.js.map +1 -0
- package/dist/nexus/discover.js +288 -0
- package/dist/nexus/discover.js.map +1 -0
- package/dist/nexus/hash.js +10 -0
- package/dist/nexus/hash.js.map +1 -0
- package/dist/nexus/index.js +40 -0
- package/dist/nexus/index.js.map +1 -0
- package/dist/nexus/migrate-json-to-sqlite.js +115 -0
- package/dist/nexus/migrate-json-to-sqlite.js.map +1 -0
- package/dist/nexus/permissions.js +105 -0
- package/dist/nexus/permissions.js.map +1 -0
- package/dist/nexus/query.js +175 -0
- package/dist/nexus/query.js.map +1 -0
- package/dist/nexus/registry.js +584 -0
- package/dist/nexus/registry.js.map +1 -0
- package/dist/nexus/sharing/index.js +288 -0
- package/dist/nexus/sharing/index.js.map +1 -0
- package/dist/nexus/transfer-types.js +8 -0
- package/dist/nexus/transfer-types.js.map +1 -0
- package/dist/nexus/transfer.js +263 -0
- package/dist/nexus/transfer.js.map +1 -0
- package/dist/nexus/workspace.js +355 -0
- package/dist/nexus/workspace.js.map +1 -0
- package/dist/observability/index.js +103 -0
- package/dist/observability/index.js.map +1 -0
- package/dist/observability/log-filter.js +63 -0
- package/dist/observability/log-filter.js.map +1 -0
- package/dist/observability/log-parser.js +99 -0
- package/dist/observability/log-parser.js.map +1 -0
- package/dist/observability/log-reader.js +139 -0
- package/dist/observability/log-reader.js.map +1 -0
- package/dist/observability/types.js +19 -0
- package/dist/observability/types.js.map +1 -0
- package/dist/orchestration/analyze.js +107 -0
- package/dist/orchestration/analyze.js.map +1 -0
- package/dist/orchestration/bootstrap.js +132 -0
- package/dist/orchestration/bootstrap.js.map +1 -0
- package/dist/orchestration/context.js +56 -0
- package/dist/orchestration/context.js.map +1 -0
- package/dist/orchestration/critical-path.js +100 -0
- package/dist/orchestration/critical-path.js.map +1 -0
- package/dist/orchestration/hierarchy.js +183 -0
- package/dist/orchestration/hierarchy.js.map +1 -0
- package/dist/orchestration/index.js +287 -0
- package/dist/orchestration/index.js.map +1 -0
- package/dist/orchestration/parallel.js +89 -0
- package/dist/orchestration/parallel.js.map +1 -0
- package/dist/orchestration/protocol-validators.js +815 -0
- package/dist/orchestration/protocol-validators.js.map +1 -0
- package/dist/orchestration/skill-ops.js +98 -0
- package/dist/orchestration/skill-ops.js.map +1 -0
- package/dist/orchestration/status.js +107 -0
- package/dist/orchestration/status.js.map +1 -0
- package/dist/orchestration/unblock.js +103 -0
- package/dist/orchestration/unblock.js.map +1 -0
- package/dist/orchestration/validate-spawn.js +67 -0
- package/dist/orchestration/validate-spawn.js.map +1 -0
- package/dist/orchestration/waves.js +86 -0
- package/dist/orchestration/waves.js.map +1 -0
- package/dist/otel/index.js +163 -0
- package/dist/otel/index.js.map +1 -0
- package/dist/output.js +164 -0
- package/dist/output.js.map +1 -0
- package/dist/pagination.js +64 -0
- package/dist/pagination.js.map +1 -0
- package/dist/paths.d.ts +39 -9
- package/dist/paths.d.ts.map +1 -1
- package/dist/paths.js +776 -0
- package/dist/paths.js.map +1 -0
- package/dist/phases/deps.js +372 -0
- package/dist/phases/deps.js.map +1 -0
- package/dist/phases/index.js +349 -0
- package/dist/phases/index.js.map +1 -0
- package/dist/pipeline/index.js +10 -0
- package/dist/pipeline/index.js.map +1 -0
- package/dist/pipeline/phase.js +45 -0
- package/dist/pipeline/phase.js.map +1 -0
- package/dist/platform.js +211 -0
- package/dist/platform.js.map +1 -0
- package/dist/project-info.js +84 -0
- package/dist/project-info.js.map +1 -0
- package/dist/reconciliation/index.js +10 -0
- package/dist/reconciliation/index.js.map +1 -0
- package/dist/reconciliation/link-store.js +129 -0
- package/dist/reconciliation/link-store.js.map +1 -0
- package/dist/reconciliation/reconciliation-engine.js +298 -0
- package/dist/reconciliation/reconciliation-engine.js.map +1 -0
- package/dist/release/artifacts.js +427 -0
- package/dist/release/artifacts.js.map +1 -0
- package/dist/release/changelog-writer.js +151 -0
- package/dist/release/changelog-writer.js.map +1 -0
- package/dist/release/channel.js +144 -0
- package/dist/release/channel.js.map +1 -0
- package/dist/release/ci.js +166 -0
- package/dist/release/ci.js.map +1 -0
- package/dist/release/github-pr.js +225 -0
- package/dist/release/github-pr.js.map +1 -0
- package/dist/release/guards.js +116 -0
- package/dist/release/guards.js.map +1 -0
- package/dist/release/index.js +22 -0
- package/dist/release/index.js.map +1 -0
- package/dist/release/release-config.js +158 -0
- package/dist/release/release-config.js.map +1 -0
- package/dist/release/release-manifest.js +1019 -0
- package/dist/release/release-manifest.js.map +1 -0
- package/dist/release/version-bump.js +255 -0
- package/dist/release/version-bump.js.map +1 -0
- package/dist/remote/index.js +257 -0
- package/dist/remote/index.js.map +1 -0
- package/dist/repair.js +130 -0
- package/dist/repair.js.map +1 -0
- package/dist/research/index.js +2 -0
- package/dist/research/index.js.map +1 -0
- package/dist/roadmap/index.js +59 -0
- package/dist/roadmap/index.js.map +1 -0
- package/dist/routing/capability-matrix.js +1556 -0
- package/dist/routing/capability-matrix.js.map +1 -0
- package/dist/routing/index.js +9 -0
- package/dist/routing/index.js.map +1 -0
- package/dist/scaffold.d.ts +15 -2
- package/dist/scaffold.d.ts.map +1 -1
- package/dist/scaffold.js +1759 -0
- package/dist/scaffold.js.map +1 -0
- package/dist/schema-management.js +295 -0
- package/dist/schema-management.js.map +1 -0
- package/dist/security/index.js +9 -0
- package/dist/security/index.js.map +1 -0
- package/dist/security/input-sanitization.js +321 -0
- package/dist/security/input-sanitization.js.map +1 -0
- package/dist/sequence/index.js +295 -0
- package/dist/sequence/index.js.map +1 -0
- package/dist/sessions/assumptions.js +54 -0
- package/dist/sessions/assumptions.js.map +1 -0
- package/dist/sessions/briefing.js +377 -0
- package/dist/sessions/briefing.js.map +1 -0
- package/dist/sessions/context-alert.js +222 -0
- package/dist/sessions/context-alert.js.map +1 -0
- package/dist/sessions/context-inject.js +61 -0
- package/dist/sessions/context-inject.js.map +1 -0
- package/dist/sessions/context-monitor.js +98 -0
- package/dist/sessions/context-monitor.js.map +1 -0
- package/dist/sessions/decisions.js +65 -0
- package/dist/sessions/decisions.js.map +1 -0
- package/dist/sessions/find.js +65 -0
- package/dist/sessions/find.js.map +1 -0
- package/dist/sessions/handoff.js +328 -0
- package/dist/sessions/handoff.js.map +1 -0
- package/dist/sessions/hitl-warnings.js +254 -0
- package/dist/sessions/hitl-warnings.js.map +1 -0
- package/dist/sessions/index.js +327 -0
- package/dist/sessions/index.js.map +1 -0
- package/dist/sessions/session-archive.js +40 -0
- package/dist/sessions/session-archive.js.map +1 -0
- package/dist/sessions/session-cleanup.js +59 -0
- package/dist/sessions/session-cleanup.js.map +1 -0
- package/dist/sessions/session-drift.js +134 -0
- package/dist/sessions/session-drift.js.map +1 -0
- package/dist/sessions/session-enforcement.js +144 -0
- package/dist/sessions/session-enforcement.js.map +1 -0
- package/dist/sessions/session-grade.js +253 -0
- package/dist/sessions/session-grade.js.map +1 -0
- package/dist/sessions/session-history.js +42 -0
- package/dist/sessions/session-history.js.map +1 -0
- package/dist/sessions/session-id.js +81 -0
- package/dist/sessions/session-id.js.map +1 -0
- package/dist/sessions/session-memory-bridge.js +52 -0
- package/dist/sessions/session-memory-bridge.js.map +1 -0
- package/dist/sessions/session-show.js +24 -0
- package/dist/sessions/session-show.js.map +1 -0
- package/dist/sessions/session-stats.js +69 -0
- package/dist/sessions/session-stats.js.map +1 -0
- package/dist/sessions/session-suspend.js +39 -0
- package/dist/sessions/session-suspend.js.map +1 -0
- package/dist/sessions/session-switch.js +51 -0
- package/dist/sessions/session-switch.js.map +1 -0
- package/dist/sessions/session-view.js +76 -0
- package/dist/sessions/session-view.js.map +1 -0
- package/dist/sessions/snapshot.js +213 -0
- package/dist/sessions/snapshot.js.map +1 -0
- package/dist/sessions/statusline-setup.js +85 -0
- package/dist/sessions/statusline-setup.js.map +1 -0
- package/dist/sessions/types.js +8 -0
- package/dist/sessions/types.js.map +1 -0
- package/dist/skills/agents/config.js +94 -0
- package/dist/skills/agents/config.js.map +1 -0
- package/dist/skills/agents/install.js +116 -0
- package/dist/skills/agents/install.js.map +1 -0
- package/dist/skills/agents/registry.js +161 -0
- package/dist/skills/agents/registry.js.map +1 -0
- package/dist/skills/discovery.js +333 -0
- package/dist/skills/discovery.js.map +1 -0
- package/dist/skills/dispatch.js +347 -0
- package/dist/skills/dispatch.js.map +1 -0
- package/dist/skills/dynamic-skill-generator.js +87 -0
- package/dist/skills/dynamic-skill-generator.js.map +1 -0
- package/dist/skills/index.js +44 -0
- package/dist/skills/index.js.map +1 -0
- package/dist/skills/injection/subagent.js +195 -0
- package/dist/skills/injection/subagent.js.map +1 -0
- package/dist/skills/injection/token.js +260 -0
- package/dist/skills/injection/token.js.map +1 -0
- package/dist/skills/install.js +40 -0
- package/dist/skills/install.js.map +1 -0
- package/dist/skills/manifests/contribution.js +175 -0
- package/dist/skills/manifests/contribution.js.map +1 -0
- package/dist/skills/manifests/research.js +281 -0
- package/dist/skills/manifests/research.js.map +1 -0
- package/dist/skills/manifests/resolver.js +146 -0
- package/dist/skills/manifests/resolver.js.map +1 -0
- package/dist/skills/marketplace.js +90 -0
- package/dist/skills/marketplace.js.map +1 -0
- package/dist/skills/orchestrator/spawn.js +178 -0
- package/dist/skills/orchestrator/spawn.js.map +1 -0
- package/dist/skills/orchestrator/startup.js +451 -0
- package/dist/skills/orchestrator/startup.js.map +1 -0
- package/dist/skills/orchestrator/validator.js +301 -0
- package/dist/skills/orchestrator/validator.js.map +1 -0
- package/dist/skills/precedence-integration.js +73 -0
- package/dist/skills/precedence-integration.js.map +1 -0
- package/dist/skills/precedence-types.js +16 -0
- package/dist/skills/precedence-types.js.map +1 -0
- package/dist/skills/routing-table.js +63 -0
- package/dist/skills/routing-table.js.map +1 -0
- package/dist/skills/skill-paths.js +217 -0
- package/dist/skills/skill-paths.js.map +1 -0
- package/dist/skills/test-utility.js +55 -0
- package/dist/skills/test-utility.js.map +1 -0
- package/dist/skills/types.js +118 -0
- package/dist/skills/types.js.map +1 -0
- package/dist/skills/validation.js +183 -0
- package/dist/skills/validation.js.map +1 -0
- package/dist/skills/version.js +57 -0
- package/dist/skills/version.js.map +1 -0
- package/dist/snapshot/index.js +188 -0
- package/dist/snapshot/index.js.map +1 -0
- package/dist/spawn/adapter-registry.js +246 -0
- package/dist/spawn/adapter-registry.js.map +1 -0
- package/dist/spawn/index.js +10 -0
- package/dist/spawn/index.js.map +1 -0
- package/dist/stats/index.js +343 -0
- package/dist/stats/index.js.map +1 -0
- package/dist/stats/workflow-telemetry.js +400 -0
- package/dist/stats/workflow-telemetry.js.map +1 -0
- package/dist/sticky/archive.js +47 -0
- package/dist/sticky/archive.js.map +1 -0
- package/dist/sticky/convert.js +235 -0
- package/dist/sticky/convert.js.map +1 -0
- package/dist/sticky/create.js +48 -0
- package/dist/sticky/create.js.map +1 -0
- package/dist/sticky/id.js +35 -0
- package/dist/sticky/id.js.map +1 -0
- package/dist/sticky/index.js +16 -0
- package/dist/sticky/index.js.map +1 -0
- package/dist/sticky/list.js +44 -0
- package/dist/sticky/list.js.map +1 -0
- package/dist/sticky/purge.js +45 -0
- package/dist/sticky/purge.js.map +1 -0
- package/dist/sticky/show.js +42 -0
- package/dist/sticky/show.js.map +1 -0
- package/dist/sticky/types.js +10 -0
- package/dist/sticky/types.js.map +1 -0
- package/dist/store/agent-registry-accessor.js +265 -0
- package/dist/store/agent-registry-accessor.js.map +1 -0
- package/dist/store/atomic.js +167 -0
- package/dist/store/atomic.js.map +1 -0
- package/dist/store/backup.js +94 -0
- package/dist/store/backup.js.map +1 -0
- package/dist/store/brain-accessor.js +397 -0
- package/dist/store/brain-accessor.js.map +1 -0
- package/dist/store/brain-schema.d.ts +8 -8
- package/dist/store/brain-schema.js +215 -0
- package/dist/store/brain-schema.js.map +1 -0
- package/dist/store/brain-sqlite.js +222 -0
- package/dist/store/brain-sqlite.js.map +1 -0
- package/dist/store/cache.js +168 -0
- package/dist/store/cache.js.map +1 -0
- package/dist/store/chain-schema.js +51 -0
- package/dist/store/chain-schema.js.map +1 -0
- package/dist/store/cleanup-legacy.d.ts +128 -0
- package/dist/store/cleanup-legacy.d.ts.map +1 -0
- package/dist/store/converters.js +124 -0
- package/dist/store/converters.js.map +1 -0
- package/dist/store/cross-db-cleanup.js +319 -0
- package/dist/store/cross-db-cleanup.js.map +1 -0
- package/dist/store/data-accessor.js +26 -0
- package/dist/store/data-accessor.js.map +1 -0
- package/dist/store/data-safety-central.js +269 -0
- package/dist/store/data-safety-central.js.map +1 -0
- package/dist/store/data-safety.js +274 -0
- package/dist/store/data-safety.js.map +1 -0
- package/dist/store/db-helpers.js +224 -0
- package/dist/store/db-helpers.js.map +1 -0
- package/dist/store/export.js +155 -0
- package/dist/store/export.js.map +1 -0
- package/dist/store/file-utils.js +270 -0
- package/dist/store/file-utils.js.map +1 -0
- package/dist/store/git-checkpoint.js +365 -0
- package/dist/store/git-checkpoint.js.map +1 -0
- package/dist/store/import-logging.js +139 -0
- package/dist/store/import-logging.js.map +1 -0
- package/dist/store/import-remap.js +145 -0
- package/dist/store/import-remap.js.map +1 -0
- package/dist/store/import-sort.js +121 -0
- package/dist/store/import-sort.js.map +1 -0
- package/dist/store/index.d.ts +1 -0
- package/dist/store/index.d.ts.map +1 -1
- package/dist/store/index.js +29 -0
- package/dist/store/index.js.map +1 -0
- package/dist/store/json.js +208 -0
- package/dist/store/json.js.map +1 -0
- package/dist/store/lifecycle-store.js +249 -0
- package/dist/store/lifecycle-store.js.map +1 -0
- package/dist/store/lock.js +70 -0
- package/dist/store/lock.js.map +1 -0
- package/dist/store/migration-manager.js +151 -0
- package/dist/store/migration-manager.js.map +1 -0
- package/dist/store/migration-sqlite.js +676 -0
- package/dist/store/migration-sqlite.js.map +1 -0
- package/dist/store/nexus-schema.js +62 -0
- package/dist/store/nexus-schema.js.map +1 -0
- package/dist/store/nexus-sqlite.d.ts +14 -2
- package/dist/store/nexus-sqlite.d.ts.map +1 -1
- package/dist/store/nexus-sqlite.js +217 -0
- package/dist/store/nexus-sqlite.js.map +1 -0
- package/dist/store/nexus-validation-schemas.js +40 -0
- package/dist/store/nexus-validation-schemas.js.map +1 -0
- package/dist/store/parsers.js +37 -0
- package/dist/store/parsers.js.map +1 -0
- package/dist/store/project-detect.js +457 -0
- package/dist/store/project-detect.js.map +1 -0
- package/dist/store/provider.js +101 -0
- package/dist/store/provider.js.map +1 -0
- package/dist/store/safety-data-accessor.js +257 -0
- package/dist/store/safety-data-accessor.js.map +1 -0
- package/dist/store/schema.js +7 -0
- package/dist/store/schema.js.map +1 -0
- package/dist/store/session-store.js +219 -0
- package/dist/store/session-store.js.map +1 -0
- package/dist/store/signaldock-sqlite.js +400 -0
- package/dist/store/signaldock-sqlite.js.map +1 -0
- package/dist/store/sqlite-backup.d.ts +121 -10
- package/dist/store/sqlite-backup.d.ts.map +1 -1
- package/dist/store/sqlite-backup.js +241 -0
- package/dist/store/sqlite-backup.js.map +1 -0
- package/dist/store/sqlite-data-accessor.js +787 -0
- package/dist/store/sqlite-data-accessor.js.map +1 -0
- package/dist/store/sqlite.d.ts.map +1 -1
- package/dist/store/sqlite.js +481 -0
- package/dist/store/sqlite.js.map +1 -0
- package/dist/store/status-registry.js +8 -0
- package/dist/store/status-registry.js.map +1 -0
- package/dist/store/task-store.js +358 -0
- package/dist/store/task-store.js.map +1 -0
- package/dist/store/tasks-schema.d.ts +8 -8
- package/dist/store/tasks-schema.js +610 -0
- package/dist/store/tasks-schema.js.map +1 -0
- package/dist/store/typed-query.js +15 -0
- package/dist/store/typed-query.js.map +1 -0
- package/dist/store/validation-schemas.d.ts +37 -37
- package/dist/store/validation-schemas.js +278 -0
- package/dist/store/validation-schemas.js.map +1 -0
- package/dist/system/archive-analytics.js +277 -0
- package/dist/system/archive-analytics.js.map +1 -0
- package/dist/system/archive-stats.js +64 -0
- package/dist/system/archive-stats.js.map +1 -0
- package/dist/system/audit.js +145 -0
- package/dist/system/audit.js.map +1 -0
- package/dist/system/backup.d.ts +91 -3
- package/dist/system/backup.d.ts.map +1 -1
- package/dist/system/backup.js +280 -0
- package/dist/system/backup.js.map +1 -0
- package/dist/system/cleanup.js +134 -0
- package/dist/system/cleanup.js.map +1 -0
- package/dist/system/health.js +1100 -0
- package/dist/system/health.js.map +1 -0
- package/dist/system/index.js +18 -0
- package/dist/system/index.js.map +1 -0
- package/dist/system/inject-generate.js +122 -0
- package/dist/system/inject-generate.js.map +1 -0
- package/dist/system/labels.js +38 -0
- package/dist/system/labels.js.map +1 -0
- package/dist/system/metrics.js +61 -0
- package/dist/system/metrics.js.map +1 -0
- package/dist/system/migrate.js +43 -0
- package/dist/system/migrate.js.map +1 -0
- package/dist/system/platform-paths.js +80 -0
- package/dist/system/platform-paths.js.map +1 -0
- package/dist/system/runtime.js +161 -0
- package/dist/system/runtime.js.map +1 -0
- package/dist/system/safestop.js +99 -0
- package/dist/system/safestop.js.map +1 -0
- package/dist/system/storage-preflight.js +123 -0
- package/dist/system/storage-preflight.js.map +1 -0
- package/dist/task-work/index.js +159 -0
- package/dist/task-work/index.js.map +1 -0
- package/dist/tasks/add.js +661 -0
- package/dist/tasks/add.js.map +1 -0
- package/dist/tasks/analyze.js +85 -0
- package/dist/tasks/analyze.js.map +1 -0
- package/dist/tasks/archive.js +90 -0
- package/dist/tasks/archive.js.map +1 -0
- package/dist/tasks/atomicity.js +83 -0
- package/dist/tasks/atomicity.js.map +1 -0
- package/dist/tasks/cancel-ops.js +83 -0
- package/dist/tasks/cancel-ops.js.map +1 -0
- package/dist/tasks/complete.js +211 -0
- package/dist/tasks/complete.js.map +1 -0
- package/dist/tasks/crossref-extract.js +73 -0
- package/dist/tasks/crossref-extract.js.map +1 -0
- package/dist/tasks/delete-preview.js +192 -0
- package/dist/tasks/delete-preview.js.map +1 -0
- package/dist/tasks/delete.js +120 -0
- package/dist/tasks/delete.js.map +1 -0
- package/dist/tasks/deletion-strategy.js +200 -0
- package/dist/tasks/deletion-strategy.js.map +1 -0
- package/dist/tasks/dependency-check.js +278 -0
- package/dist/tasks/dependency-check.js.map +1 -0
- package/dist/tasks/deps-ready.js +32 -0
- package/dist/tasks/deps-ready.js.map +1 -0
- package/dist/tasks/enforcement.js +86 -0
- package/dist/tasks/enforcement.js.map +1 -0
- package/dist/tasks/epic-enforcement.js +294 -0
- package/dist/tasks/epic-enforcement.js.map +1 -0
- package/dist/tasks/find.js +154 -0
- package/dist/tasks/find.js.map +1 -0
- package/dist/tasks/graph-cache.js +127 -0
- package/dist/tasks/graph-cache.js.map +1 -0
- package/dist/tasks/graph-ops.js +171 -0
- package/dist/tasks/graph-ops.js.map +1 -0
- package/dist/tasks/graph-rag.js +328 -0
- package/dist/tasks/graph-rag.js.map +1 -0
- package/dist/tasks/hierarchy-policy.js +149 -0
- package/dist/tasks/hierarchy-policy.js.map +1 -0
- package/dist/tasks/hierarchy.js +185 -0
- package/dist/tasks/hierarchy.js.map +1 -0
- package/dist/tasks/id-generator.js +65 -0
- package/dist/tasks/id-generator.js.map +1 -0
- package/dist/tasks/index.js +14 -0
- package/dist/tasks/index.js.map +1 -0
- package/dist/tasks/labels.js +52 -0
- package/dist/tasks/labels.js.map +1 -0
- package/dist/tasks/list.js +75 -0
- package/dist/tasks/list.js.map +1 -0
- package/dist/tasks/phase-tracking.js +133 -0
- package/dist/tasks/phase-tracking.js.map +1 -0
- package/dist/tasks/pipeline-stage.js +248 -0
- package/dist/tasks/pipeline-stage.js.map +1 -0
- package/dist/tasks/plan.js +268 -0
- package/dist/tasks/plan.js.map +1 -0
- package/dist/tasks/relates.js +89 -0
- package/dist/tasks/relates.js.map +1 -0
- package/dist/tasks/show.js +80 -0
- package/dist/tasks/show.js.map +1 -0
- package/dist/tasks/size-weighting.js +86 -0
- package/dist/tasks/size-weighting.js.map +1 -0
- package/dist/tasks/staleness.js +86 -0
- package/dist/tasks/staleness.js.map +1 -0
- package/dist/tasks/task-ops.js +1741 -0
- package/dist/tasks/task-ops.js.map +1 -0
- package/dist/tasks/update.js +277 -0
- package/dist/tasks/update.js.map +1 -0
- package/dist/templates/index.js +10 -0
- package/dist/templates/index.js.map +1 -0
- package/dist/templates/parser.js +254 -0
- package/dist/templates/parser.js.map +1 -0
- package/dist/ui/aliases.js +153 -0
- package/dist/ui/aliases.js.map +1 -0
- package/dist/ui/changelog.js +184 -0
- package/dist/ui/changelog.js.map +1 -0
- package/dist/ui/command-registry.js +168 -0
- package/dist/ui/command-registry.js.map +1 -0
- package/dist/ui/flags.js +94 -0
- package/dist/ui/flags.js.map +1 -0
- package/dist/ui/index.js +24 -0
- package/dist/ui/index.js.map +1 -0
- package/dist/upgrade.js +1148 -0
- package/dist/upgrade.js.map +1 -0
- package/dist/validation/chain-validation.js +146 -0
- package/dist/validation/chain-validation.js.map +1 -0
- package/dist/validation/compliance.js +155 -0
- package/dist/validation/compliance.js.map +1 -0
- package/dist/validation/docs-sync.js +212 -0
- package/dist/validation/docs-sync.js.map +1 -0
- package/dist/validation/doctor/checks.js +1069 -0
- package/dist/validation/doctor/checks.js.map +1 -0
- package/dist/validation/doctor/index.js +9 -0
- package/dist/validation/doctor/index.js.map +1 -0
- package/dist/validation/doctor/project-cache.js +160 -0
- package/dist/validation/doctor/project-cache.js.map +1 -0
- package/dist/validation/doctor/utils.js +155 -0
- package/dist/validation/doctor/utils.js.map +1 -0
- package/dist/validation/engine.js +902 -0
- package/dist/validation/engine.js.map +1 -0
- package/dist/validation/gap-check.js +175 -0
- package/dist/validation/gap-check.js.map +1 -0
- package/dist/validation/index.js +40 -0
- package/dist/validation/index.js.map +1 -0
- package/dist/validation/manifest.js +237 -0
- package/dist/validation/manifest.js.map +1 -0
- package/dist/validation/operation-gate-validators.js +724 -0
- package/dist/validation/operation-gate-validators.js.map +1 -0
- package/dist/validation/operation-verification-gates.js +532 -0
- package/dist/validation/operation-verification-gates.js.map +1 -0
- package/dist/validation/param-utils.js +139 -0
- package/dist/validation/param-utils.js.map +1 -0
- package/dist/validation/protocol-common.js +300 -0
- package/dist/validation/protocol-common.js.map +1 -0
- package/dist/validation/protocols/_shared.js +75 -0
- package/dist/validation/protocols/_shared.js.map +1 -0
- package/dist/validation/protocols/architecture-decision.js +31 -0
- package/dist/validation/protocols/architecture-decision.js.map +1 -0
- package/dist/validation/protocols/artifact-publish.js +28 -0
- package/dist/validation/protocols/artifact-publish.js.map +1 -0
- package/dist/validation/protocols/consensus.js +41 -0
- package/dist/validation/protocols/consensus.js.map +1 -0
- package/dist/validation/protocols/contribution.js +27 -0
- package/dist/validation/protocols/contribution.js.map +1 -0
- package/dist/validation/protocols/decomposition.js +28 -0
- package/dist/validation/protocols/decomposition.js.map +1 -0
- package/dist/validation/protocols/implementation.js +24 -0
- package/dist/validation/protocols/implementation.js.map +1 -0
- package/dist/validation/protocols/provenance.js +29 -0
- package/dist/validation/protocols/provenance.js.map +1 -0
- package/dist/validation/protocols/release.js +29 -0
- package/dist/validation/protocols/release.js.map +1 -0
- package/dist/validation/protocols/research.js +24 -0
- package/dist/validation/protocols/research.js.map +1 -0
- package/dist/validation/protocols/specification.js +27 -0
- package/dist/validation/protocols/specification.js.map +1 -0
- package/dist/validation/protocols/testing.js +30 -0
- package/dist/validation/protocols/testing.js.map +1 -0
- package/dist/validation/protocols/validation.js +30 -0
- package/dist/validation/protocols/validation.js.map +1 -0
- package/dist/validation/schema-integrity.js +170 -0
- package/dist/validation/schema-integrity.js.map +1 -0
- package/dist/validation/schema-validator.js +176 -0
- package/dist/validation/schema-validator.js.map +1 -0
- package/dist/validation/validate-ops.js +937 -0
- package/dist/validation/validate-ops.js.map +1 -0
- package/dist/validation/validation-rules.js +226 -0
- package/dist/validation/validation-rules.js.map +1 -0
- package/dist/validation/verification.js +321 -0
- package/dist/validation/verification.js.map +1 -0
- package/package.json +10 -8
- package/src/__tests__/paths-walkup.test.ts +305 -0
- package/src/__tests__/paths.test.ts +61 -17
- package/src/hooks/handlers/session-hooks.ts +42 -0
- package/src/internal.ts +19 -2
- package/src/paths.ts +91 -14
- package/src/scaffold.ts +22 -3
- package/src/store/__tests__/cleanup-legacy.test.ts +268 -0
- package/src/store/__tests__/database-topology-integration.test.ts +504 -0
- package/src/store/__tests__/sqlite-backup-global.test.ts +281 -0
- package/src/store/__tests__/sqlite-backup.test.ts +118 -10
- package/src/store/cleanup-legacy.ts +208 -0
- package/src/store/index.ts +7 -0
- package/src/store/nexus-sqlite.ts +32 -3
- package/src/store/sqlite-backup.ts +368 -37
- package/src/store/sqlite.ts +19 -3
- package/src/system/__tests__/backup.test.ts +237 -0
- package/src/system/backup.ts +248 -28
- package/templates/cleo-gitignore +19 -3
|
@@ -0,0 +1,1019 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Release manifest operations backed by the release_manifests SQLite table.
|
|
3
|
+
*
|
|
4
|
+
* Migrated from .cleo/releases.json to SQLite per T5580.
|
|
5
|
+
* All reads/writes now go through Drizzle ORM via tasks.db.
|
|
6
|
+
*
|
|
7
|
+
* @task T5580
|
|
8
|
+
* @task T4788
|
|
9
|
+
*/
|
|
10
|
+
import { execFileSync } from 'node:child_process';
|
|
11
|
+
import { existsSync, renameSync } from 'node:fs';
|
|
12
|
+
import { readFile } from 'node:fs/promises';
|
|
13
|
+
import { join } from 'node:path';
|
|
14
|
+
import { and, count, desc, eq } from 'drizzle-orm';
|
|
15
|
+
import { createPage } from '../pagination.js';
|
|
16
|
+
import { getCleoDirAbsolute, getProjectRoot } from '../paths.js';
|
|
17
|
+
import * as schema from '../store/tasks-schema.js';
|
|
18
|
+
import { parseChangelogBlocks, writeChangelogSection } from './changelog-writer.js';
|
|
19
|
+
import { resolveChannelFromBranch } from './channel.js';
|
|
20
|
+
import { detectBranchProtection } from './github-pr.js';
|
|
21
|
+
import { getChannelConfig, getGitFlowConfig, getPushMode, loadReleaseConfig, } from './release-config.js';
|
|
22
|
+
import { getVersionBumpConfig } from './version-bump.js';
|
|
23
|
+
async function getDb(cwd) {
|
|
24
|
+
const { getDb: _getDb } = await import('../store/sqlite.js');
|
|
25
|
+
return _getDb(cwd);
|
|
26
|
+
}
|
|
27
|
+
function normalizeLimit(limit) {
|
|
28
|
+
return typeof limit === 'number' && limit > 0 ? limit : undefined;
|
|
29
|
+
}
|
|
30
|
+
function normalizeOffset(offset) {
|
|
31
|
+
return typeof offset === 'number' && offset > 0 ? offset : undefined;
|
|
32
|
+
}
|
|
33
|
+
function effectivePageLimit(limit, offset) {
|
|
34
|
+
return limit ?? (offset !== undefined ? 50 : undefined);
|
|
35
|
+
}
|
|
36
|
+
// ── Internal helpers ─────────────────────────────────────────────────
|
|
37
|
+
function isValidVersion(version) {
|
|
38
|
+
return /^v?\d+\.\d+\.\d+(-[\w.]+)?(\+[\w.]+)?$/.test(version);
|
|
39
|
+
}
|
|
40
|
+
function validateCalVerWindow(version, now = new Date()) {
|
|
41
|
+
const normalized = version.startsWith('v') ? version.slice(1) : version;
|
|
42
|
+
const base = normalized.split('-')[0] ?? normalized;
|
|
43
|
+
const parts = base.split('.');
|
|
44
|
+
if (parts.length !== 3) {
|
|
45
|
+
return { valid: false, message: `Invalid CalVer format: ${version}` };
|
|
46
|
+
}
|
|
47
|
+
const tagYear = Number.parseInt(parts[0] ?? '', 10);
|
|
48
|
+
const tagMonth = Number.parseInt(parts[1] ?? '', 10);
|
|
49
|
+
if (!Number.isInteger(tagYear) || !Number.isInteger(tagMonth)) {
|
|
50
|
+
return { valid: false, message: `Invalid CalVer date components: ${version}` };
|
|
51
|
+
}
|
|
52
|
+
const currentYear = now.getUTCFullYear();
|
|
53
|
+
const currentMonth = now.getUTCMonth() + 1;
|
|
54
|
+
const nextMonth = currentMonth === 12 ? 1 : currentMonth + 1;
|
|
55
|
+
const nextYear = currentMonth === 12 ? currentYear + 1 : currentYear;
|
|
56
|
+
const isPreRelease = normalized.includes('-');
|
|
57
|
+
if (isPreRelease) {
|
|
58
|
+
const valid = (tagYear === currentYear || tagYear === nextYear) &&
|
|
59
|
+
(tagMonth === currentMonth || tagMonth === nextMonth);
|
|
60
|
+
return {
|
|
61
|
+
valid,
|
|
62
|
+
message: valid
|
|
63
|
+
? `CalVer OK (pre-release): ${version}`
|
|
64
|
+
: `Pre-release ${version} outside allowed CalVer range ${currentYear}.${currentMonth} or ${nextYear}.${nextMonth}`,
|
|
65
|
+
};
|
|
66
|
+
}
|
|
67
|
+
const valid = tagYear === currentYear && tagMonth === currentMonth;
|
|
68
|
+
return {
|
|
69
|
+
valid,
|
|
70
|
+
message: valid
|
|
71
|
+
? `CalVer OK (stable): ${version}`
|
|
72
|
+
: `${version} does not match current CalVer ${currentYear}.${currentMonth}`,
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
function normalizeVersion(version) {
|
|
76
|
+
return version.startsWith('v') ? version : `v${version}`;
|
|
77
|
+
}
|
|
78
|
+
function rowToManifest(row) {
|
|
79
|
+
return {
|
|
80
|
+
version: row.version,
|
|
81
|
+
status: row.status,
|
|
82
|
+
createdAt: row.createdAt,
|
|
83
|
+
preparedAt: row.preparedAt ?? undefined,
|
|
84
|
+
committedAt: row.committedAt ?? undefined,
|
|
85
|
+
taggedAt: row.taggedAt ?? undefined,
|
|
86
|
+
pushedAt: row.pushedAt ?? undefined,
|
|
87
|
+
tasks: JSON.parse(row.tasksJson),
|
|
88
|
+
notes: row.notes ?? undefined,
|
|
89
|
+
changelog: row.changelog ?? undefined,
|
|
90
|
+
previousVersion: row.previousVersion ?? undefined,
|
|
91
|
+
commitSha: row.commitSha ?? undefined,
|
|
92
|
+
gitTag: row.gitTag ?? undefined,
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
async function findLatestPushedVersion(cwd) {
|
|
96
|
+
const db = await getDb(cwd);
|
|
97
|
+
const rows = await db
|
|
98
|
+
.select({ version: schema.releaseManifests.version })
|
|
99
|
+
.from(schema.releaseManifests)
|
|
100
|
+
.where(eq(schema.releaseManifests.status, 'pushed'))
|
|
101
|
+
.orderBy(desc(schema.releaseManifests.pushedAt))
|
|
102
|
+
.limit(1)
|
|
103
|
+
.all();
|
|
104
|
+
return rows[0]?.version;
|
|
105
|
+
}
|
|
106
|
+
// ── Public API ───────────────────────────────────────────────────────
|
|
107
|
+
/**
|
|
108
|
+
* Prepare a release (create a release manifest entry).
|
|
109
|
+
* @task T4788
|
|
110
|
+
*/
|
|
111
|
+
export async function prepareRelease(version, tasks, notes, loadTasksFn, cwd) {
|
|
112
|
+
if (!version) {
|
|
113
|
+
throw new Error('version is required');
|
|
114
|
+
}
|
|
115
|
+
if (!isValidVersion(version)) {
|
|
116
|
+
throw new Error(`Invalid version format: ${version} (expected X.Y.Z or YYYY.M.patch)`);
|
|
117
|
+
}
|
|
118
|
+
const normalizedVersion = normalizeVersion(version);
|
|
119
|
+
const db = await getDb(cwd);
|
|
120
|
+
const existing = await db
|
|
121
|
+
.select()
|
|
122
|
+
.from(schema.releaseManifests)
|
|
123
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
124
|
+
.limit(1)
|
|
125
|
+
.all();
|
|
126
|
+
if (existing.length > 0) {
|
|
127
|
+
throw new Error(`Release ${normalizedVersion} already exists (status: ${existing[0].status})`);
|
|
128
|
+
}
|
|
129
|
+
let releaseTasks = tasks ?? [];
|
|
130
|
+
if (releaseTasks.length === 0) {
|
|
131
|
+
const allTasks = await loadTasksFn();
|
|
132
|
+
releaseTasks = allTasks.filter((t) => t.status === 'done' && t.completedAt).map((t) => t.id);
|
|
133
|
+
}
|
|
134
|
+
// Filter out epic IDs
|
|
135
|
+
const allTasks = await loadTasksFn();
|
|
136
|
+
const epicIds = new Set(allTasks.filter((t) => allTasks.some((c) => c.parentId === t.id)).map((t) => t.id));
|
|
137
|
+
releaseTasks = releaseTasks.filter((id) => !epicIds.has(id));
|
|
138
|
+
const previousVersion = await findLatestPushedVersion(cwd);
|
|
139
|
+
const now = new Date().toISOString();
|
|
140
|
+
const id = `rel-${normalizedVersion.replace(/[^a-z0-9]/gi, '-')}`;
|
|
141
|
+
await db
|
|
142
|
+
.insert(schema.releaseManifests)
|
|
143
|
+
.values({
|
|
144
|
+
id,
|
|
145
|
+
version: normalizedVersion,
|
|
146
|
+
status: 'prepared',
|
|
147
|
+
tasksJson: JSON.stringify(releaseTasks),
|
|
148
|
+
notes: notes ?? null,
|
|
149
|
+
previousVersion: previousVersion ?? null,
|
|
150
|
+
createdAt: now,
|
|
151
|
+
preparedAt: now,
|
|
152
|
+
})
|
|
153
|
+
.run();
|
|
154
|
+
return {
|
|
155
|
+
version: normalizedVersion,
|
|
156
|
+
status: 'prepared',
|
|
157
|
+
tasks: releaseTasks,
|
|
158
|
+
taskCount: releaseTasks.length,
|
|
159
|
+
};
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Generate changelog for a release.
|
|
163
|
+
* @task T4788
|
|
164
|
+
*/
|
|
165
|
+
export async function generateReleaseChangelog(version, loadTasksFn, cwd) {
|
|
166
|
+
if (!version) {
|
|
167
|
+
throw new Error('version is required');
|
|
168
|
+
}
|
|
169
|
+
const normalizedVersion = normalizeVersion(version);
|
|
170
|
+
const db = await getDb(cwd);
|
|
171
|
+
const rows = await db
|
|
172
|
+
.select()
|
|
173
|
+
.from(schema.releaseManifests)
|
|
174
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
175
|
+
.limit(1)
|
|
176
|
+
.all();
|
|
177
|
+
if (rows.length === 0) {
|
|
178
|
+
throw new Error(`Release ${normalizedVersion} not found`);
|
|
179
|
+
}
|
|
180
|
+
const row = rows[0];
|
|
181
|
+
const releaseTasks = JSON.parse(row.tasksJson);
|
|
182
|
+
const allTasks = await loadTasksFn();
|
|
183
|
+
const taskMap = new Map(allTasks.map((t) => [t.id, t]));
|
|
184
|
+
const features = [];
|
|
185
|
+
const fixes = [];
|
|
186
|
+
const chores = [];
|
|
187
|
+
const docs = [];
|
|
188
|
+
const tests = [];
|
|
189
|
+
const changes = [];
|
|
190
|
+
/**
|
|
191
|
+
* Strip conventional commit prefixes from task titles.
|
|
192
|
+
* e.g. "feat: add auth" → "Add auth", "fix(ui): button" → "Button"
|
|
193
|
+
*/
|
|
194
|
+
function stripConventionalPrefix(title) {
|
|
195
|
+
return title.replace(/^(feat|fix|docs?|test|chore|refactor|style|ci|build|perf)(\([^)]+\))?:\s*/i, '');
|
|
196
|
+
}
|
|
197
|
+
/**
|
|
198
|
+
* Capitalize the first character of a string.
|
|
199
|
+
*/
|
|
200
|
+
function capitalize(s) {
|
|
201
|
+
return s.length === 0 ? s : s[0].toUpperCase() + s.slice(1);
|
|
202
|
+
}
|
|
203
|
+
/**
|
|
204
|
+
* Build a changelog entry line for a task.
|
|
205
|
+
* Uses description to enrich the entry when it's meaningfully different from the title.
|
|
206
|
+
*/
|
|
207
|
+
function buildEntry(task) {
|
|
208
|
+
const cleanTitle = capitalize(stripConventionalPrefix(task.title));
|
|
209
|
+
// Strip newlines and collapse whitespace in description
|
|
210
|
+
const safeDesc = task.description
|
|
211
|
+
?.replace(/\r?\n/g, ' ')
|
|
212
|
+
.replace(/\s{2,}/g, ' ')
|
|
213
|
+
.trim();
|
|
214
|
+
const desc = safeDesc;
|
|
215
|
+
// Include description only when it's non-trivial and adds information beyond the title.
|
|
216
|
+
// Skip if: description is empty, identical to title, or a minor rephrasing (≤10% longer, no new words).
|
|
217
|
+
const shouldIncludeDesc = (() => {
|
|
218
|
+
if (!desc || desc.length === 0)
|
|
219
|
+
return false;
|
|
220
|
+
const titleNorm = cleanTitle
|
|
221
|
+
.toLowerCase()
|
|
222
|
+
.replace(/[^a-z0-9\s]/g, '')
|
|
223
|
+
.trim();
|
|
224
|
+
const descNorm = desc
|
|
225
|
+
.toLowerCase()
|
|
226
|
+
.replace(/[^a-z0-9\s]/g, '')
|
|
227
|
+
.trim();
|
|
228
|
+
if (titleNorm === descNorm)
|
|
229
|
+
return false;
|
|
230
|
+
if (descNorm.startsWith(titleNorm) && descNorm.length < titleNorm.length * 1.3)
|
|
231
|
+
return false;
|
|
232
|
+
// Require description to be at least 20 chars and contain different content
|
|
233
|
+
return desc.length >= 20;
|
|
234
|
+
})();
|
|
235
|
+
if (shouldIncludeDesc) {
|
|
236
|
+
// Truncate long descriptions to keep changelog readable
|
|
237
|
+
const descDisplay = desc.length > 150 ? desc.slice(0, 147) + '...' : desc;
|
|
238
|
+
return `- **${cleanTitle}**: ${descDisplay} (${task.id})`;
|
|
239
|
+
}
|
|
240
|
+
return `- ${cleanTitle} (${task.id})`;
|
|
241
|
+
}
|
|
242
|
+
/**
|
|
243
|
+
* Categorize a task into a changelog section.
|
|
244
|
+
* Priority order:
|
|
245
|
+
* 1. task.type field ('epic' → skip, others are hints for 'task'/'subtask')
|
|
246
|
+
* 2. task.labels array
|
|
247
|
+
* 3. Title keyword scan (with conventional prefix stripped)
|
|
248
|
+
*/
|
|
249
|
+
function categorizeTask(task) {
|
|
250
|
+
// Fix A: Skip epics entirely — they are parent containers, not deliverables
|
|
251
|
+
if (task.type === 'epic')
|
|
252
|
+
return 'changes'; // Will be filtered out by the caller
|
|
253
|
+
// Priority 1: task.type field is the most authoritative signal
|
|
254
|
+
const taskType = (task.type ?? '').toLowerCase();
|
|
255
|
+
if (taskType === 'test')
|
|
256
|
+
return 'tests';
|
|
257
|
+
if (taskType === 'fix' || taskType === 'bugfix')
|
|
258
|
+
return 'fixes';
|
|
259
|
+
if (taskType === 'feat' || taskType === 'feature')
|
|
260
|
+
return 'features';
|
|
261
|
+
if (taskType === 'docs' || taskType === 'doc')
|
|
262
|
+
return 'docs';
|
|
263
|
+
if (taskType === 'chore' || taskType === 'refactor')
|
|
264
|
+
return 'chores';
|
|
265
|
+
// Priority 2: conventional commit prefix in raw title
|
|
266
|
+
if (/^feat(\([^)]+\))?:/.test(task.title.toLowerCase()))
|
|
267
|
+
return 'features';
|
|
268
|
+
if (/^fix(\([^)]+\))?:/.test(task.title.toLowerCase()))
|
|
269
|
+
return 'fixes';
|
|
270
|
+
if (/^docs?(\([^)]+\))?:/.test(task.title.toLowerCase()))
|
|
271
|
+
return 'docs';
|
|
272
|
+
if (/^test(\([^)]+\))?:/.test(task.title.toLowerCase()))
|
|
273
|
+
return 'tests';
|
|
274
|
+
if (/^(chore|refactor|style|ci|build|perf)(\([^)]+\))?:/.test(task.title.toLowerCase()))
|
|
275
|
+
return 'chores';
|
|
276
|
+
// Priority 3: labels for strong category signals
|
|
277
|
+
const labels = task.labels ?? [];
|
|
278
|
+
if (labels.some((l) => ['test', 'testing'].includes(l.toLowerCase())))
|
|
279
|
+
return 'tests';
|
|
280
|
+
if (labels.some((l) => ['fix', 'bug', 'bugfix', 'regression'].includes(l.toLowerCase())))
|
|
281
|
+
return 'fixes';
|
|
282
|
+
if (labels.some((l) => ['feat', 'feature', 'enhancement', 'add'].includes(l.toLowerCase())))
|
|
283
|
+
return 'features';
|
|
284
|
+
if (labels.some((l) => ['docs', 'documentation'].includes(l.toLowerCase())))
|
|
285
|
+
return 'docs';
|
|
286
|
+
if (labels.some((l) => ['chore', 'refactor', 'cleanup', 'maintenance'].includes(l.toLowerCase())))
|
|
287
|
+
return 'chores';
|
|
288
|
+
// Priority 4: keyword scan on the cleaned title
|
|
289
|
+
const titleLower = stripConventionalPrefix(task.title).toLowerCase();
|
|
290
|
+
const rawTitleLower = task.title.toLowerCase();
|
|
291
|
+
if (titleLower.startsWith('test') ||
|
|
292
|
+
(titleLower.includes('test') && titleLower.includes('add')))
|
|
293
|
+
return 'tests';
|
|
294
|
+
if (titleLower.includes('bug') ||
|
|
295
|
+
titleLower.startsWith('fix') ||
|
|
296
|
+
titleLower.includes('regression') ||
|
|
297
|
+
titleLower.includes('broken'))
|
|
298
|
+
return 'fixes';
|
|
299
|
+
if (titleLower.startsWith('add ') ||
|
|
300
|
+
titleLower.includes('implement') ||
|
|
301
|
+
titleLower.startsWith('create ') ||
|
|
302
|
+
titleLower.startsWith('introduce '))
|
|
303
|
+
return 'features';
|
|
304
|
+
if (titleLower.startsWith('doc') ||
|
|
305
|
+
titleLower.includes('documentation') ||
|
|
306
|
+
titleLower.includes('readme') ||
|
|
307
|
+
titleLower.includes('changelog'))
|
|
308
|
+
return 'docs';
|
|
309
|
+
if (titleLower.startsWith('chore') ||
|
|
310
|
+
titleLower.includes('refactor') ||
|
|
311
|
+
titleLower.includes('cleanup') ||
|
|
312
|
+
titleLower.includes('migrate') ||
|
|
313
|
+
titleLower.includes('upgrade') ||
|
|
314
|
+
titleLower.includes('remove ') ||
|
|
315
|
+
titleLower.startsWith('audit'))
|
|
316
|
+
return 'chores';
|
|
317
|
+
// Raw title scan for backward compat
|
|
318
|
+
if (rawTitleLower.startsWith('feat'))
|
|
319
|
+
return 'features';
|
|
320
|
+
return 'changes';
|
|
321
|
+
}
|
|
322
|
+
for (const taskId of releaseTasks) {
|
|
323
|
+
const task = taskMap.get(taskId);
|
|
324
|
+
if (!task)
|
|
325
|
+
continue;
|
|
326
|
+
// Fix A: Filter out epics — they are containers, not changelog entries
|
|
327
|
+
if (task.type === 'epic')
|
|
328
|
+
continue;
|
|
329
|
+
// Also filter by label in case type field is not populated
|
|
330
|
+
if (task.labels?.some((l) => l.toLowerCase() === 'epic'))
|
|
331
|
+
continue;
|
|
332
|
+
// Heuristic: titles starting with "EPIC:" are epics even without type field
|
|
333
|
+
if (/^epic:/i.test(task.title.trim()))
|
|
334
|
+
continue;
|
|
335
|
+
// Filter out research/internal/spike/audit tasks — not user-facing deliverables
|
|
336
|
+
const labelsLower = (task.labels ?? []).map((l) => l.toLowerCase());
|
|
337
|
+
if (labelsLower.some((l) => ['research', 'internal', 'spike', 'audit'].includes(l)))
|
|
338
|
+
continue;
|
|
339
|
+
if (['spike', 'research'].includes((task.type ?? '').toLowerCase()))
|
|
340
|
+
continue;
|
|
341
|
+
if (/^(research|investigate|audit|spike)\s/i.test(task.title.trim()))
|
|
342
|
+
continue;
|
|
343
|
+
const category = categorizeTask(task);
|
|
344
|
+
const entry = buildEntry(task);
|
|
345
|
+
if (category === 'features')
|
|
346
|
+
features.push(entry);
|
|
347
|
+
else if (category === 'fixes')
|
|
348
|
+
fixes.push(entry);
|
|
349
|
+
else if (category === 'docs')
|
|
350
|
+
docs.push(entry);
|
|
351
|
+
else if (category === 'tests')
|
|
352
|
+
tests.push(entry);
|
|
353
|
+
else if (category === 'chores')
|
|
354
|
+
chores.push(entry);
|
|
355
|
+
else
|
|
356
|
+
changes.push(entry);
|
|
357
|
+
}
|
|
358
|
+
const sections = [];
|
|
359
|
+
const date = new Date().toISOString().split('T')[0];
|
|
360
|
+
sections.push(`## ${normalizedVersion} (${date})`);
|
|
361
|
+
sections.push('');
|
|
362
|
+
if (row.notes) {
|
|
363
|
+
sections.push(row.notes);
|
|
364
|
+
sections.push('');
|
|
365
|
+
}
|
|
366
|
+
if (features.length > 0) {
|
|
367
|
+
sections.push('### Features');
|
|
368
|
+
sections.push(...features);
|
|
369
|
+
sections.push('');
|
|
370
|
+
}
|
|
371
|
+
if (fixes.length > 0) {
|
|
372
|
+
sections.push('### Bug Fixes');
|
|
373
|
+
sections.push(...fixes);
|
|
374
|
+
sections.push('');
|
|
375
|
+
}
|
|
376
|
+
if (docs.length > 0) {
|
|
377
|
+
sections.push('### Documentation');
|
|
378
|
+
sections.push(...docs);
|
|
379
|
+
sections.push('');
|
|
380
|
+
}
|
|
381
|
+
if (tests.length > 0) {
|
|
382
|
+
sections.push('### Tests');
|
|
383
|
+
sections.push(...tests);
|
|
384
|
+
sections.push('');
|
|
385
|
+
}
|
|
386
|
+
if (chores.length > 0) {
|
|
387
|
+
sections.push('### Chores');
|
|
388
|
+
sections.push(...chores);
|
|
389
|
+
sections.push('');
|
|
390
|
+
}
|
|
391
|
+
if (changes.length > 0) {
|
|
392
|
+
sections.push('### Changes');
|
|
393
|
+
sections.push(...changes);
|
|
394
|
+
sections.push('');
|
|
395
|
+
}
|
|
396
|
+
const changelog = sections.join('\n');
|
|
397
|
+
await db
|
|
398
|
+
.update(schema.releaseManifests)
|
|
399
|
+
.set({ changelog })
|
|
400
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
401
|
+
.run();
|
|
402
|
+
// Write or update CHANGELOG.md with section-aware merge
|
|
403
|
+
const changelogPath = join(cwd ?? process.cwd(), 'CHANGELOG.md');
|
|
404
|
+
let existingChangelogContent = '';
|
|
405
|
+
try {
|
|
406
|
+
existingChangelogContent = await readFile(changelogPath, 'utf8');
|
|
407
|
+
}
|
|
408
|
+
catch {
|
|
409
|
+
// File doesn't exist yet — start fresh
|
|
410
|
+
}
|
|
411
|
+
const { customBlocks } = parseChangelogBlocks(existingChangelogContent);
|
|
412
|
+
// Build the changelog body (content after the ## header line)
|
|
413
|
+
const changelogBody = sections.slice(2).join('\n'); // skip header + blank line
|
|
414
|
+
await writeChangelogSection(normalizedVersion.replace(/^v/, ''), changelogBody, customBlocks, changelogPath);
|
|
415
|
+
return {
|
|
416
|
+
version: normalizedVersion,
|
|
417
|
+
changelog,
|
|
418
|
+
taskCount: releaseTasks.length,
|
|
419
|
+
sections: {
|
|
420
|
+
features: features.length,
|
|
421
|
+
fixes: fixes.length,
|
|
422
|
+
docs: docs.length,
|
|
423
|
+
tests: tests.length,
|
|
424
|
+
chores: chores.length,
|
|
425
|
+
changes: changes.length,
|
|
426
|
+
},
|
|
427
|
+
};
|
|
428
|
+
}
|
|
429
|
+
/**
|
|
430
|
+
* List all releases.
|
|
431
|
+
* @task T4788
|
|
432
|
+
*/
|
|
433
|
+
export async function listManifestReleases(optionsOrCwd, cwd) {
|
|
434
|
+
const options = typeof optionsOrCwd === 'string' || optionsOrCwd === undefined ? {} : optionsOrCwd;
|
|
435
|
+
const effectiveCwd = typeof optionsOrCwd === 'string' ? optionsOrCwd : cwd;
|
|
436
|
+
const limit = normalizeLimit(options.limit);
|
|
437
|
+
const offset = normalizeOffset(options.offset);
|
|
438
|
+
const pageLimit = effectivePageLimit(limit, offset);
|
|
439
|
+
const db = await getDb(effectiveCwd);
|
|
440
|
+
const totalRow = await db.select({ count: count() }).from(schema.releaseManifests).get();
|
|
441
|
+
const total = totalRow?.count ?? 0;
|
|
442
|
+
const conditions = options.status ? [eq(schema.releaseManifests.status, options.status)] : [];
|
|
443
|
+
const whereClause = conditions.length > 0 ? and(...conditions) : undefined;
|
|
444
|
+
const filteredRow = await db
|
|
445
|
+
.select({ count: count() })
|
|
446
|
+
.from(schema.releaseManifests)
|
|
447
|
+
.where(whereClause)
|
|
448
|
+
.get();
|
|
449
|
+
const filtered = filteredRow?.count ?? 0;
|
|
450
|
+
let query = db
|
|
451
|
+
.select()
|
|
452
|
+
.from(schema.releaseManifests)
|
|
453
|
+
.where(whereClause)
|
|
454
|
+
.orderBy(desc(schema.releaseManifests.createdAt));
|
|
455
|
+
if (pageLimit !== undefined) {
|
|
456
|
+
query = query.limit(pageLimit);
|
|
457
|
+
}
|
|
458
|
+
if (offset !== undefined) {
|
|
459
|
+
query = query.offset(offset);
|
|
460
|
+
}
|
|
461
|
+
const rows = await query.all();
|
|
462
|
+
const latest = await findLatestPushedVersion(effectiveCwd);
|
|
463
|
+
return {
|
|
464
|
+
releases: rows.map((r) => ({
|
|
465
|
+
version: r.version,
|
|
466
|
+
status: r.status,
|
|
467
|
+
createdAt: r.createdAt,
|
|
468
|
+
taskCount: JSON.parse(r.tasksJson).length,
|
|
469
|
+
})),
|
|
470
|
+
total,
|
|
471
|
+
filtered,
|
|
472
|
+
latest,
|
|
473
|
+
page: createPage({ total: filtered, limit: pageLimit, offset }),
|
|
474
|
+
};
|
|
475
|
+
}
|
|
476
|
+
/**
|
|
477
|
+
* Show release details.
|
|
478
|
+
* @task T4788
|
|
479
|
+
*/
|
|
480
|
+
export async function showManifestRelease(version, cwd) {
|
|
481
|
+
if (!version) {
|
|
482
|
+
throw new Error('version is required');
|
|
483
|
+
}
|
|
484
|
+
const normalizedVersion = normalizeVersion(version);
|
|
485
|
+
const db = await getDb(cwd);
|
|
486
|
+
const rows = await db
|
|
487
|
+
.select()
|
|
488
|
+
.from(schema.releaseManifests)
|
|
489
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
490
|
+
.limit(1)
|
|
491
|
+
.all();
|
|
492
|
+
if (rows.length === 0) {
|
|
493
|
+
throw new Error(`Release ${normalizedVersion} not found`);
|
|
494
|
+
}
|
|
495
|
+
return rowToManifest(rows[0]);
|
|
496
|
+
}
|
|
497
|
+
/**
|
|
498
|
+
* Mark release as committed (metadata only).
|
|
499
|
+
* @task T4788
|
|
500
|
+
*/
|
|
501
|
+
export async function commitRelease(version, cwd) {
|
|
502
|
+
if (!version) {
|
|
503
|
+
throw new Error('version is required');
|
|
504
|
+
}
|
|
505
|
+
const normalizedVersion = normalizeVersion(version);
|
|
506
|
+
const db = await getDb(cwd);
|
|
507
|
+
const rows = await db
|
|
508
|
+
.select()
|
|
509
|
+
.from(schema.releaseManifests)
|
|
510
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
511
|
+
.limit(1)
|
|
512
|
+
.all();
|
|
513
|
+
if (rows.length === 0) {
|
|
514
|
+
throw new Error(`Release ${normalizedVersion} not found`);
|
|
515
|
+
}
|
|
516
|
+
if (rows[0].status !== 'prepared') {
|
|
517
|
+
throw new Error(`Release ${normalizedVersion} is in state '${rows[0].status}', expected 'prepared'`);
|
|
518
|
+
}
|
|
519
|
+
const committedAt = new Date().toISOString();
|
|
520
|
+
await db
|
|
521
|
+
.update(schema.releaseManifests)
|
|
522
|
+
.set({ status: 'committed', committedAt })
|
|
523
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
524
|
+
.run();
|
|
525
|
+
return { version: normalizedVersion, status: 'committed', committedAt };
|
|
526
|
+
}
|
|
527
|
+
/**
|
|
528
|
+
* Mark release as tagged (metadata only).
|
|
529
|
+
* @task T4788
|
|
530
|
+
*/
|
|
531
|
+
export async function tagRelease(version, cwd) {
|
|
532
|
+
if (!version) {
|
|
533
|
+
throw new Error('version is required');
|
|
534
|
+
}
|
|
535
|
+
const normalizedVersion = normalizeVersion(version);
|
|
536
|
+
const db = await getDb(cwd);
|
|
537
|
+
const rows = await db
|
|
538
|
+
.select()
|
|
539
|
+
.from(schema.releaseManifests)
|
|
540
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
541
|
+
.limit(1)
|
|
542
|
+
.all();
|
|
543
|
+
if (rows.length === 0) {
|
|
544
|
+
throw new Error(`Release ${normalizedVersion} not found`);
|
|
545
|
+
}
|
|
546
|
+
const taggedAt = new Date().toISOString();
|
|
547
|
+
await db
|
|
548
|
+
.update(schema.releaseManifests)
|
|
549
|
+
.set({ status: 'tagged', taggedAt })
|
|
550
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
551
|
+
.run();
|
|
552
|
+
return { version: normalizedVersion, status: 'tagged', taggedAt };
|
|
553
|
+
}
|
|
554
|
+
/**
|
|
555
|
+
* Run release validation gates.
|
|
556
|
+
* @task T4788
|
|
557
|
+
* @task T5586
|
|
558
|
+
*/
|
|
559
|
+
export async function runReleaseGates(version, loadTasksFn, cwd, opts) {
|
|
560
|
+
if (!version) {
|
|
561
|
+
throw new Error('version is required');
|
|
562
|
+
}
|
|
563
|
+
const normalizedVersion = normalizeVersion(version);
|
|
564
|
+
const db = await getDb(cwd);
|
|
565
|
+
const rows = await db
|
|
566
|
+
.select()
|
|
567
|
+
.from(schema.releaseManifests)
|
|
568
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
569
|
+
.limit(1)
|
|
570
|
+
.all();
|
|
571
|
+
if (rows.length === 0) {
|
|
572
|
+
throw new Error(`Release ${normalizedVersion} not found`);
|
|
573
|
+
}
|
|
574
|
+
const row = rows[0];
|
|
575
|
+
const releaseTasks = JSON.parse(row.tasksJson);
|
|
576
|
+
const gates = [];
|
|
577
|
+
gates.push({
|
|
578
|
+
name: 'version_valid',
|
|
579
|
+
status: isValidVersion(normalizedVersion) ? 'passed' : 'failed',
|
|
580
|
+
message: isValidVersion(normalizedVersion)
|
|
581
|
+
? 'Version format is valid'
|
|
582
|
+
: 'Invalid version format',
|
|
583
|
+
});
|
|
584
|
+
const releaseConfig = loadReleaseConfig(cwd);
|
|
585
|
+
if (releaseConfig.versioningScheme === 'calver') {
|
|
586
|
+
const calver = validateCalVerWindow(normalizedVersion);
|
|
587
|
+
gates.push({
|
|
588
|
+
name: 'calver_window',
|
|
589
|
+
status: calver.valid ? 'passed' : 'failed',
|
|
590
|
+
message: calver.message,
|
|
591
|
+
});
|
|
592
|
+
}
|
|
593
|
+
gates.push({
|
|
594
|
+
name: 'has_tasks',
|
|
595
|
+
status: releaseTasks.length > 0 ? 'passed' : 'failed',
|
|
596
|
+
message: releaseTasks.length > 0 ? `${releaseTasks.length} tasks included` : 'No tasks in release',
|
|
597
|
+
});
|
|
598
|
+
gates.push({
|
|
599
|
+
name: 'has_changelog',
|
|
600
|
+
status: row.changelog ? 'passed' : 'failed',
|
|
601
|
+
message: row.changelog
|
|
602
|
+
? 'Changelog generated'
|
|
603
|
+
: 'No changelog generated. Run release.changelog first.',
|
|
604
|
+
});
|
|
605
|
+
const allTasks = await loadTasksFn();
|
|
606
|
+
const incompleteTasks = releaseTasks.filter((id) => {
|
|
607
|
+
const task = allTasks.find((t) => t.id === id);
|
|
608
|
+
return task && task.status !== 'done';
|
|
609
|
+
});
|
|
610
|
+
gates.push({
|
|
611
|
+
name: 'tasks_complete',
|
|
612
|
+
status: incompleteTasks.length === 0 ? 'passed' : 'failed',
|
|
613
|
+
message: incompleteTasks.length === 0
|
|
614
|
+
? 'All tasks completed'
|
|
615
|
+
: `${incompleteTasks.length} tasks not completed: ${incompleteTasks.join(', ')}`,
|
|
616
|
+
});
|
|
617
|
+
// G2: Build artifact — dist/ must exist (Node projects only)
|
|
618
|
+
// Monorepo-aware: checks packages/cleo/dist/ first, then root dist/
|
|
619
|
+
const projectRoot = cwd ?? getProjectRoot();
|
|
620
|
+
const monorepoDist = join(projectRoot, 'packages', 'cleo', 'dist', 'cli', 'index.js');
|
|
621
|
+
const rootDist = join(projectRoot, 'dist', 'cli', 'index.js');
|
|
622
|
+
const distExists = existsSync(monorepoDist) || existsSync(rootDist);
|
|
623
|
+
const isNodeProject = existsSync(join(projectRoot, 'package.json'));
|
|
624
|
+
if (isNodeProject) {
|
|
625
|
+
gates.push({
|
|
626
|
+
name: 'build_artifact',
|
|
627
|
+
status: distExists ? 'passed' : 'failed',
|
|
628
|
+
message: distExists ? 'Build artifacts present' : 'dist/ not built — run: pnpm run build',
|
|
629
|
+
});
|
|
630
|
+
}
|
|
631
|
+
// GD1: Clean working tree (CHANGELOG.md and version bump targets are allowed to be dirty)
|
|
632
|
+
// Skipped in dry-run mode — dry-run makes no commits so tree cleanliness is irrelevant.
|
|
633
|
+
// Untracked files (?? lines) are excluded from the dirty check — they do not affect git
|
|
634
|
+
// commit/tag operations and must not block releases.
|
|
635
|
+
if (opts?.dryRun) {
|
|
636
|
+
gates.push({
|
|
637
|
+
name: 'clean_working_tree',
|
|
638
|
+
status: 'passed',
|
|
639
|
+
message: 'Skipped in dry-run mode',
|
|
640
|
+
});
|
|
641
|
+
}
|
|
642
|
+
else {
|
|
643
|
+
// Dynamically build exclusion set from configured version bump targets + CHANGELOG.md
|
|
644
|
+
const bumpTargets = getVersionBumpConfig(cwd);
|
|
645
|
+
const allowedDirty = new Set(['CHANGELOG.md', ...bumpTargets.map((t) => t.file)]);
|
|
646
|
+
let workingTreeClean = true;
|
|
647
|
+
let dirtyFiles = [];
|
|
648
|
+
try {
|
|
649
|
+
const porcelain = execFileSync('git', ['status', '--porcelain'], {
|
|
650
|
+
cwd: projectRoot,
|
|
651
|
+
encoding: 'utf-8',
|
|
652
|
+
stdio: 'pipe',
|
|
653
|
+
});
|
|
654
|
+
dirtyFiles = porcelain
|
|
655
|
+
.split('\n')
|
|
656
|
+
.filter((l) => l.trim())
|
|
657
|
+
// Exclude untracked files (?? prefix) — they don't affect commits or tags
|
|
658
|
+
.filter((l) => !l.startsWith('?? '))
|
|
659
|
+
.map((l) => l.slice(3).trim())
|
|
660
|
+
.filter((f) => !allowedDirty.has(f));
|
|
661
|
+
workingTreeClean = dirtyFiles.length === 0;
|
|
662
|
+
}
|
|
663
|
+
catch {
|
|
664
|
+
/* git not available — skip */
|
|
665
|
+
}
|
|
666
|
+
const excludeList = [...allowedDirty].join(', ');
|
|
667
|
+
gates.push({
|
|
668
|
+
name: 'clean_working_tree',
|
|
669
|
+
status: workingTreeClean ? 'passed' : 'failed',
|
|
670
|
+
message: workingTreeClean
|
|
671
|
+
? `Working tree clean (excluding ${excludeList})`
|
|
672
|
+
: `Uncommitted changes in: ${dirtyFiles.slice(0, 5).join(', ')}${dirtyFiles.length > 5 ? ` (+${dirtyFiles.length - 5} more)` : ''}`,
|
|
673
|
+
});
|
|
674
|
+
}
|
|
675
|
+
// GD2: Branch target — use GitFlow config if available, else defaults
|
|
676
|
+
const isPreRelease = normalizedVersion.includes('-');
|
|
677
|
+
let currentBranch = '';
|
|
678
|
+
try {
|
|
679
|
+
currentBranch = execFileSync('git', ['rev-parse', '--abbrev-ref', 'HEAD'], {
|
|
680
|
+
cwd: projectRoot,
|
|
681
|
+
encoding: 'utf-8',
|
|
682
|
+
stdio: 'pipe',
|
|
683
|
+
}).trim();
|
|
684
|
+
}
|
|
685
|
+
catch {
|
|
686
|
+
/* git not available — skip */
|
|
687
|
+
}
|
|
688
|
+
const gitFlowCfg = getGitFlowConfig(releaseConfig);
|
|
689
|
+
const channelCfg = getChannelConfig(releaseConfig);
|
|
690
|
+
const expectedBranch = isPreRelease ? gitFlowCfg.branches.develop : gitFlowCfg.branches.main;
|
|
691
|
+
const isFeatureBranch = currentBranch.startsWith(gitFlowCfg.branches.featurePrefix) ||
|
|
692
|
+
currentBranch.startsWith(gitFlowCfg.branches.hotfixPrefix) ||
|
|
693
|
+
currentBranch.startsWith(gitFlowCfg.branches.releasePrefix);
|
|
694
|
+
const branchOk = !currentBranch || // git unavailable → pass
|
|
695
|
+
currentBranch === 'HEAD' || // detached HEAD → pass
|
|
696
|
+
currentBranch === expectedBranch || // exactly right branch → pass
|
|
697
|
+
(isPreRelease && isFeatureBranch); // feature/hotfix/release branch with pre-release → pass
|
|
698
|
+
// Resolve channel from current branch
|
|
699
|
+
const detectedChannel = currentBranch
|
|
700
|
+
? resolveChannelFromBranch(currentBranch, channelCfg)
|
|
701
|
+
: isPreRelease
|
|
702
|
+
? 'beta'
|
|
703
|
+
: 'latest';
|
|
704
|
+
gates.push({
|
|
705
|
+
name: 'branch_target',
|
|
706
|
+
status: branchOk ? 'passed' : 'failed',
|
|
707
|
+
message: branchOk
|
|
708
|
+
? `On correct branch: ${currentBranch} (channel: ${detectedChannel})`
|
|
709
|
+
: `Expected branch '${expectedBranch}' for ${isPreRelease ? 'pre-release' : 'stable'} release, but on '${currentBranch}'`,
|
|
710
|
+
});
|
|
711
|
+
// GD3: Branch protection — detect if push requires a PR (informational, never fails)
|
|
712
|
+
const pushMode = getPushMode(releaseConfig);
|
|
713
|
+
let requiresPR = false;
|
|
714
|
+
if (pushMode === 'pr') {
|
|
715
|
+
requiresPR = true;
|
|
716
|
+
}
|
|
717
|
+
else if (pushMode === 'auto') {
|
|
718
|
+
try {
|
|
719
|
+
const protectionResult = await detectBranchProtection(expectedBranch, 'origin', projectRoot);
|
|
720
|
+
requiresPR = protectionResult.protected;
|
|
721
|
+
}
|
|
722
|
+
catch {
|
|
723
|
+
// Branch protection detection is best-effort; never block release
|
|
724
|
+
requiresPR = false;
|
|
725
|
+
}
|
|
726
|
+
}
|
|
727
|
+
gates.push({
|
|
728
|
+
name: 'branch_protection',
|
|
729
|
+
status: 'passed',
|
|
730
|
+
message: requiresPR
|
|
731
|
+
? `Branch '${expectedBranch}' is protected — release.ship will create a PR`
|
|
732
|
+
: `Branch '${expectedBranch}' allows direct push`,
|
|
733
|
+
});
|
|
734
|
+
const allPassed = gates.every((g) => g.status === 'passed');
|
|
735
|
+
const metadata = {
|
|
736
|
+
channel: detectedChannel,
|
|
737
|
+
requiresPR,
|
|
738
|
+
targetBranch: expectedBranch,
|
|
739
|
+
currentBranch,
|
|
740
|
+
};
|
|
741
|
+
return {
|
|
742
|
+
version: normalizedVersion,
|
|
743
|
+
allPassed,
|
|
744
|
+
gates,
|
|
745
|
+
passedCount: gates.filter((g) => g.status === 'passed').length,
|
|
746
|
+
failedCount: gates.filter((g) => g.status === 'failed').length,
|
|
747
|
+
metadata,
|
|
748
|
+
};
|
|
749
|
+
}
|
|
750
|
+
/**
|
|
751
|
+
* Cancel and remove a release in draft or prepared state.
|
|
752
|
+
* Only releases that have not yet been committed to git can be cancelled.
|
|
753
|
+
* For committed/tagged/pushed releases, use rollbackRelease() instead.
|
|
754
|
+
*
|
|
755
|
+
* @task T5602
|
|
756
|
+
*/
|
|
757
|
+
export async function cancelRelease(version, projectRoot) {
|
|
758
|
+
if (!version) {
|
|
759
|
+
throw new Error('version is required');
|
|
760
|
+
}
|
|
761
|
+
const normalizedVersion = normalizeVersion(version);
|
|
762
|
+
const db = await getDb(projectRoot);
|
|
763
|
+
const rows = await db
|
|
764
|
+
.select()
|
|
765
|
+
.from(schema.releaseManifests)
|
|
766
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
767
|
+
.limit(1)
|
|
768
|
+
.all();
|
|
769
|
+
if (rows.length === 0) {
|
|
770
|
+
return {
|
|
771
|
+
success: false,
|
|
772
|
+
message: `Release ${normalizedVersion} not found`,
|
|
773
|
+
version: normalizedVersion,
|
|
774
|
+
};
|
|
775
|
+
}
|
|
776
|
+
const status = rows[0].status;
|
|
777
|
+
const cancellableStates = ['draft', 'prepared'];
|
|
778
|
+
if (!cancellableStates.includes(status)) {
|
|
779
|
+
return {
|
|
780
|
+
success: false,
|
|
781
|
+
message: `Cannot cancel a release in '${status}' state. Use 'release rollback' instead.`,
|
|
782
|
+
version: normalizedVersion,
|
|
783
|
+
};
|
|
784
|
+
}
|
|
785
|
+
await db
|
|
786
|
+
.delete(schema.releaseManifests)
|
|
787
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
788
|
+
.run();
|
|
789
|
+
return {
|
|
790
|
+
success: true,
|
|
791
|
+
message: `Release ${normalizedVersion} cancelled and removed`,
|
|
792
|
+
version: normalizedVersion,
|
|
793
|
+
};
|
|
794
|
+
}
|
|
795
|
+
/**
|
|
796
|
+
* Rollback a release.
|
|
797
|
+
* @task T4788
|
|
798
|
+
*/
|
|
799
|
+
export async function rollbackRelease(version, reason, cwd) {
|
|
800
|
+
if (!version) {
|
|
801
|
+
throw new Error('version is required');
|
|
802
|
+
}
|
|
803
|
+
const normalizedVersion = normalizeVersion(version);
|
|
804
|
+
const db = await getDb(cwd);
|
|
805
|
+
const rows = await db
|
|
806
|
+
.select()
|
|
807
|
+
.from(schema.releaseManifests)
|
|
808
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
809
|
+
.limit(1)
|
|
810
|
+
.all();
|
|
811
|
+
if (rows.length === 0) {
|
|
812
|
+
throw new Error(`Release ${normalizedVersion} not found`);
|
|
813
|
+
}
|
|
814
|
+
const previousStatus = rows[0].status;
|
|
815
|
+
await db
|
|
816
|
+
.update(schema.releaseManifests)
|
|
817
|
+
.set({ status: 'rolled_back' })
|
|
818
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
819
|
+
.run();
|
|
820
|
+
return {
|
|
821
|
+
version: normalizedVersion,
|
|
822
|
+
previousStatus,
|
|
823
|
+
status: 'rolled_back',
|
|
824
|
+
reason: reason ?? 'No reason provided',
|
|
825
|
+
};
|
|
826
|
+
}
|
|
827
|
+
/**
|
|
828
|
+
* Read push policy from project config.
|
|
829
|
+
* Returns undefined if no push config exists.
|
|
830
|
+
*/
|
|
831
|
+
async function readPushPolicy(cwd) {
|
|
832
|
+
const configPath = join(getCleoDirAbsolute(cwd), 'config.json');
|
|
833
|
+
let config;
|
|
834
|
+
try {
|
|
835
|
+
const raw = await readFile(configPath, 'utf-8');
|
|
836
|
+
config = JSON.parse(raw);
|
|
837
|
+
}
|
|
838
|
+
catch {
|
|
839
|
+
return undefined;
|
|
840
|
+
}
|
|
841
|
+
if (!config)
|
|
842
|
+
return undefined;
|
|
843
|
+
const release = config.release;
|
|
844
|
+
if (!release)
|
|
845
|
+
return undefined;
|
|
846
|
+
return release.push;
|
|
847
|
+
}
|
|
848
|
+
/**
|
|
849
|
+
* Push release to remote via git.
|
|
850
|
+
*
|
|
851
|
+
* Respects config.release.push policy:
|
|
852
|
+
* - remote: override default remote (fallback to 'origin')
|
|
853
|
+
* - requireCleanTree: verify git working tree is clean before push
|
|
854
|
+
* - allowedBranches: verify current branch is in the allowed list
|
|
855
|
+
* - enabled: if false and no explicit push flag, caller should skip
|
|
856
|
+
*
|
|
857
|
+
* @task T4788
|
|
858
|
+
* @task T4276
|
|
859
|
+
*/
|
|
860
|
+
export async function pushRelease(version, remote, cwd, opts) {
|
|
861
|
+
if (!version) {
|
|
862
|
+
throw new Error('version is required');
|
|
863
|
+
}
|
|
864
|
+
const normalizedVersion = normalizeVersion(version);
|
|
865
|
+
const projectRoot = getProjectRoot(cwd);
|
|
866
|
+
const pushPolicy = await readPushPolicy(cwd);
|
|
867
|
+
// Resolve effective push mode: opts.mode > pushPolicy.mode > config > 'direct'
|
|
868
|
+
const configPushMode = getPushMode(loadReleaseConfig(cwd));
|
|
869
|
+
const effectivePushMode = opts?.mode ?? pushPolicy?.mode ?? configPushMode;
|
|
870
|
+
// If branch protection detected and mode allows PR creation, signal PR required
|
|
871
|
+
if (effectivePushMode === 'pr' || effectivePushMode === 'auto') {
|
|
872
|
+
const targetRemoteForCheck = remote ?? pushPolicy?.remote ?? 'origin';
|
|
873
|
+
let branchIsProtected = effectivePushMode === 'pr'; // 'pr' always requires PR
|
|
874
|
+
if (effectivePushMode === 'auto') {
|
|
875
|
+
try {
|
|
876
|
+
const protection = await detectBranchProtection(pushPolicy?.allowedBranches?.[0] ?? 'main', targetRemoteForCheck, projectRoot);
|
|
877
|
+
branchIsProtected = protection.protected;
|
|
878
|
+
}
|
|
879
|
+
catch {
|
|
880
|
+
// Best-effort; default to direct push if detection fails
|
|
881
|
+
branchIsProtected = false;
|
|
882
|
+
}
|
|
883
|
+
}
|
|
884
|
+
if (branchIsProtected) {
|
|
885
|
+
return {
|
|
886
|
+
version: normalizedVersion,
|
|
887
|
+
status: 'requires_pr',
|
|
888
|
+
remote: targetRemoteForCheck,
|
|
889
|
+
pushedAt: new Date().toISOString(),
|
|
890
|
+
requiresPR: true,
|
|
891
|
+
};
|
|
892
|
+
}
|
|
893
|
+
}
|
|
894
|
+
// If push policy says disabled and caller didn't explicitly pass --push, skip
|
|
895
|
+
if (pushPolicy && pushPolicy.enabled === false && !opts?.explicitPush) {
|
|
896
|
+
throw new Error('Push is disabled by config (release.push.enabled=false). Use --push to override.');
|
|
897
|
+
}
|
|
898
|
+
// Determine remote: explicit param > config > 'origin'
|
|
899
|
+
const targetRemote = remote ?? pushPolicy?.remote ?? 'origin';
|
|
900
|
+
// Check requireCleanTree
|
|
901
|
+
// Untracked files (?? lines) are excluded — they do not affect push operations.
|
|
902
|
+
if (pushPolicy?.requireCleanTree) {
|
|
903
|
+
const statusOutput = execFileSync('git', ['status', '--porcelain'], {
|
|
904
|
+
cwd: projectRoot,
|
|
905
|
+
timeout: 10000,
|
|
906
|
+
encoding: 'utf-8',
|
|
907
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
908
|
+
});
|
|
909
|
+
const trackedDirty = statusOutput
|
|
910
|
+
.split('\n')
|
|
911
|
+
.filter((l) => l.trim() && !l.startsWith('?? '))
|
|
912
|
+
.join('\n');
|
|
913
|
+
if (trackedDirty.trim().length > 0) {
|
|
914
|
+
throw new Error('Git working tree is not clean. Commit or stash changes before pushing (config: release.push.requireCleanTree=true).');
|
|
915
|
+
}
|
|
916
|
+
}
|
|
917
|
+
// Check allowedBranches
|
|
918
|
+
if (pushPolicy?.allowedBranches && pushPolicy.allowedBranches.length > 0) {
|
|
919
|
+
const currentBranch = execFileSync('git', ['rev-parse', '--abbrev-ref', 'HEAD'], {
|
|
920
|
+
cwd: projectRoot,
|
|
921
|
+
timeout: 10000,
|
|
922
|
+
encoding: 'utf-8',
|
|
923
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
924
|
+
}).trim();
|
|
925
|
+
if (!pushPolicy.allowedBranches.includes(currentBranch)) {
|
|
926
|
+
throw new Error(`Current branch '${currentBranch}' is not in allowed branches: ${pushPolicy.allowedBranches.join(', ')} (config: release.push.allowedBranches).`);
|
|
927
|
+
}
|
|
928
|
+
}
|
|
929
|
+
execFileSync('git', ['push', targetRemote, '--follow-tags'], {
|
|
930
|
+
cwd: projectRoot,
|
|
931
|
+
timeout: 60000,
|
|
932
|
+
encoding: 'utf-8',
|
|
933
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
934
|
+
});
|
|
935
|
+
return {
|
|
936
|
+
version: normalizedVersion,
|
|
937
|
+
status: 'pushed',
|
|
938
|
+
remote: targetRemote,
|
|
939
|
+
pushedAt: new Date().toISOString(),
|
|
940
|
+
};
|
|
941
|
+
}
|
|
942
|
+
/**
|
|
943
|
+
* Update release status after push, with optional provenance fields.
|
|
944
|
+
* @task T4788
|
|
945
|
+
* @task T5580
|
|
946
|
+
*/
|
|
947
|
+
export async function markReleasePushed(version, pushedAt, cwd, provenance) {
|
|
948
|
+
const normalizedVersion = normalizeVersion(version);
|
|
949
|
+
const db = await getDb(cwd);
|
|
950
|
+
await db
|
|
951
|
+
.update(schema.releaseManifests)
|
|
952
|
+
.set({
|
|
953
|
+
status: 'pushed',
|
|
954
|
+
pushedAt,
|
|
955
|
+
...(provenance?.commitSha != null ? { commitSha: provenance.commitSha } : {}),
|
|
956
|
+
...(provenance?.gitTag != null ? { gitTag: provenance.gitTag } : {}),
|
|
957
|
+
})
|
|
958
|
+
.where(eq(schema.releaseManifests.version, normalizedVersion))
|
|
959
|
+
.run();
|
|
960
|
+
}
|
|
961
|
+
/**
|
|
962
|
+
* One-time migration: read .cleo/releases.json and insert each release into
|
|
963
|
+
* the release_manifests table. Renames the file to releases.json.migrated on success.
|
|
964
|
+
*
|
|
965
|
+
* @task T5580
|
|
966
|
+
*/
|
|
967
|
+
export async function migrateReleasesJsonToSqlite(projectRoot) {
|
|
968
|
+
const releasesPath = join(getCleoDirAbsolute(projectRoot), 'releases.json');
|
|
969
|
+
if (!existsSync(releasesPath)) {
|
|
970
|
+
return { migrated: 0 };
|
|
971
|
+
}
|
|
972
|
+
let raw;
|
|
973
|
+
try {
|
|
974
|
+
const contents = await readFile(releasesPath, 'utf-8');
|
|
975
|
+
raw = JSON.parse(contents);
|
|
976
|
+
}
|
|
977
|
+
catch {
|
|
978
|
+
return { migrated: 0 };
|
|
979
|
+
}
|
|
980
|
+
if (!raw || !Array.isArray(raw.releases)) {
|
|
981
|
+
return { migrated: 0 };
|
|
982
|
+
}
|
|
983
|
+
const db = await getDb(projectRoot);
|
|
984
|
+
let migrated = 0;
|
|
985
|
+
for (const r of raw.releases) {
|
|
986
|
+
// Skip if already exists by version
|
|
987
|
+
const existing = await db
|
|
988
|
+
.select({ id: schema.releaseManifests.id })
|
|
989
|
+
.from(schema.releaseManifests)
|
|
990
|
+
.where(eq(schema.releaseManifests.version, r.version))
|
|
991
|
+
.limit(1)
|
|
992
|
+
.all();
|
|
993
|
+
if (existing.length > 0)
|
|
994
|
+
continue;
|
|
995
|
+
const id = `rel-${r.version.replace(/[^a-z0-9]/gi, '-')}`;
|
|
996
|
+
await db
|
|
997
|
+
.insert(schema.releaseManifests)
|
|
998
|
+
.values({
|
|
999
|
+
id,
|
|
1000
|
+
version: r.version,
|
|
1001
|
+
status: r.status,
|
|
1002
|
+
tasksJson: JSON.stringify(r.tasks ?? []),
|
|
1003
|
+
notes: r.notes ?? null,
|
|
1004
|
+
changelog: r.changelog ?? null,
|
|
1005
|
+
previousVersion: r.previousVersion ?? null,
|
|
1006
|
+
createdAt: r.createdAt,
|
|
1007
|
+
preparedAt: r.preparedAt ?? null,
|
|
1008
|
+
committedAt: r.committedAt ?? null,
|
|
1009
|
+
taggedAt: r.taggedAt ?? null,
|
|
1010
|
+
pushedAt: r.pushedAt ?? null,
|
|
1011
|
+
})
|
|
1012
|
+
.run();
|
|
1013
|
+
migrated++;
|
|
1014
|
+
}
|
|
1015
|
+
// Rename legacy file on success
|
|
1016
|
+
renameSync(releasesPath, releasesPath + '.migrated');
|
|
1017
|
+
return { migrated };
|
|
1018
|
+
}
|
|
1019
|
+
//# sourceMappingURL=release-manifest.js.map
|