@nimiplatform/nimi-coding 0.1.0 → 0.2.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 +19 -20
- package/adapters/oh-my-codex/README.md +8 -9
- package/cli/commands/audit-sweep.mjs +10 -10
- package/cli/commands/classify-spec-tree.mjs +5 -0
- package/cli/commands/closeout.mjs +3 -0
- package/cli/commands/generate-spec-derived-docs.mjs +20 -0
- package/cli/commands/generate-spec-migration-plan.mjs +30 -0
- package/cli/commands/start.mjs +5 -1
- package/cli/commands/surface-validator-command.mjs +49 -0
- package/cli/commands/sweep-design.mjs +295 -0
- package/cli/commands/sweep.mjs +22 -0
- package/cli/commands/sync.mjs +132 -0
- package/cli/commands/topic-formatters.mjs +8 -8
- package/cli/commands/validate-ai-governance.mjs +167 -46
- package/cli/commands/validate-domain-admission.mjs +5 -0
- package/cli/commands/validate-guidance-bodies.mjs +5 -0
- package/cli/commands/validate-placement.mjs +5 -0
- package/cli/commands/validate-projection-edges.mjs +5 -0
- package/cli/commands/validate-spec-audit.mjs +5 -1
- package/cli/commands/validate-table-family.mjs +5 -0
- package/cli/commands/validate-tracked-output-admission.mjs +5 -0
- package/cli/constants.mjs +5 -49
- package/cli/help.mjs +33 -11
- package/cli/index.mjs +20 -2
- package/cli/lib/audit-sweep-runtime/admissions.mjs +38 -29
- package/cli/lib/audit-sweep-runtime/audit-validity.mjs +8 -0
- package/cli/lib/audit-sweep-runtime/chunks.mjs +11 -11
- package/cli/lib/audit-sweep-runtime/closeout.mjs +8 -8
- package/cli/lib/audit-sweep-runtime/codex-auditor-evidence.mjs +3 -3
- package/cli/lib/audit-sweep-runtime/codex-auditor.mjs +10 -10
- package/cli/lib/audit-sweep-runtime/common.mjs +7 -7
- package/cli/lib/audit-sweep-runtime/format.mjs +3 -3
- package/cli/lib/audit-sweep-runtime/ingest.mjs +8 -8
- package/cli/lib/audit-sweep-runtime/inventory-spec-chunks.mjs +24 -27
- package/cli/lib/audit-sweep-runtime/inventory.mjs +58 -18
- package/cli/lib/audit-sweep-runtime/ledger.mjs +1 -1
- package/cli/lib/audit-sweep-runtime/p0p1-profile.mjs +2 -2
- package/cli/lib/audit-sweep-runtime/remediation.mjs +6 -6
- package/cli/lib/audit-sweep-runtime/rerun.mjs +6 -6
- package/cli/lib/audit-sweep-runtime/status.mjs +1 -1
- package/cli/lib/audit-sweep-runtime/validators.mjs +2 -2
- package/cli/lib/authority-convergence.mjs +397 -2
- package/cli/lib/blueprint-audit.mjs +5 -5
- package/cli/lib/closeout.mjs +126 -3
- package/cli/lib/contracts.mjs +21 -17
- package/cli/lib/handoff.mjs +29 -11
- package/cli/lib/high-risk-admission.mjs +60 -11
- package/cli/lib/high-risk-decision.mjs +31 -2
- package/cli/lib/high-risk-ingest.mjs +5 -1
- package/cli/lib/high-risk-review.mjs +5 -1
- package/cli/lib/internal/contracts-parse.mjs +195 -24
- package/cli/lib/internal/contracts-validators.mjs +3 -2
- package/cli/lib/internal/doctor-bootstrap-surface.mjs +82 -35
- package/cli/lib/internal/doctor-delegated-surface.mjs +1 -1
- package/cli/lib/internal/doctor-finalize.mjs +12 -8
- package/cli/lib/internal/doctor-inspectors.mjs +34 -1
- package/cli/lib/internal/governance/ai/ai-context-budget-core.mjs +74 -12
- package/cli/lib/internal/governance/ai/ai-structure-budget-core.mjs +24 -6
- package/cli/lib/internal/governance/ai/check-agents-freshness.mjs +18 -23
- package/cli/lib/internal/surface-taxonomy-validators.mjs +931 -0
- package/cli/lib/internal/validators-spec.mjs +229 -20
- package/cli/lib/sweep-design-runtime/common.mjs +246 -0
- package/cli/lib/sweep-design-runtime/engine.mjs +733 -0
- package/cli/lib/sweep-design-runtime/fix-topic.mjs +414 -0
- package/cli/lib/sweep-design-runtime/lifecycle.mjs +54 -0
- package/cli/lib/sweep-design-runtime/results.mjs +324 -0
- package/cli/lib/sweep-design.mjs +8 -0
- package/cli/lib/sync.mjs +143 -0
- package/cli/lib/topic-artifacts.mjs +186 -0
- package/cli/lib/topic-authority-coverage.mjs +73 -0
- package/cli/lib/topic-closeout.mjs +560 -0
- package/cli/lib/topic-common.mjs +404 -0
- package/cli/lib/topic-decisions.mjs +332 -0
- package/cli/lib/topic-draft-packets.mjs +126 -7
- package/cli/lib/topic-execution.mjs +515 -0
- package/cli/lib/topic-goal.mjs +112 -33
- package/cli/lib/topic-ledger.mjs +281 -0
- package/cli/lib/topic-lifecycle-artifacts.mjs +173 -0
- package/cli/lib/topic-root-validation.mjs +288 -0
- package/cli/lib/topic-runner-commands.mjs +174 -0
- package/cli/lib/topic-runner-deferral.mjs +532 -0
- package/cli/lib/topic-runner-stale-gates.mjs +114 -0
- package/cli/lib/topic-runner-validation.mjs +138 -0
- package/cli/lib/topic-runner.mjs +109 -154
- package/cli/lib/topic-scaffold.mjs +252 -0
- package/cli/lib/topic-waves.mjs +403 -0
- package/cli/lib/topic.mjs +81 -93
- package/cli/lib/value-helpers.mjs +6 -1
- package/cli/seeds/bootstrap.mjs +96 -20
- package/cli/seeds/seed-policy.yaml +67 -0
- package/config/bootstrap.yaml +1 -1
- package/config/skill-manifest.yaml +4 -2
- package/config/spec-generation-inputs.yaml +41 -19
- package/contracts/audit-remediation-map.schema.yaml +1 -0
- package/contracts/audit-sweep-result.yaml +4 -0
- package/contracts/domain-admission.schema.yaml +56 -0
- package/contracts/migration-inventory.schema.yaml +80 -0
- package/contracts/negative-fixtures.yaml +91 -0
- package/contracts/placement-contract.schema.yaml +163 -0
- package/contracts/projection-edge.schema.yaml +130 -0
- package/contracts/shared-enums.yaml +68 -0
- package/contracts/spec-generation-audit.schema.yaml +19 -4
- package/contracts/spec-generation-inputs.schema.yaml +130 -29
- package/contracts/spec-reconstruction-result.yaml +9 -5
- package/contracts/surface-taxonomy.schema.yaml +201 -0
- package/contracts/sweep-design-result.yaml +349 -0
- package/contracts/table-family.schema.yaml +114 -0
- package/contracts/topic-goal.schema.yaml +10 -1
- package/contracts/tracked-output-admission.schema.yaml +70 -0
- package/contracts/workflow-consumer.schema.yaml +112 -0
- package/methodology/audit-sweep-p0p1-recall.yaml +1 -1
- package/methodology/spec-reconstruction.yaml +53 -30
- package/package.json +5 -4
- package/spec/_meta/command-gating-matrix.yaml +33 -0
- package/spec/_meta/generate-drift-migration-checklist.yaml +44 -62
- package/spec/_meta/governance-routing-cutover-checklist.yaml +3 -3
- package/spec/_meta/phase2-impacted-surface-matrix.yaml +14 -14
- package/spec/_meta/spec-authority-cutover-readiness.yaml +3 -5
- package/spec/_meta/spec-tree-model.yaml +104 -36
- package/spec/bootstrap-state.yaml +36 -36
- package/spec/product-scope.yaml +13 -10
|
@@ -0,0 +1,332 @@
|
|
|
1
|
+
import {
|
|
2
|
+
buildDispatchPrompt,
|
|
3
|
+
buildPostUpdateReviewDecision,
|
|
4
|
+
buildPreImplementationDecision,
|
|
5
|
+
loadAuthorityConvergencePolicy,
|
|
6
|
+
} from "./authority-convergence.mjs";
|
|
7
|
+
import path from "node:path";
|
|
8
|
+
|
|
9
|
+
import { findUniqueFreezableDraftPacket } from "./topic-draft-packets.mjs";
|
|
10
|
+
import { loadTopicRuntimeAuthority, toPortableRelativePath } from "./topic-common.mjs";
|
|
11
|
+
import { findDeterministicNextWave, getTopicWaves, loadTopicReport } from "./topic-scaffold.mjs";
|
|
12
|
+
import { validateTopicGraph, validateWaveAdmission } from "./topic-waves.mjs";
|
|
13
|
+
import { listWavePackets, listWaveResults } from "./topic-artifacts.mjs";
|
|
14
|
+
|
|
15
|
+
export function buildTopicStepDecision(topic, wave, values) {
|
|
16
|
+
const waveId = wave?.wave_id ?? null;
|
|
17
|
+
return {
|
|
18
|
+
decision_id: `${topic.topic_id}:${waveId ?? "topic"}:${values.reasonCode}`,
|
|
19
|
+
topic_id: topic.topic_id,
|
|
20
|
+
wave_id: waveId,
|
|
21
|
+
decision_kind: "topic_next_step",
|
|
22
|
+
stop_class: values.stopClass,
|
|
23
|
+
recommended_action: values.recommendedAction,
|
|
24
|
+
reason_code: values.reasonCode,
|
|
25
|
+
requires_human_confirmation: values.stopClass === "require_human_confirmation",
|
|
26
|
+
recommended_decision: values.recommendedDecision,
|
|
27
|
+
recommendation_rationale: values.recommendationRationale,
|
|
28
|
+
expected_artifacts: values.expectedArtifacts ?? [],
|
|
29
|
+
next_command_ref: values.nextCommandRef ?? null,
|
|
30
|
+
blocking_checks: values.blockingChecks ?? [],
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
export function commandRef(parts) {
|
|
34
|
+
return ["nimicoding", "topic", ...parts].join(" ");
|
|
35
|
+
}
|
|
36
|
+
export async function buildDecisionForSelectedWave(projectRoot, loaded, graphReport, wave) {
|
|
37
|
+
const failedGraphChecks = (graphReport.checks ?? []).filter((entry) => !entry.ok);
|
|
38
|
+
if (!graphReport.ok)
|
|
39
|
+
return buildTopicStepDecision(loaded.topic, wave, {
|
|
40
|
+
stopClass: "blocked",
|
|
41
|
+
recommendedAction: "no_action",
|
|
42
|
+
reasonCode: "topic_graph_validation_failed",
|
|
43
|
+
recommendedDecision: "fix_topic_graph_before_continuing",
|
|
44
|
+
recommendationRationale:
|
|
45
|
+
"The wave graph is the dispatch authority for topic execution and must validate before any next step is emitted.",
|
|
46
|
+
blockingChecks: failedGraphChecks,
|
|
47
|
+
});
|
|
48
|
+
if (["closed", "retired", "superseded"].includes(wave.state))
|
|
49
|
+
return buildTopicStepDecision(loaded.topic, wave, {
|
|
50
|
+
stopClass: "blocked",
|
|
51
|
+
recommendedAction: "no_action",
|
|
52
|
+
reasonCode: "selected_wave_is_terminal",
|
|
53
|
+
recommendedDecision: "select_a_non_terminal_wave_or_close_the_topic",
|
|
54
|
+
recommendationRationale: `Selected wave ${wave.wave_id} is ${wave.state} and cannot be dispatched.`,
|
|
55
|
+
});
|
|
56
|
+
if (wave.state === "needs_revision")
|
|
57
|
+
return buildTopicStepDecision(loaded.topic, wave, {
|
|
58
|
+
stopClass: "blocked",
|
|
59
|
+
recommendedAction: "open_remediation",
|
|
60
|
+
reasonCode: "revise",
|
|
61
|
+
recommendedDecision: "fix",
|
|
62
|
+
recommendationRationale: "Block.",
|
|
63
|
+
});
|
|
64
|
+
if (["candidate", "preflight_draft"].includes(wave.state)) {
|
|
65
|
+
const admission = await validateWaveAdmission(projectRoot, loaded.topicId, wave.wave_id);
|
|
66
|
+
return admission.ok
|
|
67
|
+
? buildTopicStepDecision(loaded.topic, wave, {
|
|
68
|
+
stopClass: "continue",
|
|
69
|
+
recommendedAction: "admit_wave",
|
|
70
|
+
reasonCode: "wave_admission_ready",
|
|
71
|
+
recommendedDecision: "admit_wave",
|
|
72
|
+
recommendationRationale: "Admission is mechanical.",
|
|
73
|
+
nextCommandRef: commandRef(["wave", "admit", loaded.topicId, wave.wave_id]),
|
|
74
|
+
})
|
|
75
|
+
: buildTopicStepDecision(loaded.topic, wave, {
|
|
76
|
+
stopClass: "blocked",
|
|
77
|
+
recommendedAction: "admit_wave",
|
|
78
|
+
reasonCode: "wave_admission_validation_failed",
|
|
79
|
+
recommendedDecision: "repair_admission_blockers",
|
|
80
|
+
recommendationRationale:
|
|
81
|
+
"The selected wave cannot be admitted until its admission checks pass.",
|
|
82
|
+
nextCommandRef: commandRef([
|
|
83
|
+
"validate",
|
|
84
|
+
"admission",
|
|
85
|
+
loaded.topicId,
|
|
86
|
+
wave.wave_id,
|
|
87
|
+
"--json",
|
|
88
|
+
]),
|
|
89
|
+
blockingChecks: (admission.checks ?? []).filter((entry) => !entry.ok),
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
if (
|
|
93
|
+
["preflight_admitted", "implementation_admitted", "continuation_packet_open"].includes(
|
|
94
|
+
wave.state,
|
|
95
|
+
)
|
|
96
|
+
)
|
|
97
|
+
return buildTopicStepDecision(
|
|
98
|
+
loaded.topic,
|
|
99
|
+
wave,
|
|
100
|
+
await buildPreImplementationDecision({
|
|
101
|
+
projectRoot,
|
|
102
|
+
loaded,
|
|
103
|
+
wave,
|
|
104
|
+
commandRef,
|
|
105
|
+
listWavePackets,
|
|
106
|
+
listWaveResults,
|
|
107
|
+
findUniqueFreezableDraftPacket,
|
|
108
|
+
loadTopicRuntimeAuthority,
|
|
109
|
+
}),
|
|
110
|
+
);
|
|
111
|
+
if (wave.state === "implementation_active") {
|
|
112
|
+
const waveResults = await listWaveResults(loaded.topicDir, wave.wave_id),
|
|
113
|
+
implementationResult = waveResults.find(
|
|
114
|
+
(entry) => entry.result?.result_kind === "implementation",
|
|
115
|
+
);
|
|
116
|
+
if (!implementationResult)
|
|
117
|
+
return buildTopicStepDecision(loaded.topic, wave, {
|
|
118
|
+
stopClass: "await_external_evidence",
|
|
119
|
+
recommendedAction: "record_result",
|
|
120
|
+
reasonCode: "awaiting_implementation_result",
|
|
121
|
+
recommendedDecision: "ingest_the_implementation_result_when_available",
|
|
122
|
+
recommendationRationale:
|
|
123
|
+
"The wave is active, but closeout requires explicit implementation result lineage.",
|
|
124
|
+
expectedArtifacts: ["result-<wave-id>-implementation.md"],
|
|
125
|
+
nextCommandRef: commandRef([
|
|
126
|
+
"result",
|
|
127
|
+
"record",
|
|
128
|
+
loaded.topicId,
|
|
129
|
+
"--kind",
|
|
130
|
+
"implementation",
|
|
131
|
+
"--verdict",
|
|
132
|
+
"<verdict>",
|
|
133
|
+
"--from",
|
|
134
|
+
"<path>",
|
|
135
|
+
"--verified-at",
|
|
136
|
+
"<utc>",
|
|
137
|
+
]),
|
|
138
|
+
});
|
|
139
|
+
const postUpdateReviewDecision = await buildPostUpdateReviewDecision({
|
|
140
|
+
projectRoot,
|
|
141
|
+
topicDir: loaded.topicDir,
|
|
142
|
+
topicId: loaded.topicId,
|
|
143
|
+
wave,
|
|
144
|
+
packets: await listWavePackets(loaded.topicDir, wave.wave_id),
|
|
145
|
+
results: waveResults,
|
|
146
|
+
policy: await loadAuthorityConvergencePolicy(projectRoot),
|
|
147
|
+
commandRef,
|
|
148
|
+
});
|
|
149
|
+
return postUpdateReviewDecision
|
|
150
|
+
? buildTopicStepDecision(loaded.topic, wave, postUpdateReviewDecision)
|
|
151
|
+
: buildTopicStepDecision(loaded.topic, wave, {
|
|
152
|
+
stopClass: "continue",
|
|
153
|
+
recommendedAction: "closeout_wave",
|
|
154
|
+
reasonCode: "wave_has_result_lineage_ready_for_closeout",
|
|
155
|
+
recommendedDecision: "closeout_wave",
|
|
156
|
+
recommendationRationale:
|
|
157
|
+
"Wave closeout is a deterministic phase transition once lineage-backed result evidence exists.",
|
|
158
|
+
nextCommandRef: commandRef([
|
|
159
|
+
"closeout",
|
|
160
|
+
"wave",
|
|
161
|
+
loaded.topicId,
|
|
162
|
+
wave.wave_id,
|
|
163
|
+
"--authority",
|
|
164
|
+
"closed",
|
|
165
|
+
"--semantic",
|
|
166
|
+
"closed",
|
|
167
|
+
"--consumer",
|
|
168
|
+
"closed",
|
|
169
|
+
"--drift-resistance",
|
|
170
|
+
"closed",
|
|
171
|
+
"--disposition",
|
|
172
|
+
"complete",
|
|
173
|
+
]),
|
|
174
|
+
});
|
|
175
|
+
}
|
|
176
|
+
return wave.state === "overflowed"
|
|
177
|
+
? buildTopicStepDecision(loaded.topic, wave, {
|
|
178
|
+
stopClass: "require_human_confirmation",
|
|
179
|
+
recommendedAction: "continue_overflow",
|
|
180
|
+
reasonCode: "overflow_requires_manager_judgement",
|
|
181
|
+
recommendedDecision: "approve_continuation_only_if_the_same_owner_domain_rule_still_holds",
|
|
182
|
+
recommendationRationale:
|
|
183
|
+
"Overflow is not pass or rollback; continuation requires explicit manager judgement.",
|
|
184
|
+
nextCommandRef: commandRef([
|
|
185
|
+
"overflow",
|
|
186
|
+
"continue",
|
|
187
|
+
loaded.topicId,
|
|
188
|
+
"--packet",
|
|
189
|
+
"<continuation-packet-id>",
|
|
190
|
+
"--overflowed-packet",
|
|
191
|
+
"<packet-id>",
|
|
192
|
+
"--manager-judgement",
|
|
193
|
+
"<text>",
|
|
194
|
+
"--same-owner-domain",
|
|
195
|
+
]),
|
|
196
|
+
})
|
|
197
|
+
: buildTopicStepDecision(loaded.topic, wave, {
|
|
198
|
+
stopClass: "blocked",
|
|
199
|
+
recommendedAction: "no_action",
|
|
200
|
+
reasonCode: "unsupported_wave_state",
|
|
201
|
+
recommendedDecision: "repair_or_review_the_selected_wave_state",
|
|
202
|
+
recommendationRationale: `No next-step rule is defined for wave state ${wave.state}.`,
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
export async function decideTopicNextStep(projectRoot, input = null) {
|
|
206
|
+
const loaded = await loadTopicReport(projectRoot, input);
|
|
207
|
+
if (!loaded.ok) return loaded;
|
|
208
|
+
const { validateTopicRoot } = await import("./topic-root-validation.mjs");
|
|
209
|
+
const rootValidation = await validateTopicRoot(projectRoot, input),
|
|
210
|
+
selectedWave =
|
|
211
|
+
getTopicWaves(loaded.topic).find(
|
|
212
|
+
(entry) => entry.wave_id === loaded.topic.selected_next_target,
|
|
213
|
+
) ?? null;
|
|
214
|
+
if (!rootValidation.ok)
|
|
215
|
+
return {
|
|
216
|
+
ok: true,
|
|
217
|
+
topicId: loaded.topicId,
|
|
218
|
+
topicRef: toPortableRelativePath(path.relative(projectRoot, loaded.topicDir)),
|
|
219
|
+
decision: buildTopicStepDecision(loaded.topic, selectedWave, {
|
|
220
|
+
stopClass: "blocked",
|
|
221
|
+
recommendedAction: "no_action",
|
|
222
|
+
reasonCode: "topic_root_validation_failed",
|
|
223
|
+
recommendedDecision: "repair_topic_root_before_continuing",
|
|
224
|
+
recommendationRationale:
|
|
225
|
+
"The topic root must validate before a next execution step can be selected.",
|
|
226
|
+
blockingChecks: (rootValidation.checks ?? []).filter((entry) => !entry.ok),
|
|
227
|
+
}),
|
|
228
|
+
};
|
|
229
|
+
if (loaded.topic.state === "pending")
|
|
230
|
+
return {
|
|
231
|
+
ok: true,
|
|
232
|
+
topicId: loaded.topicId,
|
|
233
|
+
topicRef: toPortableRelativePath(path.relative(projectRoot, loaded.topicDir)),
|
|
234
|
+
decision: buildTopicStepDecision(loaded.topic, selectedWave, {
|
|
235
|
+
stopClass: "await_external_evidence",
|
|
236
|
+
recommendedAction: "resume_topic",
|
|
237
|
+
reasonCode: "topic_pending_wait",
|
|
238
|
+
recommendedDecision: "resume_only_when_the_pending_note_reopen_criteria_are_met",
|
|
239
|
+
recommendationRationale:
|
|
240
|
+
"Pending is an explicit wait state and must not be bypassed by the loop.",
|
|
241
|
+
nextCommandRef: commandRef(["resume", loaded.topicId, "--criteria-met", "<text>"]),
|
|
242
|
+
}),
|
|
243
|
+
};
|
|
244
|
+
if (
|
|
245
|
+
!loaded.topic.selected_next_target ||
|
|
246
|
+
loaded.topic.selected_next_target === "topic_design_baseline"
|
|
247
|
+
) {
|
|
248
|
+
const allWavesTerminal =
|
|
249
|
+
getTopicWaves(loaded.topic).filter(
|
|
250
|
+
(entry) => !["closed", "retired", "superseded"].includes(entry.state),
|
|
251
|
+
).length === 0 && getTopicWaves(loaded.topic).length > 0,
|
|
252
|
+
deterministicNextWave = allWavesTerminal ? null : findDeterministicNextWave(loaded.topic);
|
|
253
|
+
return {
|
|
254
|
+
ok: true,
|
|
255
|
+
topicId: loaded.topicId,
|
|
256
|
+
topicRef: toPortableRelativePath(path.relative(projectRoot, loaded.topicDir)),
|
|
257
|
+
decision: buildTopicStepDecision(
|
|
258
|
+
loaded.topic,
|
|
259
|
+
deterministicNextWave,
|
|
260
|
+
allWavesTerminal
|
|
261
|
+
? {
|
|
262
|
+
stopClass: "completed",
|
|
263
|
+
recommendedAction: "closeout_topic",
|
|
264
|
+
reasonCode: "all_waves_terminal",
|
|
265
|
+
recommendedDecision: "run_topic_closeout_and_true_close_checks",
|
|
266
|
+
recommendationRationale:
|
|
267
|
+
"All waves are terminal, so the next step is topic-level closeout review.",
|
|
268
|
+
nextCommandRef: commandRef([
|
|
269
|
+
"closeout",
|
|
270
|
+
"topic",
|
|
271
|
+
loaded.topicId,
|
|
272
|
+
"--authority",
|
|
273
|
+
"closed",
|
|
274
|
+
"--semantic",
|
|
275
|
+
"closed",
|
|
276
|
+
"--consumer",
|
|
277
|
+
"closed",
|
|
278
|
+
"--drift-resistance",
|
|
279
|
+
"closed",
|
|
280
|
+
"--disposition",
|
|
281
|
+
"complete",
|
|
282
|
+
]),
|
|
283
|
+
}
|
|
284
|
+
: deterministicNextWave
|
|
285
|
+
? {
|
|
286
|
+
stopClass: "continue",
|
|
287
|
+
recommendedAction: "admit_wave",
|
|
288
|
+
reasonCode: "deterministic_next_wave_ready",
|
|
289
|
+
recommendedDecision: "admit_wave",
|
|
290
|
+
recommendationRationale:
|
|
291
|
+
"The first dependency-ready non-terminal wave in topic.yaml waves[] order is selected mechanically.",
|
|
292
|
+
nextCommandRef: commandRef([
|
|
293
|
+
"wave",
|
|
294
|
+
"admit",
|
|
295
|
+
loaded.topicId,
|
|
296
|
+
deterministicNextWave.wave_id,
|
|
297
|
+
]),
|
|
298
|
+
}
|
|
299
|
+
: {
|
|
300
|
+
stopClass: "require_human_confirmation",
|
|
301
|
+
recommendedAction: "admit_wave",
|
|
302
|
+
reasonCode: "no_selected_next_target",
|
|
303
|
+
recommendedDecision: "select_the_next_wave_or_hold_the_topic",
|
|
304
|
+
recommendationRationale:
|
|
305
|
+
"The loop cannot choose among possible next waves without manager judgement.",
|
|
306
|
+
nextCommandRef: commandRef(["wave", "select", loaded.topicId, "<wave-id>"]),
|
|
307
|
+
},
|
|
308
|
+
),
|
|
309
|
+
};
|
|
310
|
+
}
|
|
311
|
+
if (!selectedWave)
|
|
312
|
+
return {
|
|
313
|
+
ok: true,
|
|
314
|
+
topicId: loaded.topicId,
|
|
315
|
+
topicRef: toPortableRelativePath(path.relative(projectRoot, loaded.topicDir)),
|
|
316
|
+
decision: buildTopicStepDecision(loaded.topic, null, {
|
|
317
|
+
stopClass: "blocked",
|
|
318
|
+
recommendedAction: "no_action",
|
|
319
|
+
reasonCode: "selected_next_target_does_not_resolve",
|
|
320
|
+
recommendedDecision: "repair_topic_selected_next_target",
|
|
321
|
+
recommendationRationale: `selected_next_target does not resolve to a declared wave: ${loaded.topic.selected_next_target}`,
|
|
322
|
+
}),
|
|
323
|
+
};
|
|
324
|
+
const graphReport = await validateTopicGraph(projectRoot, input);
|
|
325
|
+
return {
|
|
326
|
+
ok: true,
|
|
327
|
+
topicId: loaded.topicId,
|
|
328
|
+
topicRef: toPortableRelativePath(path.relative(projectRoot, loaded.topicDir)),
|
|
329
|
+
decision: await buildDecisionForSelectedWave(projectRoot, loaded, graphReport, selectedWave),
|
|
330
|
+
};
|
|
331
|
+
}
|
|
332
|
+
const RUN_ID_PATTERN = /^[a-z0-9]+(?:-[a-z0-9]+)*$/;
|
|
@@ -1,7 +1,13 @@
|
|
|
1
|
-
import { readdir } from "node:fs/promises";
|
|
1
|
+
import { readdir, writeFile } from "node:fs/promises";
|
|
2
2
|
import path from "node:path";
|
|
3
|
+
import YAML from "yaml";
|
|
3
4
|
|
|
4
5
|
import { readTextIfFile } from "./fs-helpers.mjs";
|
|
6
|
+
import {
|
|
7
|
+
packetAuthorityCoverage,
|
|
8
|
+
packetAuthorityCoverageError,
|
|
9
|
+
sourceSweepDesignAuthorityRefs,
|
|
10
|
+
} from "./topic-authority-coverage.mjs";
|
|
5
11
|
import { parseYamlText } from "./yaml-helpers.mjs";
|
|
6
12
|
|
|
7
13
|
function toPortableRelativePath(filePath) {
|
|
@@ -19,6 +25,105 @@ function parsePacketDraft(text) {
|
|
|
19
25
|
return parseYamlText(text);
|
|
20
26
|
}
|
|
21
27
|
|
|
28
|
+
function nonEmptyStringArray(value) {
|
|
29
|
+
return Array.isArray(value)
|
|
30
|
+
? value.filter((entry) => typeof entry === "string" && entry.length > 0)
|
|
31
|
+
: [];
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
function singleStringList(value) {
|
|
35
|
+
return typeof value === "string" && value.length > 0 ? [value] : nonEmptyStringArray(value);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
function missingSweepDraftFields(wave) {
|
|
39
|
+
const source = wave?.source_sweep_design;
|
|
40
|
+
if (!source || typeof source !== "object") {
|
|
41
|
+
return ["source_sweep_design"];
|
|
42
|
+
}
|
|
43
|
+
const missing = [];
|
|
44
|
+
if (singleStringList(source.authority_owner).length === 0) missing.push("source_sweep_design.authority_owner");
|
|
45
|
+
if (nonEmptyStringArray(source.validation_commands).length === 0) missing.push("source_sweep_design.validation_commands");
|
|
46
|
+
if (nonEmptyStringArray(source.negative_checks).length === 0) missing.push("source_sweep_design.negative_checks");
|
|
47
|
+
if (nonEmptyStringArray(source.closeout_criteria).length === 0) missing.push("source_sweep_design.closeout_criteria");
|
|
48
|
+
if (nonEmptyStringArray(source.source_design_packet_refs).length === 0) missing.push("source_sweep_design.source_design_packet_refs");
|
|
49
|
+
if (nonEmptyStringArray(source.design_auditor_result_refs).length === 0) missing.push("source_sweep_design.design_auditor_result_refs");
|
|
50
|
+
if (nonEmptyStringArray(source.revision_ledger_entry_refs).length === 0) missing.push("source_sweep_design.revision_ledger_entry_refs");
|
|
51
|
+
if (nonEmptyStringArray(source.blocked_gate_refs).length > 0) missing.push("source_sweep_design.blocked_gate_refs_empty");
|
|
52
|
+
return missing;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
async function maybeGenerateSweepFixDraft(projectRoot, loaded, wave) {
|
|
56
|
+
const missing = missingSweepDraftFields(wave);
|
|
57
|
+
if (missing.length > 0) {
|
|
58
|
+
return {
|
|
59
|
+
ok: false,
|
|
60
|
+
reasonCode: missing.includes("source_sweep_design.blocked_gate_refs_empty")
|
|
61
|
+
? "admitted_wave_has_blocked_gate_refs"
|
|
62
|
+
: missing.includes("source_sweep_design.validation_commands")
|
|
63
|
+
? "admitted_wave_missing_validation_commands"
|
|
64
|
+
: "admitted_wave_requires_packet",
|
|
65
|
+
missing,
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
const draftPath = path.join(loaded.topicDir, `draft-${wave.wave_id}-implementation.yaml`);
|
|
70
|
+
if (await readTextIfFile(draftPath) !== null) {
|
|
71
|
+
return {
|
|
72
|
+
ok: false,
|
|
73
|
+
reasonCode: "admitted_wave_has_ambiguous_draft_packets",
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
const source = wave.source_sweep_design;
|
|
78
|
+
const sourceAuthorityRefs = sourceSweepDesignAuthorityRefs(source);
|
|
79
|
+
const packet = {
|
|
80
|
+
packet_id: `${wave.wave_id}-implementation`,
|
|
81
|
+
topic_id: loaded.topicId,
|
|
82
|
+
wave_id: wave.wave_id,
|
|
83
|
+
packet_kind: "implementation",
|
|
84
|
+
status: "draft",
|
|
85
|
+
authority_owner: sourceAuthorityRefs.length > 0 ? sourceAuthorityRefs : singleStringList(source.authority_owner),
|
|
86
|
+
canonical_seams: [
|
|
87
|
+
...(sourceAuthorityRefs.length > 0 ? sourceAuthorityRefs : singleStringList(source.authority_owner)),
|
|
88
|
+
...nonEmptyStringArray(source.source_design_packet_refs),
|
|
89
|
+
...nonEmptyStringArray(source.design_auditor_result_refs),
|
|
90
|
+
...nonEmptyStringArray(source.revision_ledger_entry_refs),
|
|
91
|
+
],
|
|
92
|
+
forbidden_shortcuts: nonEmptyStringArray(loaded.topic?.forbidden_shortcuts),
|
|
93
|
+
acceptance_invariants: [
|
|
94
|
+
...nonEmptyStringArray(source.closeout_criteria),
|
|
95
|
+
...nonEmptyStringArray(source.validation_commands).map((command) => `validation command required: ${command}`),
|
|
96
|
+
],
|
|
97
|
+
negative_tests: nonEmptyStringArray(source.negative_checks),
|
|
98
|
+
reopen_conditions: [
|
|
99
|
+
...nonEmptyStringArray(source.drift_resistance_checks),
|
|
100
|
+
"source_sweep_design provenance changes require packet regeneration",
|
|
101
|
+
],
|
|
102
|
+
source_sweep_design_run_id: source.run_id ?? null,
|
|
103
|
+
source_authority_refs: sourceAuthorityRefs,
|
|
104
|
+
source_authority_coverage_policy: "authority_owner_and_canonical_seams_cover_union_of_source_sweep_design_authority_refs",
|
|
105
|
+
source_design_packet_refs: nonEmptyStringArray(source.source_design_packet_refs),
|
|
106
|
+
design_auditor_result_refs: nonEmptyStringArray(source.design_auditor_result_refs),
|
|
107
|
+
revision_ledger_entry_refs: nonEmptyStringArray(source.revision_ledger_entry_refs),
|
|
108
|
+
validation_commands: nonEmptyStringArray(source.validation_commands),
|
|
109
|
+
closeout_criteria: nonEmptyStringArray(source.closeout_criteria),
|
|
110
|
+
};
|
|
111
|
+
if (packet.forbidden_shortcuts.length === 0) {
|
|
112
|
+
return {
|
|
113
|
+
ok: false,
|
|
114
|
+
reasonCode: "admitted_wave_requires_packet",
|
|
115
|
+
missing: ["topic.forbidden_shortcuts"],
|
|
116
|
+
};
|
|
117
|
+
}
|
|
118
|
+
await writeFile(draftPath, YAML.stringify(packet), "utf8");
|
|
119
|
+
return {
|
|
120
|
+
ok: true,
|
|
121
|
+
packet,
|
|
122
|
+
draftRef: toPortableRelativePath(path.relative(projectRoot, draftPath)),
|
|
123
|
+
generated: true,
|
|
124
|
+
};
|
|
125
|
+
}
|
|
126
|
+
|
|
22
127
|
export async function findUniqueFreezableDraftPacket(projectRoot, loaded, wave, authority) {
|
|
23
128
|
const matches = [];
|
|
24
129
|
for (const entry of await readdir(loaded.topicDir, { withFileTypes: true })) {
|
|
@@ -32,17 +137,31 @@ export async function findUniqueFreezableDraftPacket(projectRoot, loaded, wave,
|
|
|
32
137
|
const value = packet[field];
|
|
33
138
|
return value == null || value === "" || (Array.isArray(value) && value.length === 0);
|
|
34
139
|
})) continue;
|
|
140
|
+
const coverage = packetAuthorityCoverage(packet, wave);
|
|
141
|
+
if (!coverage.ok) {
|
|
142
|
+
return {
|
|
143
|
+
ok: false,
|
|
144
|
+
reasonCode: "admitted_wave_packet_authority_coverage_incomplete",
|
|
145
|
+
missing: [
|
|
146
|
+
...coverage.missingAuthorityOwnerRefs.map((ref) => `authority_owner:${ref}`),
|
|
147
|
+
...coverage.missingCanonicalSeamRefs.map((ref) => `canonical_seams:${ref}`),
|
|
148
|
+
],
|
|
149
|
+
error: packetAuthorityCoverageError(coverage),
|
|
150
|
+
};
|
|
151
|
+
}
|
|
35
152
|
matches.push({
|
|
36
153
|
packet,
|
|
37
154
|
draftRef: toPortableRelativePath(path.relative(projectRoot, draftPath)),
|
|
38
155
|
});
|
|
39
156
|
}
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
157
|
+
if (matches.length === 1) {
|
|
158
|
+
return { ok: true, ...matches[0] };
|
|
159
|
+
}
|
|
160
|
+
if (matches.length > 1) {
|
|
161
|
+
return {
|
|
43
162
|
ok: false,
|
|
44
|
-
reasonCode:
|
|
45
|
-
? "admitted_wave_requires_packet"
|
|
46
|
-
: "admitted_wave_has_ambiguous_draft_packets",
|
|
163
|
+
reasonCode: "admitted_wave_has_ambiguous_draft_packets",
|
|
47
164
|
};
|
|
165
|
+
}
|
|
166
|
+
return maybeGenerateSweepFixDraft(projectRoot, loaded, wave);
|
|
48
167
|
}
|