@principles/pd-cli 1.73.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +90 -0
- package/dist/commands/artifact.d.ts +14 -0
- package/dist/commands/artifact.d.ts.map +1 -0
- package/dist/commands/artifact.js +67 -0
- package/dist/commands/artifact.js.map +1 -0
- package/dist/commands/candidate.d.ts +83 -0
- package/dist/commands/candidate.d.ts.map +1 -0
- package/dist/commands/candidate.js +891 -0
- package/dist/commands/candidate.js.map +1 -0
- package/dist/commands/central-sync.d.ts +10 -0
- package/dist/commands/central-sync.d.ts.map +1 -0
- package/dist/commands/central-sync.js +32 -0
- package/dist/commands/central-sync.js.map +1 -0
- package/dist/commands/console.d.ts +9 -0
- package/dist/commands/console.d.ts.map +1 -0
- package/dist/commands/console.js +114 -0
- package/dist/commands/console.js.map +1 -0
- package/dist/commands/context.d.ts +7 -0
- package/dist/commands/context.d.ts.map +1 -0
- package/dist/commands/context.js +55 -0
- package/dist/commands/context.js.map +1 -0
- package/dist/commands/demo-story-a.d.ts +12 -0
- package/dist/commands/demo-story-a.d.ts.map +1 -0
- package/dist/commands/demo-story-a.js +175 -0
- package/dist/commands/demo-story-a.js.map +1 -0
- package/dist/commands/diagnose.d.ts +35 -0
- package/dist/commands/diagnose.d.ts.map +1 -0
- package/dist/commands/diagnose.js +390 -0
- package/dist/commands/diagnose.js.map +1 -0
- package/dist/commands/evolution-tasks-list.d.ts +15 -0
- package/dist/commands/evolution-tasks-list.d.ts.map +1 -0
- package/dist/commands/evolution-tasks-list.js +34 -0
- package/dist/commands/evolution-tasks-list.js.map +1 -0
- package/dist/commands/evolution-tasks-show.d.ts +14 -0
- package/dist/commands/evolution-tasks-show.d.ts.map +1 -0
- package/dist/commands/evolution-tasks-show.js +52 -0
- package/dist/commands/evolution-tasks-show.js.map +1 -0
- package/dist/commands/flow.d.ts +7 -0
- package/dist/commands/flow.d.ts.map +1 -0
- package/dist/commands/flow.js +57 -0
- package/dist/commands/flow.js.map +1 -0
- package/dist/commands/health.d.ts +16 -0
- package/dist/commands/health.d.ts.map +1 -0
- package/dist/commands/health.js +150 -0
- package/dist/commands/health.js.map +1 -0
- package/dist/commands/history.d.ts +11 -0
- package/dist/commands/history.d.ts.map +1 -0
- package/dist/commands/history.js +50 -0
- package/dist/commands/history.js.map +1 -0
- package/dist/commands/legacy-cleanup.d.ts +27 -0
- package/dist/commands/legacy-cleanup.d.ts.map +1 -0
- package/dist/commands/legacy-cleanup.js +171 -0
- package/dist/commands/legacy-cleanup.js.map +1 -0
- package/dist/commands/legacy-import.d.ts +7 -0
- package/dist/commands/legacy-import.d.ts.map +1 -0
- package/dist/commands/legacy-import.js +86 -0
- package/dist/commands/legacy-import.js.map +1 -0
- package/dist/commands/pain-record.d.ts +10 -0
- package/dist/commands/pain-record.d.ts.map +1 -0
- package/dist/commands/pain-record.js +162 -0
- package/dist/commands/pain-record.js.map +1 -0
- package/dist/commands/proven-channel-baseline.d.ts +12 -0
- package/dist/commands/proven-channel-baseline.d.ts.map +1 -0
- package/dist/commands/proven-channel-baseline.js +97 -0
- package/dist/commands/proven-channel-baseline.js.map +1 -0
- package/dist/commands/remediation-output.d.ts +40 -0
- package/dist/commands/remediation-output.d.ts.map +1 -0
- package/dist/commands/remediation-output.js +23 -0
- package/dist/commands/remediation-output.js.map +1 -0
- package/dist/commands/run.d.ts +10 -0
- package/dist/commands/run.d.ts.map +1 -0
- package/dist/commands/run.js +68 -0
- package/dist/commands/run.js.map +1 -0
- package/dist/commands/runtime-activation.d.ts +11 -0
- package/dist/commands/runtime-activation.d.ts.map +1 -0
- package/dist/commands/runtime-activation.js +150 -0
- package/dist/commands/runtime-activation.js.map +1 -0
- package/dist/commands/runtime-canary.d.ts +30 -0
- package/dist/commands/runtime-canary.d.ts.map +1 -0
- package/dist/commands/runtime-canary.js +343 -0
- package/dist/commands/runtime-canary.js.map +1 -0
- package/dist/commands/runtime-diagnostics-export.d.ts +20 -0
- package/dist/commands/runtime-diagnostics-export.d.ts.map +1 -0
- package/dist/commands/runtime-diagnostics-export.js +177 -0
- package/dist/commands/runtime-diagnostics-export.js.map +1 -0
- package/dist/commands/runtime-features.d.ts +26 -0
- package/dist/commands/runtime-features.d.ts.map +1 -0
- package/dist/commands/runtime-features.js +70 -0
- package/dist/commands/runtime-features.js.map +1 -0
- package/dist/commands/runtime-gfi-snapshot.d.ts +7 -0
- package/dist/commands/runtime-gfi-snapshot.d.ts.map +1 -0
- package/dist/commands/runtime-gfi-snapshot.js +101 -0
- package/dist/commands/runtime-gfi-snapshot.js.map +1 -0
- package/dist/commands/runtime-health-snapshot.d.ts +7 -0
- package/dist/commands/runtime-health-snapshot.d.ts.map +1 -0
- package/dist/commands/runtime-health-snapshot.js +93 -0
- package/dist/commands/runtime-health-snapshot.js.map +1 -0
- package/dist/commands/runtime-idle-trigger.d.ts +12 -0
- package/dist/commands/runtime-idle-trigger.d.ts.map +1 -0
- package/dist/commands/runtime-idle-trigger.js +102 -0
- package/dist/commands/runtime-idle-trigger.js.map +1 -0
- package/dist/commands/runtime-internalization-enqueue-successors.d.ts +9 -0
- package/dist/commands/runtime-internalization-enqueue-successors.d.ts.map +1 -0
- package/dist/commands/runtime-internalization-enqueue-successors.js +393 -0
- package/dist/commands/runtime-internalization-enqueue-successors.js.map +1 -0
- package/dist/commands/runtime-internalization-integrity-repair.d.ts +9 -0
- package/dist/commands/runtime-internalization-integrity-repair.d.ts.map +1 -0
- package/dist/commands/runtime-internalization-integrity-repair.js +54 -0
- package/dist/commands/runtime-internalization-integrity-repair.js.map +1 -0
- package/dist/commands/runtime-internalization-integrity.d.ts +7 -0
- package/dist/commands/runtime-internalization-integrity.d.ts.map +1 -0
- package/dist/commands/runtime-internalization-integrity.js +53 -0
- package/dist/commands/runtime-internalization-integrity.js.map +1 -0
- package/dist/commands/runtime-internalization-queue.d.ts +7 -0
- package/dist/commands/runtime-internalization-queue.d.ts.map +1 -0
- package/dist/commands/runtime-internalization-queue.js +85 -0
- package/dist/commands/runtime-internalization-queue.js.map +1 -0
- package/dist/commands/runtime-internalization-run-once.d.ts +12 -0
- package/dist/commands/runtime-internalization-run-once.d.ts.map +1 -0
- package/dist/commands/runtime-internalization-run-once.js +546 -0
- package/dist/commands/runtime-internalization-run-once.js.map +1 -0
- package/dist/commands/runtime-internalization-wake-once.d.ts +8 -0
- package/dist/commands/runtime-internalization-wake-once.d.ts.map +1 -0
- package/dist/commands/runtime-internalization-wake-once.js +72 -0
- package/dist/commands/runtime-internalization-wake-once.js.map +1 -0
- package/dist/commands/runtime-pain-flood-simulation.d.ts +10 -0
- package/dist/commands/runtime-pain-flood-simulation.d.ts.map +1 -0
- package/dist/commands/runtime-pain-flood-simulation.js +104 -0
- package/dist/commands/runtime-pain-flood-simulation.js.map +1 -0
- package/dist/commands/runtime-pruning.d.ts +45 -0
- package/dist/commands/runtime-pruning.d.ts.map +1 -0
- package/dist/commands/runtime-pruning.js +355 -0
- package/dist/commands/runtime-pruning.js.map +1 -0
- package/dist/commands/runtime-recovery.d.ts +9 -0
- package/dist/commands/runtime-recovery.d.ts.map +1 -0
- package/dist/commands/runtime-recovery.js +94 -0
- package/dist/commands/runtime-recovery.js.map +1 -0
- package/dist/commands/runtime-synthetic-baseline.d.ts +7 -0
- package/dist/commands/runtime-synthetic-baseline.d.ts.map +1 -0
- package/dist/commands/runtime-synthetic-baseline.js +59 -0
- package/dist/commands/runtime-synthetic-baseline.js.map +1 -0
- package/dist/commands/runtime-uat.d.ts +52 -0
- package/dist/commands/runtime-uat.d.ts.map +1 -0
- package/dist/commands/runtime-uat.js +274 -0
- package/dist/commands/runtime-uat.js.map +1 -0
- package/dist/commands/runtime.d.ts +20 -0
- package/dist/commands/runtime.d.ts.map +1 -0
- package/dist/commands/runtime.js +256 -0
- package/dist/commands/runtime.js.map +1 -0
- package/dist/commands/samples-list.d.ts +11 -0
- package/dist/commands/samples-list.d.ts.map +1 -0
- package/dist/commands/samples-list.js +37 -0
- package/dist/commands/samples-list.js.map +1 -0
- package/dist/commands/samples-review.d.ts +14 -0
- package/dist/commands/samples-review.d.ts.map +1 -0
- package/dist/commands/samples-review.js +22 -0
- package/dist/commands/samples-review.js.map +1 -0
- package/dist/commands/task.d.ts +14 -0
- package/dist/commands/task.d.ts.map +1 -0
- package/dist/commands/task.js +92 -0
- package/dist/commands/task.js.map +1 -0
- package/dist/commands/trace.d.ts +19 -0
- package/dist/commands/trace.d.ts.map +1 -0
- package/dist/commands/trace.js +154 -0
- package/dist/commands/trace.js.map +1 -0
- package/dist/commands/trajectory.d.ts +11 -0
- package/dist/commands/trajectory.d.ts.map +1 -0
- package/dist/commands/trajectory.js +47 -0
- package/dist/commands/trajectory.js.map +1 -0
- package/dist/index.d.ts +9 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +736 -0
- package/dist/index.js.map +1 -0
- package/dist/legacy/legacy-import.d.ts +15 -0
- package/dist/legacy/legacy-import.d.ts.map +1 -0
- package/dist/legacy/legacy-import.js +141 -0
- package/dist/legacy/legacy-import.js.map +1 -0
- package/dist/legacy/session-history-import.d.ts +26 -0
- package/dist/legacy/session-history-import.d.ts.map +1 -0
- package/dist/legacy/session-history-import.js +151 -0
- package/dist/legacy/session-history-import.js.map +1 -0
- package/dist/principle-tree-ledger-adapter.d.ts +12 -0
- package/dist/principle-tree-ledger-adapter.d.ts.map +1 -0
- package/dist/principle-tree-ledger-adapter.js +12 -0
- package/dist/principle-tree-ledger-adapter.js.map +1 -0
- package/dist/resolve-workspace.d.ts +12 -0
- package/dist/resolve-workspace.d.ts.map +1 -0
- package/dist/resolve-workspace.js +20 -0
- package/dist/resolve-workspace.js.map +1 -0
- package/dist/services/demo-story-a-runner.d.ts +8 -0
- package/dist/services/demo-story-a-runner.d.ts.map +1 -0
- package/dist/services/demo-story-a-runner.js +369 -0
- package/dist/services/demo-story-a-runner.js.map +1 -0
- package/dist/services/feature-flag-loader.d.ts +6 -0
- package/dist/services/feature-flag-loader.d.ts.map +1 -0
- package/dist/services/feature-flag-loader.js +54 -0
- package/dist/services/feature-flag-loader.js.map +1 -0
- package/dist/services/pain-flood-simulation-runner.d.ts +10 -0
- package/dist/services/pain-flood-simulation-runner.d.ts.map +1 -0
- package/dist/services/pain-flood-simulation-runner.js +289 -0
- package/dist/services/pain-flood-simulation-runner.js.map +1 -0
- package/dist/services/proven-channel-baseline-runner.d.ts +12 -0
- package/dist/services/proven-channel-baseline-runner.d.ts.map +1 -0
- package/dist/services/proven-channel-baseline-runner.js +114 -0
- package/dist/services/proven-channel-baseline-runner.js.map +1 -0
- package/dist/services/synthetic-baseline-runner.d.ts +8 -0
- package/dist/services/synthetic-baseline-runner.d.ts.map +1 -0
- package/dist/services/synthetic-baseline-runner.js +251 -0
- package/dist/services/synthetic-baseline-runner.js.map +1 -0
- package/package.json +35 -0
- package/src/commands/artifact.ts +82 -0
- package/src/commands/candidate.ts +1117 -0
- package/src/commands/central-sync.ts +44 -0
- package/src/commands/console.ts +121 -0
- package/src/commands/context.ts +72 -0
- package/src/commands/demo-story-a.ts +195 -0
- package/src/commands/diagnose.ts +452 -0
- package/src/commands/evolution-tasks-list.ts +44 -0
- package/src/commands/evolution-tasks-show.ts +60 -0
- package/src/commands/flow.ts +60 -0
- package/src/commands/health.ts +189 -0
- package/src/commands/history.ts +63 -0
- package/src/commands/legacy-cleanup.ts +206 -0
- package/src/commands/legacy-import.ts +104 -0
- package/src/commands/pain-record.ts +167 -0
- package/src/commands/proven-channel-baseline.ts +113 -0
- package/src/commands/remediation-output.ts +66 -0
- package/src/commands/run.ts +89 -0
- package/src/commands/runtime-activation.ts +176 -0
- package/src/commands/runtime-canary.ts +371 -0
- package/src/commands/runtime-diagnostics-export.ts +229 -0
- package/src/commands/runtime-features.ts +103 -0
- package/src/commands/runtime-gfi-snapshot.ts +135 -0
- package/src/commands/runtime-health-snapshot.ts +106 -0
- package/src/commands/runtime-internalization-enqueue-successors.ts +479 -0
- package/src/commands/runtime-internalization-integrity-repair.ts +69 -0
- package/src/commands/runtime-internalization-integrity.ts +63 -0
- package/src/commands/runtime-internalization-queue.ts +106 -0
- package/src/commands/runtime-internalization-run-once.ts +658 -0
- package/src/commands/runtime-internalization-wake-once.ts +87 -0
- package/src/commands/runtime-pain-flood-simulation.ts +121 -0
- package/src/commands/runtime-pruning.ts +438 -0
- package/src/commands/runtime-recovery.ts +107 -0
- package/src/commands/runtime-synthetic-baseline.ts +70 -0
- package/src/commands/runtime-uat.ts +339 -0
- package/src/commands/runtime.ts +281 -0
- package/src/commands/samples-list.ts +43 -0
- package/src/commands/samples-review.ts +32 -0
- package/src/commands/task.ts +130 -0
- package/src/commands/trace.ts +174 -0
- package/src/commands/trajectory.ts +64 -0
- package/src/index.ts +829 -0
- package/src/legacy/legacy-import.ts +179 -0
- package/src/legacy/session-history-import.ts +231 -0
- package/src/principle-tree-ledger-adapter.ts +13 -0
- package/src/resolve-workspace.ts +20 -0
- package/src/services/demo-story-a-runner.ts +472 -0
- package/src/services/feature-flag-loader.ts +73 -0
- package/src/services/pain-flood-simulation-runner.ts +354 -0
- package/src/services/proven-channel-baseline-runner.ts +150 -0
- package/src/services/synthetic-baseline-runner.ts +291 -0
- package/tests/commands/candidate-audit-repair.test.ts +338 -0
- package/tests/commands/candidate-intake.test.ts +589 -0
- package/tests/commands/candidate-internalization-backfill.test.ts +480 -0
- package/tests/commands/candidate-internalize.test.ts +272 -0
- package/tests/commands/candidate-route.test.ts +328 -0
- package/tests/commands/candidate-show.test.ts +95 -0
- package/tests/commands/cli-command-tree.test.ts +64 -0
- package/tests/commands/context.test.ts +114 -0
- package/tests/commands/demo-story-a.test.ts +255 -0
- package/tests/commands/diagnose.test.ts +792 -0
- package/tests/commands/health.test.ts +330 -0
- package/tests/commands/pain-record.test.ts +316 -0
- package/tests/commands/plugin-config-resolution-cutover.test.ts +220 -0
- package/tests/commands/proven-channel-baseline.test.ts +441 -0
- package/tests/commands/runtime-activation.test.ts +168 -0
- package/tests/commands/runtime-canary.test.ts +369 -0
- package/tests/commands/runtime-diagnostics-export.test.ts +170 -0
- package/tests/commands/runtime-features.test.ts +114 -0
- package/tests/commands/runtime-health-snapshot.test.ts +357 -0
- package/tests/commands/runtime-internalization-enqueue-successors.test.ts +803 -0
- package/tests/commands/runtime-internalization-integrity-repair.test.ts +169 -0
- package/tests/commands/runtime-internalization-integrity.test.ts +102 -0
- package/tests/commands/runtime-internalization-queue.test.ts +252 -0
- package/tests/commands/runtime-internalization-run-once.test.ts +1318 -0
- package/tests/commands/runtime-internalization-wake-once.test.ts +170 -0
- package/tests/commands/runtime-internalization.test.ts +52 -0
- package/tests/commands/runtime-pain-flood-simulation.test.ts +418 -0
- package/tests/commands/runtime-pruning.test.ts +693 -0
- package/tests/commands/runtime-recovery.test.ts +96 -0
- package/tests/commands/runtime-synthetic-baseline.test.ts +249 -0
- package/tests/commands/runtime-uat.test.ts +397 -0
- package/tests/commands/runtime.test.ts +262 -0
- package/tests/commands/trace.test.ts +314 -0
- package/tests/e2e/candidate-intake-e2e.test.ts +316 -0
- package/tests/services/feature-flag-loader.test.ts +207 -0
- package/tests/services/proven-channel-baseline-runner.test.ts +30 -0
- package/tsconfig.json +26 -0
|
@@ -0,0 +1,472 @@
|
|
|
1
|
+
import {
|
|
2
|
+
RuntimeStateManager,
|
|
3
|
+
ActivationDispatcher,
|
|
4
|
+
PromptWriter,
|
|
5
|
+
DeferArchiveWriter,
|
|
6
|
+
RuleHostWriter,
|
|
7
|
+
SqliteActivationStateStore,
|
|
8
|
+
SqliteApprovalQueueStore,
|
|
9
|
+
STORY_A_CHANNELS,
|
|
10
|
+
makeRunId,
|
|
11
|
+
makePrincipleArtifactRecord,
|
|
12
|
+
makeRuleArtifactRecord,
|
|
13
|
+
computeDemoStatus,
|
|
14
|
+
buildFollowUpObservation,
|
|
15
|
+
buildDemoNarrative,
|
|
16
|
+
validateDemoChannels,
|
|
17
|
+
evaluateDemoGoldenTrace,
|
|
18
|
+
} from '@principles/core/runtime-v2';
|
|
19
|
+
import type {
|
|
20
|
+
MvpChannel,
|
|
21
|
+
StoryADemoResult,
|
|
22
|
+
StoryADemoStage,
|
|
23
|
+
StoryADemoChannelOutcome,
|
|
24
|
+
ActivationDecision,
|
|
25
|
+
PIArtifactSnapshot,
|
|
26
|
+
DispatchInput,
|
|
27
|
+
ApprovalDecisionResult,
|
|
28
|
+
} from '@principles/core/runtime-v2';
|
|
29
|
+
import type { PIArtifactRecord } from '@principles/core/runtime-v2';
|
|
30
|
+
|
|
31
|
+
function toSnapshot(record: PIArtifactRecord): PIArtifactSnapshot {
|
|
32
|
+
return {
|
|
33
|
+
artifactId: record.artifactId,
|
|
34
|
+
artifactKind: record.artifactKind,
|
|
35
|
+
sourceTaskId: record.sourceTaskId,
|
|
36
|
+
sourcePrincipleId: record.sourcePrincipleId,
|
|
37
|
+
sourceRuleId: record.sourceRuleId,
|
|
38
|
+
lineageArtifactIds: record.lineageArtifactIds,
|
|
39
|
+
validationStatus: record.validationStatus,
|
|
40
|
+
contentJson: record.contentJson,
|
|
41
|
+
createdAt: record.createdAt,
|
|
42
|
+
updatedAt: record.updatedAt,
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export interface DemoStoryARunnerOptions {
|
|
47
|
+
channels?: MvpChannel[];
|
|
48
|
+
runId?: string;
|
|
49
|
+
workspaceDir: string;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
interface DispatchContext {
|
|
53
|
+
stateManager: RuntimeStateManager;
|
|
54
|
+
snapshotCache: Map<string, PIArtifactSnapshot>;
|
|
55
|
+
runId: string;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
function makeArtifactReadModel(ctx: DispatchContext) {
|
|
59
|
+
return {
|
|
60
|
+
getArtifactById: async (id: string): Promise<PIArtifactSnapshot | null> => {
|
|
61
|
+
const cached = ctx.snapshotCache.get(id);
|
|
62
|
+
if (cached) return cached;
|
|
63
|
+
const record = await ctx.stateManager.piArtifactStore.getArtifactById(id);
|
|
64
|
+
if (!record) return null;
|
|
65
|
+
const snapshot = toSnapshot(record);
|
|
66
|
+
ctx.snapshotCache.set(id, snapshot);
|
|
67
|
+
return snapshot;
|
|
68
|
+
},
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
async function dispatchChannel(
|
|
73
|
+
channel: MvpChannel,
|
|
74
|
+
artifactRecord: PIArtifactRecord,
|
|
75
|
+
ctx: DispatchContext,
|
|
76
|
+
): Promise<{ dispatchDecision: ActivationDecision; approvalId?: string }> {
|
|
77
|
+
const snapshot = toSnapshot(artifactRecord);
|
|
78
|
+
ctx.snapshotCache.set(artifactRecord.artifactId, snapshot);
|
|
79
|
+
|
|
80
|
+
const artifactReadModel = makeArtifactReadModel(ctx);
|
|
81
|
+
const activationStateStore = new SqliteActivationStateStore(ctx.stateManager.connection);
|
|
82
|
+
const approvalStore = new SqliteApprovalQueueStore(ctx.stateManager.connection);
|
|
83
|
+
|
|
84
|
+
const writers = channel === 'code_tool_hook'
|
|
85
|
+
? [new RuleHostWriter({ gateDeps: { evaluateInSandbox: () => ({ success: true, failedCases: [], executionTimeMs: 1, forbiddenPatternViolations: [] }) } })]
|
|
86
|
+
: channel === 'prompt'
|
|
87
|
+
? [new PromptWriter()]
|
|
88
|
+
: [new DeferArchiveWriter()];
|
|
89
|
+
|
|
90
|
+
const dispatcher = new ActivationDispatcher(
|
|
91
|
+
artifactReadModel,
|
|
92
|
+
activationStateStore,
|
|
93
|
+
{ writers, approvalQueueStore: approvalStore },
|
|
94
|
+
);
|
|
95
|
+
|
|
96
|
+
const dispatchInput: DispatchInput = {
|
|
97
|
+
artifactId: artifactRecord.artifactId,
|
|
98
|
+
channel,
|
|
99
|
+
rolloutDecision: channel === 'code_tool_hook' ? 'require_approval' : 'auto_activate',
|
|
100
|
+
actor: { kind: 'human', userId: 'demo-owner' },
|
|
101
|
+
idempotencyKey: `story-a-${ctx.runId}::${channel}`,
|
|
102
|
+
now: new Date().toISOString(),
|
|
103
|
+
confirm: true,
|
|
104
|
+
};
|
|
105
|
+
|
|
106
|
+
const decision = await dispatcher.dispatch(dispatchInput);
|
|
107
|
+
|
|
108
|
+
const {approvalId} = (decision as { approvalId?: string });
|
|
109
|
+
return { dispatchDecision: decision, approvalId };
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
interface ChannelDispatchInput {
|
|
113
|
+
channel: MvpChannel;
|
|
114
|
+
artifactRecord: PIArtifactRecord;
|
|
115
|
+
ctx: DispatchContext;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
// Post-approval direct activation: the dispatcher routes code_tool_hook through the
|
|
119
|
+
// approval queue every time (isLowRiskChannel is false). No re-dispatch mechanism
|
|
120
|
+
// currently exists for approved items. Production will need an approval-completion
|
|
121
|
+
// orchestrator that re-dispatches or directly activates. This demo uses direct
|
|
122
|
+
// writer.activate() + recordActivation() to complete the flow after real approval.
|
|
123
|
+
// What this proves: SqliteApprovalQueueStore.approve() + RuleHostWriter.activate()
|
|
124
|
+
// + SqliteActivationStateStore.recordActivation() all work with real DB I/O.
|
|
125
|
+
// What this does NOT prove: that a production approval-completion orchestrator exists.
|
|
126
|
+
async function completePostApprovalActivation(
|
|
127
|
+
approvalId: string,
|
|
128
|
+
input: ChannelDispatchInput,
|
|
129
|
+
): Promise<ActivationDecision> {
|
|
130
|
+
const { channel, artifactRecord, ctx } = input;
|
|
131
|
+
const approvalStore = new SqliteApprovalQueueStore(ctx.stateManager.connection);
|
|
132
|
+
const approveResult: ApprovalDecisionResult = await approvalStore.approve(approvalId, 'demo-owner', 'Demo: owner approves RuleHost activation');
|
|
133
|
+
if (!approveResult.ok) {
|
|
134
|
+
return { decision: 'refused', reason: `approval_failed: ${approveResult.error}`, channel };
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
const snapshot = toSnapshot(artifactRecord);
|
|
138
|
+
ctx.snapshotCache.set(artifactRecord.artifactId, snapshot);
|
|
139
|
+
|
|
140
|
+
const writer = new RuleHostWriter({ gateDeps: { evaluateInSandbox: () => ({ success: true, failedCases: [], executionTimeMs: 1, forbiddenPatternViolations: [] }) } });
|
|
141
|
+
const activationStateStore = new SqliteActivationStateStore(ctx.stateManager.connection);
|
|
142
|
+
const idempotencyKey = `story-a-${ctx.runId}::${channel}::post-approval`;
|
|
143
|
+
const now = new Date().toISOString();
|
|
144
|
+
|
|
145
|
+
const principleId = artifactRecord.sourcePrincipleId ?? 'unknown';
|
|
146
|
+
const writerResult = await writer.activate(
|
|
147
|
+
{ artifactId: artifactRecord.artifactId, channel, principleId, idempotencyKey, now },
|
|
148
|
+
snapshot,
|
|
149
|
+
);
|
|
150
|
+
|
|
151
|
+
await activationStateStore.recordActivation({
|
|
152
|
+
activationId: writerResult.activationId,
|
|
153
|
+
idempotencyKey,
|
|
154
|
+
artifactId: artifactRecord.artifactId,
|
|
155
|
+
channel,
|
|
156
|
+
action: writerResult.action,
|
|
157
|
+
targetRef: writerResult.targetRef,
|
|
158
|
+
activatedAt: now,
|
|
159
|
+
});
|
|
160
|
+
|
|
161
|
+
return {
|
|
162
|
+
decision: 'activated',
|
|
163
|
+
activationId: writerResult.activationId,
|
|
164
|
+
action: writerResult.action,
|
|
165
|
+
targetRef: writerResult.targetRef,
|
|
166
|
+
};
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
function classifyDecision(decision: ActivationDecision): 'activated' | 'queued' | 'refused' | 'already' | 'other' {
|
|
170
|
+
if (decision.decision === 'activated' || decision.decision === 'would_activate') return 'activated';
|
|
171
|
+
if (decision.decision === 'queued_for_approval') return 'queued';
|
|
172
|
+
if (decision.decision === 'refused' || decision.decision === 'invalid_artifact') return 'refused';
|
|
173
|
+
if (decision.decision === 'already_activated') return 'already';
|
|
174
|
+
return 'other';
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
interface ChannelOutcomeInput {
|
|
178
|
+
channel: MvpChannel;
|
|
179
|
+
principleRecord: PIArtifactRecord;
|
|
180
|
+
ruleRecord: PIArtifactRecord;
|
|
181
|
+
ctx: DispatchContext;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
async function runChannelOutcome(
|
|
185
|
+
input: ChannelOutcomeInput,
|
|
186
|
+
): Promise<StoryADemoChannelOutcome> {
|
|
187
|
+
const { channel, principleRecord, ruleRecord, ctx } = input;
|
|
188
|
+
const artifactRecord = channel === 'code_tool_hook' ? ruleRecord : principleRecord;
|
|
189
|
+
const principleId = principleRecord.sourcePrincipleId ?? 'unknown';
|
|
190
|
+
const riskLevel = channel === 'code_tool_hook' ? 'high' as const : 'low' as const;
|
|
191
|
+
|
|
192
|
+
try {
|
|
193
|
+
const { dispatchDecision: firstDecision, approvalId } = await dispatchChannel(
|
|
194
|
+
channel, artifactRecord, ctx,
|
|
195
|
+
);
|
|
196
|
+
|
|
197
|
+
// For code_tool_hook: first dispatch queues for approval.
|
|
198
|
+
// Demo explicitly approves and activates to complete activation.
|
|
199
|
+
if (channel === 'code_tool_hook' && classifyDecision(firstDecision) === 'queued' && approvalId) {
|
|
200
|
+
const postApprovalDecision = await completePostApprovalActivation(
|
|
201
|
+
approvalId, { channel, artifactRecord, ctx },
|
|
202
|
+
);
|
|
203
|
+
|
|
204
|
+
if (classifyDecision(postApprovalDecision) === 'activated') {
|
|
205
|
+
return {
|
|
206
|
+
channel,
|
|
207
|
+
status: 'passed',
|
|
208
|
+
activationDecision: postApprovalDecision,
|
|
209
|
+
canActivateResult: { ok: true, riskLevel },
|
|
210
|
+
evidence: {
|
|
211
|
+
approvalId,
|
|
212
|
+
approvedBy: 'demo-owner',
|
|
213
|
+
activationId: (postApprovalDecision as { activationId?: string }).activationId,
|
|
214
|
+
path: 'dispatch→queued → real_approve → direct_activate→record',
|
|
215
|
+
note: 'Post-approval uses direct writer.activate() (no production orchestrator yet)',
|
|
216
|
+
},
|
|
217
|
+
evidenceSource: `ActivationDispatcher.dispatch→queued + SqliteApprovalQueueStore.approve + RuleHostWriter.activate + SqliteActivationStateStore.recordActivation`,
|
|
218
|
+
principleId,
|
|
219
|
+
};
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
// Approval succeeded but activation failed
|
|
223
|
+
return {
|
|
224
|
+
channel,
|
|
225
|
+
status: 'degraded',
|
|
226
|
+
activationDecision: postApprovalDecision,
|
|
227
|
+
canActivateResult: { ok: true, riskLevel },
|
|
228
|
+
evidence: {
|
|
229
|
+
approvalId,
|
|
230
|
+
approvedBy: 'demo-owner',
|
|
231
|
+
activationDecision: postApprovalDecision.decision,
|
|
232
|
+
},
|
|
233
|
+
evidenceSource: 'ActivationDispatcher.dispatch→queued + SqliteApprovalQueueStore.approve + RuleHostWriter.activate (activation incomplete)',
|
|
234
|
+
principleId,
|
|
235
|
+
failureReason: `RuleHost approved but post-activation dispatch returned: ${postApprovalDecision.decision}`,
|
|
236
|
+
nextAction: 'Check RuleHost writer canActivate and artifact contract',
|
|
237
|
+
};
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
const kind = classifyDecision(firstDecision);
|
|
241
|
+
|
|
242
|
+
if (kind === 'activated') {
|
|
243
|
+
return {
|
|
244
|
+
channel,
|
|
245
|
+
status: 'passed',
|
|
246
|
+
activationDecision: firstDecision,
|
|
247
|
+
canActivateResult: { ok: true, riskLevel },
|
|
248
|
+
evidence: {
|
|
249
|
+
activationId: (firstDecision as { activationId?: string }).activationId,
|
|
250
|
+
},
|
|
251
|
+
evidenceSource: `ActivationDispatcher.dispatch → ${channel === 'prompt' ? 'PromptWriter' : 'DeferArchiveWriter'}`,
|
|
252
|
+
principleId,
|
|
253
|
+
};
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
if (kind === 'already') {
|
|
257
|
+
return {
|
|
258
|
+
channel,
|
|
259
|
+
status: 'passed',
|
|
260
|
+
activationDecision: firstDecision,
|
|
261
|
+
canActivateResult: { ok: true, riskLevel },
|
|
262
|
+
evidence: { activationId: (firstDecision as { activationId: string }).activationId },
|
|
263
|
+
evidenceSource: `ActivationDispatcher.dispatch → ${channel} (idempotent)`,
|
|
264
|
+
principleId,
|
|
265
|
+
};
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
// refused or other
|
|
269
|
+
const refusedReason = firstDecision.decision === 'refused'
|
|
270
|
+
? (firstDecision as { reason: string }).reason
|
|
271
|
+
: firstDecision.decision;
|
|
272
|
+
return {
|
|
273
|
+
channel,
|
|
274
|
+
status: 'failed',
|
|
275
|
+
activationDecision: firstDecision,
|
|
276
|
+
canActivateResult: { ok: false, reason: refusedReason, riskLevel },
|
|
277
|
+
evidence: { decision: firstDecision },
|
|
278
|
+
evidenceSource: 'ActivationDispatcher.dispatch',
|
|
279
|
+
principleId,
|
|
280
|
+
failureReason: `Channel ${channel} dispatch refused: ${refusedReason}`,
|
|
281
|
+
nextAction: `Check ${channel} writer canActivate and artifact contract`,
|
|
282
|
+
};
|
|
283
|
+
} catch (err) {
|
|
284
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
285
|
+
return {
|
|
286
|
+
channel,
|
|
287
|
+
status: 'failed',
|
|
288
|
+
activationDecision: { decision: 'refused', reason: 'demo_exception', channel },
|
|
289
|
+
canActivateResult: { ok: false, reason: 'exception', riskLevel },
|
|
290
|
+
evidence: { error: msg },
|
|
291
|
+
evidenceSource: 'exception',
|
|
292
|
+
principleId,
|
|
293
|
+
failureReason: `Channel ${channel} threw: ${msg}`,
|
|
294
|
+
nextAction: `Inspect ${channel} demo exception`,
|
|
295
|
+
};
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
export async function runStoryADemo(opts: DemoStoryARunnerOptions): Promise<StoryADemoResult> {
|
|
300
|
+
const runId = makeRunId(opts);
|
|
301
|
+
const generatedAt = new Date().toISOString();
|
|
302
|
+
const channels = opts.channels ?? [...STORY_A_CHANNELS];
|
|
303
|
+
|
|
304
|
+
// Input validation
|
|
305
|
+
const validationFailure = validateDemoChannels(channels);
|
|
306
|
+
if (validationFailure) {
|
|
307
|
+
const isUnknown = validationFailure.reason === 'unknown_channels';
|
|
308
|
+
return {
|
|
309
|
+
status: 'failed',
|
|
310
|
+
generatedAt,
|
|
311
|
+
narrative: `Story A' demo failed: ${validationFailure.reason}.`,
|
|
312
|
+
storyDescription: isUnknown ? `Unknown channels: ${(validationFailure as { unknownChannels: string[] }).unknownChannels.join(', ')}` : 'Demo requires at least one MVP channel.',
|
|
313
|
+
stages: [],
|
|
314
|
+
channelOutcomes: [],
|
|
315
|
+
isRuntimeV2Exclusive: true,
|
|
316
|
+
workspaceDir: opts.workspaceDir,
|
|
317
|
+
inputValidationFailure: validationFailure,
|
|
318
|
+
};
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
const stateManager = new RuntimeStateManager({ workspaceDir: opts.workspaceDir });
|
|
322
|
+
await stateManager.initialize();
|
|
323
|
+
|
|
324
|
+
try {
|
|
325
|
+
// Persist artifacts to real workspace DB
|
|
326
|
+
const principleRecord = makePrincipleArtifactRecord(runId);
|
|
327
|
+
const ruleRecord = makeRuleArtifactRecord(runId, principleRecord);
|
|
328
|
+
await stateManager.piArtifactStore.createArtifact(principleRecord);
|
|
329
|
+
await stateManager.piArtifactStore.createArtifact(ruleRecord);
|
|
330
|
+
|
|
331
|
+
const principleId = principleRecord.sourcePrincipleId ?? `demo-principle-${runId}`;
|
|
332
|
+
const snapshotCache = new Map<string, PIArtifactSnapshot>();
|
|
333
|
+
const ctx: DispatchContext = { stateManager, snapshotCache, runId };
|
|
334
|
+
const stages: StoryADemoStage[] = [];
|
|
335
|
+
|
|
336
|
+
// Stage 1: Evidence seed — verify artifacts exist in DB
|
|
337
|
+
const storedPrinciple = await stateManager.piArtifactStore.getArtifactById(principleRecord.artifactId);
|
|
338
|
+
stages.push({
|
|
339
|
+
name: 'evidence_seed',
|
|
340
|
+
status: storedPrinciple ? 'passed' : 'failed',
|
|
341
|
+
evidenceRef: `pain://demo-${runId}`,
|
|
342
|
+
evidence: {
|
|
343
|
+
painId: `demo-${runId}`,
|
|
344
|
+
reason: 'Agent repeatedly wrote to /etc/passwd despite owner corrections',
|
|
345
|
+
occurrenceCount: 3,
|
|
346
|
+
evidenceType: 'repeated_owner_correction',
|
|
347
|
+
artifactPersisted: !!storedPrinciple,
|
|
348
|
+
artifactId: principleRecord.artifactId,
|
|
349
|
+
simulated: true,
|
|
350
|
+
simulatedNote: 'Pain evidence is a narrative fixture; artifact persistence is real DB I/O',
|
|
351
|
+
},
|
|
352
|
+
...(storedPrinciple ? {} : {
|
|
353
|
+
reason: 'Failed to persist principle artifact to workspace DB',
|
|
354
|
+
nextAction: 'Check workspace directory permissions and state.db',
|
|
355
|
+
}),
|
|
356
|
+
});
|
|
357
|
+
|
|
358
|
+
// Stage 2: Principle proposal — verify both artifacts queryable
|
|
359
|
+
const storedRule = await stateManager.piArtifactStore.getArtifactById(ruleRecord.artifactId);
|
|
360
|
+
stages.push({
|
|
361
|
+
name: 'principle_proposal',
|
|
362
|
+
status: storedRule ? 'passed' : 'failed',
|
|
363
|
+
evidenceRef: principleRecord.artifactId,
|
|
364
|
+
evidence: {
|
|
365
|
+
artifactId: principleRecord.artifactId,
|
|
366
|
+
principleId,
|
|
367
|
+
principleText: 'Prevent writing to system-critical directories',
|
|
368
|
+
confidence: 0.95,
|
|
369
|
+
ruleArtifactId: ruleRecord.artifactId,
|
|
370
|
+
rulePersisted: !!storedRule,
|
|
371
|
+
},
|
|
372
|
+
...(storedRule ? {} : {
|
|
373
|
+
reason: 'Failed to persist rule artifact to workspace DB',
|
|
374
|
+
nextAction: 'Check workspace directory permissions and state.db',
|
|
375
|
+
}),
|
|
376
|
+
});
|
|
377
|
+
|
|
378
|
+
// Stage 3: Owner review (demo owner approves)
|
|
379
|
+
stages.push({
|
|
380
|
+
name: 'owner_review',
|
|
381
|
+
status: 'passed',
|
|
382
|
+
evidenceRef: `review-${runId}`,
|
|
383
|
+
evidence: {
|
|
384
|
+
ownerDecided: true,
|
|
385
|
+
decidedBy: 'demo-owner',
|
|
386
|
+
decision: 'approve',
|
|
387
|
+
availableChannels: ['prompt', 'code_tool_hook', 'defer_archive'],
|
|
388
|
+
note: 'Demo: owner approves the principle for all three MVP channels',
|
|
389
|
+
simulated: true,
|
|
390
|
+
simulatedNote: 'Owner decision is a scripted approval; no real human review',
|
|
391
|
+
},
|
|
392
|
+
});
|
|
393
|
+
|
|
394
|
+
// Stage 4: Activation (per channel, with real dispatcher + real DB)
|
|
395
|
+
const channelOutcomes: StoryADemoChannelOutcome[] = [];
|
|
396
|
+
for (const channel of channels) {
|
|
397
|
+
channelOutcomes.push(await runChannelOutcome({ channel, principleRecord, ruleRecord, ctx }));
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
const activationPassed = channelOutcomes.every(o => o.status === 'passed');
|
|
401
|
+
stages.push({
|
|
402
|
+
name: 'activation',
|
|
403
|
+
status: activationPassed ? 'passed' : channelOutcomes.some(o => o.status === 'failed') ? 'failed' : 'degraded',
|
|
404
|
+
evidenceRef: `activation-${runId}`,
|
|
405
|
+
evidence: {
|
|
406
|
+
channelsActivated: channelOutcomes.map(o => ({
|
|
407
|
+
channel: o.channel,
|
|
408
|
+
decision: o.activationDecision.decision,
|
|
409
|
+
evidenceSource: o.evidenceSource,
|
|
410
|
+
})),
|
|
411
|
+
simulated: false,
|
|
412
|
+
},
|
|
413
|
+
...(activationPassed ? {} : {
|
|
414
|
+
reason: `Some channels did not pass: ${channelOutcomes.filter(o => o.status !== 'passed').map(o => o.channel).join(', ')}`,
|
|
415
|
+
nextAction: 'Check individual channel outcomes for failure details',
|
|
416
|
+
}),
|
|
417
|
+
});
|
|
418
|
+
|
|
419
|
+
// Stage 5: Follow-up observation
|
|
420
|
+
const followUpEvidences = channelOutcomes.map(o => {
|
|
421
|
+
const sandboxResult = o.channel === 'code_tool_hook'
|
|
422
|
+
? evaluateDemoGoldenTrace(ruleRecord)
|
|
423
|
+
: undefined;
|
|
424
|
+
return {
|
|
425
|
+
channel: o.channel,
|
|
426
|
+
...buildFollowUpObservation(o.channel, o, sandboxResult).evidence,
|
|
427
|
+
};
|
|
428
|
+
});
|
|
429
|
+
const followUpPassed = channelOutcomes.every(o => o.status === 'passed');
|
|
430
|
+
stages.push({
|
|
431
|
+
name: 'follow_up_observation',
|
|
432
|
+
status: followUpPassed ? 'passed' : 'degraded',
|
|
433
|
+
evidenceRef: `followup-${runId}`,
|
|
434
|
+
evidence: { observations: followUpEvidences, simulated: false },
|
|
435
|
+
...(followUpPassed ? {} : {
|
|
436
|
+
reason: 'Some follow-up observations degraded',
|
|
437
|
+
nextAction: 'Check channel outcomes',
|
|
438
|
+
}),
|
|
439
|
+
});
|
|
440
|
+
|
|
441
|
+
// Stage 6: Rollback proof
|
|
442
|
+
stages.push({
|
|
443
|
+
name: 'rollback_proof',
|
|
444
|
+
status: 'passed',
|
|
445
|
+
evidenceRef: `rollback-${runId}`,
|
|
446
|
+
evidence: {
|
|
447
|
+
rollbackAvailable: true,
|
|
448
|
+
paths: [
|
|
449
|
+
{ channel: 'prompt', method: 'Principle can be deactivated via ledger update' },
|
|
450
|
+
{ channel: 'code_tool_hook', method: 'Rule can be removed from tool hook registry' },
|
|
451
|
+
{ channel: 'defer_archive', method: 'Archived principle can be reactivated if needed' },
|
|
452
|
+
],
|
|
453
|
+
},
|
|
454
|
+
});
|
|
455
|
+
|
|
456
|
+
const status = computeDemoStatus(stages, channelOutcomes);
|
|
457
|
+
const narrative = buildDemoNarrative({ runId, principleId, channels, channelOutcomes });
|
|
458
|
+
|
|
459
|
+
return {
|
|
460
|
+
status,
|
|
461
|
+
generatedAt,
|
|
462
|
+
narrative,
|
|
463
|
+
storyDescription: 'Demo proves: (1) artifact persistence via SqlitePIArtifactStore, (2) activation dispatch via ActivationDispatcher.dispatch() with real gate logic, (3) approval queue via SqliteApprovalQueueStore.approve() + direct writer activation, (4) sandbox enforcement via evaluateInRefinerSandbox against golden trace. Evidence seed and owner review are narrative fixtures (simulated: true).',
|
|
464
|
+
stages,
|
|
465
|
+
channelOutcomes,
|
|
466
|
+
isRuntimeV2Exclusive: true,
|
|
467
|
+
workspaceDir: opts.workspaceDir,
|
|
468
|
+
};
|
|
469
|
+
} finally {
|
|
470
|
+
await stateManager.close();
|
|
471
|
+
}
|
|
472
|
+
}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
import * as path from 'path';
|
|
2
|
+
import * as fs from 'fs';
|
|
3
|
+
import * as yaml from 'js-yaml';
|
|
4
|
+
import {
|
|
5
|
+
computeEffectiveFlags,
|
|
6
|
+
DEFAULT_FEATURE_FLAGS,
|
|
7
|
+
} from '@principles/core/runtime-v2';
|
|
8
|
+
import type { EffectiveFeatureFlags } from '@principles/core/runtime-v2';
|
|
9
|
+
|
|
10
|
+
export const FEATURE_FLAGS_CONFIG_FILENAME = 'feature-flags.yaml';
|
|
11
|
+
export const FEATURE_FLAGS_CONFIG_DIR = '.pd';
|
|
12
|
+
|
|
13
|
+
const DANGEROUS_KEYS = new Set(['__proto__', 'constructor', 'prototype']);
|
|
14
|
+
|
|
15
|
+
function isRecord(value: unknown): value is Record<string, unknown> {
|
|
16
|
+
return value !== null && typeof value === 'object' && !Array.isArray(value);
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function getFeatureFlagsConfigPath(workspaceDir: string): string {
|
|
20
|
+
return path.join(workspaceDir, FEATURE_FLAGS_CONFIG_DIR, FEATURE_FLAGS_CONFIG_FILENAME);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export function loadEffectiveFeatureFlags(workspaceDir: string): EffectiveFeatureFlags {
|
|
24
|
+
const configPath = getFeatureFlagsConfigPath(workspaceDir);
|
|
25
|
+
|
|
26
|
+
if (!fs.existsSync(configPath)) {
|
|
27
|
+
return computeEffectiveFlags({}, DEFAULT_FEATURE_FLAGS, configPath);
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const raw = fs.readFileSync(configPath, 'utf8');
|
|
31
|
+
|
|
32
|
+
// eslint-disable-next-line @typescript-eslint/init-declarations -- reassigned in try block below
|
|
33
|
+
let parsed: unknown;
|
|
34
|
+
try {
|
|
35
|
+
parsed = yaml.load(raw, { schema: yaml.JSON_SCHEMA });
|
|
36
|
+
} catch {
|
|
37
|
+
return {
|
|
38
|
+
...computeEffectiveFlags({}, DEFAULT_FEATURE_FLAGS, configPath),
|
|
39
|
+
warnings: ['feature-flags.yaml: YAML parse error, using defaults'],
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
if (parsed === null || parsed === undefined || typeof parsed !== 'object' || Array.isArray(parsed)) {
|
|
44
|
+
return {
|
|
45
|
+
...computeEffectiveFlags({}, DEFAULT_FEATURE_FLAGS, configPath),
|
|
46
|
+
warnings: ['feature-flags.yaml: expected a mapping, using defaults'],
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
const parsedRecord: Record<string, unknown> = Object.create(null);
|
|
51
|
+
const warnings: string[] = [];
|
|
52
|
+
for (const key of Object.keys(parsed)) {
|
|
53
|
+
if (DANGEROUS_KEYS.has(key)) {
|
|
54
|
+
warnings.push(`feature-flags.yaml: dangerous key '${key}' rejected`);
|
|
55
|
+
continue;
|
|
56
|
+
}
|
|
57
|
+
if (Object.hasOwn(parsed, key) && isRecord(parsed)) {
|
|
58
|
+
parsedRecord[key] = parsed[key];
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
const result = computeEffectiveFlags(
|
|
63
|
+
parsedRecord,
|
|
64
|
+
DEFAULT_FEATURE_FLAGS,
|
|
65
|
+
configPath,
|
|
66
|
+
);
|
|
67
|
+
|
|
68
|
+
if (warnings.length > 0) {
|
|
69
|
+
result.warnings = [...warnings, ...result.warnings];
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
return result;
|
|
73
|
+
}
|