@nimiplatform/nimi-coding 0.1.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.
Files changed (186) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +348 -0
  3. package/adapters/README.md +25 -0
  4. package/adapters/claude/README.md +89 -0
  5. package/adapters/claude/profile.yaml +70 -0
  6. package/adapters/codex/README.md +53 -0
  7. package/adapters/codex/profile.yaml +78 -0
  8. package/adapters/oh-my-codex/README.md +185 -0
  9. package/adapters/oh-my-codex/profile.yaml +46 -0
  10. package/bin/nimicoding.mjs +6 -0
  11. package/cli/commands/admit-high-risk-decision.mjs +108 -0
  12. package/cli/commands/audit-sweep.mjs +341 -0
  13. package/cli/commands/blueprint-audit.mjs +91 -0
  14. package/cli/commands/clear.mjs +168 -0
  15. package/cli/commands/closeout.mjs +183 -0
  16. package/cli/commands/decide-high-risk-execution.mjs +124 -0
  17. package/cli/commands/doctor.mjs +53 -0
  18. package/cli/commands/generate-spec-derived-docs.mjs +131 -0
  19. package/cli/commands/handoff.mjs +123 -0
  20. package/cli/commands/ingest-high-risk-execution.mjs +95 -0
  21. package/cli/commands/review-high-risk-execution.mjs +95 -0
  22. package/cli/commands/start.mjs +717 -0
  23. package/cli/commands/topic-formatters.mjs +382 -0
  24. package/cli/commands/topic-goal.mjs +33 -0
  25. package/cli/commands/topic-options-shared.mjs +27 -0
  26. package/cli/commands/topic-options-workflow.mjs +767 -0
  27. package/cli/commands/topic-options.mjs +626 -0
  28. package/cli/commands/topic-runner.mjs +169 -0
  29. package/cli/commands/topic.mjs +795 -0
  30. package/cli/commands/validate-acceptance.mjs +5 -0
  31. package/cli/commands/validate-ai-governance.mjs +214 -0
  32. package/cli/commands/validate-execution-packet.mjs +5 -0
  33. package/cli/commands/validate-orchestration-state.mjs +5 -0
  34. package/cli/commands/validate-prompt.mjs +5 -0
  35. package/cli/commands/validate-spec-audit.mjs +27 -0
  36. package/cli/commands/validate-spec-governance.mjs +124 -0
  37. package/cli/commands/validate-spec-tree.mjs +27 -0
  38. package/cli/commands/validate-worker-output.mjs +5 -0
  39. package/cli/constants.mjs +489 -0
  40. package/cli/help.mjs +134 -0
  41. package/cli/index.mjs +103 -0
  42. package/cli/lib/adapter-profiles.mjs +403 -0
  43. package/cli/lib/audit-execution.mjs +52 -0
  44. package/cli/lib/audit-sweep-runtime/admissions.mjs +381 -0
  45. package/cli/lib/audit-sweep-runtime/audit-validity.mjs +333 -0
  46. package/cli/lib/audit-sweep-runtime/chunks.mjs +697 -0
  47. package/cli/lib/audit-sweep-runtime/closeout.mjs +144 -0
  48. package/cli/lib/audit-sweep-runtime/codex-auditor-evidence.mjs +639 -0
  49. package/cli/lib/audit-sweep-runtime/codex-auditor.mjs +515 -0
  50. package/cli/lib/audit-sweep-runtime/common.mjs +329 -0
  51. package/cli/lib/audit-sweep-runtime/coverage-quality.mjs +172 -0
  52. package/cli/lib/audit-sweep-runtime/evidence-assignment.mjs +152 -0
  53. package/cli/lib/audit-sweep-runtime/format.mjs +57 -0
  54. package/cli/lib/audit-sweep-runtime/ingest.mjs +486 -0
  55. package/cli/lib/audit-sweep-runtime/inventory-spec-chunks.mjs +198 -0
  56. package/cli/lib/audit-sweep-runtime/inventory.mjs +728 -0
  57. package/cli/lib/audit-sweep-runtime/ledger.mjs +315 -0
  58. package/cli/lib/audit-sweep-runtime/p0p1-profile.mjs +101 -0
  59. package/cli/lib/audit-sweep-runtime/remediation.mjs +349 -0
  60. package/cli/lib/audit-sweep-runtime/rerun.mjs +129 -0
  61. package/cli/lib/audit-sweep-runtime/risk-budget.mjs +300 -0
  62. package/cli/lib/audit-sweep-runtime/status.mjs +62 -0
  63. package/cli/lib/audit-sweep-runtime/validators-ledger.mjs +215 -0
  64. package/cli/lib/audit-sweep-runtime/validators.mjs +758 -0
  65. package/cli/lib/audit-sweep.mjs +18 -0
  66. package/cli/lib/authority-convergence.mjs +309 -0
  67. package/cli/lib/blueprint-audit.mjs +370 -0
  68. package/cli/lib/bootstrap.mjs +228 -0
  69. package/cli/lib/closeout.mjs +623 -0
  70. package/cli/lib/codex-sdk-runner.mjs +76 -0
  71. package/cli/lib/contracts.mjs +180 -0
  72. package/cli/lib/doctor.mjs +18 -0
  73. package/cli/lib/entrypoints.mjs +274 -0
  74. package/cli/lib/external-execution.mjs +101 -0
  75. package/cli/lib/fs-helpers.mjs +33 -0
  76. package/cli/lib/handoff.mjs +785 -0
  77. package/cli/lib/high-risk-admission.mjs +442 -0
  78. package/cli/lib/high-risk-decision.mjs +324 -0
  79. package/cli/lib/high-risk-ingest.mjs +317 -0
  80. package/cli/lib/high-risk-review.mjs +263 -0
  81. package/cli/lib/internal/contracts-loaders.mjs +132 -0
  82. package/cli/lib/internal/contracts-parse-high-risk.mjs +131 -0
  83. package/cli/lib/internal/contracts-parse.mjs +457 -0
  84. package/cli/lib/internal/contracts-validators.mjs +398 -0
  85. package/cli/lib/internal/doctor-bootstrap-surface.mjs +359 -0
  86. package/cli/lib/internal/doctor-delegated-surface.mjs +256 -0
  87. package/cli/lib/internal/doctor-finalize.mjs +385 -0
  88. package/cli/lib/internal/doctor-format.mjs +286 -0
  89. package/cli/lib/internal/doctor-inspectors.mjs +294 -0
  90. package/cli/lib/internal/doctor-state.mjs +205 -0
  91. package/cli/lib/internal/governance/ai/ai-context-budget-core.mjs +315 -0
  92. package/cli/lib/internal/governance/ai/ai-structure-budget-core.mjs +358 -0
  93. package/cli/lib/internal/governance/ai/check-agents-freshness.mjs +155 -0
  94. package/cli/lib/internal/governance/ai/check-high-risk-doc-metadata-core.mjs +173 -0
  95. package/cli/lib/internal/governance/config.mjs +150 -0
  96. package/cli/lib/internal/governance/runner.mjs +35 -0
  97. package/cli/lib/internal/governance/shared/read-yaml-with-fragments.mjs +49 -0
  98. package/cli/lib/internal/validators-artifacts.mjs +515 -0
  99. package/cli/lib/internal/validators-shared.mjs +28 -0
  100. package/cli/lib/internal/validators-spec-helpers.mjs +186 -0
  101. package/cli/lib/internal/validators-spec.mjs +410 -0
  102. package/cli/lib/shared.mjs +83 -0
  103. package/cli/lib/topic-draft-packets.mjs +48 -0
  104. package/cli/lib/topic-goal.mjs +361 -0
  105. package/cli/lib/topic-runner.mjs +772 -0
  106. package/cli/lib/topic.mjs +93 -0
  107. package/cli/lib/ui.mjs +178 -0
  108. package/cli/lib/validators.mjs +78 -0
  109. package/cli/lib/value-helpers.mjs +24 -0
  110. package/cli/lib/yaml-helpers.mjs +133 -0
  111. package/cli/nimicoding.mjs +1 -0
  112. package/cli/seeds/bootstrap.mjs +47 -0
  113. package/config/audit-execution-artifacts.yaml +20 -0
  114. package/config/bootstrap.yaml +6 -0
  115. package/config/external-execution-artifacts.yaml +16 -0
  116. package/config/host-adapter.yaml +30 -0
  117. package/config/host-profile.yaml +29 -0
  118. package/config/installer-evidence.yaml +31 -0
  119. package/config/skill-installer.yaml +23 -0
  120. package/config/skill-manifest.yaml +46 -0
  121. package/config/skills.yaml +30 -0
  122. package/config/spec-generation-inputs.yaml +25 -0
  123. package/contracts/acceptance.schema.yaml +16 -0
  124. package/contracts/admission-checklist.schema.yaml +15 -0
  125. package/contracts/audit-chunk.schema.yaml +110 -0
  126. package/contracts/audit-closeout.schema.yaml +51 -0
  127. package/contracts/audit-finding.schema.yaml +61 -0
  128. package/contracts/audit-ledger.schema.yaml +138 -0
  129. package/contracts/audit-plan.schema.yaml +123 -0
  130. package/contracts/audit-remediation-map.schema.yaml +51 -0
  131. package/contracts/audit-rerun.schema.yaml +31 -0
  132. package/contracts/audit-sweep-result.yaml +49 -0
  133. package/contracts/authority-convergence-audit.schema.yaml +19 -0
  134. package/contracts/closeout.schema.yaml +25 -0
  135. package/contracts/decision-review.schema.yaml +16 -0
  136. package/contracts/doc-spec-audit-result.yaml +19 -0
  137. package/contracts/execution-packet.schema.yaml +49 -0
  138. package/contracts/external-host-compatibility.yaml +22 -0
  139. package/contracts/forbidden-shortcuts.catalog.yaml +23 -0
  140. package/contracts/high-risk-admission.schema.yaml +23 -0
  141. package/contracts/high-risk-execution-result.yaml +20 -0
  142. package/contracts/orchestration-state.schema.yaml +41 -0
  143. package/contracts/overflow-continuation.schema.yaml +12 -0
  144. package/contracts/packet.schema.yaml +30 -0
  145. package/contracts/pending-note.schema.yaml +17 -0
  146. package/contracts/prompt.schema.yaml +12 -0
  147. package/contracts/remediation.schema.yaml +16 -0
  148. package/contracts/result.schema.yaml +24 -0
  149. package/contracts/spec-generation-audit.schema.yaml +31 -0
  150. package/contracts/spec-generation-inputs.schema.yaml +39 -0
  151. package/contracts/spec-reconstruction-result.yaml +37 -0
  152. package/contracts/topic-goal.schema.yaml +78 -0
  153. package/contracts/topic-run-ledger.schema.yaml +72 -0
  154. package/contracts/topic-step-decision.schema.yaml +45 -0
  155. package/contracts/topic.schema.yaml +65 -0
  156. package/contracts/true-close.schema.yaml +15 -0
  157. package/contracts/wave.schema.yaml +29 -0
  158. package/contracts/worker-output.schema.yaml +15 -0
  159. package/methodology/audit-sweep-p0p1-recall.yaml +45 -0
  160. package/methodology/authority-convergence-policy.yaml +42 -0
  161. package/methodology/core.yaml +25 -0
  162. package/methodology/four-closure-policy.yaml +28 -0
  163. package/methodology/overflow-continuation-policy.yaml +14 -0
  164. package/methodology/role-separation-policy.yaml +28 -0
  165. package/methodology/skill-exchange-projection.yaml +114 -0
  166. package/methodology/skill-handoff.yaml +34 -0
  167. package/methodology/skill-installer-result.yaml +27 -0
  168. package/methodology/skill-installer-summary-projection.yaml +181 -0
  169. package/methodology/skill-runtime.yaml +23 -0
  170. package/methodology/spec-reconstruction.yaml +63 -0
  171. package/methodology/spec-target-truth-profile.yaml +53 -0
  172. package/methodology/topic-lifecycle-report.yaml +144 -0
  173. package/methodology/topic-lifecycle.yaml +37 -0
  174. package/methodology/topic-naming-ontology.yaml +21 -0
  175. package/methodology/topic-ontology.yaml +38 -0
  176. package/methodology/topic-validation-policy.yaml +9 -0
  177. package/methodology/wave-dag-policy.yaml +14 -0
  178. package/package.json +50 -0
  179. package/spec/_meta/command-gating-matrix.yaml +110 -0
  180. package/spec/_meta/generate-drift-migration-checklist.yaml +155 -0
  181. package/spec/_meta/governance-routing-cutover-checklist.yaml +35 -0
  182. package/spec/_meta/phase2-impacted-surface-matrix.yaml +44 -0
  183. package/spec/_meta/spec-authority-cutover-readiness.yaml +104 -0
  184. package/spec/_meta/spec-tree-model.yaml +72 -0
  185. package/spec/bootstrap-state.yaml +99 -0
  186. package/spec/product-scope.yaml +56 -0
@@ -0,0 +1,382 @@
1
+ import { localize, styleHeading, styleLabel, styleMuted } from "../lib/ui.mjs";
2
+
3
+ export function buildJsonReport(command, report) {
4
+ return {
5
+ contract: "nimicoding.topic-command-result.v1",
6
+ command,
7
+ ...report,
8
+ };
9
+ }
10
+ export function writeJson(report) {
11
+ process.stdout.write(`${JSON.stringify(report, null, 2)}\n`);
12
+ }
13
+ export function formatTopicGoalRefusal(report) {
14
+ const blockingChecks = report.readiness?.checks?.filter((entry) => entry.status === "fail") ?? [];
15
+ const lines = [
16
+ "nimicoding topic goal refused: topic is not goal-ready.",
17
+ "",
18
+ `Topic: ${report.topic_id ?? "unknown"}`,
19
+ `State: ${report.topic_state ?? "unknown"}`,
20
+ `Selected Next Target: ${report.selected_next_target ?? "none"}`,
21
+ "",
22
+ "Blocking checks:",
23
+ ...blockingChecks.map((entry) => `- ${entry.id}: ${entry.message}`),
24
+ "",
25
+ "Fix the topic admission artifacts first, then rerun:",
26
+ `pnpm exec nimicoding topic goal ${report.topic_id ?? "<topic-id>"}`,
27
+ ];
28
+ return `${lines.join("\n")}\n`;
29
+ }
30
+ export function formatTopicStatus(report) {
31
+ const lines = [
32
+ styleHeading(`nimicoding topic status: ${report.topicId}`),
33
+ "",
34
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
35
+ `${styleLabel(localize("State", "状态"))}: ${report.state}`,
36
+ `${styleLabel(localize("Schema", "Schema"))}: ${report.schemaMode}`,
37
+ `${styleLabel(localize("Title", "标题"))}: ${report.title ?? localize("none", "无")}`,
38
+ `${styleLabel(localize("Selected Next Target", "当前下一目标"))}: ${report.selectedNextTarget ?? localize("none", "无")}`,
39
+ `${styleLabel(localize("True-Close", "True-Close"))}: ${report.currentTrueCloseStatus ?? localize("none", "无")}`,
40
+ `${styleLabel(localize("Pending Note", "Pending Note"))}: ${report.pendingNoteStatus ?? localize("none", "无")}`,
41
+ `${styleLabel(localize("Migration Posture", "迁移姿态"))}: ${report.migrationPosture ?? localize("none", "无")}`,
42
+ `${styleLabel(localize("Validation Disposition", "校验姿态"))}: ${report.validationDisposition ?? "strict"}`,
43
+ `${styleLabel(localize("Canonical Validated", "Canonical 校验"))}: ${report.canonicalValidated ? "true" : "false"}`,
44
+ `${styleLabel(localize("Ignored By Policy", "策略忽略"))}: ${report.ignoredByPolicy ? "true" : "false"}`,
45
+ ];
46
+ if (report.ignoredByPolicy && report.ignorePolicyReason) {
47
+ lines.push(`${styleLabel(localize("Ignore Reason", "忽略原因"))}: ${report.ignorePolicyReason}`);
48
+ }
49
+ if (report.artifactSummary) {
50
+ lines.push(
51
+ "",
52
+ styleLabel(localize("Artifacts", "Artifacts")),
53
+ `- files=${report.artifactSummary.files} packets=${report.artifactSummary.packets} results=${report.artifactSummary.results} closeouts=${report.artifactSummary.closeouts}`,
54
+ `- decision_reviews=${report.artifactSummary.decision_reviews} remediations=${report.artifactSummary.remediations} overflow_continuations=${report.artifactSummary.overflow_continuations}`,
55
+ `- exec_packs=${report.artifactSummary.exec_packs} true_close_artifacts=${report.artifactSummary.true_close_artifacts}`,
56
+ );
57
+ }
58
+ if (report.featureFlags) {
59
+ lines.push(
60
+ "",
61
+ styleLabel(localize("Feature Flags", "Feature Flags")),
62
+ ...Object.entries(report.featureFlags).map(([key, value]) => `- ${key}: ${value ? "true" : "false"}`),
63
+ );
64
+ }
65
+ if (Array.isArray(report.legacyObservedWaves) && report.legacyObservedWaves.length > 0) {
66
+ lines.push(
67
+ "",
68
+ styleLabel(localize("Observed Waves", "Observed Waves")),
69
+ ...report.legacyObservedWaves.slice(0, 8).map((entry) => (
70
+ `- ${entry.wave_id}: ${entry.observed_lineage} packets=${entry.packets} results=${entry.results} closeouts=${entry.closeouts} exec_packs=${entry.exec_packs}`
71
+ )),
72
+ );
73
+ if (report.legacyObservedWaves.length > 8) {
74
+ lines.push(styleMuted(`- ... ${report.legacyObservedWaves.length - 8} more wave observations`));
75
+ }
76
+ }
77
+ if (report.warnings.length > 0) {
78
+ lines.push(
79
+ "",
80
+ styleLabel(localize("Warnings", "警告")),
81
+ ...report.warnings.map((entry) => styleMuted(`- ${entry}`)),
82
+ );
83
+ }
84
+ return `${lines.join("\n")}\n`;
85
+ }
86
+ export function formatTopicValidate(report) {
87
+ const lines = [
88
+ styleHeading(`nimicoding topic validate: ${report.topicId}`),
89
+ "",
90
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
91
+ `${styleLabel(localize("State", "状态"))}: ${report.state}`,
92
+ `${styleLabel(localize("Schema", "Schema"))}: ${report.schemaMode}`,
93
+ `${styleLabel(localize("Result", "结果"))}: ${report.ok ? localize("ok", "通过") : localize("failed", "失败")}`,
94
+ `${styleLabel(localize("Pending Note", "Pending Note"))}: ${report.pendingNoteStatus ?? localize("none", "无")}`,
95
+ `${styleLabel(localize("Migration Posture", "迁移姿态"))}: ${report.migrationPosture ?? localize("none", "无")}`,
96
+ `${styleLabel(localize("Validation Disposition", "校验姿态"))}: ${report.validationDisposition ?? "strict"}`,
97
+ `${styleLabel(localize("Canonical Validated", "Canonical 校验"))}: ${report.canonicalValidated ? "true" : "false"}`,
98
+ `${styleLabel(localize("Ignored By Policy", "策略忽略"))}: ${report.ignoredByPolicy ? "true" : "false"}`,
99
+ "",
100
+ styleLabel(localize("Checks", "检查项")),
101
+ ...report.checks.map((entry) => `- [${entry.ok ? "ok" : "fail"}] ${entry.id}: ${entry.reason}`),
102
+ ];
103
+ if (report.ignoredByPolicy && report.ignorePolicyReason) {
104
+ lines.push(`${styleLabel(localize("Ignore Reason", "忽略原因"))}: ${report.ignorePolicyReason}`);
105
+ }
106
+ if (report.artifactSummary) {
107
+ lines.push(
108
+ "",
109
+ styleLabel(localize("Artifacts", "Artifacts")),
110
+ `- files=${report.artifactSummary.files} packets=${report.artifactSummary.packets} results=${report.artifactSummary.results} closeouts=${report.artifactSummary.closeouts}`,
111
+ `- decision_reviews=${report.artifactSummary.decision_reviews} remediations=${report.artifactSummary.remediations} overflow_continuations=${report.artifactSummary.overflow_continuations}`,
112
+ `- exec_packs=${report.artifactSummary.exec_packs} true_close_artifacts=${report.artifactSummary.true_close_artifacts}`,
113
+ );
114
+ }
115
+ if (report.featureFlags) {
116
+ lines.push(
117
+ "",
118
+ styleLabel(localize("Feature Flags", "Feature Flags")),
119
+ ...Object.entries(report.featureFlags).map(([key, value]) => `- ${key}: ${value ? "true" : "false"}`),
120
+ );
121
+ }
122
+ if (Array.isArray(report.legacyObservedWaves) && report.legacyObservedWaves.length > 0) {
123
+ lines.push(
124
+ "",
125
+ styleLabel(localize("Observed Waves", "Observed Waves")),
126
+ ...report.legacyObservedWaves.slice(0, 8).map((entry) => (
127
+ `- ${entry.wave_id}: ${entry.observed_lineage} packets=${entry.packets} results=${entry.results} closeouts=${entry.closeouts} exec_packs=${entry.exec_packs}`
128
+ )),
129
+ );
130
+ if (report.legacyObservedWaves.length > 8) {
131
+ lines.push(styleMuted(`- ... ${report.legacyObservedWaves.length - 8} more wave observations`));
132
+ }
133
+ }
134
+ if (report.warnings.length > 0) {
135
+ lines.push(
136
+ "",
137
+ styleLabel(localize("Warnings", "警告")),
138
+ ...report.warnings.map((entry) => styleMuted(`- ${entry}`)),
139
+ );
140
+ }
141
+ return `${lines.join("\n")}\n`;
142
+ }
143
+ export function formatNextStep(report) {
144
+ const decision = report.decision;
145
+ const lines = [
146
+ styleHeading(`nimicoding topic run-next-step: ${report.topicId}`),
147
+ "",
148
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
149
+ `${styleLabel(localize("Stop Class", "Stop Class"))}: ${decision.stop_class}`,
150
+ `${styleLabel(localize("Action", "Action"))}: ${decision.recommended_action}`,
151
+ `${styleLabel(localize("Reason", "Reason"))}: ${decision.reason_code}`,
152
+ `${styleLabel(localize("Human Confirmation", "Human Confirmation"))}: ${decision.requires_human_confirmation ? "true" : "false"}`,
153
+ `${styleLabel(localize("Recommendation", "Recommendation"))}: ${decision.recommended_decision}`,
154
+ `${styleLabel(localize("Rationale", "Rationale"))}: ${decision.recommendation_rationale}`,
155
+ ];
156
+ if (decision.next_command_ref) {
157
+ lines.push(`${styleLabel(localize("Next Command", "Next Command"))}: ${decision.next_command_ref}`);
158
+ }
159
+ if (Array.isArray(decision.expected_artifacts) && decision.expected_artifacts.length > 0) {
160
+ lines.push("", styleLabel(localize("Expected Artifacts", "Expected Artifacts")));
161
+ lines.push(...decision.expected_artifacts.map((entry) => `- ${entry}`));
162
+ }
163
+ if (Array.isArray(decision.blocking_checks) && decision.blocking_checks.length > 0) {
164
+ lines.push("", styleLabel(localize("Blocking Checks", "Blocking Checks")));
165
+ lines.push(...decision.blocking_checks.map((entry) => `- [${entry.ok ? "ok" : "fail"}] ${entry.id}: ${entry.reason}`));
166
+ }
167
+ return `${lines.join("\n")}\n`;
168
+ }
169
+ export function formatRunLedger(report, action) {
170
+ const ledger = report.ledger ?? {};
171
+ const lines = [
172
+ styleHeading(`nimicoding topic run-ledger ${action}: ${report.topicId}`),
173
+ "",
174
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
175
+ `${styleLabel(localize("Run", "Run"))}: ${report.runId}`,
176
+ `${styleLabel(localize("Ledger", "Ledger"))}: ${report.ledgerRef}`,
177
+ `${styleLabel(localize("Status", "Status"))}: ${report.runStatus ?? ledger.run_status}`,
178
+ `${styleLabel(localize("Events", "Events"))}: ${report.eventCount ?? ledger.event_count ?? 0}`,
179
+ ];
180
+ if (report.eventRef) {
181
+ lines.push(`${styleLabel(localize("Event", "Event"))}: ${report.eventRef}`);
182
+ }
183
+ if (ledger.current_human_gate) {
184
+ lines.push(`${styleLabel(localize("Human Gate", "Human Gate"))}: ${ledger.current_human_gate.summary}`);
185
+ }
186
+ return `${lines.join("\n")}\n`;
187
+ }
188
+ export function formatGraphValidate(report) {
189
+ const lines = [
190
+ styleHeading(`nimicoding topic validate graph: ${report.topicId}`),
191
+ "",
192
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
193
+ `${styleLabel(localize("Wave Count", "Wave 数量"))}: ${report.waveCount ?? 0}`,
194
+ `${styleLabel(localize("Result", "结果"))}: ${report.ok ? localize("ok", "通过") : localize("failed", "失败")}`,
195
+ "",
196
+ styleLabel(localize("Checks", "检查项")),
197
+ ...report.checks.map((entry) => `- [${entry.ok ? "ok" : "fail"}] ${entry.id}: ${entry.reason}`),
198
+ ];
199
+ if (report.warnings.length > 0) {
200
+ lines.push("", styleLabel(localize("Warnings", "警告")), ...report.warnings.map((entry) => styleMuted(`- ${entry}`)));
201
+ }
202
+ return `${lines.join("\n")}\n`;
203
+ }
204
+ export function formatAdmissionValidate(report, waveId) {
205
+ const lines = [
206
+ styleHeading(`nimicoding topic validate admission: ${report.topicId} / ${waveId}`),
207
+ "",
208
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
209
+ `${styleLabel(localize("Result", "结果"))}: ${report.ok ? localize("ok", "通过") : localize("failed", "失败")}`,
210
+ "",
211
+ styleLabel(localize("Checks", "检查项")),
212
+ ...report.checks.map((entry) => `- [${entry.ok ? "ok" : "fail"}] ${entry.id}: ${entry.reason}`),
213
+ ];
214
+ if (report.warnings.length > 0) {
215
+ lines.push("", styleLabel(localize("Warnings", "警告")), ...report.warnings.map((entry) => styleMuted(`- ${entry}`)));
216
+ }
217
+ return `${lines.join("\n")}\n`;
218
+ }
219
+ export function formatClosureValidate(report, waveId) {
220
+ const lines = [
221
+ styleHeading(`nimicoding topic validate closure: ${report.topicId} / ${waveId}`),
222
+ "",
223
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
224
+ `${styleLabel(localize("Result", "结果"))}: ${report.ok ? localize("ok", "通过") : localize("failed", "失败")}`,
225
+ "",
226
+ styleLabel(localize("Checks", "检查项")),
227
+ ...report.checks.map((entry) => `- [${entry.ok ? "ok" : "fail"}] ${entry.id}: ${entry.reason}`),
228
+ ];
229
+ if (report.closeoutRef) {
230
+ lines.push("", `${styleLabel(localize("Closeout Ref", "Closeout 路径"))}: ${report.closeoutRef}`);
231
+ }
232
+ if (report.warnings.length > 0) {
233
+ lines.push("", styleLabel(localize("Warnings", "警告")), ...report.warnings.map((entry) => styleMuted(`- ${entry}`)));
234
+ }
235
+ return `${lines.join("\n")}\n`;
236
+ }
237
+ export function formatWaveMutation(report, action) {
238
+ const lines = [
239
+ styleHeading(`nimicoding topic ${action}: ${report.topicId}`),
240
+ "",
241
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
242
+ `${styleLabel(localize("Wave", "Wave"))}: ${report.waveId}`,
243
+ ];
244
+ if (report.waveState) {
245
+ lines.push(`${styleLabel(localize("Wave State", "Wave 状态"))}: ${report.waveState}`);
246
+ }
247
+ if (report.selectedNextTarget) {
248
+ lines.push(`${styleLabel(localize("Selected Next Target", "当前下一目标"))}: ${report.selectedNextTarget}`);
249
+ }
250
+ if (report.state) {
251
+ lines.push(`${styleLabel(localize("Topic State", "Topic 状态"))}: ${report.state}`);
252
+ }
253
+ return `${lines.join("\n")}\n`;
254
+ }
255
+ export function formatPacketFreeze(report) {
256
+ return `${styleHeading(`nimicoding topic packet freeze: ${report.topicId}`)}
257
+ ${styleLabel(localize("Path", "路径"))}: ${report.topicRef}
258
+ ${styleLabel(localize("Packet", "Packet"))}: ${report.packetId}
259
+ ${styleLabel(localize("Wave", "Wave"))}: ${report.waveId}
260
+ ${styleLabel(localize("Packet Ref", "Packet 路径"))}: ${report.packetRef}
261
+ ${styleLabel(localize("Status", "状态"))}: ${report.status}
262
+ `;
263
+ }
264
+ export function formatDispatch(report) {
265
+ return `${styleHeading(`nimicoding topic ${report.role} dispatch: ${report.topicId}`)}
266
+ ${styleLabel(localize("Path", "路径"))}: ${report.topicRef}
267
+ ${styleLabel(localize("Packet", "Packet"))}: ${report.packetId}
268
+ ${styleLabel(localize("Wave", "Wave"))}: ${report.waveId}
269
+ ${styleLabel(localize("Packet Ref", "Packet 路径"))}: ${report.packetRef}
270
+ ${styleLabel(localize("Prompt Ref", "Prompt 路径"))}: ${report.promptRef}
271
+ ${styleLabel(localize("Wave State", "Wave 状态"))}: ${report.waveState}
272
+ `;
273
+ }
274
+ export function formatResultRecord(report) {
275
+ return `${styleHeading(`nimicoding topic result record: ${report.topicId}`)}
276
+ ${styleLabel(localize("Path", "路径"))}: ${report.topicRef}
277
+ ${styleLabel(localize("Result", "Result"))}: ${report.resultId}
278
+ ${styleLabel(localize("Wave", "Wave"))}: ${report.waveId}
279
+ ${styleLabel(localize("Kind", "类别"))}: ${report.resultKind}
280
+ ${styleLabel(localize("Verdict", "结论"))}: ${report.verdict}
281
+ ${styleLabel(localize("Result Ref", "Result 路径"))}: ${report.resultRef}
282
+ ${styleLabel(localize("Wave State", "Wave 状态"))}: ${report.waveState}
283
+ `;
284
+ }
285
+ export function formatDecisionReview(report) {
286
+ const lines = [
287
+ styleHeading(`nimicoding topic decision-review: ${report.topicId}`),
288
+ "",
289
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
290
+ `${styleLabel(localize("Decision Review", "Decision Review"))}: ${report.decisionReviewId}`,
291
+ `${styleLabel(localize("Disposition", "Disposition"))}: ${report.disposition}`,
292
+ `${styleLabel(localize("Review Ref", "Review 路径"))}: ${report.decisionReviewRef}`,
293
+ ];
294
+ if (report.targetWaveId) {
295
+ lines.push(`${styleLabel(localize("Target Wave", "目标 Wave"))}: ${report.targetWaveId}`);
296
+ }
297
+ return `${lines.join("\n")}\n`;
298
+ }
299
+ export function formatRemediation(report) {
300
+ return `${styleHeading(`nimicoding topic remediation open: ${report.topicId}`)}
301
+ ${styleLabel(localize("Path", "路径"))}: ${report.topicRef}
302
+ ${styleLabel(localize("Remediation", "Remediation"))}: ${report.remediationId}
303
+ ${styleLabel(localize("Wave", "Wave"))}: ${report.waveId}
304
+ ${styleLabel(localize("Kind", "类别"))}: ${report.kind}
305
+ ${styleLabel(localize("Reason", "原因"))}: ${report.reason}
306
+ ${styleLabel(localize("Remediation Ref", "Remediation 路径"))}: ${report.remediationRef}
307
+ ${styleLabel(localize("Wave State", "Wave 状态"))}: ${report.waveState}
308
+ `;
309
+ }
310
+ export function formatOverflowContinuation(report) {
311
+ return `${styleHeading(`nimicoding topic overflow continue: ${report.topicId}`)}
312
+ ${styleLabel(localize("Path", "路径"))}: ${report.topicRef}
313
+ ${styleLabel(localize("Wave", "Wave"))}: ${report.waveId}
314
+ ${styleLabel(localize("Overflowed Packet", "Overflowed Packet"))}: ${report.overflowedPacketId}
315
+ ${styleLabel(localize("Continuation Packet", "Continuation Packet"))}: ${report.continuationPacketId}
316
+ ${styleLabel(localize("Continuation Ref", "Continuation 路径"))}: ${report.continuationRef}
317
+ ${styleLabel(localize("Wave State", "Wave 状态"))}: ${report.waveState}
318
+ `;
319
+ }
320
+ export function formatPendingTransition(report, action) {
321
+ const lines = [
322
+ styleHeading(`nimicoding topic ${action}: ${report.topicId}`),
323
+ "",
324
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
325
+ `${styleLabel(localize("Topic State", "Topic 状态"))}: ${report.state}`,
326
+ `${styleLabel(localize("Pending Note Ref", "Pending Note 路径"))}: ${report.pendingNoteRef}`,
327
+ ];
328
+ if (report.reason) {
329
+ lines.push(`${styleLabel(localize("Reason", "原因"))}: ${report.reason}`);
330
+ }
331
+ if (report.criteriaMet) {
332
+ lines.push(`${styleLabel(localize("Criteria Met", "条件满足"))}: ${report.criteriaMet}`);
333
+ }
334
+ return `${lines.join("\n")}\n`;
335
+ }
336
+ export function formatCloseout(report, scope) {
337
+ const lines = [
338
+ styleHeading(`nimicoding topic closeout ${scope}: ${report.topicId}`),
339
+ "",
340
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
341
+ `${styleLabel(localize("Closeout Ref", "Closeout 路径"))}: ${report.closeoutRef}`,
342
+ ];
343
+ if (report.waveId) {
344
+ lines.push(`${styleLabel(localize("Wave", "Wave"))}: ${report.waveId}`);
345
+ lines.push(`${styleLabel(localize("Wave State", "Wave 状态"))}: ${report.waveState}`);
346
+ }
347
+ if (report.state) {
348
+ lines.push(`${styleLabel(localize("Topic State", "Topic 状态"))}: ${report.state}`);
349
+ }
350
+ if (report.currentTrueCloseStatus) {
351
+ lines.push(`${styleLabel(localize("True-Close", "True-Close"))}: ${report.currentTrueCloseStatus}`);
352
+ }
353
+ if (report.trueCloseRef) {
354
+ lines.push(`${styleLabel(localize("True-Close Ref", "True-Close 路径"))}: ${report.trueCloseRef}`);
355
+ }
356
+ return `${lines.join("\n")}\n`;
357
+ }
358
+ export function formatTrueCloseAudit(report) {
359
+ const lines = [
360
+ styleHeading(`nimicoding topic true-close-audit: ${report.topicId}`),
361
+ "",
362
+ `${styleLabel(localize("Path", "路径"))}: ${report.topicRef}`,
363
+ `${styleLabel(localize("Status", "状态"))}: ${report.status}`,
364
+ `${styleLabel(localize("Audit Ref", "Audit 路径"))}: ${report.auditRef}`,
365
+ `${styleLabel(localize("Judgement Ref", "Judgement 路径"))}: ${report.judgementRef}`,
366
+ "",
367
+ styleLabel(localize("Checks", "检查项")),
368
+ ...report.checks.map((entry) => `- [${entry.ok ? "ok" : "fail"}] ${entry.id}: ${entry.reason}`),
369
+ ];
370
+ return `${lines.join("\n")}\n`;
371
+ }
372
+ export function formatTopicCreate(report) {
373
+ return `${styleHeading(`nimicoding topic create: ${report.topicId}`)}
374
+ ${styleLabel(localize("Created", "已创建"))}: ${report.topicRef}
375
+ ${styleLabel(localize("State", "状态"))}: ${report.state}
376
+ ${styleLabel(localize("Title", "标题"))}: ${report.title}
377
+ ${styleMuted(localize(
378
+ "Next step: freeze the first bounded wave before admitted execution.",
379
+ "下一步:在 admitted execution 之前冻结第一个 bounded wave。",
380
+ ))}
381
+ `;
382
+ }
@@ -0,0 +1,33 @@
1
+ import process from "node:process";
2
+
3
+ import { buildTopicGoal } from "../lib/topic-goal.mjs";
4
+ import { resolveTopicProjectRoot } from "../lib/topic.mjs";
5
+ import { formatTopicGoalRefusal, writeJson } from "./topic-formatters.mjs";
6
+ import { parseTopicGoalOptions } from "./topic-options.mjs";
7
+
8
+ export async function runTopicGoal(args) {
9
+ const parsed = parseTopicGoalOptions(args);
10
+ if (!parsed.ok) {
11
+ process.stderr.write(parsed.error);
12
+ return 3;
13
+ }
14
+ try {
15
+ const projectRoot = await resolveTopicProjectRoot(process.cwd());
16
+ const report = await buildTopicGoal(projectRoot, parsed.options);
17
+ if (report.inputError) {
18
+ process.stderr.write(`${report.error}\n`);
19
+ return 3;
20
+ }
21
+ if (parsed.options.format === "json") {
22
+ writeJson(report);
23
+ } else if (report.ok) {
24
+ process.stdout.write(`${report.goal_command}\n`);
25
+ } else {
26
+ process.stderr.write(formatTopicGoalRefusal(report));
27
+ }
28
+ return report.ok ? 0 : 2;
29
+ } catch (error) {
30
+ process.stderr.write(`nimicoding topic goal failed: ${error?.message ?? String(error)}\n`);
31
+ return 1;
32
+ }
33
+ }
@@ -0,0 +1,27 @@
1
+ import { localize } from "../lib/ui.mjs";
2
+
3
+ export function requireOptionValue(name, next, errorPrefix) {
4
+ if (!next || next.startsWith("--")) {
5
+ return {
6
+ ok: false,
7
+ error: `${localize(
8
+ `${errorPrefix}: ${name} requires a value.`,
9
+ `${errorPrefix}:${name} 需要一个值。`,
10
+ )}\n`,
11
+ };
12
+ }
13
+ return { ok: true };
14
+ }
15
+
16
+ export function validateEnumOption(name, value, allowed, errorPrefix) {
17
+ if (!allowed.includes(value)) {
18
+ return {
19
+ ok: false,
20
+ error: `${localize(
21
+ `${errorPrefix}: unsupported ${name} value ${value}.`,
22
+ `${errorPrefix}:不支持的 ${name} 值 ${value}。`,
23
+ )}\n`,
24
+ };
25
+ }
26
+ return { ok: true };
27
+ }