@bastani/atomic 0.6.5 → 0.6.6-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 (125) hide show
  1. package/.agents/skills/ado-commit/SKILL.md +2 -0
  2. package/.agents/skills/ado-create-pr/SKILL.md +2 -0
  3. package/.agents/skills/advanced-evaluation/SKILL.md +2 -0
  4. package/.agents/skills/ast-grep/SKILL.md +2 -0
  5. package/.agents/skills/bdi-mental-states/SKILL.md +2 -0
  6. package/.agents/skills/bun/SKILL.md +156 -122
  7. package/.agents/skills/context-compression/SKILL.md +2 -0
  8. package/.agents/skills/context-degradation/SKILL.md +2 -0
  9. package/.agents/skills/context-fundamentals/SKILL.md +2 -0
  10. package/.agents/skills/context-optimization/SKILL.md +2 -0
  11. package/.agents/skills/create-spec/SKILL.md +2 -0
  12. package/.agents/skills/docx/SKILL.md +2 -0
  13. package/.agents/skills/evaluation/SKILL.md +2 -0
  14. package/.agents/skills/explain-code/SKILL.md +2 -0
  15. package/.agents/skills/filesystem-context/SKILL.md +2 -0
  16. package/.agents/skills/find-skills/SKILL.md +2 -0
  17. package/.agents/skills/gh-commit/SKILL.md +2 -0
  18. package/.agents/skills/gh-create-pr/SKILL.md +2 -0
  19. package/.agents/skills/hosted-agents/SKILL.md +2 -0
  20. package/.agents/skills/impeccable/SKILL.md +117 -304
  21. package/.agents/skills/impeccable/agents/openai.yaml +4 -0
  22. package/.agents/skills/{adapt/SKILL.md → impeccable/reference/adapt.md} +2 -11
  23. package/.agents/skills/{animate/SKILL.md → impeccable/reference/animate.md} +15 -15
  24. package/.agents/skills/{audit/SKILL.md → impeccable/reference/audit.md} +8 -22
  25. package/.agents/skills/{bolder/SKILL.md → impeccable/reference/bolder.md} +9 -13
  26. package/.agents/skills/impeccable/reference/brand.md +114 -0
  27. package/.agents/skills/{clarify/SKILL.md → impeccable/reference/clarify.md} +2 -11
  28. package/.agents/skills/{colorize/SKILL.md → impeccable/reference/colorize.md} +23 -12
  29. package/.agents/skills/impeccable/reference/craft.md +152 -29
  30. package/.agents/skills/{critique/SKILL.md → impeccable/reference/critique.md} +25 -37
  31. package/.agents/skills/{delight/SKILL.md → impeccable/reference/delight.md} +9 -11
  32. package/.agents/skills/{distill/SKILL.md → impeccable/reference/distill.md} +2 -13
  33. package/.agents/skills/impeccable/reference/document.md +427 -0
  34. package/.agents/skills/impeccable/reference/extract.md +1 -1
  35. package/.agents/skills/{harden/SKILL.md → impeccable/reference/harden.md} +1 -43
  36. package/.agents/skills/{layout/SKILL.md → impeccable/reference/layout.md} +27 -11
  37. package/.agents/skills/impeccable/reference/live.md +594 -0
  38. package/.agents/skills/impeccable/reference/motion-design.md +12 -2
  39. package/.agents/skills/impeccable/reference/onboard.md +234 -0
  40. package/.agents/skills/{optimize/SKILL.md → impeccable/reference/optimize.md} +4 -12
  41. package/.agents/skills/{overdrive/SKILL.md → impeccable/reference/overdrive.md} +9 -21
  42. package/.agents/skills/{critique → impeccable}/reference/personas.md +1 -1
  43. package/.agents/skills/{polish/SKILL.md → impeccable/reference/polish.md} +31 -23
  44. package/.agents/skills/impeccable/reference/product.md +62 -0
  45. package/.agents/skills/{quieter/SKILL.md → impeccable/reference/quieter.md} +7 -11
  46. package/.agents/skills/impeccable/reference/shape.md +151 -0
  47. package/.agents/skills/impeccable/reference/teach.md +156 -0
  48. package/.agents/skills/{typeset/SKILL.md → impeccable/reference/typeset.md} +19 -11
  49. package/.agents/skills/impeccable/reference/typography.md +31 -14
  50. package/.agents/skills/impeccable/scripts/cleanup-deprecated.mjs +87 -17
  51. package/.agents/skills/impeccable/scripts/command-metadata.json +94 -0
  52. package/.agents/skills/impeccable/scripts/design-parser.mjs +820 -0
  53. package/.agents/skills/impeccable/scripts/detect-csp.mjs +198 -0
  54. package/.agents/skills/impeccable/scripts/is-generated.mjs +69 -0
  55. package/.agents/skills/impeccable/scripts/live-accept.mjs +595 -0
  56. package/.agents/skills/impeccable/scripts/live-browser.js +4781 -0
  57. package/.agents/skills/impeccable/scripts/live-inject.mjs +445 -0
  58. package/.agents/skills/impeccable/scripts/live-poll.mjs +186 -0
  59. package/.agents/skills/impeccable/scripts/live-server.mjs +694 -0
  60. package/.agents/skills/impeccable/scripts/live-wrap.mjs +571 -0
  61. package/.agents/skills/impeccable/scripts/live.mjs +247 -0
  62. package/.agents/skills/impeccable/scripts/load-context.mjs +141 -0
  63. package/.agents/skills/impeccable/scripts/modern-screenshot.umd.js +14 -0
  64. package/.agents/skills/impeccable/scripts/pin.mjs +214 -0
  65. package/.agents/skills/init/SKILL.md +2 -0
  66. package/.agents/skills/liteparse/SKILL.md +1 -0
  67. package/.agents/skills/memory-systems/SKILL.md +2 -0
  68. package/.agents/skills/multi-agent-patterns/SKILL.md +2 -0
  69. package/.agents/skills/opentui/SKILL.md +1 -0
  70. package/.agents/skills/pdf/SKILL.md +2 -0
  71. package/.agents/skills/playwright-cli/SKILL.md +51 -5
  72. package/.agents/skills/playwright-cli/references/playwright-tests.md +1 -1
  73. package/.agents/skills/playwright-cli/references/running-code.md +10 -0
  74. package/.agents/skills/playwright-cli/references/session-management.md +56 -0
  75. package/.agents/skills/playwright-cli/references/spec-driven-testing.md +305 -0
  76. package/.agents/skills/playwright-cli/references/test-generation.md +49 -3
  77. package/.agents/skills/pptx/SKILL.md +2 -0
  78. package/.agents/skills/project-development/SKILL.md +2 -0
  79. package/.agents/skills/prompt-engineer/SKILL.md +2 -0
  80. package/.agents/skills/research-codebase/SKILL.md +2 -0
  81. package/.agents/skills/ripgrep/SKILL.md +2 -0
  82. package/.agents/skills/skill-creator/LICENSE.txt +1 -1
  83. package/.agents/skills/skill-creator/SKILL.md +2 -0
  84. package/.agents/skills/sl-commit/SKILL.md +2 -0
  85. package/.agents/skills/sl-submit-diff/SKILL.md +2 -0
  86. package/.agents/skills/tdd/SKILL.md +4 -0
  87. package/.agents/skills/tool-design/SKILL.md +2 -0
  88. package/.agents/skills/typescript-advanced-types/SKILL.md +2 -1
  89. package/.agents/skills/typescript-expert/SKILL.md +7 -1
  90. package/.agents/skills/typescript-react-reviewer/SKILL.md +2 -1
  91. package/.agents/skills/workflow-creator/SKILL.md +75 -72
  92. package/.agents/skills/workflow-creator/references/session-config.md +48 -1
  93. package/.agents/skills/xlsx/SKILL.md +2 -0
  94. package/.opencode/opencode.json +4 -2
  95. package/dist/sdk/runtime/executor.d.ts +8 -0
  96. package/dist/sdk/runtime/executor.d.ts.map +1 -1
  97. package/dist/sdk/runtime/port-discovery.d.ts +71 -0
  98. package/dist/sdk/runtime/port-discovery.d.ts.map +1 -0
  99. package/dist/sdk/runtime/tmux.d.ts +10 -0
  100. package/dist/sdk/runtime/tmux.d.ts.map +1 -1
  101. package/dist/sdk/types.d.ts +1 -0
  102. package/dist/sdk/types.d.ts.map +1 -1
  103. package/dist/sdk/workflows/builtin/deep-research-codebase/opencode/index.d.ts.map +1 -1
  104. package/dist/sdk/workflows/builtin/open-claude-design/opencode/index.d.ts.map +1 -1
  105. package/dist/sdk/workflows/builtin/ralph/claude/index.d.ts.map +1 -1
  106. package/dist/sdk/workflows/builtin/ralph/copilot/index.d.ts.map +1 -1
  107. package/dist/sdk/workflows/builtin/ralph/helpers/prompts.d.ts +15 -0
  108. package/dist/sdk/workflows/builtin/ralph/helpers/prompts.d.ts.map +1 -1
  109. package/dist/sdk/workflows/builtin/ralph/opencode/index.d.ts.map +1 -1
  110. package/package.json +1 -1
  111. package/src/sdk/runtime/executor.test.ts +254 -1
  112. package/src/sdk/runtime/executor.ts +135 -89
  113. package/src/sdk/runtime/port-discovery.test.ts +573 -0
  114. package/src/sdk/runtime/port-discovery.ts +496 -0
  115. package/src/sdk/runtime/tmux.ts +16 -0
  116. package/src/sdk/types.ts +1 -0
  117. package/src/sdk/workflows/builtin/deep-research-codebase/opencode/index.ts +24 -6
  118. package/src/sdk/workflows/builtin/open-claude-design/opencode/index.ts +52 -13
  119. package/src/sdk/workflows/builtin/ralph/claude/index.ts +31 -3
  120. package/src/sdk/workflows/builtin/ralph/copilot/index.ts +16 -0
  121. package/src/sdk/workflows/builtin/ralph/helpers/prompts.ts +70 -3
  122. package/src/sdk/workflows/builtin/ralph/opencode/index.ts +50 -6
  123. package/.agents/skills/shape/SKILL.md +0 -96
  124. /package/.agents/skills/{critique → impeccable}/reference/cognitive-load.md +0 -0
  125. /package/.agents/skills/{critique → impeccable}/reference/heuristics-scoring.md +0 -0
@@ -0,0 +1,820 @@
1
+ // Parse a DESIGN.md (Stitch-spec format) into a structured JSON model that
2
+ // the live-mode design-system panel can render. Deterministic, dependency-free.
3
+ //
4
+ // Two-layer: YAML frontmatter (machine-readable tokens) + markdown body
5
+ // (prose with six canonical H2 sections). When frontmatter is present, it's
6
+ // exposed on `model.frontmatter` alongside the prose-scraped sections;
7
+ // consumers can prefer frontmatter values and fall back to prose.
8
+
9
+ const CANONICAL_SECTIONS = [
10
+ 'Overview',
11
+ 'Colors',
12
+ 'Typography',
13
+ 'Elevation',
14
+ 'Components',
15
+ "Do's and Don'ts",
16
+ ];
17
+
18
+ // ---------- Frontmatter (Stitch YAML subset) ----------
19
+
20
+ function parseFrontmatter(md) {
21
+ const lines = md.split(/\r?\n/);
22
+ if (lines[0]?.trim() !== '---') return { frontmatter: null, body: md };
23
+
24
+ let end = -1;
25
+ for (let i = 1; i < lines.length; i++) {
26
+ if (lines[i].trim() === '---') { end = i; break; }
27
+ }
28
+ if (end === -1) return { frontmatter: null, body: md };
29
+
30
+ const yaml = lines.slice(1, end).join('\n');
31
+ const body = lines.slice(end + 1).join('\n');
32
+ try {
33
+ return { frontmatter: parseYamlSubset(yaml), body };
34
+ } catch {
35
+ return { frontmatter: null, body: md };
36
+ }
37
+ }
38
+
39
+ // Minimal YAML reader for the Stitch frontmatter subset: scalar maps with
40
+ // one level of nested objects (typography roles, components). Indent-based,
41
+ // 2-space convention. No arrays, no anchors, no multi-line scalars — Stitch's
42
+ // schema doesn't need them and accepting them would require a real YAML
43
+ // dependency we don't want to vendor.
44
+ function parseYamlSubset(yaml) {
45
+ const lines = yaml.split(/\r?\n/);
46
+ const root = {};
47
+ const stack = [{ indent: -1, obj: root }];
48
+
49
+ for (const raw of lines) {
50
+ // Skip blanks and line-only comments. Don't strip inline comments:
51
+ // unquoted hex values start with `#` and can't be safely distinguished
52
+ // from a comment after whitespace.
53
+ if (!raw.trim() || /^\s*#/.test(raw)) continue;
54
+
55
+ const indent = raw.match(/^\s*/)[0].length;
56
+ const content = raw.slice(indent);
57
+
58
+ const colonIdx = findTopLevelColon(content);
59
+ if (colonIdx === -1) continue;
60
+
61
+ while (stack.length > 1 && stack[stack.length - 1].indent >= indent) {
62
+ stack.pop();
63
+ }
64
+
65
+ const key = content.slice(0, colonIdx).trim();
66
+ const rest = content.slice(colonIdx + 1).trim();
67
+ const parent = stack[stack.length - 1].obj;
68
+
69
+ if (rest === '') {
70
+ const obj = {};
71
+ parent[key] = obj;
72
+ stack.push({ indent, obj });
73
+ } else {
74
+ parent[key] = parseScalar(rest);
75
+ }
76
+ }
77
+
78
+ return root;
79
+ }
80
+
81
+ function findTopLevelColon(s) {
82
+ let inQuote = null;
83
+ for (let i = 0; i < s.length; i++) {
84
+ const ch = s[i];
85
+ if (inQuote) {
86
+ if (ch === inQuote && s[i - 1] !== '\\') inQuote = null;
87
+ } else if (ch === '"' || ch === "'") {
88
+ inQuote = ch;
89
+ } else if (ch === ':') {
90
+ return i;
91
+ }
92
+ }
93
+ return -1;
94
+ }
95
+
96
+ function parseScalar(raw) {
97
+ const s = raw.trim();
98
+ if ((s.startsWith('"') && s.endsWith('"')) || (s.startsWith("'") && s.endsWith("'"))) {
99
+ return s.slice(1, -1);
100
+ }
101
+ if (s === 'true') return true;
102
+ if (s === 'false') return false;
103
+ if (s === 'null' || s === '~') return null;
104
+ if (/^-?\d+$/.test(s)) return Number(s);
105
+ if (/^-?\d*\.\d+$/.test(s)) return Number(s);
106
+ return s;
107
+ }
108
+
109
+ const HEX_RE = /#[0-9a-fA-F]{3,8}\b/g;
110
+ const OKLCH_RE = /oklch\([^)]+\)/gi;
111
+ const RGBA_RE = /rgba?\([^)]+\)/gi;
112
+ const BOX_SHADOW_RE = /(?:box-shadow:\s*)?((?:-?\d[\w\d\s\-.,/()#%]*)+)/;
113
+ const NAMED_RULE_RE = /\*\*(The [^*]+?Rule)\.\*\*\s*(.+)/;
114
+
115
+ // ---------- Section splitting ----------
116
+
117
+ function splitSections(md) {
118
+ const lines = md.split(/\r?\n/);
119
+ let title = null;
120
+ const sections = {};
121
+ let current = null;
122
+
123
+ for (const raw of lines) {
124
+ const line = raw.trimEnd();
125
+
126
+ if (!title && line.startsWith('# ') && !line.startsWith('## ')) {
127
+ title = line.replace(/^#\s+/, '').trim();
128
+ continue;
129
+ }
130
+
131
+ const h2 = line.match(/^##\s+(?:\d+\.\s*)?([^:\n]+?)(?::\s*(.+))?$/);
132
+ if (h2) {
133
+ const rawName = normalizeApostrophes(h2[1].trim());
134
+ const subtitle = h2[2] ? h2[2].trim() : null;
135
+ const canonical = matchCanonicalSection(rawName);
136
+ if (canonical) {
137
+ current = { name: canonical, subtitle, lines: [] };
138
+ sections[canonical] = current;
139
+ continue;
140
+ }
141
+ // non-canonical H2 — ignore but stop feeding into current
142
+ current = null;
143
+ continue;
144
+ }
145
+
146
+ if (current) current.lines.push(raw);
147
+ }
148
+
149
+ return { title, sections };
150
+ }
151
+
152
+ function normalizeApostrophes(s) {
153
+ return s.replace(/[\u2018\u2019]/g, "'");
154
+ }
155
+
156
+ function matchCanonicalSection(name) {
157
+ const normalized = normalizeApostrophes(name).toLowerCase();
158
+ // Exact match first
159
+ for (const c of CANONICAL_SECTIONS) {
160
+ if (normalizeApostrophes(c).toLowerCase() === normalized) return c;
161
+ }
162
+ // Keyword-contained match: "Overview & Creative North Star" -> "Overview",
163
+ // "Elevation & Depth" -> "Elevation", etc.
164
+ for (const c of CANONICAL_SECTIONS) {
165
+ const key = normalizeApostrophes(c).toLowerCase();
166
+ const pattern = new RegExp(`\\b${key.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}\\b`);
167
+ if (pattern.test(normalized)) return c;
168
+ }
169
+ return null;
170
+ }
171
+
172
+ // ---------- Subsection splitting (inside a canonical section) ----------
173
+
174
+ function splitSubsections(lines) {
175
+ const subs = [];
176
+ let current = { name: null, lines: [] };
177
+ subs.push(current);
178
+
179
+ for (const raw of lines) {
180
+ const h3 = raw.match(/^###\s+(.+?)\s*$/);
181
+ if (h3) {
182
+ current = { name: h3[1].trim(), lines: [] };
183
+ subs.push(current);
184
+ continue;
185
+ }
186
+ current.lines.push(raw);
187
+ }
188
+
189
+ return subs;
190
+ }
191
+
192
+ // ---------- Generic helpers ----------
193
+
194
+ function collectParagraphs(lines) {
195
+ const paragraphs = [];
196
+ let buf = [];
197
+ const flush = () => {
198
+ if (buf.length) {
199
+ paragraphs.push(buf.join(' ').trim());
200
+ buf = [];
201
+ }
202
+ };
203
+ for (const raw of lines) {
204
+ const trimmed = raw.trim();
205
+ if (trimmed === '') { flush(); continue; }
206
+ // Horizontal rules (---, ***) and headings/bullets end a paragraph.
207
+ if (/^(?:-{3,}|\*{3,}|_{3,})$/.test(trimmed)) { flush(); continue; }
208
+ if (raw.startsWith('#') || raw.match(/^[-*]\s/)) { flush(); continue; }
209
+ buf.push(trimmed);
210
+ }
211
+ flush();
212
+ return paragraphs.filter(Boolean);
213
+ }
214
+
215
+ function collectBullets(lines) {
216
+ const bullets = [];
217
+ let current = null;
218
+ for (const raw of lines) {
219
+ const m = raw.match(/^\s*[-*]\s+(.+)$/);
220
+ if (m) {
221
+ if (current) bullets.push(current);
222
+ current = m[1];
223
+ continue;
224
+ }
225
+ // continuation of a bullet (indented line)
226
+ if (current && raw.match(/^\s{2,}\S/)) {
227
+ current += ' ' + raw.trim();
228
+ continue;
229
+ }
230
+ // blank line ends a bullet
231
+ if (raw.trim() === '' && current) {
232
+ bullets.push(current);
233
+ current = null;
234
+ }
235
+ }
236
+ if (current) bullets.push(current);
237
+ return bullets;
238
+ }
239
+
240
+ function stripBold(s) {
241
+ return s.replace(/\*\*(.+?)\*\*/g, '$1');
242
+ }
243
+
244
+ function extractNamedRules(lines) {
245
+ const rules = [];
246
+ const seen = new Set();
247
+
248
+ // Style A (Impeccable): "**The X Rule.** body body body" — can span lines.
249
+ const joined = lines.join('\n');
250
+ const inlineStart = /\*\*(The [^*]+?Rule)\.\*\*/g;
251
+ const inlineMatches = [];
252
+ let m;
253
+ while ((m = inlineStart.exec(joined)) !== null) {
254
+ inlineMatches.push({ name: m[1], start: m.index, end: inlineStart.lastIndex });
255
+ }
256
+ for (let i = 0; i < inlineMatches.length; i++) {
257
+ const mm = inlineMatches[i];
258
+ const bodyEnd = i + 1 < inlineMatches.length ? inlineMatches[i + 1].start : joined.length;
259
+ const body = joined
260
+ .slice(mm.end, bodyEnd)
261
+ .replace(/\n##[^\n]*$/s, '')
262
+ .replace(/\n###[^\n]*$/s, '')
263
+ .trim();
264
+ const name = stripBold(mm.name).trim();
265
+ seen.add(name.toLowerCase());
266
+ rules.push({ name, body: stripBold(body) });
267
+ }
268
+
269
+ // Style B (Stitch): `### The "X" Rule` or `### The X Fallback`, body is the
270
+ // bullets/paragraphs until the next heading. Accept Rule / Fallback / Principle.
271
+ for (let i = 0; i < lines.length; i++) {
272
+ const h3 = lines[i].match(/^###\s+(.+?)\s*$/);
273
+ if (!h3) continue;
274
+ const headerName = stripBold(h3[1]).replace(/["“”]/g, '').trim();
275
+ if (!/^The\b.*\b(Rule|Fallback|Principle)\b/i.test(headerName)) continue;
276
+ if (seen.has(headerName.toLowerCase())) continue;
277
+
278
+ const bodyLines = [];
279
+ for (let j = i + 1; j < lines.length; j++) {
280
+ if (/^##\s|^###\s/.test(lines[j])) break;
281
+ bodyLines.push(lines[j]);
282
+ }
283
+ const body = stripBold(bodyLines.join('\n').replace(/\n+/g, ' ')).trim();
284
+ if (body) {
285
+ seen.add(headerName.toLowerCase());
286
+ rules.push({ name: headerName, body });
287
+ }
288
+ }
289
+
290
+ // Style C (Stitch bullet form): "* **The Layering Principle:** body"
291
+ // Colon/period lives inside the bold, so match "**...**" then inspect.
292
+ for (const b of collectBullets(lines)) {
293
+ const mm = b.match(/^\*\*([^*]+?)\*\*\s*(.+)$/);
294
+ if (!mm) continue;
295
+ const nameRaw = mm[1].replace(/[.:]\s*$/, '').replace(/["“”]/g, '').trim();
296
+ if (!/^The\b.+\b(Rule|Fallback|Principle)$/i.test(nameRaw)) continue;
297
+ if (seen.has(nameRaw.toLowerCase())) continue;
298
+ seen.add(nameRaw.toLowerCase());
299
+ rules.push({ name: nameRaw, body: stripBold(mm[2]).trim() });
300
+ }
301
+
302
+ return rules;
303
+ }
304
+
305
+ // ---------- Per-section extractors ----------
306
+
307
+ function extractOverview(section) {
308
+ if (!section) return null;
309
+ const text = section.lines.join('\n');
310
+ const northStar = text.match(/\*\*Creative North Star:\s*"([^"]+)"\*\*/);
311
+ const keyChars = [];
312
+ const keyCharMatch = text.match(/\*\*Key Characteristics:\*\*\s*\n([\s\S]+?)(?:\n##|\n###|$)/);
313
+ if (keyCharMatch) {
314
+ for (const line of keyCharMatch[1].split('\n')) {
315
+ const m = line.match(/^\s*[-*]\s+(.+)$/);
316
+ if (m) keyChars.push(stripBold(m[1].trim()));
317
+ }
318
+ }
319
+
320
+ // Philosophy paragraphs: everything that isn't a rule header or key-char block
321
+ const paragraphs = collectParagraphs(section.lines).filter(
322
+ (p) =>
323
+ !p.startsWith('**Creative North Star') &&
324
+ !p.startsWith('**Key Characteristics')
325
+ );
326
+
327
+ return {
328
+ subtitle: section.subtitle,
329
+ creativeNorthStar: northStar ? northStar[1] : null,
330
+ philosophy: paragraphs,
331
+ keyCharacteristics: keyChars,
332
+ };
333
+ }
334
+
335
+ function extractColors(section) {
336
+ if (!section) return null;
337
+ const subs = splitSubsections(section.lines);
338
+
339
+ const description = collectParagraphs(subs[0].lines).join(' ');
340
+ const groups = [];
341
+ const ROLE_KEYWORDS = /^(primary|secondary|tertiary|neutral|accent)\b/i;
342
+
343
+ for (const sub of subs.slice(1)) {
344
+ if (!sub.name || /Named Rules?/i.test(sub.name) || /^The\s/i.test(sub.name)) continue;
345
+
346
+ const bullets = collectBullets(sub.lines);
347
+ const parsed = bullets.map((b) => parseColorBullet(b)).filter(Boolean);
348
+ if (parsed.length === 0) continue;
349
+
350
+ // If every bullet starts with a role keyword (Primary/Secondary/...), promote
351
+ // each bullet to its own group. Otherwise keep the subsection as the group.
352
+ const allRoleBullets =
353
+ parsed.length > 0 && parsed.every((p) => p.name && ROLE_KEYWORDS.test(p.name));
354
+
355
+ if (allRoleBullets) {
356
+ for (const p of parsed) {
357
+ groups.push({ role: p.name, colors: [p] });
358
+ }
359
+ } else {
360
+ groups.push({ role: sub.name, colors: parsed });
361
+ }
362
+ }
363
+
364
+ // If the Colors section has no subsections at all (unlikely), fall back to
365
+ // scanning the whole section as a flat bullet list.
366
+ if (groups.length === 0) {
367
+ const flat = collectBullets(section.lines)
368
+ .map((b) => parseColorBullet(b))
369
+ .filter(Boolean);
370
+ if (flat.length) {
371
+ for (const p of flat) {
372
+ if (p.name && ROLE_KEYWORDS.test(p.name)) {
373
+ groups.push({ role: p.name, colors: [p] });
374
+ } else {
375
+ const fallback = groups.find((g) => g.role === 'Palette');
376
+ if (fallback) fallback.colors.push(p);
377
+ else groups.push({ role: 'Palette', colors: [p] });
378
+ }
379
+ }
380
+ }
381
+ }
382
+
383
+ return {
384
+ subtitle: section.subtitle,
385
+ description: description || null,
386
+ groups,
387
+ rules: extractNamedRules(section.lines),
388
+ };
389
+ }
390
+
391
+ function parseColorBullet(bullet) {
392
+ const text = bullet.trim();
393
+
394
+ // Case 1 (Impeccable): **Name** (value-with-maybe-nested-parens): description
395
+ const bold = text.match(/^\*\*(.+?)\*\*\s*(.*)$/);
396
+ if (bold && bold[2].startsWith('(')) {
397
+ const value = extractParenGroup(bold[2]);
398
+ if (value !== null) {
399
+ const after = bold[2].slice(value.length + 2).trimStart();
400
+ if (after.startsWith(':')) {
401
+ return buildColor(bold[1], value, after.slice(1).trim());
402
+ }
403
+ }
404
+ }
405
+
406
+ // Case 2 (Stitch): **Name (values):** description — value embedded in bold.
407
+ const stitch = text.match(/^\*\*([^*]+?)\s*\(([^)]+)\):\*\*\s*(.*)$/);
408
+ if (stitch) {
409
+ return buildColor(stitch[1].trim(), stitch[2], stitch[3]);
410
+ }
411
+
412
+ // Case 3: bullet without bold, just hex/oklch inside.
413
+ const values = collectColorValues(text);
414
+ if (values.length) {
415
+ return buildColor(null, values.join(' to '), text);
416
+ }
417
+ return null;
418
+ }
419
+
420
+ function extractParenGroup(s) {
421
+ if (s[0] !== '(') return null;
422
+ let depth = 0;
423
+ for (let i = 0; i < s.length; i++) {
424
+ if (s[i] === '(') depth++;
425
+ else if (s[i] === ')') {
426
+ depth--;
427
+ if (depth === 0) return s.slice(1, i);
428
+ }
429
+ }
430
+ return null;
431
+ }
432
+
433
+ function buildColor(name, rawValue, description) {
434
+ const values = collectColorValues(rawValue);
435
+ const primary = values[0] ?? rawValue.trim();
436
+ return {
437
+ name: name ? stripBold(name).trim() : null,
438
+ value: primary,
439
+ valueRange: values.length > 1 ? values : null,
440
+ format: detectFormat(primary),
441
+ description: stripBold(description || '').trim() || null,
442
+ };
443
+ }
444
+
445
+ function collectColorValues(s) {
446
+ const out = [];
447
+ s.replace(HEX_RE, (v) => {
448
+ out.push(v);
449
+ return v;
450
+ });
451
+ s.replace(OKLCH_RE, (v) => {
452
+ out.push(v);
453
+ return v;
454
+ });
455
+ return out;
456
+ }
457
+
458
+ function detectFormat(v) {
459
+ if (!v) return 'unknown';
460
+ if (v.startsWith('#')) return 'hex';
461
+ if (/^oklch/i.test(v)) return 'oklch';
462
+ if (/^rgb/i.test(v)) return 'rgb';
463
+ return 'unknown';
464
+ }
465
+
466
+ function scanInlineColors(lines) {
467
+ const out = [];
468
+ for (const line of lines) {
469
+ if (!/^\s*[-*]\s/.test(line)) continue;
470
+ const trimmed = line.replace(/^\s*[-*]\s+/, '');
471
+ const color = parseColorBullet(trimmed);
472
+ if (color) out.push(color);
473
+ }
474
+ return out;
475
+ }
476
+
477
+ function parseStitchInlineGroups(lines) {
478
+ // Stitch writes: `* **Primary (`#00478d` to `#005eb8`):** Use for "..."`
479
+ // Each bullet IS its own role. Group them under the spoken role name.
480
+ const out = [];
481
+ for (const line of lines) {
482
+ if (!/^\s*[-*]\s/.test(line)) continue;
483
+ const trimmed = line.replace(/^\s*[-*]\s+/, '').trim();
484
+ const m = trimmed.match(
485
+ /^\*\*([A-Z][a-zA-Z]+)\s*\(([^)]+)\):\*\*\s*(.*)$/
486
+ );
487
+ if (m) {
488
+ const role = m[1];
489
+ const color = buildColor(role, m[2], m[3]);
490
+ out.push({ role, colors: [color] });
491
+ }
492
+ }
493
+ return out;
494
+ }
495
+
496
+ function extractTypography(section) {
497
+ if (!section) return null;
498
+ const text = section.lines.join('\n');
499
+
500
+ const fonts = {};
501
+ // Pattern A: **Display Font:** Family (with fallback)
502
+ const fontLineRe = /\*\*([\w\s/]+?)Font:\*\*\s*([^\n(]+?)(?:\s*\(with\s+([^)]+)\))?\s*$/gm;
503
+ let fm;
504
+ while ((fm = fontLineRe.exec(text)) !== null) {
505
+ const rawRole = fm[1].trim().toLowerCase().replace(/\s+/g, '-');
506
+ const role = normalizeFontRole(rawRole) || 'display';
507
+ fonts[role] = {
508
+ family: fm[2].trim(),
509
+ fallback: fm[3] ? fm[3].trim() : null,
510
+ };
511
+ }
512
+
513
+ // Pattern B (Stitch): * **Display & Headlines (Noto Serif):** description
514
+ if (Object.keys(fonts).length === 0) {
515
+ const stitchRe = /\*\*([\w\s&/]+?)\s*\(([^)]+)\):\*\*\s*(.+)/g;
516
+ let sm;
517
+ while ((sm = stitchRe.exec(text)) !== null) {
518
+ const rawRole = sm[1]
519
+ .trim()
520
+ .toLowerCase()
521
+ .replace(/\s*&\s*/g, '-')
522
+ .replace(/\s+/g, '-');
523
+ const role = normalizeFontRole(rawRole) || rawRole;
524
+ fonts[role] = { family: sm[2].trim(), fallback: null, purpose: sm[3].trim() };
525
+ }
526
+ }
527
+
528
+ // Character paragraph — either a **Character:** label, or fall back to the
529
+ // first free paragraph under the section header (Stitch style).
530
+ const characterMatch = text.match(/\*\*Character:\*\*\s*([^\n]+(?:\n[^\n]+)*?)(?=\n\n|\n###|\n##|$)/);
531
+ let character = characterMatch ? characterMatch[1].replace(/\n/g, ' ').trim() : null;
532
+ if (!character) {
533
+ const paragraphs = collectParagraphs(section.lines).filter(
534
+ (p) => !/^\*\*[\w\s/&]+Font/i.test(p) && !/^\*\*[\w\s/&]+\([^)]+\)/.test(p)
535
+ );
536
+ if (paragraphs.length) character = paragraphs[0];
537
+ }
538
+
539
+ // Hierarchy bullets under ### Hierarchy
540
+ const subs = splitSubsections(section.lines);
541
+ let hierarchy = [];
542
+ const hierSub = subs.find((s) => s.name && /hierarch/i.test(s.name));
543
+ if (hierSub) {
544
+ const bullets = collectBullets(hierSub.lines);
545
+ hierarchy = bullets.map(parseTypeBullet).filter(Boolean);
546
+ }
547
+
548
+ return {
549
+ subtitle: section.subtitle,
550
+ fonts,
551
+ character,
552
+ hierarchy,
553
+ rules: extractNamedRules(section.lines),
554
+ };
555
+ }
556
+
557
+ function normalizeFontRole(raw) {
558
+ // Canonical roles the panel cares about: display, body, label, mono.
559
+ // Stitch often writes compound roles like "display-&-headlines" or "ui-&-body"
560
+ // — collapse them to the first canonical role present.
561
+ const tokens = raw.split(/[-/&\s]+/).filter(Boolean);
562
+ const priority = ['display', 'headline', 'body', 'ui', 'label', 'mono'];
563
+ const canonical = { headline: 'display', ui: 'body' };
564
+ for (const p of priority) {
565
+ if (tokens.includes(p)) return canonical[p] || p;
566
+ }
567
+ return null;
568
+ }
569
+
570
+ function parseTypeBullet(bullet) {
571
+ // - **Display** (family, weight 300, italic, clamp(...), line-height 1): purpose
572
+ const m = bullet.match(/^\*\*(.+?)\*\*\s*\(([^)]+)\):\s*(.*)$/);
573
+ if (!m) return null;
574
+ const name = m[1].trim();
575
+ const specs = m[2].split(',').map((s) => s.trim());
576
+ return {
577
+ name,
578
+ specs,
579
+ purpose: stripBold(m[3] || '').trim() || null,
580
+ };
581
+ }
582
+
583
+ function extractElevation(section) {
584
+ if (!section) return null;
585
+ const subs = splitSubsections(section.lines);
586
+
587
+ const description = collectParagraphs(subs[0].lines).join(' ') || null;
588
+
589
+ const shadows = [];
590
+ const seen = new Set();
591
+ const dedupe = (entry) => {
592
+ const key = (entry.name || '') + '::' + entry.value;
593
+ if (seen.has(key)) return;
594
+ seen.add(key);
595
+ shadows.push(entry);
596
+ };
597
+
598
+ for (const b of collectBullets(section.lines)) {
599
+ const parsed = parseShadowBullet(b);
600
+ if (parsed) dedupe(parsed);
601
+ }
602
+
603
+ // Fallback: extract shadows written inline in prose. Stitch style is
604
+ // "...use an extra-diffused shadow: `box-shadow: 0 12px 40px rgba(...)`."
605
+ for (const p of collectParagraphs(section.lines)) {
606
+ for (const inline of extractInlineShadows(p)) dedupe(inline);
607
+ }
608
+ for (const b of collectBullets(section.lines)) {
609
+ for (const inline of extractInlineShadows(b)) dedupe(inline);
610
+ }
611
+
612
+ return {
613
+ subtitle: section.subtitle,
614
+ description,
615
+ shadows,
616
+ rules: extractNamedRules(section.lines),
617
+ };
618
+ }
619
+
620
+ function extractInlineShadows(text) {
621
+ // Find `box-shadow: ...` anywhere in prose and capture the value. Work on the
622
+ // raw string so it handles both backtick-fenced and unfenced variants.
623
+ const out = [];
624
+ const re = /box-shadow\s*:\s*([^`;\n]+)/gi;
625
+ let m;
626
+ while ((m = re.exec(text)) !== null) {
627
+ const value = m[1].replace(/[`.)]+$/, '').trim();
628
+ if (!value) continue;
629
+ // Name heuristic: the noun immediately before the shadow phrase.
630
+ // e.g. "an extra-diffused shadow: ..." -> "extra-diffused shadow"
631
+ const before = text.slice(0, m.index);
632
+ const nameMatch = before.match(/\b([A-Za-z][A-Za-z\- ]{2,40})\s+shadow\b[^A-Za-z0-9]*$/i);
633
+ let name = null;
634
+ if (nameMatch) {
635
+ const stripped = nameMatch[1]
636
+ .replace(/^(?:use|using|apply|applying|is|are|looks? like)\s+/i, '')
637
+ .replace(/^(?:a|an|the)\s+/i, '')
638
+ .trim();
639
+ if (stripped) {
640
+ name =
641
+ stripped.charAt(0).toUpperCase() + stripped.slice(1) + ' shadow';
642
+ }
643
+ }
644
+ out.push({
645
+ name,
646
+ value,
647
+ purpose: null,
648
+ });
649
+ }
650
+ return out;
651
+ }
652
+
653
+ function parseShadowBullet(bullet) {
654
+ // - **Name** (`box-shadow: value`): purpose
655
+ // - **Name** (`value`): purpose
656
+ // Only accept if the paren content looks like a shadow value (contains px,
657
+ // rem, rgba, or box-shadow). This filters out `**Rule Name:**` bullets.
658
+ const m = bullet.match(/^\*\*(.+?)\*\*\s*\(`?([^`]+?)`?\):\s*(.*)$/);
659
+ if (!m) return null;
660
+ const rawValue = m[2].replace(/^box-shadow:\s*/i, '').trim();
661
+ const looksLikeShadow =
662
+ /box-shadow|rgba?\(|\bpx\b|\brem\b|^-?\d+\s/i.test(rawValue) &&
663
+ /\d/.test(rawValue);
664
+ if (!looksLikeShadow) return null;
665
+ const name = stripBold(m[1]).trim();
666
+ return {
667
+ name,
668
+ value: rawValue,
669
+ purpose: stripBold(m[3] || '').trim() || null,
670
+ };
671
+ }
672
+
673
+ function extractComponents(section) {
674
+ if (!section) return null;
675
+ const subs = splitSubsections(section.lines);
676
+ const components = [];
677
+
678
+ for (const sub of subs.slice(1)) {
679
+ if (!sub.name) continue;
680
+
681
+ const bullets = collectBullets(sub.lines);
682
+ const paragraphs = collectParagraphs(sub.lines);
683
+
684
+ const variants = [];
685
+ const properties = {};
686
+
687
+ for (const b of bullets) {
688
+ // - **Key:** value
689
+ const m = b.match(/^\*\*(.+?):?\*\*:?\s*(.+)$/);
690
+ if (m) {
691
+ const key = stripBold(m[1]).trim();
692
+ const value = stripBold(m[2]).trim();
693
+ // Heuristic: "Primary", "Secondary", "Hover", "Focus" etc are variants;
694
+ // "Shape", "Background", "Padding" are properties.
695
+ if (/^(primary|secondary|tertiary|ghost|hover|focus|active|disabled|default|error|selected|unselected|state)$/i.test(key.split(/[\s/]/)[0])) {
696
+ variants.push({ name: key, description: value });
697
+ } else {
698
+ properties[key.toLowerCase()] = value;
699
+ }
700
+ }
701
+ }
702
+
703
+ components.push({
704
+ name: sub.name,
705
+ description: paragraphs.join(' ') || null,
706
+ properties,
707
+ variants,
708
+ });
709
+ }
710
+
711
+ return {
712
+ subtitle: section.subtitle,
713
+ components,
714
+ };
715
+ }
716
+
717
+ function extractDosDonts(section) {
718
+ if (!section) return null;
719
+ const subs = splitSubsections(section.lines);
720
+ const dos = [];
721
+ const donts = [];
722
+
723
+ for (const sub of subs.slice(1)) {
724
+ if (!sub.name) continue;
725
+ const subName = normalizeApostrophes(sub.name);
726
+ const bullets = collectBullets(sub.lines).map((b) => stripBold(b).trim());
727
+ if (/^do'?t?:?$/i.test(subName) || /^do:?$/i.test(subName)) {
728
+ dos.push(...bullets);
729
+ } else if (/^don'?t:?$/i.test(subName)) {
730
+ donts.push(...bullets);
731
+ }
732
+ }
733
+
734
+ // Classify by bullet prefix as a backup (catches loose bullets outside H3 wrappers)
735
+ for (const b of collectBullets(section.lines)) {
736
+ const stripped = normalizeApostrophes(stripBold(b).trim());
737
+ if (/^don'?t\b/i.test(stripped)) {
738
+ if (!donts.some((d) => normalizeApostrophes(d) === stripped)) donts.push(stripped);
739
+ } else if (/^do\b/i.test(stripped)) {
740
+ if (!dos.some((d) => normalizeApostrophes(d) === stripped)) dos.push(stripped);
741
+ }
742
+ }
743
+
744
+ return { dos, donts };
745
+ }
746
+
747
+ // ---------- Coverage assessment ----------
748
+
749
+ function assessCoverage(model) {
750
+ const report = {};
751
+
752
+ report.overview = model.overview
753
+ ? {
754
+ northStar: Boolean(model.overview.creativeNorthStar),
755
+ philosophy: model.overview.philosophy.length > 0,
756
+ keyCharacteristics: model.overview.keyCharacteristics.length,
757
+ }
758
+ : 'missing';
759
+
760
+ report.colors = model.colors
761
+ ? {
762
+ groups: model.colors.groups.length,
763
+ totalColors: model.colors.groups.reduce((n, g) => n + g.colors.length, 0),
764
+ rules: model.colors.rules.length,
765
+ }
766
+ : 'missing';
767
+
768
+ report.typography = model.typography
769
+ ? {
770
+ fonts: Object.keys(model.typography.fonts).length,
771
+ hierarchyEntries: model.typography.hierarchy.length,
772
+ character: Boolean(model.typography.character),
773
+ rules: model.typography.rules.length,
774
+ }
775
+ : 'missing';
776
+
777
+ report.elevation = model.elevation
778
+ ? {
779
+ shadows: model.elevation.shadows.length,
780
+ rules: model.elevation.rules.length,
781
+ description: Boolean(model.elevation.description),
782
+ }
783
+ : 'missing';
784
+
785
+ report.components = model.components
786
+ ? {
787
+ count: model.components.components.length,
788
+ variantTotal: model.components.components.reduce((n, c) => n + c.variants.length, 0),
789
+ }
790
+ : 'missing';
791
+
792
+ report.dosDonts = model.dosDonts
793
+ ? {
794
+ dos: model.dosDonts.dos.length,
795
+ donts: model.dosDonts.donts.length,
796
+ }
797
+ : 'missing';
798
+
799
+ return report;
800
+ }
801
+
802
+ // ---------- Main ----------
803
+
804
+ export function parseDesignMd(md) {
805
+ const { frontmatter, body } = parseFrontmatter(md);
806
+ const { title, sections } = splitSections(body);
807
+ return {
808
+ schemaVersion: 2,
809
+ title,
810
+ frontmatter,
811
+ overview: extractOverview(sections['Overview']),
812
+ colors: extractColors(sections['Colors']),
813
+ typography: extractTypography(sections['Typography']),
814
+ elevation: extractElevation(sections['Elevation']),
815
+ components: extractComponents(sections['Components']),
816
+ dosDonts: extractDosDonts(sections["Do's and Don'ts"]),
817
+ };
818
+ }
819
+
820
+ export { assessCoverage };