@atlashub/smartstack-cli 3.21.0 → 3.23.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 (36) hide show
  1. package/dist/index.js +17 -5
  2. package/dist/index.js.map +1 -1
  3. package/dist/mcp-entry.mjs +155 -162
  4. package/dist/mcp-entry.mjs.map +1 -1
  5. package/package.json +1 -1
  6. package/templates/skills/apex/SKILL.md +21 -0
  7. package/templates/skills/apex/references/smartstack-api.md +481 -0
  8. package/templates/skills/apex/references/smartstack-layers.md +85 -15
  9. package/templates/skills/apex/steps/step-00-init.md +27 -14
  10. package/templates/skills/apex/steps/step-01-analyze.md +18 -0
  11. package/templates/skills/apex/steps/step-03-execute.md +8 -6
  12. package/templates/skills/apex/steps/step-04-validate.md +92 -0
  13. package/templates/skills/apex/steps/step-07-tests.md +29 -5
  14. package/templates/skills/application/references/application-roles-template.md +2 -2
  15. package/templates/skills/application/steps/step-05-frontend.md +40 -35
  16. package/templates/skills/application/templates-frontend.md +64 -36
  17. package/templates/skills/business-analyse/html/ba-interactive.html +80 -6
  18. package/templates/skills/business-analyse/html/src/scripts/05-render-specs.js +38 -6
  19. package/templates/skills/business-analyse/html/src/styles/06-wireframes.css +42 -0
  20. package/templates/skills/business-analyse/references/acceptance-criteria.md +169 -0
  21. package/templates/skills/business-analyse/references/deploy-data-build.md +5 -3
  22. package/templates/skills/business-analyse/references/handoff-file-templates.md +2 -1
  23. package/templates/skills/business-analyse/references/naming-conventions.md +245 -0
  24. package/templates/skills/business-analyse/references/validate-incremental-html.md +26 -4
  25. package/templates/skills/business-analyse/references/validation-checklist.md +31 -11
  26. package/templates/skills/business-analyse/references/wireframe-svg-style-guide.md +335 -0
  27. package/templates/skills/business-analyse/steps/step-03b-ui.md +59 -0
  28. package/templates/skills/business-analyse/steps/step-03c-compile.md +114 -0
  29. package/templates/skills/business-analyse/steps/step-03d-validate.md +144 -22
  30. package/templates/skills/business-analyse/steps/step-05a-handoff.md +114 -2
  31. package/templates/skills/business-analyse/steps/step-05b-deploy.md +28 -0
  32. package/templates/skills/ralph-loop/references/category-rules.md +5 -2
  33. package/templates/skills/ralph-loop/references/compact-loop.md +52 -1
  34. package/templates/skills/ralph-loop/references/core-seed-data.md +232 -21
  35. package/templates/skills/ralph-loop/steps/step-01-task.md +36 -4
  36. package/templates/skills/ralph-loop/steps/step-02-execute.md +81 -0
@@ -993,6 +993,48 @@ body {
993
993
  .svg-wireframe svg {
994
994
  max-width: 100%;
995
995
  height: auto;
996
+ display: block;
997
+ }
998
+
999
+ /* ============================================
1000
+ WIREFRAME VIEW TOGGLE (SVG / ASCII)
1001
+ ============================================ */
1002
+ .wireframe-toggle {
1003
+ display: flex;
1004
+ gap: 2px;
1005
+ background: var(--bg-dark);
1006
+ border-radius: 6px;
1007
+ padding: 2px;
1008
+ border: 1px solid var(--border);
1009
+ margin-left: auto;
1010
+ }
1011
+
1012
+ .wireframe-toggle-btn {
1013
+ padding: 0.25rem 0.6rem;
1014
+ font-size: 0.7rem;
1015
+ font-weight: 500;
1016
+ letter-spacing: 0.02em;
1017
+ border: none;
1018
+ border-radius: 4px;
1019
+ cursor: pointer;
1020
+ background: transparent;
1021
+ color: var(--text-muted);
1022
+ transition: all 0.15s ease;
1023
+ font-family: inherit;
1024
+ }
1025
+
1026
+ .wireframe-toggle-btn:hover {
1027
+ color: var(--text-bright);
1028
+ background: var(--bg-hover);
1029
+ }
1030
+
1031
+ .wireframe-toggle-btn.active {
1032
+ background: var(--primary);
1033
+ color: #ffffff;
1034
+ }
1035
+
1036
+ .wireframe-view:not(.active) {
1037
+ display: none;
996
1038
  }
997
1039
 
998
1040
  .wireframe-description {
@@ -3279,18 +3321,30 @@ function renderModuleMockups(code) {
3279
3321
  </div>`;
3280
3322
  }
3281
3323
 
3282
- return wireframes.map((wf, i) => `
3324
+ return wireframes.map((wf, i) => {
3325
+ const hasSvg = !!wf.svgContent;
3326
+ const wireframeId = `wf-${code}-${i}`;
3327
+
3328
+ return `
3283
3329
  <div class="mockup-frame" style="${i > 0 ? 'margin-top:1.5rem;' : ''}">
3284
3330
  <div class="mockup-toolbar">
3285
3331
  <div class="mockup-dot mockup-dot-red"></div>
3286
3332
  <div class="mockup-dot mockup-dot-yellow"></div>
3287
3333
  <div class="mockup-dot mockup-dot-green"></div>
3288
3334
  <span class="mockup-title">${wf.screen || wf.section}</span>
3335
+ ${hasSvg ? `
3336
+ <div class="wireframe-toggle">
3337
+ <button class="wireframe-toggle-btn active" data-target="${wireframeId}" data-view="svg" onclick="toggleWireframeView('${wireframeId}', 'svg')">SVG</button>
3338
+ <button class="wireframe-toggle-btn" data-target="${wireframeId}" data-view="ascii" onclick="toggleWireframeView('${wireframeId}', 'ascii')">ASCII</button>
3339
+ </div>` : ''}
3289
3340
  </div>
3290
- <div class="mockup-content">
3291
- ${wf.format === 'ascii'
3292
- ? `<pre class="ascii-wireframe">${wf.content || ''}</pre>`
3293
- : `<div class="svg-wireframe">${wf.content || ''}</div>`}
3341
+ <div class="mockup-content" id="${wireframeId}">
3342
+ ${hasSvg
3343
+ ? `<div class="svg-wireframe wireframe-view active" data-view="svg">${wf.svgContent}</div>
3344
+ <pre class="ascii-wireframe wireframe-view" data-view="ascii" style="display:none;">${wf.content || ''}</pre>`
3345
+ : (wf.format === 'ascii'
3346
+ ? `<pre class="ascii-wireframe">${wf.content || ''}</pre>`
3347
+ : `<div class="svg-wireframe">${wf.content || ''}</div>`)}
3294
3348
  </div>
3295
3349
  ${wf.description ? `
3296
3350
  <div class="wireframe-description">
@@ -3329,7 +3383,27 @@ function renderModuleMockups(code) {
3329
3383
  >${getWireframeComment(code, wf.screen)}</textarea>
3330
3384
  </div>
3331
3385
  </div>
3332
- `).join('');
3386
+ `}).join('');
3387
+ }
3388
+
3389
+ function toggleWireframeView(wireframeId, view) {
3390
+ const container = document.getElementById(wireframeId);
3391
+ if (!container) return;
3392
+
3393
+ // Toggle visibility of wireframe views
3394
+ container.querySelectorAll('.wireframe-view').forEach(el => {
3395
+ const isTarget = el.dataset.view === view;
3396
+ el.style.display = isTarget ? '' : 'none';
3397
+ el.classList.toggle('active', isTarget);
3398
+ });
3399
+
3400
+ // Toggle button active states
3401
+ const toolbar = container.closest('.mockup-frame').querySelector('.wireframe-toggle');
3402
+ if (toolbar) {
3403
+ toolbar.querySelectorAll('.wireframe-toggle-btn').forEach(btn => {
3404
+ btn.classList.toggle('active', btn.dataset.view === view);
3405
+ });
3406
+ }
3333
3407
  }
3334
3408
 
3335
3409
  function getPermRoles() {
@@ -387,18 +387,30 @@ function renderModuleMockups(code) {
387
387
  </div>`;
388
388
  }
389
389
 
390
- return wireframes.map((wf, i) => `
390
+ return wireframes.map((wf, i) => {
391
+ const hasSvg = !!wf.svgContent;
392
+ const wireframeId = `wf-${code}-${i}`;
393
+
394
+ return `
391
395
  <div class="mockup-frame" style="${i > 0 ? 'margin-top:1.5rem;' : ''}">
392
396
  <div class="mockup-toolbar">
393
397
  <div class="mockup-dot mockup-dot-red"></div>
394
398
  <div class="mockup-dot mockup-dot-yellow"></div>
395
399
  <div class="mockup-dot mockup-dot-green"></div>
396
400
  <span class="mockup-title">${wf.screen || wf.section}</span>
401
+ ${hasSvg ? `
402
+ <div class="wireframe-toggle">
403
+ <button class="wireframe-toggle-btn active" data-target="${wireframeId}" data-view="svg" onclick="toggleWireframeView('${wireframeId}', 'svg')">SVG</button>
404
+ <button class="wireframe-toggle-btn" data-target="${wireframeId}" data-view="ascii" onclick="toggleWireframeView('${wireframeId}', 'ascii')">ASCII</button>
405
+ </div>` : ''}
397
406
  </div>
398
- <div class="mockup-content">
399
- ${wf.format === 'ascii'
400
- ? `<pre class="ascii-wireframe">${wf.content || ''}</pre>`
401
- : `<div class="svg-wireframe">${wf.content || ''}</div>`}
407
+ <div class="mockup-content" id="${wireframeId}">
408
+ ${hasSvg
409
+ ? `<div class="svg-wireframe wireframe-view active" data-view="svg">${wf.svgContent}</div>
410
+ <pre class="ascii-wireframe wireframe-view" data-view="ascii" style="display:none;">${wf.content || ''}</pre>`
411
+ : (wf.format === 'ascii'
412
+ ? `<pre class="ascii-wireframe">${wf.content || ''}</pre>`
413
+ : `<div class="svg-wireframe">${wf.content || ''}</div>`)}
402
414
  </div>
403
415
  ${wf.description ? `
404
416
  <div class="wireframe-description">
@@ -437,7 +449,27 @@ function renderModuleMockups(code) {
437
449
  >${getWireframeComment(code, wf.screen)}</textarea>
438
450
  </div>
439
451
  </div>
440
- `).join('');
452
+ `}).join('');
453
+ }
454
+
455
+ function toggleWireframeView(wireframeId, view) {
456
+ const container = document.getElementById(wireframeId);
457
+ if (!container) return;
458
+
459
+ // Toggle visibility of wireframe views
460
+ container.querySelectorAll('.wireframe-view').forEach(el => {
461
+ const isTarget = el.dataset.view === view;
462
+ el.style.display = isTarget ? '' : 'none';
463
+ el.classList.toggle('active', isTarget);
464
+ });
465
+
466
+ // Toggle button active states
467
+ const toolbar = container.closest('.mockup-frame').querySelector('.wireframe-toggle');
468
+ if (toolbar) {
469
+ toolbar.querySelectorAll('.wireframe-toggle-btn').forEach(btn => {
470
+ btn.classList.toggle('active', btn.dataset.view === view);
471
+ });
472
+ }
441
473
  }
442
474
 
443
475
  function getPermRoles() {
@@ -61,6 +61,48 @@
61
61
  .svg-wireframe svg {
62
62
  max-width: 100%;
63
63
  height: auto;
64
+ display: block;
65
+ }
66
+
67
+ /* ============================================
68
+ WIREFRAME VIEW TOGGLE (SVG / ASCII)
69
+ ============================================ */
70
+ .wireframe-toggle {
71
+ display: flex;
72
+ gap: 2px;
73
+ background: var(--bg-dark);
74
+ border-radius: 6px;
75
+ padding: 2px;
76
+ border: 1px solid var(--border);
77
+ margin-left: auto;
78
+ }
79
+
80
+ .wireframe-toggle-btn {
81
+ padding: 0.25rem 0.6rem;
82
+ font-size: 0.7rem;
83
+ font-weight: 500;
84
+ letter-spacing: 0.02em;
85
+ border: none;
86
+ border-radius: 4px;
87
+ cursor: pointer;
88
+ background: transparent;
89
+ color: var(--text-muted);
90
+ transition: all 0.15s ease;
91
+ font-family: inherit;
92
+ }
93
+
94
+ .wireframe-toggle-btn:hover {
95
+ color: var(--text-bright);
96
+ background: var(--bg-hover);
97
+ }
98
+
99
+ .wireframe-toggle-btn.active {
100
+ background: var(--primary);
101
+ color: #ffffff;
102
+ }
103
+
104
+ .wireframe-view:not(.active) {
105
+ display: none;
64
106
  }
65
107
 
66
108
  .wireframe-description {
@@ -0,0 +1,169 @@
1
+ # Module Acceptance Criteria (Shared Reference)
2
+
3
+ > **Loaded by:** step-03d-validate (section 11-POST-CHECK-BASH), step-05a-handoff (pre-handoff gate), ralph-loop step-01-task (input validation)
4
+ > **Purpose:** Define measurable, bash-verifiable acceptance criteria that a module MUST pass before being marked "specified".
5
+ > **Key principle:** These checks read the REAL feature.json file on disk — NOT in-memory data the model "thinks" it wrote.
6
+
7
+ ---
8
+
9
+ ## Why Bash-Verifiable?
10
+
11
+ All previous checks were **pseudocode interpreted by the model**. The model can "decide" a check passes even when it fails (hallucination of validation). Bash checks are **objective** — they read the actual file and count real elements.
12
+
13
+ ---
14
+
15
+ ## Acceptance Criteria Table
16
+
17
+ | # | Criterion | Minimum | Field Path | Blocking | Category |
18
+ |---|-----------|---------|------------|----------|----------|
19
+ | AC-01 | Entities present | 1 | `analysis.entities[]` | YES | Data Model |
20
+ | AC-02 | Entity attributes have `type` field | ALL | `analysis.entities[].attributes[].type` | YES | Data Model |
21
+ | AC-03 | Use cases present | 2 | `specification.useCases[]` | YES | Requirements |
22
+ | AC-04 | Functional requirements present | 4 | `specification.functionalRequirements[]` | YES | Requirements |
23
+ | AC-05 | Wireframes present | 1 | `specification.uiWireframes[] \|\| specification.wireframes[]` | YES | UI |
24
+ | AC-06 | Wireframes >= sections | count match | wireframes.length >= sections.length | YES | UI |
25
+ | AC-07 | Wireframe content non-empty | ALL | `wireframe.mockup \|\| wireframe.ascii \|\| wireframe.content` | YES | UI |
26
+ | AC-08 | Sections present | 1 | `specification.sections[]` | YES | UI |
27
+ | AC-09 | SeedDataCore 7 arrays non-empty | 7/7 | `specification.seedDataCore.*` | YES | Seed Data |
28
+ | AC-10 | Gherkin scenarios is array | `Array.isArray` | `specification.gherkinScenarios` | YES | Format |
29
+ | AC-11 | API endpoints present | 1 | `specification.apiEndpoints[]` | YES | API |
30
+ | AC-12 | Messages present | 4 | `specification.messages[]` | YES | Messages |
31
+ | AC-13 | Validations present | 1 | `specification.validations[]` | YES | Validation |
32
+ | AC-14 | Wireframe field naming | canonical | `screen` not `title`, `mockup` not `ascii` | WARNING | Convention |
33
+ | AC-15 | Validation rules format | ALL array | `specification.validations[].rules` is `Array` | YES | Format |
34
+ | AC-16 | Messages have `message` field | ALL present | `specification.messages[].message` truthy | YES | Format |
35
+ | AC-17 | Gherkin content structure | ALL valid | Each element has `feature` (string) + `scenarios` (array) | YES | Format |
36
+
37
+ ---
38
+
39
+ ## Bash Verification Script (node -e)
40
+
41
+ > **Usage:** Called from step-03d section 11-POST-CHECK-BASH after writing module feature.json.
42
+ > Also callable standalone for debugging: `node -e "..." path/to/feature.json`
43
+
44
+ ```bash
45
+ MODULE_JSON="{module_feature_json_path}"
46
+ node -e "
47
+ const fs = require('fs');
48
+ const data = JSON.parse(fs.readFileSync(process.argv[1], 'utf-8'));
49
+ const spec = data.specification || {};
50
+ const analysis = data.analysis || {};
51
+ const wf = spec.uiWireframes || spec.wireframes || [];
52
+ const sections = spec.sections || [];
53
+ const sdc = spec.seedDataCore || {};
54
+
55
+ const checks = [
56
+ ['AC-01: entities >= 1', (analysis.entities||[]).length, 1],
57
+ ['AC-03: useCases >= 2', (spec.useCases||[]).length, 2],
58
+ ['AC-04: FRs >= 4', (spec.functionalRequirements||[]).length, 4],
59
+ ['AC-05: wireframes >= 1', wf.length, 1],
60
+ ['AC-06: wireframes >= sections', wf.length, sections.length],
61
+ ['AC-08: sections >= 1', sections.length, 1],
62
+ ['AC-09: seedDataCore 7 arrays', Object.keys(sdc).filter(k => Array.isArray(sdc[k]) && sdc[k].length > 0).length, 7],
63
+ ['AC-10: gherkin is array', Array.isArray(spec.gherkinScenarios) ? 1 : 0, 1],
64
+ ['AC-11: apiEndpoints >= 1', (spec.apiEndpoints||[]).length, 1],
65
+ ['AC-12: messages >= 4', (spec.messages||[]).length, 4],
66
+ ['AC-13: validations >= 1', (spec.validations||[]).length, 1]
67
+ ];
68
+
69
+ const fails = [];
70
+ checks.forEach(c => {
71
+ if (c[1] < c[2]) {
72
+ fails.push(c);
73
+ console.error('FAIL: ' + c[0] + ' = ' + c[1] + ' (min: ' + c[2] + ')');
74
+ }
75
+ });
76
+
77
+ // AC-02: Entity attribute types
78
+ const badAttrs = (analysis.entities||[]).flatMap(e =>
79
+ (e.attributes||[]).filter(a => !a.type).map(a => e.name + '.' + a.name)
80
+ );
81
+ if (badAttrs.length > 0) {
82
+ fails.push(['AC-02: attr.type missing', badAttrs.length, 0]);
83
+ console.error('FAIL: AC-02: ' + badAttrs.length + ' attributes without type: ' + badAttrs.slice(0, 5).join(', ') + (badAttrs.length > 5 ? '...' : ''));
84
+ }
85
+
86
+ // AC-07: Wireframe content non-empty
87
+ const emptyWf = wf.filter(w => !w.mockup && !w.ascii && !w.content);
88
+ if (emptyWf.length > 0) {
89
+ fails.push(['AC-07: wireframe content empty', emptyWf.length, 0]);
90
+ console.error('FAIL: AC-07: ' + emptyWf.length + ' wireframes have EMPTY content');
91
+ }
92
+
93
+ // AC-14: Wireframe field naming (WARNING only)
94
+ const badFields = wf.filter(w => w.title || w.ascii || (w.name && !w.screen));
95
+ if (badFields.length > 0) {
96
+ console.warn('WARNING: AC-14: ' + badFields.length + ' wireframes use non-canonical field names (title/ascii/name)');
97
+ }
98
+
99
+ // AC-15: Validation rules format (rules must be array, not string)
100
+ const badRules = (spec.validations||[]).filter(v => v.rules && !Array.isArray(v.rules));
101
+ if (badRules.length > 0) {
102
+ fails.push(['AC-15: validations[].rules not array', badRules.length, 0]);
103
+ console.error('FAIL: AC-15: ' + badRules.length + ' validations have rules as string instead of array');
104
+ }
105
+
106
+ // AC-16: Messages must have 'message' field
107
+ const noMsg = (spec.messages||[]).filter(m => !m.message);
108
+ if (noMsg.length > 0) {
109
+ fails.push(['AC-16: messages[].message missing', noMsg.length, 0]);
110
+ console.error('FAIL: AC-16: ' + noMsg.length + ' messages missing "message" field');
111
+ }
112
+
113
+ // AC-17: Gherkin content structure (each element needs feature + scenarios[])
114
+ if (Array.isArray(spec.gherkinScenarios)) {
115
+ const badGherkin = spec.gherkinScenarios.filter(g => !g.feature || !Array.isArray(g.scenarios));
116
+ if (badGherkin.length > 0) {
117
+ fails.push(['AC-17: gherkin content invalid', badGherkin.length, 0]);
118
+ console.error('FAIL: AC-17: ' + badGherkin.length + ' gherkin entries missing feature or scenarios[]');
119
+ }
120
+ }
121
+
122
+ // Summary
123
+ if (fails.length > 0) {
124
+ console.error('\\nBLOCKING: ' + fails.length + ' acceptance criteria FAILED');
125
+ process.exit(1);
126
+ }
127
+ console.log('PASS: All acceptance criteria met (AC-01 to AC-17)');
128
+ " "$MODULE_JSON"
129
+ ```
130
+
131
+ ---
132
+
133
+ ## Integration Points
134
+
135
+ ### 1. step-03d-validate.md (section 11-POST-CHECK-BASH)
136
+
137
+ After writing module feature.json, run the bash script above.
138
+ IF FAIL → re-execute section 11 write with corrected data → re-run until PASS.
139
+
140
+ ### 2. step-05a-handoff.md (pre-handoff gate)
141
+
142
+ Before writing handoff for each module (section 7a step 3), run the bash script on the module feature.json.
143
+ IF FAIL → STOP handoff for this module, report which criteria failed.
144
+
145
+ ### 3. ralph-loop step-01-task.md (input validation)
146
+
147
+ At PRD load time, run the bash script on each module feature.json.
148
+ IF FAIL → BLOCKING: "Module {name} does not meet acceptance criteria — run /business-analyse to fix".
149
+ This prevents ralph-loop from generating code from incomplete specifications.
150
+
151
+ ---
152
+
153
+ ## Recovery Actions per Criterion
154
+
155
+ | Criterion | Recovery |
156
+ |-----------|----------|
157
+ | AC-01/02 | Re-run step-03a2-analysis (entity definition) |
158
+ | AC-03/04 | Re-run step-03c-compile sections 8b/8c |
159
+ | AC-05/06/07 | Re-run step-03b-ui (wireframe generation) — step-03b now writes intermediately |
160
+ | AC-08 | Re-run step-03b section 3a-bis (section definition) |
161
+ | AC-09 | Re-run step-03c section 8f-bis (seedDataCore transform) |
162
+ | AC-10 | Auto-fix: wrap in array `[gherkinScenarios]` |
163
+ | AC-11 | Re-run step-03c section 8k |
164
+ | AC-12 | Re-run step-03c section 8i |
165
+ | AC-13 | Re-run step-03c section 8h |
166
+ | AC-14 | Auto-fix: rename title→screen, ascii→mockup (done by step-03c/03d auto-fix) |
167
+ | AC-15 | Auto-fix: wrap string in array `[rules]` (done by step-03c ABSOLUTE FORMAT CHECKS) |
168
+ | AC-16 | Auto-fix: copy `description` → `message` (done by step-03c ABSOLUTE FORMAT CHECKS) |
169
+ | AC-17 | Re-run step-03c section 8g — ensure each gherkin entry has `feature` string + `scenarios` array |
@@ -127,10 +127,11 @@ const EMBEDDED_ARTIFACTS = {
127
127
  // FOR EACH module: extract from specification.uiWireframes[] OR specification.wireframes[] (SAFETY NET)
128
128
  // IMPORTANT: The agent may write wireframes under either key name — always check BOTH
129
129
  [moduleCode]: (moduleFeature.specification.uiWireframes || moduleFeature.specification.wireframes || []).map(wf => ({
130
- screen: wf.screen || wf.name || wf.id || "", // SAFETY NET: fallback name/id → screen
130
+ screen: wf.screen || wf.name || wf.title || wf.id || "", // SAFETY NET: fallback name/title/id → screen
131
131
  section: wf.section || "", // e.g. "list"
132
132
  format: wf.mockupFormat || "ascii", // RENAME: mockupFormat → format
133
- content: wf.mockup, // RENAME: mockup → content (ASCII art string)
133
+ content: wf.mockup || wf.ascii || wf.content || "", // SAFETY NET: mockup/ascii/content → content
134
+ svgContent: null, // Populated by SVG generation step (see wireframe-svg-style-guide.md)
134
135
  description: wf.description || "",
135
136
  elements: wf.elements || [], // [{ id, type, label }] or ["DataGrid", ...]
136
137
  actions: wf.actions || [], // [{ trigger, action }] or ["filter", ...]
@@ -163,10 +164,11 @@ const EMBEDDED_ARTIFACTS = {
163
164
 
164
165
  ### Artifact Gathering
165
166
 
166
- 1. For EACH module: read `specification.uiWireframes[]` OR `specification.wireframes[]` (check BOTH keys — agent may use either), **rename fields** (`mockupFormat`→`format`, `mockup`→`content`), store under `wireframes[moduleCode]`
167
+ 1. For EACH module: read `specification.uiWireframes[]` OR `specification.wireframes[]` (check BOTH keys — agent may use either), **rename fields** (`mockupFormat`→`format`, `mockup`/`ascii`/`content`→`content`, `screen`/`name`/`title`→`screen`), store under `wireframes[moduleCode]`
167
168
  2. Read master's `consolidation.e2eFlows[]` and build e2eFlows array with diagram generation
168
169
  3. Read master's `dependencyGraph` and build nodes/edges
169
170
  4. Serialize as JSON with 2-space indentation
171
+ 5. **SVG Wireframe Enrichment** (after wireframe extraction): For EACH wireframe with `content` (ASCII) and `svgContent === null`, generate SVG using parallel Task(sonnet) agents. See [wireframe-svg-style-guide.md](wireframe-svg-style-guide.md) for the complete prompt template and orchestration process. If generation fails for any wireframe → leave `svgContent` as null (graceful fallback to ASCII).
170
172
 
171
173
  ## Placeholder Replacement
172
174
 
@@ -82,10 +82,11 @@ From `specification.uiWireframes[]`, `specification.dashboards[]` and `analysis.
82
82
 
83
83
  ## 4.6 SeedData Files
84
84
 
85
- **OBLIGATORY: 5 CORE + business per module:**
85
+ **OBLIGATORY: 6 CORE (1 app-level + 5 per module) + business per module:**
86
86
 
87
87
  ```json
88
88
  "seedData": [
89
+ { "path": "src/Infrastructure/Persistence/Seeding/Data/NavigationApplicationSeedData.cs", "type": "SeedData", "category": "core", "source": "specification.seedDataCore.navigationApplications", "module": "shared", "description": "Application-level navigation seed data. Created ONCE per application (FIRST). Provides ApplicationId for modules and roles." },
89
90
  { "path": "src/Infrastructure/Persistence/Seeding/Data/{ModuleName}/NavigationModuleSeedData.cs", "type": "SeedData", "category": "core", "source": "specification.seedDataCore.navigationModules", "module": "{moduleCode}" },
90
91
  { "path": "src/Infrastructure/Persistence/Seeding/Data/{ModuleName}/PermissionsSeedData.cs", "type": "SeedData", "category": "core", "source": "specification.seedDataCore.permissions", "module": "{moduleCode}" },
91
92
  { "path": "src/Infrastructure/Persistence/Seeding/Data/{ModuleName}/RolesSeedData.cs", "type": "SeedData", "category": "core", "source": "specification.seedDataCore.roles", "module": "{moduleCode}" },