@topogram/cli 0.3.64 → 0.3.66
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/package.json +1 -1
- package/src/adoption/plan/index.js +716 -0
- package/src/adoption/plan.js +12 -703
- package/src/adoption/reporting.js +1 -1
- package/src/agent-brief.js +7 -21
- package/src/agent-ops/query-builders/auth.js +375 -0
- package/src/agent-ops/query-builders/change-risk/change-plan.js +123 -0
- package/src/agent-ops/query-builders/change-risk/import-plan.js +49 -0
- package/src/agent-ops/query-builders/change-risk/maintained.js +286 -0
- package/src/agent-ops/query-builders/change-risk/review-packets.js +123 -0
- package/src/agent-ops/query-builders/change-risk/risk.js +189 -0
- package/src/agent-ops/query-builders/change-risk.js +25 -0
- package/src/agent-ops/query-builders/common.js +149 -0
- package/src/agent-ops/query-builders/maintained-risk.js +539 -0
- package/src/agent-ops/query-builders/maintained-shared.js +120 -0
- package/src/agent-ops/query-builders/multi-agent.js +547 -0
- package/src/agent-ops/query-builders/projection-impacts.js +514 -0
- package/src/agent-ops/query-builders/work-packets.js +417 -0
- package/src/agent-ops/query-builders/workflow-context-shared.js +300 -0
- package/src/agent-ops/query-builders/workflow-context.js +398 -0
- package/src/agent-ops/query-builders/workflow-presets-core.js +677 -0
- package/src/agent-ops/query-builders/workflow-presets.js +341 -0
- package/src/agent-ops/query-builders.d.ts +26 -26
- package/src/agent-ops/query-builders.js +42 -5021
- package/src/archive/jsonl.js +2 -2
- package/src/archive/resolver-bridge.js +1 -1
- package/src/archive/unarchive.js +2 -1
- package/src/catalog/constants.js +10 -0
- package/src/catalog/copy.js +65 -0
- package/src/catalog/diagnostics.js +15 -0
- package/src/catalog/entries.js +42 -0
- package/src/catalog/files.js +67 -0
- package/src/catalog/provenance.js +123 -0
- package/src/catalog/source.js +150 -0
- package/src/catalog/validation.js +252 -0
- package/src/catalog.d.ts +2 -0
- package/src/catalog.js +18 -746
- package/src/cli/command-parsers/project.js +3 -0
- package/src/cli/command-parsers/shared.js +1 -1
- package/src/cli/commands/agent.js +2 -2
- package/src/cli/commands/catalog/check.js +31 -0
- package/src/cli/commands/catalog/copy.js +59 -0
- package/src/cli/commands/catalog/doctor.js +248 -0
- package/src/cli/commands/catalog/help.js +21 -0
- package/src/cli/commands/catalog/list.js +52 -0
- package/src/cli/commands/catalog/runner.js +92 -0
- package/src/cli/commands/catalog/shared.js +17 -0
- package/src/cli/commands/catalog/show.js +134 -0
- package/src/cli/commands/catalog.js +30 -615
- package/src/cli/commands/check.js +3 -3
- package/src/cli/commands/doctor.js +2 -9
- package/src/cli/commands/generator-policy/package-info.js +162 -0
- package/src/cli/commands/generator-policy/payloads.js +372 -0
- package/src/cli/commands/generator-policy/printers.js +159 -0
- package/src/cli/commands/generator-policy/runner.js +81 -0
- package/src/cli/commands/generator-policy/shared.js +39 -0
- package/src/cli/commands/generator-policy.js +15 -783
- package/src/cli/commands/import/adopt.js +170 -0
- package/src/cli/commands/import/check.js +91 -0
- package/src/cli/commands/import/diff.js +84 -0
- package/src/cli/commands/import/help.js +47 -0
- package/src/cli/commands/import/paths.js +269 -0
- package/src/cli/commands/import/plan.js +292 -0
- package/src/cli/commands/import/refresh.js +471 -0
- package/src/cli/commands/import/status-history.js +196 -0
- package/src/cli/commands/import/workspace.js +233 -0
- package/src/cli/commands/import.js +33 -1732
- package/src/cli/commands/migrate.js +153 -0
- package/src/cli/commands/package/constants.js +17 -0
- package/src/cli/commands/package/doctor.js +240 -0
- package/src/cli/commands/package/help.js +27 -0
- package/src/cli/commands/package/lockfile.js +135 -0
- package/src/cli/commands/package/npm.js +97 -0
- package/src/cli/commands/package/reporting.js +35 -0
- package/src/cli/commands/package/runner.js +33 -0
- package/src/cli/commands/package/shared.js +9 -0
- package/src/cli/commands/package/update-cli.js +252 -0
- package/src/cli/commands/package/versions.js +35 -0
- package/src/cli/commands/package.js +29 -813
- package/src/cli/commands/query/change-plan.js +68 -0
- package/src/cli/commands/query/definitions.js +202 -0
- package/src/cli/commands/query/import-adopt.js +121 -0
- package/src/cli/commands/query/runner/artifacts.js +102 -0
- package/src/cli/commands/query/runner/boundaries.js +211 -0
- package/src/cli/commands/query/runner/change.js +182 -0
- package/src/cli/commands/query/runner/import-adopt.js +111 -0
- package/src/cli/commands/query/runner/index.js +31 -0
- package/src/cli/commands/query/runner/output.js +12 -0
- package/src/cli/commands/query/runner/workflow.js +241 -0
- package/src/cli/commands/query/runner.js +3 -0
- package/src/cli/commands/query/workflow-context.js +5 -0
- package/src/cli/commands/query/workspace.js +270 -0
- package/src/cli/commands/query.js +9 -1300
- package/src/cli/commands/source.js +3 -12
- package/src/cli/commands/template/baseline.js +100 -0
- package/src/cli/commands/template/check.js +467 -0
- package/src/cli/commands/template/constants.js +8 -0
- package/src/cli/commands/template/diagnostics.js +26 -0
- package/src/cli/commands/template/help.js +28 -0
- package/src/cli/commands/template/lifecycle.js +404 -0
- package/src/cli/commands/template/list-show.js +287 -0
- package/src/cli/commands/template/policy.js +422 -0
- package/src/cli/commands/template/shared.js +127 -0
- package/src/cli/commands/template/updates.js +352 -0
- package/src/cli/commands/template-runner.js +6 -6
- package/src/cli/commands/template.js +41 -2143
- package/src/cli/commands/trust.js +1 -1
- package/src/cli/commands/workflow.js +6 -1
- package/src/cli/dispatcher.js +6 -1
- package/src/cli/help.js +15 -14
- package/src/cli/migration-guidance.js +1 -1
- package/src/cli/output-safety.js +2 -1
- package/src/cli/path-normalization.js +3 -13
- package/src/generator/api/contracts.js +497 -0
- package/src/generator/api/metadata.js +221 -0
- package/src/generator/api/openapi.js +559 -0
- package/src/generator/api/schema.js +124 -0
- package/src/generator/api/types.d.ts +98 -0
- package/src/generator/api.js +3 -1195
- package/src/generator/context/domain-page.js +1 -1
- package/src/generator/context/shared/domain-sdlc.js +282 -0
- package/src/generator/context/shared/maintained-boundary.js +665 -0
- package/src/generator/context/shared/metrics.js +85 -0
- package/src/generator/context/shared/primitives.js +64 -0
- package/src/generator/context/shared/relationships.js +453 -0
- package/src/generator/context/shared/summaries.js +263 -0
- package/src/generator/context/shared/types.d.ts +207 -0
- package/src/generator/context/shared.d.ts +42 -0
- package/src/generator/context/shared.js +80 -1390
- package/src/generator/context/slice/core.js +397 -0
- package/src/generator/context/slice/sdlc.js +417 -0
- package/src/generator/context/slice/ui-packets.js +183 -0
- package/src/generator/context/slice.js +2 -859
- package/src/generator/context/task-mode.js +2 -2
- package/src/generator/registry/index.js +507 -0
- package/src/generator/registry.js +18 -504
- package/src/generator/runtime/environment/index.js +666 -0
- package/src/generator/runtime/environment.js +4 -666
- package/src/generator/runtime/runtime-check/index.js +554 -0
- package/src/generator/runtime/runtime-check.js +4 -554
- package/src/generator/runtime/shared/index.js +572 -0
- package/src/generator/runtime/shared.js +19 -570
- package/src/generator/sdlc/doc-page.js +1 -1
- package/src/generator/shared.d.ts +2 -0
- package/src/generator/surfaces/databases/lifecycle-shared.js +1 -1
- package/src/generator/surfaces/native/swiftui-templates/README.generated.md +1 -1
- package/src/generator/surfaces/shared.d.ts +3 -0
- package/src/generator/widget-conformance/behavior-report.js +258 -0
- package/src/generator/widget-conformance/checks.js +371 -0
- package/src/generator/widget-conformance/projection-context.js +200 -0
- package/src/generator/widget-conformance/report.js +166 -0
- package/src/generator/widget-conformance/types.d.ts +121 -0
- package/src/generator/widget-conformance.js +3 -824
- package/src/import/core/context.d.ts +3 -0
- package/src/import/core/context.js +5 -7
- package/src/import/core/contracts.d.ts +1 -0
- package/src/import/core/registry.d.ts +4 -0
- package/src/import/core/runner/candidates.js +337 -0
- package/src/import/core/runner/options.js +22 -0
- package/src/import/core/runner/reports.js +51 -0
- package/src/import/core/runner/run.js +79 -0
- package/src/import/core/runner/tracks.js +150 -0
- package/src/import/core/runner/ui-drafts.js +393 -0
- package/src/import/core/runner.js +3 -698
- package/src/import/core/shared/api-routes.js +221 -0
- package/src/import/core/shared/candidates.js +97 -0
- package/src/import/core/shared/files.js +177 -0
- package/src/import/core/shared/next-app.js +389 -0
- package/src/import/core/shared/types.d.ts +51 -0
- package/src/import/core/shared/ui-routes.js +230 -0
- package/src/import/core/shared.js +60 -861
- package/src/new-project/constants.js +128 -0
- package/src/new-project/create.js +90 -0
- package/src/new-project/json.js +28 -0
- package/src/new-project/metadata.js +96 -0
- package/src/new-project/package-spec.js +161 -0
- package/src/new-project/project-files.js +351 -0
- package/src/new-project/template-policy.js +269 -0
- package/src/new-project/template-resolution.js +370 -0
- package/src/new-project/template-snapshots.js +442 -0
- package/src/new-project/template-updates.js +512 -0
- package/src/new-project/types.d.ts +83 -0
- package/src/new-project.js +6 -2277
- package/src/parser.d.ts +87 -1
- package/src/parser.js +118 -0
- package/src/policy/review-boundaries.d.ts +15 -0
- package/src/project-config/index.js +591 -0
- package/src/project-config.js +19 -561
- package/src/resolver/enrich/acceptance-criterion.js +2 -0
- package/src/resolver/enrich/bug.js +2 -0
- package/src/resolver/enrich/pitch.js +2 -0
- package/src/resolver/enrich/requirement.js +2 -0
- package/src/resolver/enrich/task.js +2 -0
- package/src/resolver/index.js +19 -2089
- package/src/resolver/normalize.js +384 -1
- package/src/resolver/plans.js +168 -0
- package/src/resolver/projections-api.js +494 -0
- package/src/resolver/projections-db.js +133 -0
- package/src/resolver/projections-ui.js +317 -0
- package/src/resolver/shapes.js +251 -0
- package/src/resolver/shared.js +278 -0
- package/src/resolver/widgets.js +132 -0
- package/src/sdlc/adopt.js +6 -5
- package/src/sdlc/paths.js +3 -5
- package/src/sdlc/scaffold.js +2 -1
- package/src/template-trust/constants.js +62 -0
- package/src/template-trust/content.js +258 -0
- package/src/template-trust/diff.js +92 -0
- package/src/template-trust/policy.js +61 -0
- package/src/template-trust/record.js +90 -0
- package/src/template-trust/status.js +182 -0
- package/src/template-trust.js +24 -687
- package/src/text-helpers.d.ts +1 -0
- package/src/topogram-types.d.ts +69 -0
- package/src/validator/common.js +488 -0
- package/src/validator/data-model.js +237 -0
- package/src/validator/docs.js +167 -0
- package/src/validator/expressions.js +146 -1
- package/src/validator/index.d.ts +23 -0
- package/src/validator/index.js +32 -3585
- package/src/validator/kinds.d.ts +41 -0
- package/src/validator/kinds.js +2 -0
- package/src/validator/model-helpers.js +46 -0
- package/src/validator/per-kind/acceptance-criterion.js +5 -0
- package/src/validator/per-kind/bug.js +6 -0
- package/src/validator/per-kind/domain.js +15 -2
- package/src/validator/per-kind/pitch.js +7 -0
- package/src/validator/per-kind/requirement.js +5 -0
- package/src/validator/per-kind/task.js +7 -0
- package/src/validator/per-kind/widget.js +14 -0
- package/src/validator/projections/api-http-async.js +410 -0
- package/src/validator/projections/api-http-authz.js +88 -0
- package/src/validator/projections/api-http-core.js +205 -0
- package/src/validator/projections/api-http-policies.js +339 -0
- package/src/validator/projections/api-http-responses.js +233 -0
- package/src/validator/projections/api-http.js +44 -0
- package/src/validator/projections/db.js +353 -0
- package/src/validator/projections/generator-defaults.js +45 -0
- package/src/validator/projections/helpers.js +87 -0
- package/src/validator/projections/ui-helpers.js +214 -0
- package/src/validator/projections/ui-navigation.js +344 -0
- package/src/validator/projections/ui-structure.js +364 -0
- package/src/validator/projections/ui-widgets.js +493 -0
- package/src/validator/projections/ui.js +46 -0
- package/src/validator/registry.js +48 -1
- package/src/validator/utils.d.ts +20 -0
- package/src/validator/utils.js +115 -12
- package/src/widget-behavior.d.ts +1 -0
- package/src/workflows/import-app/api/collect.js +221 -0
- package/src/workflows/import-app/api/openapi.js +257 -0
- package/src/workflows/import-app/api/routes.js +327 -0
- package/src/workflows/import-app/api/sources.js +22 -0
- package/src/workflows/import-app/api.js +2 -797
- package/src/workflows/reconcile/adoption-plan/build.js +212 -0
- package/src/workflows/reconcile/adoption-plan/dependencies.js +75 -0
- package/src/workflows/reconcile/adoption-plan/outputs.js +153 -0
- package/src/workflows/reconcile/adoption-plan/paths.js +58 -0
- package/src/workflows/reconcile/adoption-plan/projection-patches.js +177 -0
- package/src/workflows/reconcile/adoption-plan/reasons.js +107 -0
- package/src/workflows/reconcile/adoption-plan.js +30 -740
- package/src/workflows/reconcile/auth/closures.js +115 -0
- package/src/workflows/reconcile/auth/formatters.js +142 -0
- package/src/workflows/reconcile/auth/inference.js +330 -0
- package/src/workflows/reconcile/auth/roles.js +122 -0
- package/src/workflows/reconcile/auth.js +35 -690
- package/src/workflows/reconcile/bundle-core/index.js +600 -0
- package/src/workflows/reconcile/bundle-core.js +12 -598
- package/src/workflows/reconcile/candidate-model.js +18 -2
- package/src/workflows/reconcile/canonical-surface.js +1 -1
- package/src/workflows/reconcile/impacts/adoption-plan.js +196 -0
- package/src/workflows/reconcile/impacts/indexes.js +105 -0
- package/src/workflows/reconcile/impacts/patches.js +252 -0
- package/src/workflows/reconcile/impacts/reports.js +80 -0
- package/src/workflows/reconcile/impacts.js +14 -623
- package/src/workflows/reconcile/renderers.js +41 -6
- package/src/workflows/shared.js +5 -11
- package/src/workspace-docs.d.ts +29 -0
- package/src/workspace-paths.js +328 -0
|
@@ -0,0 +1,278 @@
|
|
|
1
|
+
import { blockEntries, getFieldValue, symbolValues, valueAsArray } from "../validator.js";
|
|
2
|
+
|
|
3
|
+
export function groupBy(items, keyFn) {
|
|
4
|
+
const grouped = {};
|
|
5
|
+
for (const item of items) {
|
|
6
|
+
const key = keyFn(item);
|
|
7
|
+
if (!Object.hasOwn(grouped, key)) {
|
|
8
|
+
grouped[key] = [];
|
|
9
|
+
}
|
|
10
|
+
grouped[key].push(item);
|
|
11
|
+
}
|
|
12
|
+
return grouped;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function resolveReference(registry, id) {
|
|
16
|
+
return registry.get(id) || null;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function toRef(target) {
|
|
20
|
+
if (!target) {
|
|
21
|
+
return null;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
return {
|
|
25
|
+
id: target.id,
|
|
26
|
+
kind: target.kind
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export function resolveReferenceList(registry, value) {
|
|
31
|
+
return symbolValues(value).map((id) => ({
|
|
32
|
+
id,
|
|
33
|
+
target: toRef(resolveReference(registry, id))
|
|
34
|
+
}));
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export function resolveDomainTag(statement, registry) {
|
|
38
|
+
const domainField = statement.fields.find((field) => field.key === "domain");
|
|
39
|
+
if (!domainField || domainField.value.type !== "symbol") {
|
|
40
|
+
return null;
|
|
41
|
+
}
|
|
42
|
+
const id = domainField.value.value;
|
|
43
|
+
return {
|
|
44
|
+
id,
|
|
45
|
+
target: toRef(resolveReference(registry, id))
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
export function normalizeDomainScopeList(statement, key) {
|
|
50
|
+
const field = statement.fields.find((f) => f.key === key);
|
|
51
|
+
if (!field) return [];
|
|
52
|
+
const items = field.value.type === "list" || field.value.type === "sequence"
|
|
53
|
+
? field.value.items
|
|
54
|
+
: [field.value];
|
|
55
|
+
return items
|
|
56
|
+
.map((item) => (item.type === "string" || item.type === "symbol" ? item.value : null))
|
|
57
|
+
.filter((value) => value !== null);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
export function normalizeSequence(items) {
|
|
61
|
+
return items.map((item) => {
|
|
62
|
+
if (item.type === "symbol" || item.type === "string") {
|
|
63
|
+
return item.value;
|
|
64
|
+
}
|
|
65
|
+
if (item.type === "list") {
|
|
66
|
+
return item.items.map((nested) => nested.value);
|
|
67
|
+
}
|
|
68
|
+
return item.type;
|
|
69
|
+
});
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
export function normalizeFieldsBlock(statement, key = "fields") {
|
|
73
|
+
return blockEntries(getFieldValue(statement, key)).map((entry) => {
|
|
74
|
+
const [name, type, requiredness, ...rest] = entry.items;
|
|
75
|
+
let defaultValue = null;
|
|
76
|
+
|
|
77
|
+
for (let i = 0; i < rest.length - 1; i += 1) {
|
|
78
|
+
if (rest[i].type === "symbol" && rest[i].value === "default") {
|
|
79
|
+
defaultValue = rest[i + 1]?.value ?? null;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
return {
|
|
84
|
+
name: name?.value ?? null,
|
|
85
|
+
fieldType: type?.value ?? null,
|
|
86
|
+
requiredness: requiredness?.value ?? null,
|
|
87
|
+
defaultValue,
|
|
88
|
+
raw: normalizeSequence(entry.items),
|
|
89
|
+
loc: entry.loc
|
|
90
|
+
};
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
export function normalizeGenericBlock(statement, key) {
|
|
95
|
+
return blockEntries(getFieldValue(statement, key)).map((entry) => ({
|
|
96
|
+
raw: normalizeSequence(entry.items),
|
|
97
|
+
loc: entry.loc
|
|
98
|
+
}));
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
export function tokenValue(token) {
|
|
102
|
+
return token?.value ?? null;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
export function parseDefaultLiteral(token) {
|
|
106
|
+
if (!token) return null;
|
|
107
|
+
if (token.type === "list") {
|
|
108
|
+
return (token.items || []).map((item) => parseDefaultLiteral(item));
|
|
109
|
+
}
|
|
110
|
+
if (token.type === "string") {
|
|
111
|
+
return token.value ?? null;
|
|
112
|
+
}
|
|
113
|
+
if (token.type === "symbol") {
|
|
114
|
+
if (token.value === "true") return true;
|
|
115
|
+
if (token.value === "false") return false;
|
|
116
|
+
if (token.value === "null") return null;
|
|
117
|
+
if (/^-?\d+$/.test(token.value)) return Number.parseInt(token.value, 10);
|
|
118
|
+
if (/^-?\d+\.\d+$/.test(token.value)) return Number.parseFloat(token.value);
|
|
119
|
+
return token.value;
|
|
120
|
+
}
|
|
121
|
+
return token.value ?? null;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
export function parseLiteralToken(token) {
|
|
125
|
+
const value = tokenValue(token);
|
|
126
|
+
if (value == null) {
|
|
127
|
+
return null;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
if (value === "true") {
|
|
131
|
+
return { kind: "boolean", value: true };
|
|
132
|
+
}
|
|
133
|
+
if (value === "false") {
|
|
134
|
+
return { kind: "boolean", value: false };
|
|
135
|
+
}
|
|
136
|
+
if (value === "null") {
|
|
137
|
+
return { kind: "null", value: null };
|
|
138
|
+
}
|
|
139
|
+
if (/^-?\d+$/.test(value)) {
|
|
140
|
+
return { kind: "integer", value: Number.parseInt(value, 10) };
|
|
141
|
+
}
|
|
142
|
+
if (/^-?\d+\.\d+$/.test(value)) {
|
|
143
|
+
return { kind: "number", value: Number.parseFloat(value) };
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
return { kind: "symbol", value };
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
export function parseComparison(left, operator, right) {
|
|
150
|
+
if (!left || !operator || !right) {
|
|
151
|
+
return null;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
return {
|
|
155
|
+
type: "comparison",
|
|
156
|
+
left: tokenValue(left),
|
|
157
|
+
operator,
|
|
158
|
+
right: parseLiteralToken(right)
|
|
159
|
+
};
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
export function parseInvariantEntry(entry) {
|
|
163
|
+
const tokens = entry.items;
|
|
164
|
+
const values = tokens.map((item) => item.value);
|
|
165
|
+
const [a, b, c, d, e, f, g] = tokens;
|
|
166
|
+
|
|
167
|
+
if (tokenValue(b) === "requires") {
|
|
168
|
+
return {
|
|
169
|
+
type: "requires",
|
|
170
|
+
field: tokenValue(a),
|
|
171
|
+
predicate: parseComparison(c, tokenValue(d), e),
|
|
172
|
+
raw: values,
|
|
173
|
+
loc: entry.loc
|
|
174
|
+
};
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
if (tokenValue(b) === "length") {
|
|
178
|
+
return {
|
|
179
|
+
type: "length_check",
|
|
180
|
+
field: tokenValue(a),
|
|
181
|
+
operator: tokenValue(c),
|
|
182
|
+
value: parseLiteralToken(d),
|
|
183
|
+
raw: values,
|
|
184
|
+
loc: entry.loc
|
|
185
|
+
};
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
if (tokenValue(b) === "format") {
|
|
189
|
+
return {
|
|
190
|
+
type: "format_check",
|
|
191
|
+
field: tokenValue(a),
|
|
192
|
+
operator: tokenValue(c),
|
|
193
|
+
format: tokenValue(d),
|
|
194
|
+
raw: values,
|
|
195
|
+
loc: entry.loc
|
|
196
|
+
};
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
if (tokenValue(d) === "implies") {
|
|
200
|
+
return {
|
|
201
|
+
type: "implication",
|
|
202
|
+
when: parseComparison(a, tokenValue(b), c),
|
|
203
|
+
then:
|
|
204
|
+
tokenValue(f) === "is"
|
|
205
|
+
? {
|
|
206
|
+
type: "state_check",
|
|
207
|
+
field: tokenValue(e),
|
|
208
|
+
operator: "is",
|
|
209
|
+
value: parseLiteralToken(g)
|
|
210
|
+
}
|
|
211
|
+
: parseComparison(e, tokenValue(f), g),
|
|
212
|
+
raw: values,
|
|
213
|
+
loc: entry.loc
|
|
214
|
+
};
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
if (["==", "!=", "<", "<=", ">", ">="].includes(tokenValue(b))) {
|
|
218
|
+
return {
|
|
219
|
+
type: "comparison",
|
|
220
|
+
left: tokenValue(a),
|
|
221
|
+
operator: tokenValue(b),
|
|
222
|
+
right: parseLiteralToken(c),
|
|
223
|
+
raw: values,
|
|
224
|
+
loc: entry.loc
|
|
225
|
+
};
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
return {
|
|
229
|
+
type: "unknown",
|
|
230
|
+
raw: values,
|
|
231
|
+
loc: entry.loc
|
|
232
|
+
};
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
export function parseRuleExpression(value) {
|
|
236
|
+
const text = valueAsArray(value).map((item) => item.value).join(" ").trim();
|
|
237
|
+
const match = text.match(/^(.+?)\s*(==|!=|<=|>=|<|>)\s*(.+)$/);
|
|
238
|
+
if (!match) {
|
|
239
|
+
return {
|
|
240
|
+
type: "unknown",
|
|
241
|
+
raw: text
|
|
242
|
+
};
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
return {
|
|
246
|
+
type: "comparison",
|
|
247
|
+
left: match[1].trim(),
|
|
248
|
+
operator: match[2],
|
|
249
|
+
right: {
|
|
250
|
+
kind: "symbol",
|
|
251
|
+
value: match[3].trim()
|
|
252
|
+
},
|
|
253
|
+
raw: text
|
|
254
|
+
};
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
export function parseSymbolNode(value) {
|
|
258
|
+
return {
|
|
259
|
+
type: "symbol",
|
|
260
|
+
id: value
|
|
261
|
+
};
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
export function parseSymbolNodes(values) {
|
|
265
|
+
return values.map((value, index) => ({
|
|
266
|
+
...parseSymbolNode(value),
|
|
267
|
+
order: index
|
|
268
|
+
}));
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
export function parseReferenceNodes(values) {
|
|
272
|
+
return values.map((entry, index) => ({
|
|
273
|
+
type: "reference",
|
|
274
|
+
id: entry.id,
|
|
275
|
+
target: entry.target || { id: entry.id, kind: null },
|
|
276
|
+
order: index
|
|
277
|
+
}));
|
|
278
|
+
}
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
import { blockEntries, getFieldValue, symbolValues } from "../validator.js";
|
|
2
|
+
import { normalizeSequence, parseDefaultLiteral, parseReferenceNodes, tokenValue } from "./shared.js";
|
|
3
|
+
|
|
4
|
+
export function normalizeWidgetProps(statement) {
|
|
5
|
+
return blockEntries(getFieldValue(statement, "props")).map((entry) => {
|
|
6
|
+
const [name, type, requiredness, ...rest] = entry.items;
|
|
7
|
+
let defaultValue = null;
|
|
8
|
+
|
|
9
|
+
for (let i = 0; i < rest.length - 1; i += 1) {
|
|
10
|
+
if (rest[i].type === "symbol" && rest[i].value === "default") {
|
|
11
|
+
defaultValue = parseDefaultLiteral(rest[i + 1]);
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
return {
|
|
16
|
+
name: name?.value ?? null,
|
|
17
|
+
fieldType: type?.value ?? null,
|
|
18
|
+
requiredness: requiredness?.value ?? null,
|
|
19
|
+
defaultValue,
|
|
20
|
+
raw: normalizeSequence(entry.items),
|
|
21
|
+
loc: entry.loc
|
|
22
|
+
};
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function normalizeWidgetEvents(statement, registry) {
|
|
27
|
+
return blockEntries(getFieldValue(statement, "events")).map((entry) => {
|
|
28
|
+
const [eventName, shapeRef] = entry.items;
|
|
29
|
+
const shapeId = tokenValue(shapeRef);
|
|
30
|
+
return {
|
|
31
|
+
id: tokenValue(eventName),
|
|
32
|
+
shape: shapeId
|
|
33
|
+
? {
|
|
34
|
+
id: shapeId,
|
|
35
|
+
kind: registry.get(shapeId)?.kind || null
|
|
36
|
+
}
|
|
37
|
+
: null,
|
|
38
|
+
raw: normalizeSequence(entry.items),
|
|
39
|
+
loc: entry.loc
|
|
40
|
+
};
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export function normalizeWidgetSlots(statement) {
|
|
45
|
+
return blockEntries(getFieldValue(statement, "slots")).map((entry) => ({
|
|
46
|
+
id: tokenValue(entry.items[0]),
|
|
47
|
+
description: tokenValue(entry.items[1]),
|
|
48
|
+
raw: normalizeSequence(entry.items),
|
|
49
|
+
loc: entry.loc
|
|
50
|
+
}));
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
export function normalizeBehaviorValue(token) {
|
|
54
|
+
if (!token) {
|
|
55
|
+
return null;
|
|
56
|
+
}
|
|
57
|
+
if (token.type === "list") {
|
|
58
|
+
return token.items.map((item) => normalizeBehaviorValue(item));
|
|
59
|
+
}
|
|
60
|
+
return parseDefaultLiteral(token);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export function normalizeWidgetBehaviors(statement) {
|
|
64
|
+
const structured = blockEntries(getFieldValue(statement, "behaviors")).map((entry) => {
|
|
65
|
+
const kind = tokenValue(entry.items[0]);
|
|
66
|
+
const directives = {};
|
|
67
|
+
for (let i = 1; i < entry.items.length; i += 2) {
|
|
68
|
+
const key = tokenValue(entry.items[i]);
|
|
69
|
+
if (!key) {
|
|
70
|
+
continue;
|
|
71
|
+
}
|
|
72
|
+
directives[key] = normalizeBehaviorValue(entry.items[i + 1]);
|
|
73
|
+
}
|
|
74
|
+
return {
|
|
75
|
+
kind,
|
|
76
|
+
directives,
|
|
77
|
+
source: "structured",
|
|
78
|
+
raw: normalizeSequence(entry.items),
|
|
79
|
+
loc: entry.loc
|
|
80
|
+
};
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
const structuredKinds = new Set(structured.map((entry) => entry.kind));
|
|
84
|
+
const shorthand = symbolValues(getFieldValue(statement, "behavior"))
|
|
85
|
+
.filter((kind) => !structuredKinds.has(kind))
|
|
86
|
+
.map((kind) => ({
|
|
87
|
+
kind,
|
|
88
|
+
directives: {},
|
|
89
|
+
source: "shorthand",
|
|
90
|
+
raw: [kind],
|
|
91
|
+
loc: null
|
|
92
|
+
}));
|
|
93
|
+
|
|
94
|
+
return [...structured, ...shorthand];
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
export function buildWidgetContract(statement) {
|
|
98
|
+
return {
|
|
99
|
+
type: "ui_widget_contract",
|
|
100
|
+
id: statement.id,
|
|
101
|
+
name: statement.name || statement.id,
|
|
102
|
+
description: statement.description || null,
|
|
103
|
+
category: statement.category || null,
|
|
104
|
+
version: statement.version || null,
|
|
105
|
+
status: statement.status || null,
|
|
106
|
+
props: (statement.props || []).map((prop) => ({
|
|
107
|
+
name: prop.name,
|
|
108
|
+
type: prop.fieldType,
|
|
109
|
+
requiredness: prop.requiredness,
|
|
110
|
+
defaultValue: prop.defaultValue ?? null
|
|
111
|
+
})),
|
|
112
|
+
events: (statement.events || []).map((event) => ({
|
|
113
|
+
id: event.id,
|
|
114
|
+
shape: event.shape || null
|
|
115
|
+
})),
|
|
116
|
+
slots: (statement.slots || []).map((slot) => ({
|
|
117
|
+
id: slot.id,
|
|
118
|
+
description: slot.description || null
|
|
119
|
+
})),
|
|
120
|
+
behavior: [...(statement.behavior || [])],
|
|
121
|
+
behaviors: (statement.behaviors || []).map((behavior) => ({
|
|
122
|
+
kind: behavior.kind,
|
|
123
|
+
directives: { ...(behavior.directives || {}) },
|
|
124
|
+
source: behavior.source || null
|
|
125
|
+
})),
|
|
126
|
+
patterns: [...(statement.patterns || [])],
|
|
127
|
+
regions: [...(statement.regions || [])],
|
|
128
|
+
approvals: [...(statement.approvals || [])],
|
|
129
|
+
lookups: parseReferenceNodes(statement.lookups || []),
|
|
130
|
+
dependencies: parseReferenceNodes(statement.dependencies || [])
|
|
131
|
+
};
|
|
132
|
+
}
|
package/src/sdlc/adopt.js
CHANGED
|
@@ -6,6 +6,7 @@
|
|
|
6
6
|
|
|
7
7
|
import { existsSync, mkdirSync, readdirSync, statSync } from "node:fs";
|
|
8
8
|
import path from "node:path";
|
|
9
|
+
import { DEFAULT_TOPO_FOLDER_NAME, resolveTopoRoot } from "../workspace-paths.js";
|
|
9
10
|
|
|
10
11
|
const SDLC_FOLDERS = [
|
|
11
12
|
"pitches",
|
|
@@ -17,7 +18,7 @@ const SDLC_FOLDERS = [
|
|
|
17
18
|
];
|
|
18
19
|
|
|
19
20
|
function ensureFolder(root, name) {
|
|
20
|
-
const dir = path.join(root,
|
|
21
|
+
const dir = path.join(resolveTopoRoot(root), name);
|
|
21
22
|
if (!existsSync(dir)) {
|
|
22
23
|
mkdirSync(dir, { recursive: true });
|
|
23
24
|
return { name, created: true };
|
|
@@ -29,8 +30,8 @@ function scanPressure(root) {
|
|
|
29
30
|
// Pressure scan: count statements per kind so the operator knows the
|
|
30
31
|
// workspace's current shape. We do not attempt to backfill — that's the
|
|
31
32
|
// operator's job after they pick a starting point.
|
|
32
|
-
const tg =
|
|
33
|
-
if (!existsSync(tg)) return { error: `No '
|
|
33
|
+
const tg = resolveTopoRoot(root);
|
|
34
|
+
if (!existsSync(tg)) return { error: `No '${DEFAULT_TOPO_FOLDER_NAME}/' workspace folder found at ${root}` };
|
|
34
35
|
let totalFiles = 0;
|
|
35
36
|
function walk(dir) {
|
|
36
37
|
for (const entry of readdirSync(dir)) {
|
|
@@ -50,8 +51,8 @@ function scanPressure(root) {
|
|
|
50
51
|
|
|
51
52
|
export function sdlcAdopt(workspaceRoot) {
|
|
52
53
|
const root = path.resolve(workspaceRoot);
|
|
53
|
-
if (!existsSync(
|
|
54
|
-
return { ok: false, error: `No '
|
|
54
|
+
if (!existsSync(resolveTopoRoot(root))) {
|
|
55
|
+
return { ok: false, error: `No '${DEFAULT_TOPO_FOLDER_NAME}/' workspace folder at ${root}; run 'topogram new' first` };
|
|
55
56
|
}
|
|
56
57
|
const folders = SDLC_FOLDERS.map((name) => ensureFolder(root, name));
|
|
57
58
|
const pressure = scanPressure(root);
|
package/src/sdlc/paths.js
CHANGED
|
@@ -1,11 +1,9 @@
|
|
|
1
|
-
import
|
|
1
|
+
import { resolveTopoRoot, resolveWorkspaceContext } from "../workspace-paths.js";
|
|
2
2
|
|
|
3
3
|
export function topogramRootForSdlc(inputPath) {
|
|
4
|
-
|
|
5
|
-
return path.basename(absolute) === "topogram" ? absolute : path.join(absolute, "topogram");
|
|
4
|
+
return resolveTopoRoot(inputPath);
|
|
6
5
|
}
|
|
7
6
|
|
|
8
7
|
export function projectRootForSdlc(inputPath) {
|
|
9
|
-
|
|
10
|
-
return path.basename(absolute) === "topogram" ? path.dirname(absolute) : absolute;
|
|
8
|
+
return resolveWorkspaceContext(inputPath).projectRoot;
|
|
11
9
|
}
|
package/src/sdlc/scaffold.js
CHANGED
|
@@ -3,6 +3,7 @@
|
|
|
3
3
|
|
|
4
4
|
import { existsSync, mkdirSync, writeFileSync } from "node:fs";
|
|
5
5
|
import path from "node:path";
|
|
6
|
+
import { resolveTopoRoot } from "../workspace-paths.js";
|
|
6
7
|
|
|
7
8
|
const TEMPLATES = {
|
|
8
9
|
pitch: (slug) => `pitch pitch_${slug} {
|
|
@@ -73,7 +74,7 @@ export function scaffoldNew(workspaceRoot, kind, slug) {
|
|
|
73
74
|
if (!slug || !/^[a-z][a-z0-9_]*$/.test(slug)) {
|
|
74
75
|
return { ok: false, error: `Invalid slug '${slug}' — must match /^[a-z][a-z0-9_]*$/` };
|
|
75
76
|
}
|
|
76
|
-
const targetDir = path.join(workspaceRoot,
|
|
77
|
+
const targetDir = path.join(resolveTopoRoot(workspaceRoot), `${kind === "acceptance_criterion" ? "acceptance_criteria" : kind + "s"}`);
|
|
77
78
|
if (!existsSync(targetDir)) mkdirSync(targetDir, { recursive: true });
|
|
78
79
|
const targetFile = path.join(targetDir, `${slug}.tg`);
|
|
79
80
|
if (existsSync(targetFile)) {
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
|
|
3
|
+
export const TEMPLATE_TRUST_FILE = ".topogram-template-trust.json";
|
|
4
|
+
export const TEMPLATE_TRUST_POLICY = "topogram-template-executable-implementation-v1";
|
|
5
|
+
export const IGNORED_IMPLEMENTATION_ENTRIES = new Set([".DS_Store", "node_modules", ".tmp"]);
|
|
6
|
+
export const MAX_TEXT_DIFF_BYTES = 256 * 1024;
|
|
7
|
+
export const TRUST_REVIEW_COMMANDS = "`topogram trust status`, `topogram trust diff`, and `topogram trust template`";
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* @param {string} value
|
|
11
|
+
* @returns {string}
|
|
12
|
+
*/
|
|
13
|
+
export function normalizeRoot(value) {
|
|
14
|
+
return String(value || "").replace(/\\/g, "/");
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* @param {string} value
|
|
19
|
+
* @returns {string}
|
|
20
|
+
*/
|
|
21
|
+
export function normalizeRelativePath(value) {
|
|
22
|
+
return value.replace(/\\/g, "/");
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* @param {string} relativePath
|
|
27
|
+
* @returns {string}
|
|
28
|
+
*/
|
|
29
|
+
export function unsupportedImplementationSymlinkMessage(relativePath) {
|
|
30
|
+
return `Template implementation contains unsupported symlink '${relativePath}'. Implementation trust hashes real files under implementation/; symlinks can point outside the trusted root. Replace symlinks with real files under implementation/, then run ${TRUST_REVIEW_COMMANDS} after review.`;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* @param {string} modulePath
|
|
35
|
+
* @returns {string}
|
|
36
|
+
*/
|
|
37
|
+
export function implementationOutsideRootMessage(modulePath) {
|
|
38
|
+
return `Template implementation module '${modulePath}' must be under implementation/ for template-attached projects. Keep executable template code inside implementation/ so the trust record covers what topogram generate may load. Move the module back under implementation/, then run ${TRUST_REVIEW_COMMANDS} after review.`;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* @param {string|string[]} issueOrIssues
|
|
43
|
+
* @returns {string}
|
|
44
|
+
*/
|
|
45
|
+
export function templateTrustRecoveryGuidance(issueOrIssues) {
|
|
46
|
+
const issues = Array.isArray(issueOrIssues) ? issueOrIssues : [issueOrIssues];
|
|
47
|
+
const text = issues.join("\n");
|
|
48
|
+
if (issues.length > 0 && issues.every((issue) =>
|
|
49
|
+
issue.includes("topogram trust status") &&
|
|
50
|
+
issue.includes("topogram trust diff") &&
|
|
51
|
+
issue.includes("topogram trust template")
|
|
52
|
+
)) {
|
|
53
|
+
return "";
|
|
54
|
+
}
|
|
55
|
+
if (text.includes("unsupported symlink")) {
|
|
56
|
+
return `Replace symlinks with real files under implementation/, then run ${TRUST_REVIEW_COMMANDS} after review.`;
|
|
57
|
+
}
|
|
58
|
+
if (text.includes("must be under implementation/")) {
|
|
59
|
+
return `Keep executable template code under implementation/ so it can be hashed and trusted; move the module back under implementation/, then run ${TRUST_REVIEW_COMMANDS} after review.`;
|
|
60
|
+
}
|
|
61
|
+
return `Run \`topogram trust status\` and \`topogram trust diff\` to review implementation changes; after review, run \`topogram trust template\` to trust the current files.`;
|
|
62
|
+
}
|