@lumenflow/cli 3.1.1 → 3.1.3
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/README.md +36 -35
- package/dist/agent-issues-query.js +13 -8
- package/dist/agent-log-issue.js +15 -4
- package/dist/agent-session-end.js +15 -4
- package/dist/agent-session.js +18 -6
- package/dist/backlog-prune.js +1 -1
- package/dist/commands/integrate.js +32 -18
- package/dist/config-get.js +27 -15
- package/dist/config-set.js +104 -37
- package/dist/delegation-list.js +1 -1
- package/dist/doctor.js +19 -13
- package/dist/file-delete.js +1 -1
- package/dist/file-edit.js +1 -1
- package/dist/file-read.js +1 -1
- package/dist/file-write.js +1 -1
- package/dist/flow-bottlenecks.js +1 -1
- package/dist/flow-report.js +10 -9
- package/dist/gates.js +3 -2
- package/dist/git-branch.js +1 -1
- package/dist/git-diff.js +1 -1
- package/dist/git-log.js +1 -1
- package/dist/init.js +244 -61
- package/dist/initiative-add-wu.js +1 -1
- package/dist/initiative-bulk-assign-wus.js +1 -1
- package/dist/initiative-create.js +2 -2
- package/dist/initiative-edit.js +1 -1
- package/dist/initiative-list.js +1 -1
- package/dist/initiative-plan.js +1 -3
- package/dist/initiative-status.js +47 -6
- package/dist/lane-edit.js +19 -10
- package/dist/lane-health.js +13 -24
- package/dist/lane-lock.js +4 -5
- package/dist/lane-setup.js +5 -5
- package/dist/lane-status.js +4 -5
- package/dist/lane-suggest.js +9 -7
- package/dist/lane-validate.js +4 -5
- package/dist/lumenflow-upgrade.js +17 -11
- package/dist/mem-checkpoint.js +1 -1
- package/dist/mem-cleanup.js +6 -23
- package/dist/mem-context.js +1 -1
- package/dist/mem-create.js +1 -1
- package/dist/mem-delete.js +1 -1
- package/dist/mem-export.js +1 -1
- package/dist/mem-inbox.js +1 -1
- package/dist/mem-init.js +1 -1
- package/dist/mem-ready.js +1 -1
- package/dist/mem-recover.js +1 -1
- package/dist/mem-signal.js +1 -1
- package/dist/mem-start.js +1 -1
- package/dist/mem-summarize.js +8 -7
- package/dist/mem-triage.js +7 -5
- package/dist/metrics-cli.js +1 -1
- package/dist/metrics-snapshot.js +1 -1
- package/dist/onboard.js +295 -120
- package/dist/orchestrate-init-status.js +12 -7
- package/dist/orchestrate-initiative.js +23 -12
- package/dist/orchestrate-monitor.js +20 -8
- package/dist/pack-scaffold.js +1 -1
- package/dist/plan-create.js +1 -3
- package/dist/plan-edit.js +1 -3
- package/dist/plan-link.js +1 -3
- package/dist/plan-promote.js +1 -3
- package/dist/release.js +1 -3
- package/dist/signal-cleanup.js +4 -18
- package/dist/state-bootstrap.js +11 -8
- package/dist/state-cleanup.js +5 -19
- package/dist/state-doctor.js +213 -9
- package/dist/task-claim.js +1 -1
- package/dist/validate.js +1 -1
- package/dist/workspace-init.js +61 -61
- package/dist/wu-block.js +1 -1
- package/dist/wu-brief.js +1 -1
- package/dist/wu-claim.js +1 -1
- package/dist/wu-cleanup.js +1 -1
- package/dist/wu-create.js +3 -3
- package/dist/wu-delegate.js +1 -1
- package/dist/wu-deps.js +1 -1
- package/dist/wu-done.js +66 -34
- package/dist/wu-edit.js +1 -1
- package/dist/wu-infer-lane.js +1 -1
- package/dist/wu-preflight.js +1 -1
- package/dist/wu-prep.js +1 -1
- package/dist/wu-proto.js +1 -1
- package/dist/wu-prune.js +1 -1
- package/dist/wu-recover.js +1 -1
- package/dist/wu-release.js +1 -1
- package/dist/wu-repair.js +1 -1
- package/dist/wu-sandbox.js +40 -27
- package/dist/wu-status.js +1 -1
- package/dist/wu-unblock.js +1 -1
- package/dist/wu-unlock-lane.js +1 -1
- package/dist/wu-validate.js +1 -1
- package/package.json +12 -8
- package/packs/software-delivery/constants.ts +10 -0
- package/packs/software-delivery/extensions.ts +140 -0
- package/packs/software-delivery/gate-policies.ts +134 -0
- package/packs/software-delivery/index.ts +8 -0
- package/packs/software-delivery/manifest-schema.ts +236 -0
- package/packs/software-delivery/manifest.ts +417 -0
- package/packs/software-delivery/manifest.yaml +711 -0
- package/packs/software-delivery/pack-registration.ts +113 -0
- package/packs/software-delivery/tool-impl/agent-tools.ts +263 -0
- package/packs/software-delivery/tool-impl/delegation-tools.ts +66 -0
- package/packs/software-delivery/tool-impl/flow-metrics-tools.ts +219 -0
- package/packs/software-delivery/tool-impl/git-runner.ts +113 -0
- package/packs/software-delivery/tool-impl/git-tools.ts +316 -0
- package/packs/software-delivery/tool-impl/index.ts +15 -0
- package/packs/software-delivery/tool-impl/initiative-orchestration-tools.ts +720 -0
- package/packs/software-delivery/tool-impl/lane-lock.ts +246 -0
- package/packs/software-delivery/tool-impl/memory-tools.ts +415 -0
- package/packs/software-delivery/tool-impl/pending-runtime-tools.ts +21 -0
- package/packs/software-delivery/tool-impl/runtime-cli-adapter.ts +328 -0
- package/packs/software-delivery/tool-impl/runtime-native-tools.ts +687 -0
- package/packs/software-delivery/tool-impl/worker-loader.ts +52 -0
- package/packs/software-delivery/tool-impl/worktree-tools.ts +46 -0
- package/packs/software-delivery/tool-impl/wu-lifecycle-tools.ts +759 -0
- package/packs/software-delivery/tools/delegation-tools.ts +23 -0
- package/packs/software-delivery/tools/git-tools.ts +55 -0
- package/packs/software-delivery/tools/index.ts +8 -0
- package/packs/software-delivery/tools/lane-lock-tool.ts +37 -0
- package/packs/software-delivery/tools/types.ts +71 -0
- package/packs/software-delivery/tools/worktree-tools.ts +49 -0
- package/templates/core/LUMENFLOW.md.template +3 -3
- package/templates/core/ai/onboarding/agent-invocation-guide.md.template +1 -1
- package/templates/core/ai/onboarding/lumenflow-force-usage.md.template +1 -1
- package/templates/core/ai/onboarding/quick-ref-commands.md.template +5 -5
- package/templates/core/ai/onboarding/starting-prompt.md.template +3 -3
- package/templates/core/ai/onboarding/vendor-support.md.template +1 -1
- package/templates/core/ai/onboarding/wu-create-checklist.md.template +1 -1
- package/dist/agent-issues-query.js.map +0 -1
- package/dist/agent-log-issue.js.map +0 -1
- package/dist/agent-session-end.js.map +0 -1
- package/dist/agent-session.js.map +0 -1
- package/dist/backlog-prune.js.map +0 -1
- package/dist/cli-entry-point.js +0 -149
- package/dist/cli-entry-point.js.map +0 -1
- package/dist/commands/integrate.js.map +0 -1
- package/dist/commands.js.map +0 -1
- package/dist/config-get.js.map +0 -1
- package/dist/config-set.js.map +0 -1
- package/dist/delegation-list.js.map +0 -1
- package/dist/deps-add.js +0 -259
- package/dist/deps-add.js.map +0 -1
- package/dist/deps-remove.js +0 -105
- package/dist/deps-remove.js.map +0 -1
- package/dist/docs-sync.js.map +0 -1
- package/dist/doctor.js.map +0 -1
- package/dist/file-delete.js.map +0 -1
- package/dist/file-edit.js.map +0 -1
- package/dist/file-read.js.map +0 -1
- package/dist/file-write.js.map +0 -1
- package/dist/flow-bottlenecks.js.map +0 -1
- package/dist/flow-report.js.map +0 -1
- package/dist/formatters.js +0 -151
- package/dist/formatters.js.map +0 -1
- package/dist/gate-defaults.js +0 -131
- package/dist/gate-defaults.js.map +0 -1
- package/dist/gate-registry.js +0 -73
- package/dist/gate-registry.js.map +0 -1
- package/dist/gates-graceful-degradation.js +0 -153
- package/dist/gates-graceful-degradation.js.map +0 -1
- package/dist/gates-plan-resolvers.js +0 -152
- package/dist/gates-plan-resolvers.js.map +0 -1
- package/dist/gates-runners.js +0 -509
- package/dist/gates-runners.js.map +0 -1
- package/dist/gates-types.js +0 -4
- package/dist/gates-types.js.map +0 -1
- package/dist/gates-utils.js +0 -323
- package/dist/gates-utils.js.map +0 -1
- package/dist/gates.js.map +0 -1
- package/dist/git-branch.js.map +0 -1
- package/dist/git-diff.js.map +0 -1
- package/dist/git-log.js.map +0 -1
- package/dist/git-status.js.map +0 -1
- package/dist/guard-locked.js +0 -172
- package/dist/guard-locked.js.map +0 -1
- package/dist/guard-main-branch.js +0 -217
- package/dist/guard-main-branch.js.map +0 -1
- package/dist/guard-worktree-commit.js +0 -163
- package/dist/guard-worktree-commit.js.map +0 -1
- package/dist/hooks/auto-checkpoint-utils.js +0 -54
- package/dist/hooks/auto-checkpoint-utils.js.map +0 -1
- package/dist/hooks/enforcement-checks.js +0 -399
- package/dist/hooks/enforcement-checks.js.map +0 -1
- package/dist/hooks/enforcement-generator.js +0 -139
- package/dist/hooks/enforcement-generator.js.map +0 -1
- package/dist/hooks/enforcement-sync.js +0 -385
- package/dist/hooks/enforcement-sync.js.map +0 -1
- package/dist/hooks/generators/auto-checkpoint.js +0 -125
- package/dist/hooks/generators/auto-checkpoint.js.map +0 -1
- package/dist/hooks/generators/enforce-worktree.js +0 -190
- package/dist/hooks/generators/enforce-worktree.js.map +0 -1
- package/dist/hooks/generators/index.js +0 -18
- package/dist/hooks/generators/index.js.map +0 -1
- package/dist/hooks/generators/pre-compact-checkpoint.js +0 -136
- package/dist/hooks/generators/pre-compact-checkpoint.js.map +0 -1
- package/dist/hooks/generators/require-wu.js +0 -117
- package/dist/hooks/generators/require-wu.js.map +0 -1
- package/dist/hooks/generators/session-start-recovery.js +0 -103
- package/dist/hooks/generators/session-start-recovery.js.map +0 -1
- package/dist/hooks/generators/signal-utils.js +0 -54
- package/dist/hooks/generators/signal-utils.js.map +0 -1
- package/dist/hooks/generators/warn-incomplete.js +0 -67
- package/dist/hooks/generators/warn-incomplete.js.map +0 -1
- package/dist/hooks/index.js +0 -10
- package/dist/hooks/index.js.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/init-detection.js +0 -230
- package/dist/init-detection.js.map +0 -1
- package/dist/init-lane-validation.js +0 -143
- package/dist/init-lane-validation.js.map +0 -1
- package/dist/init-scaffolding.js +0 -158
- package/dist/init-scaffolding.js.map +0 -1
- package/dist/init-templates.js +0 -1964
- package/dist/init-templates.js.map +0 -1
- package/dist/init.js.map +0 -1
- package/dist/initiative-add-wu.js.map +0 -1
- package/dist/initiative-bulk-assign-wus.js.map +0 -1
- package/dist/initiative-create.js.map +0 -1
- package/dist/initiative-edit.js.map +0 -1
- package/dist/initiative-list.js.map +0 -1
- package/dist/initiative-plan.js.map +0 -1
- package/dist/initiative-remove-wu.js.map +0 -1
- package/dist/initiative-status.js.map +0 -1
- package/dist/lane-edit.js.map +0 -1
- package/dist/lane-health.js.map +0 -1
- package/dist/lane-lifecycle-process.js +0 -366
- package/dist/lane-lifecycle-process.js.map +0 -1
- package/dist/lane-lock.js.map +0 -1
- package/dist/lane-setup.js.map +0 -1
- package/dist/lane-status.js.map +0 -1
- package/dist/lane-suggest.js.map +0 -1
- package/dist/lane-validate.js.map +0 -1
- package/dist/lifecycle-regression-harness.js +0 -181
- package/dist/lifecycle-regression-harness.js.map +0 -1
- package/dist/lumenflow-upgrade.js.map +0 -1
- package/dist/mem-checkpoint.js.map +0 -1
- package/dist/mem-cleanup.js.map +0 -1
- package/dist/mem-context.js.map +0 -1
- package/dist/mem-create.js.map +0 -1
- package/dist/mem-delete.js.map +0 -1
- package/dist/mem-export.js.map +0 -1
- package/dist/mem-inbox.js.map +0 -1
- package/dist/mem-index.js +0 -214
- package/dist/mem-index.js.map +0 -1
- package/dist/mem-init.js.map +0 -1
- package/dist/mem-profile.js +0 -210
- package/dist/mem-profile.js.map +0 -1
- package/dist/mem-promote.js +0 -257
- package/dist/mem-promote.js.map +0 -1
- package/dist/mem-ready.js.map +0 -1
- package/dist/mem-recover.js.map +0 -1
- package/dist/mem-signal.js.map +0 -1
- package/dist/mem-start.js.map +0 -1
- package/dist/mem-summarize.js.map +0 -1
- package/dist/mem-triage.js.map +0 -1
- package/dist/merge-block.js +0 -225
- package/dist/merge-block.js.map +0 -1
- package/dist/metrics-cli.js.map +0 -1
- package/dist/metrics-snapshot.js.map +0 -1
- package/dist/onboard.js.map +0 -1
- package/dist/onboarding-smoke-test.js +0 -418
- package/dist/onboarding-smoke-test.js.map +0 -1
- package/dist/orchestrate-init-status.js.map +0 -1
- package/dist/orchestrate-initiative.js.map +0 -1
- package/dist/orchestrate-monitor.js.map +0 -1
- package/dist/pack-author.js.map +0 -1
- package/dist/pack-hash.js.map +0 -1
- package/dist/pack-install.js.map +0 -1
- package/dist/pack-publish.js.map +0 -1
- package/dist/pack-scaffold.js.map +0 -1
- package/dist/pack-search.js.map +0 -1
- package/dist/pack-validate.js.map +0 -1
- package/dist/plan-create.js.map +0 -1
- package/dist/plan-edit.js.map +0 -1
- package/dist/plan-link.js.map +0 -1
- package/dist/plan-promote.js.map +0 -1
- package/dist/public-manifest.js +0 -920
- package/dist/public-manifest.js.map +0 -1
- package/dist/release.js.map +0 -1
- package/dist/rotate-progress.js +0 -253
- package/dist/rotate-progress.js.map +0 -1
- package/dist/session-coordinator.js +0 -303
- package/dist/session-coordinator.js.map +0 -1
- package/dist/shared-validators.js +0 -81
- package/dist/shared-validators.js.map +0 -1
- package/dist/signal-cleanup.js.map +0 -1
- package/dist/state-bootstrap.js.map +0 -1
- package/dist/state-cleanup.js.map +0 -1
- package/dist/state-doctor-fix.js +0 -226
- package/dist/state-doctor-fix.js.map +0 -1
- package/dist/state-doctor-stamps.js +0 -23
- package/dist/state-doctor-stamps.js.map +0 -1
- package/dist/state-doctor.js.map +0 -1
- package/dist/strict-progress.js +0 -255
- package/dist/strict-progress.js.map +0 -1
- package/dist/sync-templates.js.map +0 -1
- package/dist/task-claim.js.map +0 -1
- package/dist/trace-gen.js +0 -401
- package/dist/trace-gen.js.map +0 -1
- package/dist/validate-agent-skills.js +0 -223
- package/dist/validate-agent-skills.js.map +0 -1
- package/dist/validate-agent-sync.js +0 -151
- package/dist/validate-agent-sync.js.map +0 -1
- package/dist/validate-backlog-sync.js +0 -77
- package/dist/validate-backlog-sync.js.map +0 -1
- package/dist/validate-skills-spec.js +0 -211
- package/dist/validate-skills-spec.js.map +0 -1
- package/dist/validate.js.map +0 -1
- package/dist/validator-defaults.js +0 -107
- package/dist/validator-defaults.js.map +0 -1
- package/dist/validator-registry.js +0 -71
- package/dist/validator-registry.js.map +0 -1
- package/dist/workspace-init.js.map +0 -1
- package/dist/wu-block.js.map +0 -1
- package/dist/wu-brief.js.map +0 -1
- package/dist/wu-claim-branch.js +0 -123
- package/dist/wu-claim-branch.js.map +0 -1
- package/dist/wu-claim-cloud.js +0 -79
- package/dist/wu-claim-cloud.js.map +0 -1
- package/dist/wu-claim-mode.js +0 -82
- package/dist/wu-claim-mode.js.map +0 -1
- package/dist/wu-claim-output.js +0 -85
- package/dist/wu-claim-output.js.map +0 -1
- package/dist/wu-claim-repair-guidance.js +0 -12
- package/dist/wu-claim-repair-guidance.js.map +0 -1
- package/dist/wu-claim-resume-handler.js +0 -87
- package/dist/wu-claim-resume-handler.js.map +0 -1
- package/dist/wu-claim-state.js +0 -581
- package/dist/wu-claim-state.js.map +0 -1
- package/dist/wu-claim-validation.js +0 -457
- package/dist/wu-claim-validation.js.map +0 -1
- package/dist/wu-claim-worktree.js +0 -223
- package/dist/wu-claim-worktree.js.map +0 -1
- package/dist/wu-claim.js.map +0 -1
- package/dist/wu-cleanup-cloud.js +0 -78
- package/dist/wu-cleanup-cloud.js.map +0 -1
- package/dist/wu-cleanup.js.map +0 -1
- package/dist/wu-code-path-coverage.js +0 -83
- package/dist/wu-code-path-coverage.js.map +0 -1
- package/dist/wu-create-cloud.js +0 -30
- package/dist/wu-create-cloud.js.map +0 -1
- package/dist/wu-create-content.js +0 -264
- package/dist/wu-create-content.js.map +0 -1
- package/dist/wu-create-readiness.js +0 -59
- package/dist/wu-create-readiness.js.map +0 -1
- package/dist/wu-create-validation.js +0 -128
- package/dist/wu-create-validation.js.map +0 -1
- package/dist/wu-create.js.map +0 -1
- package/dist/wu-delegate.js.map +0 -1
- package/dist/wu-delete.js.map +0 -1
- package/dist/wu-deps.js.map +0 -1
- package/dist/wu-done-auto-cleanup.js +0 -203
- package/dist/wu-done-auto-cleanup.js.map +0 -1
- package/dist/wu-done-check.js +0 -38
- package/dist/wu-done-check.js.map +0 -1
- package/dist/wu-done-cloud.js +0 -48
- package/dist/wu-done-cloud.js.map +0 -1
- package/dist/wu-done-decay.js +0 -86
- package/dist/wu-done-decay.js.map +0 -1
- package/dist/wu-done.js.map +0 -1
- package/dist/wu-edit-operations.js +0 -399
- package/dist/wu-edit-operations.js.map +0 -1
- package/dist/wu-edit-validators.js +0 -282
- package/dist/wu-edit-validators.js.map +0 -1
- package/dist/wu-edit.js.map +0 -1
- package/dist/wu-infer-lane.js.map +0 -1
- package/dist/wu-preflight.js.map +0 -1
- package/dist/wu-prep.js.map +0 -1
- package/dist/wu-proto.js.map +0 -1
- package/dist/wu-prune.js.map +0 -1
- package/dist/wu-recover.js.map +0 -1
- package/dist/wu-release.js.map +0 -1
- package/dist/wu-repair.js.map +0 -1
- package/dist/wu-sandbox.js.map +0 -1
- package/dist/wu-spawn-completion.js +0 -33
- package/dist/wu-spawn-completion.js.map +0 -1
- package/dist/wu-spawn-prompt-builders.js +0 -1197
- package/dist/wu-spawn-prompt-builders.js.map +0 -1
- package/dist/wu-spawn-strategy-resolver.js +0 -322
- package/dist/wu-spawn-strategy-resolver.js.map +0 -1
- package/dist/wu-spawn.js +0 -59
- package/dist/wu-spawn.js.map +0 -1
- package/dist/wu-state-cloud.js +0 -41
- package/dist/wu-state-cloud.js.map +0 -1
- package/dist/wu-status.js.map +0 -1
- package/dist/wu-unblock.js.map +0 -1
- package/dist/wu-unlock-lane.js.map +0 -1
- package/dist/wu-validate.js.map +0 -1
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
// Copyright (c) 2026 Hellmai Ltd
|
|
2
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
3
|
+
|
|
4
|
+
import { SOFTWARE_DELIVERY_PACK_ID, SOFTWARE_DELIVERY_POLICY_ID_PREFIX } from './constants.js';
|
|
5
|
+
|
|
6
|
+
export const SOFTWARE_DELIVERY_GATE_NAMES = [
|
|
7
|
+
'format',
|
|
8
|
+
'lint',
|
|
9
|
+
'typecheck',
|
|
10
|
+
'test',
|
|
11
|
+
'coverage',
|
|
12
|
+
] as const;
|
|
13
|
+
|
|
14
|
+
export type SoftwareDeliveryGateName = (typeof SOFTWARE_DELIVERY_GATE_NAMES)[number];
|
|
15
|
+
|
|
16
|
+
export interface PolicyProvider {
|
|
17
|
+
id: string;
|
|
18
|
+
gate: SoftwareDeliveryGateName;
|
|
19
|
+
trigger: 'on_completion';
|
|
20
|
+
command: string;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export interface CompletionPolicyContext {
|
|
24
|
+
trigger: 'on_completion';
|
|
25
|
+
run_id: string;
|
|
26
|
+
task_id: string;
|
|
27
|
+
gate: SoftwareDeliveryGateName;
|
|
28
|
+
pack_id: typeof SOFTWARE_DELIVERY_PACK_ID;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export interface CompletionPolicyEvaluation {
|
|
32
|
+
decision: 'allow' | 'deny';
|
|
33
|
+
decisions: Array<{
|
|
34
|
+
policy_id: string;
|
|
35
|
+
decision: 'allow' | 'deny';
|
|
36
|
+
reason?: string;
|
|
37
|
+
}>;
|
|
38
|
+
warnings: string[];
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export interface CompletionPolicyEngine {
|
|
42
|
+
evaluate(context: CompletionPolicyContext): Promise<CompletionPolicyEvaluation>;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export interface CompletionPolicyExecution {
|
|
46
|
+
ok: boolean;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
export interface CompletionPolicyRunResult {
|
|
50
|
+
gate: SoftwareDeliveryGateName;
|
|
51
|
+
policy_id: string;
|
|
52
|
+
decision: 'allow' | 'deny';
|
|
53
|
+
command: string;
|
|
54
|
+
executed: boolean;
|
|
55
|
+
ok: boolean;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
const SOFTWARE_DELIVERY_GATE_COMMANDS: Record<SoftwareDeliveryGateName, string> = {
|
|
59
|
+
format: 'pnpm format:check',
|
|
60
|
+
lint: 'pnpm lint',
|
|
61
|
+
typecheck: 'pnpm typecheck',
|
|
62
|
+
test: 'pnpm turbo run test',
|
|
63
|
+
coverage: 'pnpm vitest run --coverage',
|
|
64
|
+
};
|
|
65
|
+
|
|
66
|
+
export const SOFTWARE_DELIVERY_GATE_POLICIES: readonly PolicyProvider[] =
|
|
67
|
+
SOFTWARE_DELIVERY_GATE_NAMES.map((gate) => ({
|
|
68
|
+
id: `${SOFTWARE_DELIVERY_POLICY_ID_PREFIX}.${gate}`,
|
|
69
|
+
gate,
|
|
70
|
+
trigger: 'on_completion',
|
|
71
|
+
command: SOFTWARE_DELIVERY_GATE_COMMANDS[gate],
|
|
72
|
+
}));
|
|
73
|
+
|
|
74
|
+
export const SOFTWARE_DELIVERY_STATE_ALIASES = {
|
|
75
|
+
active: 'in_progress',
|
|
76
|
+
} as const;
|
|
77
|
+
|
|
78
|
+
export function resolveSoftwareDeliveryStateAlias(
|
|
79
|
+
state: string,
|
|
80
|
+
aliases: Record<string, string> = SOFTWARE_DELIVERY_STATE_ALIASES,
|
|
81
|
+
): string {
|
|
82
|
+
return aliases[state] ?? state;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
export async function runCompletionGatePolicies(options: {
|
|
86
|
+
engine: CompletionPolicyEngine;
|
|
87
|
+
runId: string;
|
|
88
|
+
taskId: string;
|
|
89
|
+
providers?: readonly PolicyProvider[];
|
|
90
|
+
executePolicy?: (provider: PolicyProvider) => Promise<CompletionPolicyExecution>;
|
|
91
|
+
}): Promise<CompletionPolicyRunResult[]> {
|
|
92
|
+
const providers = options.providers ?? SOFTWARE_DELIVERY_GATE_POLICIES;
|
|
93
|
+
const executePolicy =
|
|
94
|
+
options.executePolicy ??
|
|
95
|
+
(async () => ({
|
|
96
|
+
ok: true,
|
|
97
|
+
}));
|
|
98
|
+
|
|
99
|
+
const results: CompletionPolicyRunResult[] = [];
|
|
100
|
+
|
|
101
|
+
for (const provider of providers) {
|
|
102
|
+
const evaluation = await options.engine.evaluate({
|
|
103
|
+
trigger: provider.trigger,
|
|
104
|
+
run_id: options.runId,
|
|
105
|
+
task_id: options.taskId,
|
|
106
|
+
gate: provider.gate,
|
|
107
|
+
pack_id: SOFTWARE_DELIVERY_PACK_ID,
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
if (evaluation.decision === 'deny') {
|
|
111
|
+
results.push({
|
|
112
|
+
gate: provider.gate,
|
|
113
|
+
policy_id: provider.id,
|
|
114
|
+
decision: 'deny',
|
|
115
|
+
command: provider.command,
|
|
116
|
+
executed: false,
|
|
117
|
+
ok: false,
|
|
118
|
+
});
|
|
119
|
+
continue;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
const execution = await executePolicy(provider);
|
|
123
|
+
results.push({
|
|
124
|
+
gate: provider.gate,
|
|
125
|
+
policy_id: provider.id,
|
|
126
|
+
decision: 'allow',
|
|
127
|
+
command: provider.command,
|
|
128
|
+
executed: true,
|
|
129
|
+
ok: execution.ok,
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
return results;
|
|
134
|
+
}
|
|
@@ -0,0 +1,236 @@
|
|
|
1
|
+
// Copyright (c) 2026 Hellmai Ltd
|
|
2
|
+
// SPDX-License-Identifier: AGPL-3.0-only
|
|
3
|
+
|
|
4
|
+
import {
|
|
5
|
+
TOOL_PERMISSIONS,
|
|
6
|
+
TOOL_SCOPE_ACCESS,
|
|
7
|
+
TOOL_SCOPE_TYPES,
|
|
8
|
+
type PathScope,
|
|
9
|
+
type ToolPermission,
|
|
10
|
+
} from './tools/types.js';
|
|
11
|
+
|
|
12
|
+
interface Parser<T> {
|
|
13
|
+
parse(input: unknown): T;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export const MANIFEST_POLICY_TRIGGERS = {
|
|
17
|
+
ON_TOOL_REQUEST: 'on_tool_request',
|
|
18
|
+
ON_CLAIM: 'on_claim',
|
|
19
|
+
ON_COMPLETION: 'on_completion',
|
|
20
|
+
ON_EVIDENCE_ADDED: 'on_evidence_added',
|
|
21
|
+
} as const;
|
|
22
|
+
|
|
23
|
+
export type ManifestPolicyTrigger =
|
|
24
|
+
(typeof MANIFEST_POLICY_TRIGGERS)[keyof typeof MANIFEST_POLICY_TRIGGERS];
|
|
25
|
+
|
|
26
|
+
export const MANIFEST_POLICY_DECISIONS = {
|
|
27
|
+
ALLOW: 'allow',
|
|
28
|
+
DENY: 'deny',
|
|
29
|
+
} as const;
|
|
30
|
+
|
|
31
|
+
export type ManifestPolicyDecision =
|
|
32
|
+
(typeof MANIFEST_POLICY_DECISIONS)[keyof typeof MANIFEST_POLICY_DECISIONS];
|
|
33
|
+
|
|
34
|
+
export interface SoftwareDeliveryManifestTool {
|
|
35
|
+
name: string;
|
|
36
|
+
entry: string;
|
|
37
|
+
permission: ToolPermission;
|
|
38
|
+
required_scopes: PathScope[];
|
|
39
|
+
internal_only?: boolean;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export interface SoftwareDeliveryManifestPolicy {
|
|
43
|
+
id: string;
|
|
44
|
+
trigger: ManifestPolicyTrigger;
|
|
45
|
+
decision: ManifestPolicyDecision;
|
|
46
|
+
reason?: string;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
export interface SoftwareDeliveryPackManifest {
|
|
50
|
+
id: string;
|
|
51
|
+
version: string;
|
|
52
|
+
task_types: string[];
|
|
53
|
+
tools: SoftwareDeliveryManifestTool[];
|
|
54
|
+
policies: SoftwareDeliveryManifestPolicy[];
|
|
55
|
+
evidence_types: string[];
|
|
56
|
+
state_aliases: Record<string, string>;
|
|
57
|
+
lane_templates: Array<{ id: string }>;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
function asRecord(input: unknown, label: string): Record<string, unknown> {
|
|
61
|
+
if (!input || typeof input !== 'object' || Array.isArray(input)) {
|
|
62
|
+
throw new Error(`${label} must be an object.`);
|
|
63
|
+
}
|
|
64
|
+
return input as Record<string, unknown>;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
function parseNonEmptyString(value: unknown, label: string): string {
|
|
68
|
+
if (typeof value !== 'string' || value.trim().length === 0) {
|
|
69
|
+
throw new Error(`${label} must be a non-empty string.`);
|
|
70
|
+
}
|
|
71
|
+
return value;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
function parseStringArray(value: unknown, label: string): string[] {
|
|
75
|
+
if (!Array.isArray(value)) {
|
|
76
|
+
throw new Error(`${label} must be an array.`);
|
|
77
|
+
}
|
|
78
|
+
return value.map((entry, index) => parseNonEmptyString(entry, `${label}[${index}]`));
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
function isSemver(value: string): boolean {
|
|
82
|
+
let core = value;
|
|
83
|
+
const prereleaseIndex = core.indexOf('-');
|
|
84
|
+
if (prereleaseIndex >= 0) {
|
|
85
|
+
core = core.slice(0, prereleaseIndex);
|
|
86
|
+
}
|
|
87
|
+
const metadataIndex = core.indexOf('+');
|
|
88
|
+
if (metadataIndex >= 0) {
|
|
89
|
+
core = core.slice(0, metadataIndex);
|
|
90
|
+
}
|
|
91
|
+
const parts = core.split('.');
|
|
92
|
+
if (parts.length !== 3) {
|
|
93
|
+
return false;
|
|
94
|
+
}
|
|
95
|
+
return parts.every(
|
|
96
|
+
(part) => part.length > 0 && [...part].every((char) => char >= '0' && char <= '9'),
|
|
97
|
+
);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
const ALLOWED_POLICY_TRIGGERS = new Set<string>(Object.values(MANIFEST_POLICY_TRIGGERS));
|
|
101
|
+
const ALLOWED_POLICY_DECISIONS = new Set<string>(Object.values(MANIFEST_POLICY_DECISIONS));
|
|
102
|
+
const ALLOWED_TOOL_PERMISSIONS = new Set<string>(Object.values(TOOL_PERMISSIONS));
|
|
103
|
+
|
|
104
|
+
function parsePathScope(input: unknown, label: string): PathScope {
|
|
105
|
+
const scope = asRecord(input, label);
|
|
106
|
+
const type = parseNonEmptyString(scope.type, `${label}.type`);
|
|
107
|
+
const pattern = parseNonEmptyString(scope.pattern, `${label}.pattern`);
|
|
108
|
+
const access = parseNonEmptyString(scope.access, `${label}.access`);
|
|
109
|
+
|
|
110
|
+
if (type !== TOOL_SCOPE_TYPES.PATH) {
|
|
111
|
+
throw new Error(`${label}.type must be "${TOOL_SCOPE_TYPES.PATH}".`);
|
|
112
|
+
}
|
|
113
|
+
if (access !== TOOL_SCOPE_ACCESS.READ && access !== TOOL_SCOPE_ACCESS.WRITE) {
|
|
114
|
+
throw new Error(
|
|
115
|
+
`${label}.access must be "${TOOL_SCOPE_ACCESS.READ}" or "${TOOL_SCOPE_ACCESS.WRITE}".`,
|
|
116
|
+
);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
return {
|
|
120
|
+
type: TOOL_SCOPE_TYPES.PATH,
|
|
121
|
+
pattern,
|
|
122
|
+
access,
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
function parseRequiredScopes(value: unknown, label: string): PathScope[] {
|
|
127
|
+
if (!Array.isArray(value) || value.length === 0) {
|
|
128
|
+
throw new Error(`${label} must be a non-empty array.`);
|
|
129
|
+
}
|
|
130
|
+
return value.map((entry, index) => parsePathScope(entry, `${label}[${index}]`));
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
function parsePolicy(input: unknown, index: number): SoftwareDeliveryManifestPolicy {
|
|
134
|
+
const policy = asRecord(input, `policies[${index}]`);
|
|
135
|
+
const trigger = parseNonEmptyString(policy.trigger, `policies[${index}].trigger`);
|
|
136
|
+
const decision = parseNonEmptyString(policy.decision, `policies[${index}].decision`);
|
|
137
|
+
|
|
138
|
+
if (!ALLOWED_POLICY_TRIGGERS.has(trigger)) {
|
|
139
|
+
throw new Error(`policies[${index}].trigger is invalid.`);
|
|
140
|
+
}
|
|
141
|
+
if (!ALLOWED_POLICY_DECISIONS.has(decision)) {
|
|
142
|
+
throw new Error(`policies[${index}].decision is invalid.`);
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
return {
|
|
146
|
+
id: parseNonEmptyString(policy.id, `policies[${index}].id`),
|
|
147
|
+
trigger: trigger as ManifestPolicyTrigger,
|
|
148
|
+
decision: decision as ManifestPolicyDecision,
|
|
149
|
+
reason:
|
|
150
|
+
policy.reason === undefined
|
|
151
|
+
? undefined
|
|
152
|
+
: parseNonEmptyString(policy.reason, `policies[${index}].reason`),
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
function parseTool(input: unknown, index: number): SoftwareDeliveryManifestTool {
|
|
157
|
+
const tool = asRecord(input, `tools[${index}]`);
|
|
158
|
+
const permission =
|
|
159
|
+
tool.permission === undefined
|
|
160
|
+
? TOOL_PERMISSIONS.READ
|
|
161
|
+
: parseNonEmptyString(tool.permission, `tools[${index}].permission`);
|
|
162
|
+
|
|
163
|
+
if (!ALLOWED_TOOL_PERMISSIONS.has(permission)) {
|
|
164
|
+
throw new Error(`tools[${index}].permission is invalid.`);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
return {
|
|
168
|
+
name: parseNonEmptyString(tool.name, `tools[${index}].name`),
|
|
169
|
+
entry: parseNonEmptyString(tool.entry, `tools[${index}].entry`),
|
|
170
|
+
permission: permission as ToolPermission,
|
|
171
|
+
required_scopes: parseRequiredScopes(tool.required_scopes, `tools[${index}].required_scopes`),
|
|
172
|
+
internal_only:
|
|
173
|
+
tool.internal_only === undefined
|
|
174
|
+
? undefined
|
|
175
|
+
: (() => {
|
|
176
|
+
if (typeof tool.internal_only !== 'boolean') {
|
|
177
|
+
throw new Error(`tools[${index}].internal_only must be boolean.`);
|
|
178
|
+
}
|
|
179
|
+
return tool.internal_only;
|
|
180
|
+
})(),
|
|
181
|
+
};
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
function parseStateAliases(input: unknown): Record<string, string> {
|
|
185
|
+
const aliases = asRecord(input ?? {}, 'state_aliases');
|
|
186
|
+
const parsed: Record<string, string> = {};
|
|
187
|
+
for (const [key, value] of Object.entries(aliases)) {
|
|
188
|
+
parsed[parseNonEmptyString(key, 'state_aliases key')] = parseNonEmptyString(
|
|
189
|
+
value,
|
|
190
|
+
`state_aliases.${key}`,
|
|
191
|
+
);
|
|
192
|
+
}
|
|
193
|
+
return parsed;
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
export const SoftwareDeliveryManifestSchema: Parser<SoftwareDeliveryPackManifest> = {
|
|
197
|
+
parse(input: unknown): SoftwareDeliveryPackManifest {
|
|
198
|
+
const manifest = asRecord(input, 'manifest');
|
|
199
|
+
const version = parseNonEmptyString(manifest.version, 'version');
|
|
200
|
+
if (!isSemver(version)) {
|
|
201
|
+
throw new Error('version must be semver.');
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
const taskTypes = parseStringArray(manifest.task_types, 'task_types');
|
|
205
|
+
if (taskTypes.length === 0) {
|
|
206
|
+
throw new Error('task_types must include at least one item.');
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
const toolsValue = manifest.tools ?? [];
|
|
210
|
+
if (!Array.isArray(toolsValue)) {
|
|
211
|
+
throw new Error('tools must be an array.');
|
|
212
|
+
}
|
|
213
|
+
const policiesValue = manifest.policies ?? [];
|
|
214
|
+
if (!Array.isArray(policiesValue)) {
|
|
215
|
+
throw new Error('policies must be an array.');
|
|
216
|
+
}
|
|
217
|
+
const laneTemplatesValue = manifest.lane_templates ?? [];
|
|
218
|
+
if (!Array.isArray(laneTemplatesValue)) {
|
|
219
|
+
throw new Error('lane_templates must be an array.');
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
return {
|
|
223
|
+
id: parseNonEmptyString(manifest.id, 'id'),
|
|
224
|
+
version,
|
|
225
|
+
task_types: taskTypes,
|
|
226
|
+
tools: toolsValue.map((tool, index) => parseTool(tool, index)),
|
|
227
|
+
policies: policiesValue.map((policy, index) => parsePolicy(policy, index)),
|
|
228
|
+
evidence_types: parseStringArray(manifest.evidence_types ?? [], 'evidence_types'),
|
|
229
|
+
state_aliases: parseStateAliases(manifest.state_aliases),
|
|
230
|
+
lane_templates: laneTemplatesValue.map((laneTemplate, index) => {
|
|
231
|
+
const entry = asRecord(laneTemplate, `lane_templates[${index}]`);
|
|
232
|
+
return { id: parseNonEmptyString(entry.id, `lane_templates[${index}].id`) };
|
|
233
|
+
}),
|
|
234
|
+
};
|
|
235
|
+
},
|
|
236
|
+
};
|