oh-my-design-cli 1.7.1 → 1.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/hooks/lib/preferences-parser.cjs +91 -0
- package/.claude/hooks/lib/preferences-writer.cjs +118 -0
- package/.claude/hooks/post-edit-watch.cjs +232 -36
- package/.claude/hooks/session-end-foldin.cjs +72 -19
- package/.claude/hooks/session-state-loader.cjs +52 -3
- package/.claude/hooks/skill-activation.cjs +40 -7
- package/README.ja.md +32 -103
- package/README.ko.md +45 -206
- package/README.md +33 -151
- package/README.zh-TW.md +32 -103
- package/agents/AGENT.md +3 -3
- package/agents/omd-master.md +16 -7
- package/agents/omd-microcopy.md +1 -1
- package/agents/omd-ux-engineer.md +9 -7
- package/agents/omd-ux-researcher.md +1 -1
- package/agents/omd-ux-writer.md +1 -1
- package/dist/bin/oh-my-design.js +3 -3
- package/dist/bin/oh-my-design.js.map +1 -1
- package/dist/{install-skills-YYHEC4CS.js → install-skills-7UUDOLG2.js} +152 -20
- package/dist/install-skills-7UUDOLG2.js.map +1 -0
- package/package.json +3 -1
- package/skills/omd-designer-review/SKILL.md +34 -0
- package/skills/omd-final-qa/SKILL.md +29 -0
- package/skills/omd-harness/SKILL.md +30 -9
- package/skills/omd-init/SKILL.md +52 -6
- package/skills/omd-kr-writer/SKILL.md +73 -3
- package/skills/omd-learn/SKILL.md +20 -0
- package/skills/omd-reference-capture/SKILL.md +15 -4
- package/skills/omd-taste/SKILL.md +79 -0
- package/dist/install-skills-YYHEC4CS.js.map +0 -1
|
@@ -13,6 +13,7 @@ import {
|
|
|
13
13
|
} from "fs";
|
|
14
14
|
import { join as join2, dirname, relative } from "path";
|
|
15
15
|
import { homedir } from "os";
|
|
16
|
+
import { createHash } from "crypto";
|
|
16
17
|
import { fileURLToPath } from "url";
|
|
17
18
|
|
|
18
19
|
// src/core/agent-detect.ts
|
|
@@ -156,7 +157,32 @@ function planForTarget(projectRoot, target) {
|
|
|
156
157
|
};
|
|
157
158
|
}
|
|
158
159
|
}
|
|
160
|
+
var CHANNEL_DATA_DIRS = {
|
|
161
|
+
"claude-code": ".claude",
|
|
162
|
+
codex: ".codex",
|
|
163
|
+
opencode: null,
|
|
164
|
+
cursor: null
|
|
165
|
+
};
|
|
166
|
+
function dataDirFor(target, targets) {
|
|
167
|
+
if (target === "cursor") {
|
|
168
|
+
return targets.includes("claude-code") ? null : ".claude";
|
|
169
|
+
}
|
|
170
|
+
return CHANNEL_DATA_DIRS[target];
|
|
171
|
+
}
|
|
159
172
|
var MANAGED_HEADER = "<!-- omd:installed-skill \u2014 managed by `omd install-skills`. Do not edit; rerun the command to refresh. -->";
|
|
173
|
+
var MANAGED_MARKER_SUBSTR = "omd:installed-skill";
|
|
174
|
+
function withManagedMarker(src) {
|
|
175
|
+
const fm = /^(---\r?\n[\s\S]*?\r?\n---\r?\n)([\s\S]*)$/.exec(src);
|
|
176
|
+
if (!fm) {
|
|
177
|
+
return MANAGED_HEADER + "\n\n" + src;
|
|
178
|
+
}
|
|
179
|
+
return fm[1] + MANAGED_HEADER + "\n\n" + fm[2];
|
|
180
|
+
}
|
|
181
|
+
function isManagedSkillFile(content) {
|
|
182
|
+
if (!content) return false;
|
|
183
|
+
const head = content.split("\n", 30).join("\n");
|
|
184
|
+
return head.includes(MANAGED_MARKER_SUBSTR);
|
|
185
|
+
}
|
|
160
186
|
var IGNORED_SKILL_ENTRIES = /* @__PURE__ */ new Set([".runtime", "__pycache__", ".DS_Store"]);
|
|
161
187
|
function parseSkillChannels(skillMd) {
|
|
162
188
|
const fm = /^---\n([\s\S]*?)\n---/.exec(skillMd);
|
|
@@ -173,7 +199,7 @@ function skillSupportedChannels(packageRoot, skill) {
|
|
|
173
199
|
function installOne(packageRoot, plan, skill, force) {
|
|
174
200
|
const skillDir = join2(packageRoot, "skills", skill);
|
|
175
201
|
const src = readFileSync(join2(skillDir, "SKILL.md"), "utf8");
|
|
176
|
-
const managed =
|
|
202
|
+
const managed = withManagedMarker(src);
|
|
177
203
|
const channels = parseSkillChannels(src);
|
|
178
204
|
if (channels && !channels.includes(plan.target)) {
|
|
179
205
|
return {
|
|
@@ -201,7 +227,7 @@ function installOne(packageRoot, plan, skill, force) {
|
|
|
201
227
|
if (exists && existing === managed && !isMultiFile) {
|
|
202
228
|
return { target: plan.target, skill, destPath, status: "unchanged" };
|
|
203
229
|
}
|
|
204
|
-
if (exists && !existing
|
|
230
|
+
if (exists && !isManagedSkillFile(existing) && !force) {
|
|
205
231
|
return { target: plan.target, skill, destPath, status: "skipped-drift" };
|
|
206
232
|
}
|
|
207
233
|
mkdirSync(dirname(destPath), { recursive: true });
|
|
@@ -272,8 +298,7 @@ function installSettingsJson(packageRoot, projectRoot, force) {
|
|
|
272
298
|
writeFileSync(destPath, src);
|
|
273
299
|
return { target, skill: skillLabel, destPath, status: exists ? "updated" : "created" };
|
|
274
300
|
}
|
|
275
|
-
function installDataFile(packageRoot, projectRoot, channelDir, dataFilename, force) {
|
|
276
|
-
const target = channelDir === ".claude" ? "claude-code" : "codex";
|
|
301
|
+
function installDataFile(packageRoot, projectRoot, channelDir, dataFilename, force, target) {
|
|
277
302
|
const skillLabel = `data:${dataFilename}`;
|
|
278
303
|
const srcPath = join2(packageRoot, "data", dataFilename);
|
|
279
304
|
const destPath = join2(projectRoot, channelDir, "data", dataFilename);
|
|
@@ -329,6 +354,75 @@ function installAgentFile(packageRoot, projectRoot, channel, filename, force) {
|
|
|
329
354
|
status: exists ? "updated" : "created"
|
|
330
355
|
};
|
|
331
356
|
}
|
|
357
|
+
function installReferenceCatalog(packageRoot, installRoot, channelDir, force) {
|
|
358
|
+
const srcRoot = join2(packageRoot, "web", "references");
|
|
359
|
+
if (!existsSync2(srcRoot)) return 0;
|
|
360
|
+
const destRoot = join2(installRoot, channelDir, "data", "references");
|
|
361
|
+
let written = 0;
|
|
362
|
+
for (const id of readdirSync(srcRoot)) {
|
|
363
|
+
const srcDesign = join2(srcRoot, id, "DESIGN.md");
|
|
364
|
+
if (!existsSync2(srcDesign)) continue;
|
|
365
|
+
const destDesign = join2(destRoot, id, "DESIGN.md");
|
|
366
|
+
const src = readFileSync(srcDesign, "utf8");
|
|
367
|
+
if (existsSync2(destDesign)) {
|
|
368
|
+
const existing = readFileSync(destDesign, "utf8");
|
|
369
|
+
if (existing === src) continue;
|
|
370
|
+
if (!force) continue;
|
|
371
|
+
}
|
|
372
|
+
mkdirSync(dirname(destDesign), { recursive: true });
|
|
373
|
+
writeFileSync(destDesign, src, "utf8");
|
|
374
|
+
written++;
|
|
375
|
+
}
|
|
376
|
+
return written;
|
|
377
|
+
}
|
|
378
|
+
var CURSOR_RULE_BODY = [
|
|
379
|
+
"The authoritative design spec lives at `@DESIGN.md` (repo root). Open and read before generating/modifying UI.",
|
|
380
|
+
"",
|
|
381
|
+
"Pending preference corrections: `@.omd/preferences.md`.",
|
|
382
|
+
"",
|
|
383
|
+
"Precedence: DESIGN.md > preferences.md > framework defaults."
|
|
384
|
+
].join("\n");
|
|
385
|
+
function renderCursorRule() {
|
|
386
|
+
const hash = createHash("sha256").update(CURSOR_RULE_BODY).digest("hex").slice(0, 12);
|
|
387
|
+
return [
|
|
388
|
+
"---",
|
|
389
|
+
"description: Authoritative brand & UI design system. Read DESIGN.md before UI work.",
|
|
390
|
+
"globs:",
|
|
391
|
+
' - "**/*.tsx"',
|
|
392
|
+
' - "**/*.jsx"',
|
|
393
|
+
' - "**/*.vue"',
|
|
394
|
+
' - "**/*.svelte"',
|
|
395
|
+
' - "**/*.css"',
|
|
396
|
+
' - "**/*.scss"',
|
|
397
|
+
' - "**/tailwind.config.*"',
|
|
398
|
+
' - "**/components/**"',
|
|
399
|
+
' - "**/app/**/page.*"',
|
|
400
|
+
"alwaysApply: false",
|
|
401
|
+
"---",
|
|
402
|
+
"",
|
|
403
|
+
`<!-- omd:start v=1 hash=${hash} -->`,
|
|
404
|
+
CURSOR_RULE_BODY,
|
|
405
|
+
"<!-- omd:end -->",
|
|
406
|
+
""
|
|
407
|
+
].join("\n");
|
|
408
|
+
}
|
|
409
|
+
function installCursorRule(installRoot, force) {
|
|
410
|
+
const target = "cursor";
|
|
411
|
+
const skillLabel = "rule:omd-design.mdc";
|
|
412
|
+
const destPath = join2(installRoot, ".cursor", "rules", "omd-design.mdc");
|
|
413
|
+
const rendered = renderCursorRule();
|
|
414
|
+
const exists = existsSync2(destPath);
|
|
415
|
+
const existing = exists ? readFileSync(destPath, "utf8") : "";
|
|
416
|
+
if (exists && existing === rendered) {
|
|
417
|
+
return { target, skill: skillLabel, destPath, status: "unchanged" };
|
|
418
|
+
}
|
|
419
|
+
if (exists && !existing.includes("<!-- omd:start") && !force) {
|
|
420
|
+
return { target, skill: skillLabel, destPath, status: "skipped-drift" };
|
|
421
|
+
}
|
|
422
|
+
mkdirSync(dirname(destPath), { recursive: true });
|
|
423
|
+
writeFileSync(destPath, rendered, "utf8");
|
|
424
|
+
return { target, skill: skillLabel, destPath, status: exists ? "updated" : "created" };
|
|
425
|
+
}
|
|
332
426
|
var STATUS_LABEL = {
|
|
333
427
|
created: pc.green("created"),
|
|
334
428
|
updated: pc.cyan("updated"),
|
|
@@ -342,6 +436,7 @@ function autoDetectTargets(projectRoot) {
|
|
|
342
436
|
if (presence.claudeCode) targets.push("claude-code");
|
|
343
437
|
if (presence.codex) targets.push("codex");
|
|
344
438
|
if (presence.opencode) targets.push("opencode");
|
|
439
|
+
if (presence.cursor) targets.push("cursor");
|
|
345
440
|
if (targets.length === 0) {
|
|
346
441
|
return ["claude-code", "codex", "opencode"];
|
|
347
442
|
}
|
|
@@ -373,7 +468,8 @@ async function runInstallSkills(opts = {}) {
|
|
|
373
468
|
const actuallyDetected = [
|
|
374
469
|
presence.claudeCode ? "claude-code" : null,
|
|
375
470
|
presence.codex ? "codex" : null,
|
|
376
|
-
presence.opencode ? "opencode" : null
|
|
471
|
+
presence.opencode ? "opencode" : null,
|
|
472
|
+
presence.cursor ? "cursor" : null
|
|
377
473
|
].filter((x) => x !== null);
|
|
378
474
|
if (!opts.global && interactive) {
|
|
379
475
|
const scopeResult = await p.select({
|
|
@@ -430,17 +526,20 @@ async function runInstallSkills(opts = {}) {
|
|
|
430
526
|
}
|
|
431
527
|
const supportedTargets = (() => {
|
|
432
528
|
const set = new Set(skills.flatMap((s) => skillSupportedChannels(packageRoot, s)));
|
|
433
|
-
|
|
529
|
+
set.add("cursor");
|
|
530
|
+
return ["claude-code", "codex", "opencode", "cursor"].filter((t) => set.has(t));
|
|
434
531
|
})();
|
|
435
532
|
const channelLabel = {
|
|
436
533
|
"claude-code": "Claude Code",
|
|
437
534
|
codex: "Codex",
|
|
438
|
-
opencode: "OpenCode"
|
|
535
|
+
opencode: "OpenCode",
|
|
536
|
+
cursor: "Cursor"
|
|
439
537
|
};
|
|
440
538
|
const channelDir = {
|
|
441
539
|
"claude-code": ".claude",
|
|
442
540
|
codex: ".codex",
|
|
443
|
-
opencode: ".opencode"
|
|
541
|
+
opencode: ".opencode",
|
|
542
|
+
cursor: ".cursor"
|
|
444
543
|
};
|
|
445
544
|
let targets;
|
|
446
545
|
if (opts.agents) {
|
|
@@ -468,7 +567,10 @@ async function runInstallSkills(opts = {}) {
|
|
|
468
567
|
if (narrowed.length > 0) targets = narrowed;
|
|
469
568
|
}
|
|
470
569
|
const installRoot = scope === "global" ? homedir() : projectRoot;
|
|
471
|
-
const
|
|
570
|
+
const skillChannelTargets = targets.filter(
|
|
571
|
+
(t) => t !== "cursor"
|
|
572
|
+
);
|
|
573
|
+
const plans = skillChannelTargets.map((t) => planForTarget(installRoot, t));
|
|
472
574
|
p.log.message(
|
|
473
575
|
pc.bold("Scope: ") + pc.cyan(scope) + pc.dim(scope === "global" ? " (~/.claude)" : ` (${relative(process.cwd(), projectRoot) || "."})`)
|
|
474
576
|
);
|
|
@@ -487,6 +589,7 @@ async function runInstallSkills(opts = {}) {
|
|
|
487
589
|
pc.bold("Targets: ") + targets.map((t) => pc.cyan(t)).join(", ")
|
|
488
590
|
);
|
|
489
591
|
const results = [];
|
|
592
|
+
let catalogCount = 0;
|
|
490
593
|
for (const plan of plans) {
|
|
491
594
|
for (const skill of skills) {
|
|
492
595
|
results.push(installOne(packageRoot, plan, skill, force));
|
|
@@ -504,6 +607,9 @@ async function runInstallSkills(opts = {}) {
|
|
|
504
607
|
}
|
|
505
608
|
}
|
|
506
609
|
if (!minimal) {
|
|
610
|
+
if (targets.includes("cursor")) {
|
|
611
|
+
results.push(installCursorRule(installRoot, force));
|
|
612
|
+
}
|
|
507
613
|
const dataFiles = [
|
|
508
614
|
"reference-fingerprints.json",
|
|
509
615
|
"reference-tags.md",
|
|
@@ -512,14 +618,29 @@ async function runInstallSkills(opts = {}) {
|
|
|
512
618
|
"opt-out-corpus.json"
|
|
513
619
|
];
|
|
514
620
|
for (const target of targets) {
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
}
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
621
|
+
const dataDir = dataDirFor(target, targets);
|
|
622
|
+
if (!dataDir) continue;
|
|
623
|
+
for (const dataFile of dataFiles) {
|
|
624
|
+
results.push(installDataFile(packageRoot, installRoot, dataDir, dataFile, force, target));
|
|
625
|
+
}
|
|
626
|
+
}
|
|
627
|
+
for (const target of targets) {
|
|
628
|
+
const dataDir = dataDirFor(target, targets);
|
|
629
|
+
if (dataDir) {
|
|
630
|
+
catalogCount += installReferenceCatalog(packageRoot, installRoot, dataDir, force);
|
|
631
|
+
}
|
|
632
|
+
}
|
|
633
|
+
for (const target of targets) {
|
|
634
|
+
const cd = CHANNEL_DATA_DIRS[target];
|
|
635
|
+
if (!cd) continue;
|
|
636
|
+
for (const helper of ["ctx-prime.cjs", "context.cjs"]) {
|
|
637
|
+
const srcHelper = join2(packageRoot, "scripts", helper);
|
|
638
|
+
if (!existsSync2(srcHelper)) continue;
|
|
639
|
+
const destHelper = join2(installRoot, cd, "data", "scripts", helper);
|
|
640
|
+
const srcTxt = readFileSync(srcHelper, "utf8");
|
|
641
|
+
if (existsSync2(destHelper) && readFileSync(destHelper, "utf8") === srcTxt) continue;
|
|
642
|
+
mkdirSync(dirname(destHelper), { recursive: true });
|
|
643
|
+
writeFileSync(destHelper, srcTxt, "utf8");
|
|
523
644
|
}
|
|
524
645
|
}
|
|
525
646
|
if (scope === "project" && targets.includes("claude-code")) {
|
|
@@ -527,7 +648,12 @@ async function runInstallSkills(opts = {}) {
|
|
|
527
648
|
"skill-activation.cjs",
|
|
528
649
|
"session-state-loader.cjs",
|
|
529
650
|
"post-edit-watch.cjs",
|
|
530
|
-
"session-end-foldin.cjs"
|
|
651
|
+
"session-end-foldin.cjs",
|
|
652
|
+
// Shared modules required by the fold-in / state-loader / watch hooks.
|
|
653
|
+
// Live in a lib/ subdir; installHookFile preserves the relative path
|
|
654
|
+
// under .claude/hooks/.
|
|
655
|
+
join2("lib", "preferences-parser.cjs"),
|
|
656
|
+
join2("lib", "preferences-writer.cjs")
|
|
531
657
|
]) {
|
|
532
658
|
results.push(installHookFile(packageRoot, installRoot, hookFile, force));
|
|
533
659
|
}
|
|
@@ -589,14 +715,20 @@ async function runInstallSkills(opts = {}) {
|
|
|
589
715
|
].join("\n");
|
|
590
716
|
p.note(nextSteps, "Next");
|
|
591
717
|
const hookCount = scope === "project" && targets.includes("claude-code") ? 4 : 0;
|
|
718
|
+
if (catalogCount > 0) {
|
|
719
|
+
p.log.message(
|
|
720
|
+
pc.bold("Reference catalog: ") + pc.cyan(`${catalogCount}`) + pc.dim(" DESIGN.md copied \u2192 .claude/data/references/<id>/DESIGN.md")
|
|
721
|
+
);
|
|
722
|
+
}
|
|
592
723
|
p.outro(
|
|
593
724
|
pc.green(
|
|
594
|
-
`Done. ${skills.length} skills \xB7 ${canonicalAgents.length} sub-agents \xB7 ${hookCount} hooks installed (${writtenCount} files)${scope === "global" ? " globally (~/.claude)" : ""}.`
|
|
725
|
+
`Done. ${skills.length} skills \xB7 ${canonicalAgents.length} sub-agents \xB7 ${hookCount} hooks \xB7 ${catalogCount} catalog refs installed (${writtenCount} files)${scope === "global" ? " globally (~/.claude)" : ""}.`
|
|
595
726
|
)
|
|
596
727
|
);
|
|
597
728
|
return 0;
|
|
598
729
|
}
|
|
599
730
|
export {
|
|
731
|
+
dataDirFor,
|
|
600
732
|
runInstallSkills
|
|
601
733
|
};
|
|
602
|
-
//# sourceMappingURL=install-skills-
|
|
734
|
+
//# sourceMappingURL=install-skills-7UUDOLG2.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/cli/install-skills.ts","../src/core/agent-detect.ts"],"sourcesContent":["import * as p from '@clack/prompts';\nimport pc from 'picocolors';\nimport {\n readFileSync,\n readdirSync,\n writeFileSync,\n existsSync,\n mkdirSync,\n cpSync,\n} from 'node:fs';\nimport { join, dirname, relative } from 'node:path';\nimport { homedir } from 'node:os';\nimport { createHash } from 'node:crypto';\nimport { fileURLToPath } from 'node:url';\nimport { detectInstalledAgents } from '../core/agent-detect.js';\n\nexport type SkillTarget = 'claude-code' | 'codex' | 'opencode' | 'cursor';\n\n/** Channels that host SKILL.md trees. Cursor is NOT one — it consumes a\n * `.cursor/rules` shim + the shared `.claude/data` catalog (issue #20). */\ntype SkillChannel = Exclude<SkillTarget, 'cursor'>;\n\nexport interface InstallSkillsOptions {\n dir?: string;\n agents?: SkillTarget[];\n force?: boolean;\n /** Non-interactive: install all skills + all agents without TUI prompt.\n * Default false → interactive multiselect when TTY available. */\n all?: boolean;\n /** Pre-select specific skill names from CLI flag (`--skills omd-init,omd-apply`).\n * Overrides interactive prompt when set. */\n skillsFilter?: string[];\n /** Pre-select specific agent names. Overrides interactive prompt when set. */\n agentsFilter?: string[];\n /** Minimal install: only the named skill files — skip sub-agents, data files,\n * hooks, and settings.json. Ideal for shipping a single standalone skill. */\n skillsOnly?: boolean;\n /** Install to the user-level dir (~/.claude/skills) instead of this project.\n * Writes skills + sub-agents (+ data); never touches global hooks/settings.\n * When unset and interactive, the TUI asks project-vs-global. */\n global?: boolean;\n}\n\ninterface InstallPlan {\n target: SkillChannel;\n destDir: string;\n layout: 'folder' | 'flat';\n}\n\nfunction findPackageRoot(): string | null {\n let cur = dirname(fileURLToPath(import.meta.url));\n for (let i = 0; i < 8; i++) {\n if (existsSync(join(cur, 'skills'))) return cur;\n const parent = dirname(cur);\n if (parent === cur) break;\n cur = parent;\n }\n return null;\n}\n\nfunction listShippedSkills(packageRoot: string): string[] {\n const skillsDir = join(packageRoot, 'skills');\n if (!existsSync(skillsDir)) return [];\n return readdirSync(skillsDir)\n .filter((name) => existsSync(join(skillsDir, name, 'SKILL.md')))\n .sort();\n}\n\n/**\n * Canonical agent definitions live at `agents/<name>.md` (markdown with YAML\n * frontmatter). Channel-specific files (.claude/agents/*.md, .codex/agents/*.toml)\n * are generated artifacts — never the source of truth.\n *\n * The package ships only `agents/` and the generator emits per-channel files\n * into the user's project on `omd install-skills`.\n */\nfunction listCanonicalAgents(packageRoot: string): string[] {\n const dir = join(packageRoot, 'agents');\n if (!existsSync(dir)) return [];\n return readdirSync(dir)\n .filter((name) => name.startsWith('omd-') && name.endsWith('.md'))\n .sort();\n}\n\ninterface ParsedAgent {\n name: string;\n description: string;\n tools: string[];\n model: string;\n body: string;\n}\n\n/** Parse `agents/<name>.md` YAML frontmatter + body into structured form. */\nfunction parseCanonicalAgent(packageRoot: string, filename: string): ParsedAgent {\n const src = readFileSync(join(packageRoot, 'agents', filename), 'utf8');\n const match = /^---\\n([\\s\\S]*?)\\n---\\n([\\s\\S]*)$/.exec(src);\n if (!match) {\n throw new Error(`agents/${filename}: missing YAML frontmatter`);\n }\n const fm = match[1];\n const body = match[2];\n const grab = (key: string): string => {\n const re = new RegExp(`^${key}:\\\\s*(.+)$`, 'm');\n const m = re.exec(fm);\n return m ? m[1].trim().replace(/^[\"']|[\"']$/g, '') : '';\n };\n return {\n name: grab('name') || filename.replace(/\\.md$/, ''),\n description: grab('description'),\n tools: grab('tools')\n .split(',')\n .map((t) => t.trim())\n .filter(Boolean),\n model: grab('model') || 'sonnet',\n body,\n };\n}\n\n/** Map Claude tool names to Codex tool names (best-effort). */\nfunction claudeToolsToCodex(tools: string[]): string[] {\n const m: Record<string, string> = {\n Read: 'read_file',\n Write: 'write_file',\n Edit: 'edit_file',\n Bash: 'shell',\n Glob: 'search',\n Grep: 'search',\n WebFetch: 'web_fetch',\n WebSearch: 'search',\n Agent: 'spawn_agent',\n TaskCreate: 'task',\n TaskUpdate: 'task',\n TaskList: 'task',\n };\n const out = new Set<string>();\n for (const t of tools) out.add(m[t] ?? t.toLowerCase());\n return [...out].sort();\n}\n\n/** Map Claude model alias to Codex/OpenAI model id (best-effort). */\nfunction claudeModelToCodex(model: string): string {\n const m: Record<string, string> = {\n haiku: 'gpt-4.1-mini',\n sonnet: 'gpt-4.1',\n opus: 'gpt-4.1',\n };\n return m[model.toLowerCase()] ?? 'gpt-4.1';\n}\n\n/** Render a canonical agent as a Claude Code subagent file.\n * IMPORTANT: Claude Code's subagent parser requires YAML frontmatter (`---`)\n * as the FIRST line of the file. Any preceding content (HTML comments, blank\n * lines) breaks discovery. So we encode the managed-by-omd marker as a\n * custom frontmatter field (`omd_managed: true`) instead of an HTML comment.\n */\nfunction renderClaudeAgent(a: ParsedAgent): string {\n const fm = [\n '---',\n `name: ${a.name}`,\n `description: ${a.description}`,\n `tools: ${a.tools.join(', ')}`,\n `model: ${a.model}`,\n `omd_managed: true`,\n '---',\n '',\n ].join('\\n');\n return fm + a.body;\n}\n\n/** Render a canonical agent as a Codex TOML file (declarative pointer). */\nfunction renderCodexAgent(a: ParsedAgent): string {\n const tools = claudeToolsToCodex(a.tools);\n const model = claudeModelToCodex(a.model);\n const desc = a.description.replace(/\"/g, '\\\\\"');\n return [\n `[agent]`,\n `name = \"${a.name}\"`,\n `description = \"${desc}\"`,\n `model = \"${model}\"`,\n `max_threads = 1`,\n `allowed_tools = [${tools.map((t) => `\"${t}\"`).join(', ')}]`,\n '',\n `instructions = \"\"\"`,\n `Source of truth: agents/${a.name}.md (canonical). The full role spec is`,\n `mirrored to .claude/agents/${a.name}.md when installed for Claude Code.`,\n `Follow that spec verbatim regardless of channel.`,\n '',\n `Codex notes:`,\n `- Spawn sub-agents via spawn_agent with names matching .codex/agents/<name>.toml`,\n `- Use shell to invoke CLI helpers (omd init prepare, omd remember, git apply, npx axe-core, npx lighthouse)`,\n `- All artifacts go inside .omd/runs/run-<latest>/ (or skills/omd-lab-02-design-harness/runs/<lab-version>-...)`,\n `\"\"\"`,\n '',\n ].join('\\n');\n}\n\nfunction planForTarget(projectRoot: string, target: SkillChannel): InstallPlan {\n switch (target) {\n case 'claude-code':\n return {\n target,\n destDir: join(projectRoot, '.claude', 'skills'),\n layout: 'folder',\n };\n case 'codex':\n // Official Codex skill discovery path is `.agents/skills/<name>/SKILL.md`\n // (developers.openai.com/codex/skills) — NOT `.codex/skills`. Folder layout\n // so multi-file skills (scripts/, references/) install + run.\n return {\n target,\n destDir: join(projectRoot, '.agents', 'skills'),\n layout: 'folder',\n };\n case 'opencode':\n // OpenCode loads `.opencode/skills/<name>/SKILL.md` (opencode.ai/docs/skills)\n // as folder skills — the old flat `.opencode/agents/<name>.md` couldn't host\n // a skill's scripts/references.\n return {\n target,\n destDir: join(projectRoot, '.opencode', 'skills'),\n layout: 'folder',\n };\n }\n}\n\n/**\n * Channel → shared data dir (`<dir>/data/…`) for read-only data assets\n * (catalog JSONs, reference DESIGN.md catalog, ctx-prime helper scripts).\n * Single lookup table replacing the repeated if-else/ternary chains (issue #28).\n * `null` = the channel hosts no data dir of its own:\n * - opencode ships skills only (no data channel yet);\n * - cursor reads the SHARED `.claude/data` path (issue #20) — resolved by\n * `dataDirFor()` below, which also applies the claude-code dedup guard.\n * (Helper scripts intentionally stay claude-code/codex only — a cursor-only\n * install gets the catalog + JSONs but no ctx-prime, matching the shim's scope.)\n */\nconst CHANNEL_DATA_DIRS: Record<SkillTarget, '.claude' | '.codex' | null> = {\n 'claude-code': '.claude',\n codex: '.codex',\n opencode: null,\n cursor: null,\n};\n\n/**\n * Where a target's data assets (data JSONs + reference catalog) land. Cursor\n * reuses `.claude/data` so the catalog location stays single (issue #20) —\n * but ONLY when claude-code isn't also selected: the claude-code pass already\n * writes there, and a second pass would double-copy the catalog.\n */\nexport function dataDirFor(\n target: SkillTarget,\n targets: SkillTarget[]\n): '.claude' | '.codex' | null {\n if (target === 'cursor') {\n return targets.includes('claude-code') ? null : '.claude';\n }\n return CHANNEL_DATA_DIRS[target];\n}\n\nconst MANAGED_HEADER =\n '<!-- omd:installed-skill — managed by `omd install-skills`. Do not edit; rerun the command to refresh. -->';\n\n// Substring shared by old (line 1) and new (after-frontmatter) marker formats.\n// Used for detection so upgrades from pre-v1.7.2 installs still refresh.\nconst MANAGED_MARKER_SUBSTR = 'omd:installed-skill';\n\n/**\n * Write the managed marker AFTER the YAML frontmatter block so the very first\n * line of the installed file is `---`. Claude Code's skill loader reads the\n * frontmatter `name`/`description` only when `---` is line 1 — a leading HTML\n * comment makes it register the comment as the description (issue #17).\n *\n * If the source has no frontmatter (shouldn't happen for SKILL.md, but be\n * defensive), fall back to prepending the marker.\n */\nfunction withManagedMarker(src: string): string {\n // \\r?\\n: a CRLF checkout (Windows core.autocrlf) must not miss the\n // frontmatter and fall back to a line-1 marker — that reintroduces #17.\n const fm = /^(---\\r?\\n[\\s\\S]*?\\r?\\n---\\r?\\n)([\\s\\S]*)$/.exec(src);\n if (!fm) {\n return MANAGED_HEADER + '\\n\\n' + src;\n }\n return fm[1] + MANAGED_HEADER + '\\n\\n' + fm[2];\n}\n\n/**\n * Detect an omd-managed installed-skill file. Matches both the new format\n * (marker after frontmatter) and the legacy format (marker on line 1) by\n * scanning the first ~30 lines for the marker substring. This keeps upgrades\n * working: a pre-v1.7.2 file with the marker at line 1 is still recognized as\n * managed and gets refreshed rather than skipped as user-edited drift.\n */\nfunction isManagedSkillFile(content: string): boolean {\n if (!content) return false;\n const head = content.split('\\n', 30).join('\\n');\n return head.includes(MANAGED_MARKER_SUBSTR);\n}\n\ninterface InstallResult {\n target: SkillTarget;\n skill: string;\n destPath: string;\n status: 'created' | 'updated' | 'unchanged' | 'skipped-drift' | 'skipped-incompat';\n}\n\n// Skill-tree entries that must never be installed (runtime state, caches, OS cruft).\nconst IGNORED_SKILL_ENTRIES = new Set(['.runtime', '__pycache__', '.DS_Store']);\n\n/**\n * A skill may restrict itself to specific agent channels via a frontmatter line\n * `x-omd-channels: claude-code` (comma/space separated). Returns the allowed\n * channels, or null when channel-agnostic (installs anywhere). Used by skills that\n * depend on a particular agent runtime — e.g. claude-design needs Claude Code's\n * claude-in-chrome MCP + Bash/python/node and is therefore claude-code only.\n */\nfunction parseSkillChannels(skillMd: string): SkillChannel[] | null {\n const fm = /^---\\n([\\s\\S]*?)\\n---/.exec(skillMd);\n if (!fm) return null;\n const m = /^x-omd-channels:\\s*(.+)$/m.exec(fm[1]);\n if (!m) return null;\n const valid: SkillChannel[] = ['claude-code', 'codex', 'opencode'];\n const list = m[1]\n .split(/[,\\s]+/)\n .map((s) => s.trim())\n .filter((s): s is SkillChannel => (valid as string[]).includes(s));\n return list.length > 0 ? list : null;\n}\n\n/**\n * The agent channels a skill can install into: its declared `x-omd-channels`\n * (if any), else all channels. All three channels now use folder layout\n * (.claude/skills, .agents/skills, .opencode/skills) so multi-file skills with\n * scripts/references install everywhere — the only restriction is what the skill\n * itself declares (e.g. claude-design needs a browser-driving runtime).\n */\nfunction skillSupportedChannels(packageRoot: string, skill: string): SkillChannel[] {\n return (\n parseSkillChannels(readFileSync(join(packageRoot, 'skills', skill, 'SKILL.md'), 'utf8')) ??\n (['claude-code', 'codex', 'opencode'] as SkillChannel[])\n );\n}\n\nfunction installOne(\n packageRoot: string,\n plan: InstallPlan,\n skill: string,\n force: boolean\n): InstallResult {\n const skillDir = join(packageRoot, 'skills', skill);\n const src = readFileSync(join(skillDir, 'SKILL.md'), 'utf8');\n // Marker goes AFTER frontmatter so `---` stays line 1 (issue #17).\n const managed = withManagedMarker(src);\n\n // Respect a skill's declared channel restriction (frontmatter `x-omd-channels:`).\n const channels = parseSkillChannels(src);\n if (channels && !channels.includes(plan.target)) {\n return {\n target: plan.target,\n skill,\n destPath: join(plan.destDir, skill + '.md'),\n status: 'skipped-incompat',\n };\n }\n\n // A skill is \"multi-file\" when it ships more than SKILL.md (scripts/, references/, …).\n const extras = readdirSync(skillDir).filter(\n (n) => n !== 'SKILL.md' && !IGNORED_SKILL_ENTRIES.has(n)\n );\n const isMultiFile = extras.length > 0;\n\n // Flat channels (codex/opencode) store a skill as a single <skill>.md and cannot\n // host a multi-file skill's scripts/references — such skills are claude-code only.\n if (plan.layout !== 'folder' && isMultiFile) {\n return {\n target: plan.target,\n skill,\n destPath: join(plan.destDir, skill + '.md'),\n status: 'skipped-incompat',\n };\n }\n\n const destPath =\n plan.layout === 'folder'\n ? join(plan.destDir, skill, 'SKILL.md')\n : join(plan.destDir, skill + '.md');\n\n const exists = existsSync(destPath);\n const existing = exists ? readFileSync(destPath, 'utf8') : '';\n\n // Drift protection guards the user-editable SKILL.md. Single-file skills can\n // short-circuit on \"unchanged\"; multi-file skills always re-sync their tree.\n if (exists && existing === managed && !isMultiFile) {\n return { target: plan.target, skill, destPath, status: 'unchanged' };\n }\n // Drift = a file we didn't write. Detect the marker anywhere in the head\n // (new after-frontmatter position OR legacy line-1 position) so pre-v1.7.2\n // installs are recognized as managed and refreshed, not skipped.\n if (exists && !isManagedSkillFile(existing) && !force) {\n return { target: plan.target, skill, destPath, status: 'skipped-drift' };\n }\n\n mkdirSync(dirname(destPath), { recursive: true });\n writeFileSync(destPath, managed, 'utf8');\n\n // Copy the rest of the skill tree (scripts/, references/, …) for folder layout.\n if (plan.layout === 'folder' && isMultiFile) {\n const destSkillDir = join(plan.destDir, skill);\n for (const entry of extras) {\n cpSync(join(skillDir, entry), join(destSkillDir, entry), {\n recursive: true,\n filter: (s) => !/(\\/__pycache__|\\/\\.runtime|\\.pyc$|\\.DS_Store$)/.test(s),\n });\n }\n }\n\n return {\n target: plan.target,\n skill,\n destPath,\n status: exists ? 'updated' : 'created',\n };\n}\n\n/** Install a hook script from package's .claude/hooks/ to project. */\nfunction installHookFile(\n packageRoot: string,\n projectRoot: string,\n filename: string,\n force: boolean\n): InstallResult {\n const target: SkillTarget = 'claude-code';\n const skillLabel = `hook:${filename}`;\n const srcPath = join(packageRoot, '.claude', 'hooks', filename);\n const destPath = join(projectRoot, '.claude', 'hooks', filename);\n\n if (!existsSync(srcPath)) {\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n const src = readFileSync(srcPath, 'utf8');\n const exists = existsSync(destPath);\n const existing = exists ? readFileSync(destPath, 'utf8') : '';\n if (exists && existing === src) {\n return { target, skill: skillLabel, destPath, status: 'unchanged' };\n }\n if (exists && !force) {\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n mkdirSync(dirname(destPath), { recursive: true });\n writeFileSync(destPath, src);\n return { target, skill: skillLabel, destPath, status: exists ? 'updated' : 'created' };\n}\n\n/**\n * Install / merge .claude/settings.json. We MERGE hooks (don't clobber user\n * customizations) by checking if the omd-managed `_doc` field is present.\n * Without --force, if a user-edited settings.json exists (no _doc field),\n * we skip with `skipped-drift`.\n */\nfunction installSettingsJson(\n packageRoot: string,\n projectRoot: string,\n force: boolean\n): InstallResult {\n const target: SkillTarget = 'claude-code';\n const skillLabel = 'settings:.claude/settings.json';\n const srcPath = join(packageRoot, '.claude', 'settings.json');\n const destPath = join(projectRoot, '.claude', 'settings.json');\n if (!existsSync(srcPath)) {\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n const src = readFileSync(srcPath, 'utf8');\n const exists = existsSync(destPath);\n const existing = exists ? readFileSync(destPath, 'utf8') : '';\n if (exists && existing === src) {\n return { target, skill: skillLabel, destPath, status: 'unchanged' };\n }\n if (exists && !force) {\n // Check if it's the omd-managed version\n try {\n const parsed = JSON.parse(existing);\n if (typeof parsed._doc === 'string' && parsed._doc.includes('OmD')) {\n // managed — overwrite\n } else {\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n } catch {\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n }\n mkdirSync(dirname(destPath), { recursive: true });\n writeFileSync(destPath, src);\n return { target, skill: skillLabel, destPath, status: exists ? 'updated' : 'created' };\n}\n\n/**\n * Copy a read-only data asset (reference-fingerprints.json, vocabulary.json, …)\n * from the package's `data/` into the project's `.claude/data/` or `.codex/data/`.\n * The skill reads these at runtime — they replace the deprecated `omd init recommend` CLI.\n */\nfunction installDataFile(\n packageRoot: string,\n projectRoot: string,\n channelDir: string,\n dataFilename: string,\n force: boolean,\n // Cursor reuses the `.claude` data dir (single catalog path) — callers pass\n // an explicit target so the results table reports the real channel.\n target: SkillTarget\n): InstallResult {\n const skillLabel = `data:${dataFilename}`;\n\n const srcPath = join(packageRoot, 'data', dataFilename);\n const destPath = join(projectRoot, channelDir, 'data', dataFilename);\n\n if (!existsSync(srcPath)) {\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n\n const src = readFileSync(srcPath, 'utf8');\n const exists = existsSync(destPath);\n const existing = exists ? readFileSync(destPath, 'utf8') : '';\n\n // Data files are pure copies — no managed header (would corrupt JSON).\n if (exists && existing === src) {\n return { target, skill: skillLabel, destPath, status: 'unchanged' };\n }\n if (exists && !force) {\n // Honor user customization unless --force\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n\n mkdirSync(dirname(destPath), { recursive: true });\n writeFileSync(destPath, src, 'utf8');\n return {\n target,\n skill: skillLabel,\n destPath,\n status: exists ? 'updated' : 'created',\n };\n}\n\n/**\n * Generate a per-channel agent file from the canonical `agents/<name>.md`.\n *\n * Channel = 'claude' → emits `.claude/agents/<name>.md` (markdown w/ frontmatter)\n * Channel = 'codex' → emits `.codex/agents/<name>.toml` (TOML pointer)\n */\nfunction installAgentFile(\n packageRoot: string,\n projectRoot: string,\n channel: 'claude' | 'codex',\n filename: string,\n force: boolean\n): InstallResult {\n const target: SkillTarget = channel === 'claude' ? 'claude-code' : 'codex';\n const skillLabel = `agent:${filename}`;\n\n const parsed = parseCanonicalAgent(packageRoot, filename);\n const rendered =\n channel === 'claude' ? renderClaudeAgent(parsed) : renderCodexAgent(parsed);\n\n const destFilename =\n channel === 'claude' ? filename : filename.replace(/\\.md$/, '.toml');\n const destPath = join(\n projectRoot,\n channel === 'claude' ? '.claude' : '.codex',\n 'agents',\n destFilename\n );\n\n // For Claude Code: managed marker is encoded as `omd_managed: true` INSIDE the\n // frontmatter (rendered above) — no HTML comment can precede `---` or the\n // subagent loader rejects the file.\n // For Codex: TOML allows leading comments, so `# omd:installed-agent ...` is fine.\n const managed =\n channel === 'claude'\n ? rendered\n : '# omd:installed-agent — generated from agents/' +\n filename +\n ' by `omd install-skills`. Do not edit; rerun the command to refresh.\\n\\n' +\n rendered;\n\n const exists = existsSync(destPath);\n const existing = exists ? readFileSync(destPath, 'utf8') : '';\n\n if (exists && existing === managed) {\n return { target, skill: skillLabel, destPath, status: 'unchanged' };\n }\n\n // Drift detection sentinels:\n // Claude — look for `omd_managed: true` line inside frontmatter\n // Codex — look for `# omd:installed-agent` comment\n const isManaged =\n channel === 'claude'\n ? /\\nomd_managed:\\s*true\\b/.test(existing)\n : existing.startsWith('# omd:installed-agent');\n\n if (exists && !isManaged && !force) {\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n\n mkdirSync(dirname(destPath), { recursive: true });\n writeFileSync(destPath, managed, 'utf8');\n return {\n target,\n skill: skillLabel,\n destPath,\n status: exists ? 'updated' : 'created',\n };\n}\n\n/**\n * Copy the reference catalog (`web/references/<id>/DESIGN.md`) into the project's\n * `.claude/data/references/<id>/DESIGN.md` so it's reachable on clean npx installs\n * — where there is no `node_modules` and no dev `web/references` (issue #16).\n *\n * Only DESIGN.md per id is copied (not _promo.json/_research.md/screenshots) to\n * keep the install lean. Idempotent: skips ids whose DESIGN.md already matches.\n * Returns the number of catalog files written (created or updated).\n */\nfunction installReferenceCatalog(\n packageRoot: string,\n installRoot: string,\n channelDir: string,\n force: boolean\n): number {\n const srcRoot = join(packageRoot, 'web', 'references');\n if (!existsSync(srcRoot)) return 0;\n const destRoot = join(installRoot, channelDir, 'data', 'references');\n\n let written = 0;\n for (const id of readdirSync(srcRoot)) {\n const srcDesign = join(srcRoot, id, 'DESIGN.md');\n if (!existsSync(srcDesign)) continue;\n const destDesign = join(destRoot, id, 'DESIGN.md');\n const src = readFileSync(srcDesign, 'utf8');\n if (existsSync(destDesign)) {\n const existing = readFileSync(destDesign, 'utf8');\n if (existing === src) continue;\n if (!force) continue; // honor user edits unless --force\n }\n mkdirSync(dirname(destDesign), { recursive: true });\n writeFileSync(destDesign, src, 'utf8');\n written++;\n }\n return written;\n}\n\n/**\n * Cursor channel shim — Cursor has no skill/agent surface; it consumes a\n * project rule at `.cursor/rules/omd-design.mdc`. Frontmatter, body, and the\n * body-hash marker below mirror the omd:sync skill's cursor template EXACTLY\n * (skills/omd-sync/SKILL.md, \"whole\" mode: hash = sha256 of the body text,\n * 12-char hex prefix), so a later omd:sync run reads the installer-written\n * file as `clean` rather than drifted (issue #20).\n */\nconst CURSOR_RULE_BODY = [\n 'The authoritative design spec lives at `@DESIGN.md` (repo root). Open and read before generating/modifying UI.',\n '',\n 'Pending preference corrections: `@.omd/preferences.md`.',\n '',\n 'Precedence: DESIGN.md > preferences.md > framework defaults.',\n].join('\\n');\n\nfunction renderCursorRule(): string {\n const hash = createHash('sha256').update(CURSOR_RULE_BODY).digest('hex').slice(0, 12);\n return [\n '---',\n 'description: Authoritative brand & UI design system. Read DESIGN.md before UI work.',\n 'globs:',\n ' - \"**/*.tsx\"',\n ' - \"**/*.jsx\"',\n ' - \"**/*.vue\"',\n ' - \"**/*.svelte\"',\n ' - \"**/*.css\"',\n ' - \"**/*.scss\"',\n ' - \"**/tailwind.config.*\"',\n ' - \"**/components/**\"',\n ' - \"**/app/**/page.*\"',\n 'alwaysApply: false',\n '---',\n '',\n `<!-- omd:start v=1 hash=${hash} -->`,\n CURSOR_RULE_BODY,\n '<!-- omd:end -->',\n '',\n ].join('\\n');\n}\n\nfunction installCursorRule(installRoot: string, force: boolean): InstallResult {\n const target: SkillTarget = 'cursor';\n const skillLabel = 'rule:omd-design.mdc';\n const destPath = join(installRoot, '.cursor', 'rules', 'omd-design.mdc');\n const rendered = renderCursorRule();\n\n const exists = existsSync(destPath);\n const existing = exists ? readFileSync(destPath, 'utf8') : '';\n if (exists && existing === rendered) {\n return { target, skill: skillLabel, destPath, status: 'unchanged' };\n }\n // The omd marker block doubles as the managed sentinel (matching omd:sync's\n // whole-mode rules): a file without it is user content → drift unless --force.\n if (exists && !existing.includes('<!-- omd:start') && !force) {\n return { target, skill: skillLabel, destPath, status: 'skipped-drift' };\n }\n mkdirSync(dirname(destPath), { recursive: true });\n writeFileSync(destPath, rendered, 'utf8');\n return { target, skill: skillLabel, destPath, status: exists ? 'updated' : 'created' };\n}\n\nconst STATUS_LABEL: Record<InstallResult['status'], string> = {\n created: pc.green('created'),\n updated: pc.cyan('updated'),\n unchanged: pc.dim('unchanged'),\n 'skipped-drift': pc.yellow('skipped'),\n 'skipped-incompat': pc.yellow('skipped (claude-code only)'),\n};\n\nfunction autoDetectTargets(projectRoot: string): SkillTarget[] {\n const presence = detectInstalledAgents(projectRoot);\n const targets: SkillTarget[] = [];\n if (presence.claudeCode) targets.push('claude-code');\n if (presence.codex) targets.push('codex');\n if (presence.opencode) targets.push('opencode');\n // Cursor hosts no skills — its channel writes the .cursor/rules shim + the\n // shared .claude/data catalog (issue #20). Only when .cursor is detected;\n // the no-signal fallback below stays skill-channel-only so we never drop a\n // .cursor dir into projects that don't use Cursor.\n if (presence.cursor) targets.push('cursor');\n if (targets.length === 0) {\n // Fallback: install for all three skill channels so user gets coverage\n // even without explicit signal. Idempotent so cost is low.\n return ['claude-code', 'codex', 'opencode'];\n }\n return targets;\n}\n\nexport async function runInstallSkills(\n opts: InstallSkillsOptions = {}\n): Promise<number> {\n const projectRoot = opts.dir ?? process.cwd();\n const packageRoot = findPackageRoot();\n if (!packageRoot) {\n console.error(pc.red('omd install-skills: package data not found'));\n return 1;\n }\n\n const allSkills = listShippedSkills(packageRoot);\n if (allSkills.length === 0) {\n console.error(pc.red('omd install-skills: no skills found in package'));\n return 1;\n }\n const allAgents = listCanonicalAgents(packageRoot);\n\n const force = opts.force ?? false;\n const minimal = opts.skillsOnly === true;\n // Install scope: 'project' (<cwd>/.claude/…) or 'global' (~/.claude/…). --global\n // forces it; otherwise the interactive TUI asks. Global writes skills + sub-agents\n // (+ data) to the user-level dir but never touches global hooks/settings.json.\n let scope: 'project' | 'global' = opts.global ? 'global' : 'project';\n\n p.intro(\n pc.bold('omd install-skills') +\n pc.dim(` (${relative(process.cwd(), projectRoot) || '.'})`)\n );\n\n // Each dimension (scope / skills / sub-agents / channels) is resolved\n // independently: a CLI flag pins it; otherwise we prompt — but only when stdin\n // is a TTY and --all wasn't passed. This is the key fix: `--skills X` or\n // `--skills-only` no longer suppress the *channel* (where to install) prompt —\n // they only pin the dimension they name.\n const isTTY = Boolean(process.stdin.isTTY && process.stdout.isTTY);\n const interactive = isTTY && !opts.all;\n\n const detected = autoDetectTargets(projectRoot);\n // Real presence (not the all-3 fallback) — used for hint labels + prompt defaults.\n const presence = detectInstalledAgents(projectRoot);\n const actuallyDetected: SkillTarget[] = [\n presence.claudeCode ? 'claude-code' : null,\n presence.codex ? 'codex' : null,\n presence.opencode ? 'opencode' : null,\n presence.cursor ? 'cursor' : null,\n ].filter((x): x is SkillTarget => x !== null);\n\n // --- Scope (project vs global) — --global pins it, else ask / default project.\n if (!opts.global && interactive) {\n const scopeResult = await p.select({\n message: 'Install scope · 어디에 설치할까요?',\n options: [\n { value: 'project', label: 'Project', hint: `${relative(process.cwd(), projectRoot) || '.'}/.claude/skills · 이 프로젝트만` },\n { value: 'global', label: 'Global', hint: '~/.claude/skills · 모든 프로젝트 (skills + sub-agents, hooks/settings 제외)' },\n ],\n initialValue: 'project',\n });\n if (p.isCancel(scopeResult)) { p.cancel('Install cancelled.'); return 130; }\n scope = scopeResult as 'project' | 'global';\n }\n\n // --- Skills — --skills pins it, else ask / default ALL.\n let skills: string[];\n if (opts.skillsFilter) {\n skills = allSkills.filter((s) => opts.skillsFilter!.includes(s));\n } else if (interactive) {\n const skillResult = await p.multiselect({\n message: 'Skills · space = 토글 · a = 전체 · enter = 확인 (default ALL)',\n options: allSkills.map((s) => ({ value: s, label: s, hint: 'omd skill' })),\n initialValues: allSkills,\n required: true,\n });\n if (p.isCancel(skillResult)) { p.cancel('Install cancelled.'); return 130; }\n skills = skillResult as string[];\n } else {\n skills = allSkills;\n }\n\n // --- Sub-agents — dropped by --skills-only, else --agents pins, else ask / ALL.\n let canonicalAgents: string[];\n if (minimal) {\n canonicalAgents = [];\n } else if (opts.agentsFilter) {\n canonicalAgents = allAgents.filter((a) => opts.agentsFilter!.includes(a.replace(/\\.md$/, '')));\n } else if (interactive && allAgents.length > 0) {\n const agentResult = await p.multiselect({\n message: 'Sub-agents · space = 토글 · a = 전체 · enter = 확인 (default ALL)',\n options: allAgents.map((a) => ({ value: a, label: a.replace(/\\.md$/, ''), hint: 'subagent' })),\n initialValues: allAgents,\n required: false,\n });\n if (p.isCancel(agentResult)) { p.cancel('Install cancelled.'); return 130; }\n canonicalAgents = agentResult as string[];\n } else {\n canonicalAgents = allAgents;\n }\n\n // --- Channels / targets — the \"where do I install\" choice.\n // --agent pins it. Otherwise, in a TTY we ASK — limited to the channels the\n // selected skills actually support (claude-design is claude-code only, so its\n // picker shows just Claude Code). Non-TTY / --all falls back to auto-resolution.\n const supportedTargets = ((): SkillTarget[] => {\n const set = new Set<SkillTarget>(skills.flatMap((s) => skillSupportedChannels(packageRoot, s)));\n // Cursor consumes no skills — its channel install (.cursor/rules shim +\n // shared .claude/data catalog) is skill-independent, so always offer it.\n set.add('cursor');\n return (['claude-code', 'codex', 'opencode', 'cursor'] as SkillTarget[]).filter((t) => set.has(t));\n })();\n const channelLabel: Record<SkillTarget, string> = {\n 'claude-code': 'Claude Code',\n codex: 'Codex',\n opencode: 'OpenCode',\n cursor: 'Cursor',\n };\n const channelDir: Record<SkillTarget, string> = {\n 'claude-code': '.claude',\n codex: '.codex',\n opencode: '.opencode',\n cursor: '.cursor',\n };\n let targets: SkillTarget[];\n if (opts.agents) {\n targets = opts.agents;\n } else if (interactive) {\n const defaults = actuallyDetected.filter((t) => supportedTargets.includes(t));\n const targetResult = await p.multiselect({\n message: 'Agent channels · 어디에 설치할까요? · space = 토글 · enter = 확인',\n options: supportedTargets.map((t) => ({\n value: t,\n label: channelLabel[t],\n hint: actuallyDetected.includes(t) ? `${channelDir[t]}/ detected` : '',\n })) as { value: SkillTarget; label: string; hint?: string }[],\n initialValues: defaults.length > 0 ? defaults : supportedTargets,\n required: true,\n });\n if (p.isCancel(targetResult)) { p.cancel('Install cancelled.'); return 130; }\n targets = targetResult as SkillTarget[];\n } else {\n // Non-interactive (CI / piped / --all): resolve from flags + detection,\n // narrowed to channels the selected skills support.\n targets = opts.all\n ? (['claude-code', 'codex', 'opencode'] as SkillTarget[])\n : minimal\n ? (actuallyDetected.length > 0 ? actuallyDetected : (['claude-code'] as SkillTarget[]))\n : detected;\n const narrowed = targets.filter((t) => supportedTargets.includes(t));\n if (narrowed.length > 0) targets = narrowed;\n }\n\n // Global scope roots everything at the home dir, so plan dirs resolve to\n // ~/.claude/skills, ~/.claude/agents, etc. Project scope uses cwd (or --dir).\n const installRoot = scope === 'global' ? homedir() : projectRoot;\n // Cursor hosts no SKILL.md tree — it's excluded from skill plans and handled\n // below via the .cursor/rules shim + shared data copies (issue #20).\n const skillChannelTargets = targets.filter(\n (t): t is SkillChannel => t !== 'cursor'\n );\n const plans = skillChannelTargets.map((t) => planForTarget(installRoot, t));\n\n p.log.message(\n pc.bold('Scope: ') +\n pc.cyan(scope) +\n pc.dim(scope === 'global' ? ' (~/.claude)' : ` (${relative(process.cwd(), projectRoot) || '.'})`)\n );\n p.log.message(\n pc.bold(`Skills (${skills.length}): `) +\n skills.map((s) => pc.cyan(s)).join(', ')\n );\n if (minimal) {\n // --skills-only: sub-agents are intentionally skipped (minimal single-skill\n // install). Clear BEFORE the summary so we never print agents we won't write.\n canonicalAgents = [];\n p.log.message(pc.bold('Agents: ') + pc.dim('skipped (--skills-only)'));\n } else if (canonicalAgents.length > 0) {\n p.log.message(\n pc.bold(`Agents (${canonicalAgents.length}): `) +\n canonicalAgents.map((a) => pc.cyan(a.replace(/\\.md$/, ''))).join(', ')\n );\n }\n p.log.message(\n pc.bold('Targets: ') + targets.map((t) => pc.cyan(t)).join(', ')\n );\n\n const results: InstallResult[] = [];\n // Count of reference-catalog DESIGN.md files copied (issue #16) — surfaced in\n // the install summary. Declared here so the outro (outside `if (!minimal)`) sees it.\n let catalogCount = 0;\n for (const plan of plans) {\n for (const skill of skills) {\n results.push(installOne(packageRoot, plan, skill, force));\n }\n }\n\n // Generate per-channel sub-agent definitions from the canonical `agents/`.\n // This is the v2 portable source-of-truth pattern (oh-my-agent style).\n // `canonicalAgents` is already resolved above by the TUI / --agents filter.\n for (const target of targets) {\n if (target === 'claude-code') {\n for (const filename of canonicalAgents) {\n results.push(installAgentFile(packageRoot, installRoot, 'claude', filename, force));\n }\n } else if (target === 'codex') {\n for (const filename of canonicalAgents) {\n results.push(installAgentFile(packageRoot, installRoot, 'codex', filename, force));\n }\n }\n // OpenCode currently has no agent-definition channel — skills only.\n }\n\n if (!minimal) {\n // Cursor channel: write the `.cursor/rules` shim (the exact content omd:sync\n // renders for .cursor/rules/omd-design.mdc) so Cursor reads DESIGN.md before\n // UI work. No skills/agents/hooks — the shim plus the shared .claude/data\n // copies below are the whole Cursor install (issue #20).\n if (targets.includes('cursor')) {\n results.push(installCursorRule(installRoot, force));\n }\n\n // Ship the read-only data assets (reference fingerprints, controlled vocab,\n // human-readable tag matrix, opt-out corpus) so skills + hooks can run entirely\n // on the host CLI's own model — no external API keys.\n const dataFiles = [\n 'reference-fingerprints.json',\n 'reference-tags.md',\n 'vocabulary.json',\n 'synonyms.json',\n 'opt-out-corpus.json',\n ];\n // Channel→dir resolution (incl. the cursor shared-`.claude/data` dedup guard,\n // issue #20) lives in dataDirFor — single source for all three copy loops.\n for (const target of targets) {\n const dataDir = dataDirFor(target, targets);\n if (!dataDir) continue;\n for (const dataFile of dataFiles) {\n results.push(installDataFile(packageRoot, installRoot, dataDir, dataFile, force, target));\n }\n }\n\n // Ship the reference catalog (DESIGN.md per id) into .claude/data/references\n // so omd:init can resolve a reference on clean npx installs — no node_modules,\n // no dev web/references (issue #16). Skipped under --skills-only (handled by the\n // enclosing `if (!minimal)`). Codex gets the same copy under .codex/data.\n // Same dataDirFor single-path rule as the data JSONs above — Cursor reads\n // .claude/data/references, never a second catalog location.\n for (const target of targets) {\n const dataDir = dataDirFor(target, targets);\n if (dataDir) {\n catalogCount += installReferenceCatalog(packageRoot, installRoot, dataDir, force);\n }\n }\n\n // Copy ctx-prime.cjs (+ its companion context.cjs) into .claude/data/scripts/\n // so /omd-harness CTX-PRIME works without the package dir on npx installs\n // (issue #18 / harness OMD_DIR resolution).\n // Note: base table (no cursor special-case) — helper scripts are\n // claude-code/codex only; a cursor-only install ships no ctx-prime.\n for (const target of targets) {\n const cd = CHANNEL_DATA_DIRS[target];\n if (!cd) continue;\n for (const helper of ['ctx-prime.cjs', 'context.cjs']) {\n const srcHelper = join(packageRoot, 'scripts', helper);\n if (!existsSync(srcHelper)) continue;\n const destHelper = join(installRoot, cd, 'data', 'scripts', helper);\n const srcTxt = readFileSync(srcHelper, 'utf8');\n if (existsSync(destHelper) && readFileSync(destHelper, 'utf8') === srcTxt) continue;\n mkdirSync(dirname(destHelper), { recursive: true });\n writeFileSync(destHelper, srcTxt, 'utf8');\n }\n }\n\n // Hooks + settings.json are PROJECT-SCOPED only — a global install must not\n // mutate the user's global Claude config / make hooks fire in every project.\n if (scope === 'project' && targets.includes('claude-code')) {\n for (const hookFile of [\n 'skill-activation.cjs',\n 'session-state-loader.cjs',\n 'post-edit-watch.cjs',\n 'session-end-foldin.cjs',\n // Shared modules required by the fold-in / state-loader / watch hooks.\n // Live in a lib/ subdir; installHookFile preserves the relative path\n // under .claude/hooks/.\n join('lib', 'preferences-parser.cjs'),\n join('lib', 'preferences-writer.cjs'),\n ]) {\n results.push(installHookFile(packageRoot, installRoot, hookFile, force));\n }\n // settings.json (with merge, never clobber user)\n results.push(installSettingsJson(packageRoot, installRoot, force));\n }\n } // !minimal — skills-only skips data files, hooks, and settings.json\n\n p.log.message(pc.bold('\\nResults:'));\n for (const r of results) {\n const rel = relative(installRoot, r.destPath);\n p.log.message(\n ` ${STATUS_LABEL[r.status]} ${pc.dim(r.target.padEnd(12))} ${rel}`\n );\n }\n\n const driftCount = results.filter((r) => r.status === 'skipped-drift').length;\n const writtenCount = results.filter(\n (r) => r.status === 'created' || r.status === 'updated'\n ).length;\n\n if (driftCount > 0) {\n p.outro(\n pc.yellow(\n `${writtenCount} written, ${driftCount} skipped (existing files lack the omd marker — rerun with --force to overwrite).`\n )\n );\n return 0;\n }\n\n // Minimal single-skill install (--skills-only): no omd onboarding, no agents/hooks.\n // Ideal for shipping a standalone skill (e.g. claude-design) to people who don't\n // want the rest of the omd toolchain.\n if (minimal) {\n for (const r of results.filter((x) => x.status === 'skipped-incompat')) {\n p.log.warn(\n `${pc.bold(r.skill)} ${pc.dim('skipped for ')}${pc.cyan(r.target)}${pc.dim(' — declares x-omd-channels (channel not supported).')}`\n );\n }\n const installed = results.filter(\n (r) => r.status === 'created' || r.status === 'updated'\n );\n if (installed.length === 0) {\n p.outro(pc.yellow('Nothing installed — no compatible skill/channel match.'));\n return 0;\n }\n p.outro(\n pc.green(\n `Done. Installed ${skills.map((s) => pc.bold(s)).join(', ')} ${scope === 'global' ? 'globally (~/.claude/skills)' : `for ${targets.join(', ')}`}.`\n ) +\n pc.dim(' → restart your agent, then use the skill (e.g. ') +\n pc.cyan('/claude-design') +\n pc.dim(').')\n );\n return 0;\n }\n\n // Friendly next-step nudge after successful install.\n // The first prompt is kept identical to the README's \"Your first 60 seconds\"\n // block so the README, the terminal, and the postinstall message all teach\n // the same activation moment. Bilingual (EN + KR) so an English reader is not\n // handed a Korean-only outro.\n const nextSteps = [\n `${pc.bold('Restart your agent, then type your first prompt:')}`,\n '',\n ` ${pc.cyan('EN')} ${pc.dim('Set up our design system — Toss-style, for a family meal-tracking app.')}`,\n ` ${pc.cyan('KR')} ${pc.dim('토스 스타일로 가족 식단 공유 앱 디자인 시스템 잡아줘')}`,\n '',\n `${pc.dim('Your agent runs omd:init and writes DESIGN.md. Then build against it:')}`,\n ` ${pc.cyan('EN')} ${pc.dim('Design the home screen.')} ${pc.cyan('KR')} ${pc.dim('홈 화면 디자인해줘')}`,\n '',\n `${pc.dim('Full walkthrough → \"Your first 60 seconds\" in the README. Routing is automatic — no slash command needed.')}`,\n `${pc.dim('Power user: ')}${pc.cyan('/omd-harness <task>')}${pc.dim(' — jump straight into the pipeline.')}`,\n '',\n `${pc.yellow('⚠ Already-running session?')} ${pc.dim('Run `/agents` to reload — or quit (Cmd+Q on macOS) and relaunch. Without reload, hooks/agents do not load.')}`,\n ].join('\\n');\n p.note(nextSteps, 'Next');\n\n // Counts derived from what was actually resolved/installed — never hardcoded,\n // so the outro can't drift from the real skill/agent/hook set (or the README).\n const hookCount = scope === 'project' && targets.includes('claude-code') ? 4 : 0;\n if (catalogCount > 0) {\n p.log.message(\n pc.bold('Reference catalog: ') +\n pc.cyan(`${catalogCount}`) +\n pc.dim(' DESIGN.md copied → .claude/data/references/<id>/DESIGN.md'),\n );\n }\n p.outro(\n pc.green(\n `Done. ${skills.length} skills · ${canonicalAgents.length} sub-agents · ${hookCount} hooks · ${catalogCount} catalog refs installed (${writtenCount} files)${scope === 'global' ? ' globally (~/.claude)' : ''}.`,\n ),\n );\n return 0;\n}\n\n","import { existsSync } from 'node:fs';\nimport { join } from 'node:path';\n\nexport type AgentId = 'claude-code' | 'codex' | 'opencode' | 'cursor' | 'unknown';\n\nexport function detectCallingAgent(): AgentId {\n const env = process.env;\n\n if (env.CLAUDECODE === '1' || env.CLAUDE_CODE === '1' || env.CLAUDE_CODE_TASK_ID) {\n return 'claude-code';\n }\n if (env.CODEX_SESSION_ID || env.CODEX || env.OPENAI_CODEX) {\n return 'codex';\n }\n if (env.OPENCODE || env.OPENCODE_SESSION) {\n return 'opencode';\n }\n if (env.CURSOR_SESSION_ID || env.CURSOR_AGENT) {\n return 'cursor';\n }\n\n return 'unknown';\n}\n\nexport interface AgentPresence {\n claudeCode: boolean;\n codex: boolean;\n opencode: boolean;\n cursor: boolean;\n}\n\nexport function detectInstalledAgents(projectRoot: string): AgentPresence {\n return {\n claudeCode:\n existsSync(join(projectRoot, '.claude')) ||\n existsSync(join(projectRoot, 'CLAUDE.md')),\n codex:\n existsSync(join(projectRoot, '.codex')) ||\n existsSync(join(projectRoot, 'AGENTS.md')) ||\n existsSync(join(projectRoot, 'AGENTS.override.md')),\n opencode:\n existsSync(join(projectRoot, '.opencode')) ||\n existsSync(join(projectRoot, 'opencode.json')) ||\n existsSync(join(projectRoot, 'opencode.jsonc')),\n cursor:\n existsSync(join(projectRoot, '.cursor')) ||\n existsSync(join(projectRoot, '.cursorrules')),\n };\n}\n"],"mappings":"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","names":["existsSync","join","existsSync","join"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "oh-my-design-cli",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.8.0",
|
|
4
4
|
"description": "Bootstrap oh-my-design skills + agents into your project. After install, talk to your AI coding agent in natural language — no other CLI commands.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"bin": {
|
|
@@ -15,6 +15,7 @@
|
|
|
15
15
|
"skills/omd-learn",
|
|
16
16
|
"skills/omd-remember",
|
|
17
17
|
"skills/omd-sync",
|
|
18
|
+
"skills/omd-taste",
|
|
18
19
|
"skills/omd-reference-capture",
|
|
19
20
|
"skills/omd-asset-fetch",
|
|
20
21
|
"skills/omd-experiment-gallery",
|
|
@@ -33,6 +34,7 @@
|
|
|
33
34
|
"data/reference-tags.md",
|
|
34
35
|
"web/references/*/DESIGN.md",
|
|
35
36
|
".claude/hooks/*.cjs",
|
|
37
|
+
".claude/hooks/lib/*.cjs",
|
|
36
38
|
".claude/settings.json",
|
|
37
39
|
"AGENTS.md",
|
|
38
40
|
"scripts/postinstall.cjs",
|
|
@@ -144,3 +144,37 @@ input에 `prior_report_path` 포함되면:
|
|
|
144
144
|
- 항목별로 RESOLVED / UNRESOLVED / NEW로 표시
|
|
145
145
|
|
|
146
146
|
Round 2에도 UNRESOLVED BLOCK 있으면 orchestrator로 BLOCK escalation.
|
|
147
|
+
|
|
148
|
+
## 8. 취향 캡처 — review → taste loop (최종 phase)
|
|
149
|
+
|
|
150
|
+
review report를 emit한 **후에** 이번 run의 finding들을 한 번 스캔해, 반복 패턴을 취향 후보로 제안한다. report 자체는 advisory 그대로 — 이 phase가 유일하게 쓰기를 일으킬 수 있는 지점이고, 그것도 **사용자가 동의한 경우에만**.
|
|
151
|
+
|
|
152
|
+
### 후보 조건 (둘 중 하나)
|
|
153
|
+
|
|
154
|
+
1. **같은 axis ≥2회** — 이번 run의 finding을 axis로 분류(radius / color / spacing / typo / voice)했을 때 같은 axis가 2회 이상 등장
|
|
155
|
+
2. **기존 pending preference와 매칭** — `.omd/preferences.md`가 존재하면 read해서, finding이 `status: pending` 엔트리의 scope와 같은 축이면 1회여도 후보 (반복의 증거가 이미 파일에 있으므로)
|
|
156
|
+
|
|
157
|
+
`.omd/preferences.md`가 없으면 조건 2는 생략 — 파일을 만들지 않는다.
|
|
158
|
+
|
|
159
|
+
### 제안 (run당 질문 1개 max)
|
|
160
|
+
|
|
161
|
+
후보가 1개 이상이면 **단 한 번** 묻는다: "이 패턴, 취향으로 기록할까요?"
|
|
162
|
+
|
|
163
|
+
- **Claude Code**: AskUserQuestion — 후보가 여러 개면 multiSelect 옵션으로 묶어서 한 질문에 (후보당 한 줄: axis + 발생 횟수 + 요약)
|
|
164
|
+
- **다른 채널 (Codex / OpenCode)**: 같은 내용을 산문 질문 하나로
|
|
165
|
+
|
|
166
|
+
### 동의 시 기록
|
|
167
|
+
|
|
168
|
+
선택된 후보는 **omd:remember 스킬의 기록 절차를 그대로 수행**해 기록한다 — 포맷을 손으로 흉내 내 직접 append하지 말 것 (writer는 omd:remember 하나; id 생성·scope 매핑·frontmatter 생성·heading 규칙 전부 그 절차를 따른다):
|
|
169
|
+
|
|
170
|
+
- `signal: review` / `confidence: inferred` / `status: pending`
|
|
171
|
+
- `source_context`: 이번 review report 경로 (예: `.reviews/designer-review-round-1.md`)
|
|
172
|
+
|
|
173
|
+
### 금지
|
|
174
|
+
|
|
175
|
+
- **자동 기록 금지** — 질문 없이 append 절대 불가 (omd:remember의 "묻지 말고 기록" 룰은 사용자 발화용 — review 추론에는 적용되지 않는다)
|
|
176
|
+
- **자동 fold 금지** — `status: pending`으로만 기록. DESIGN.md 반영은 평소의 fold-in 임계/게이트(omd:learn)가 결정
|
|
177
|
+
- run당 질문 2개 이상 금지 — 후보가 많아도 multiSelect 하나로 배칭
|
|
178
|
+
- 거절된 후보를 같은 세션에서 재제안 금지
|
|
179
|
+
|
|
180
|
+
> **수동 검증**: 같은 artifact에서 radius WARN 2건이 나오는 review를 돌리면, report 출력 후 "이 패턴, 취향으로 기록할까요?" 질문이 정확히 1회 뜨고, 동의 시 `.omd/preferences.md`에 `signal: review` / `confidence: inferred` / `status: pending` 엔트리 1개가 append되어야 한다 (DESIGN.md는 변경 없음).
|
|
@@ -151,3 +151,32 @@ round 2:
|
|
|
151
151
|
|
|
152
152
|
- `omd-designer-review` 보고서 ← `prior_reviews`로 참조 (해소 여부 확인)
|
|
153
153
|
- `omd-locale-adapter` 결과물 ← rubric [5]에서 검증
|
|
154
|
+
|
|
155
|
+
## 8. 취향 캡처 — QA → taste loop (최종 phase)
|
|
156
|
+
|
|
157
|
+
verdict를 emit한 **후에**, 이번 run의 rubric FAIL들을 한 번 스캔해 반복 위반을 취향 후보로 제안한다. read-only 원칙은 그대로 — artifact·DESIGN.md는 절대 건드리지 않으며, 이 스킬이 트리거할 수 있는 **유일한 쓰기**는 사용자가 명시적으로 동의한 뒤의 preference append이고, 그것도 `omd:remember`의 canonical 절차 그대로다.
|
|
158
|
+
|
|
159
|
+
### 후보 조건 (둘 중 하나)
|
|
160
|
+
|
|
161
|
+
1. **반복 rubric 위반** — 같은 rubric item의 FAIL이 이번 run에서 2회 이상 (여러 artifact/locale에 걸쳐, 또는 round 1·2 연속 동일 item). item을 axis로 환원: [1] Brand consistency → 위반 토큰 종류에 따라 color/spacing/radius, [2] Typography hierarchy → typo, [3] Voice register → voice
|
|
162
|
+
2. **기존 pending preference와 매칭** — `.omd/preferences.md`가 존재하면 read해서, FAIL이 `status: pending` 엔트리의 scope와 같은 축이면 1회여도 후보
|
|
163
|
+
|
|
164
|
+
`.omd/preferences.md`가 없으면 조건 2는 생략 — 파일을 만들지 않는다. a11y/performance/links([6]-[8])는 취향이 아니라 hard rule — 후보에서 제외.
|
|
165
|
+
|
|
166
|
+
### 제안 (run당 질문 1개 max)
|
|
167
|
+
|
|
168
|
+
후보가 1개 이상이면 **단 한 번** 묻는다: "이 패턴, 취향으로 기록할까요?"
|
|
169
|
+
|
|
170
|
+
- **Claude Code**: AskUserQuestion — 후보 여러 개는 multiSelect 옵션으로 배칭 (후보당 한 줄: axis + FAIL 횟수 + 요약)
|
|
171
|
+
- **다른 채널**: 같은 내용을 산문 질문 하나로
|
|
172
|
+
|
|
173
|
+
동의된 후보는 **omd:remember 스킬의 기록 절차를 그대로 수행**해 기록 (직접 포맷 모방 금지 — writer는 omd:remember 하나) — `signal: review` / `confidence: inferred` / `status: pending`, `source_context`는 이번 QA report 경로 (예: `.reviews/final-qa-round-1.md`).
|
|
174
|
+
|
|
175
|
+
### 금지
|
|
176
|
+
|
|
177
|
+
- 질문 없는 자동 기록 절대 불가 — review 추론은 사용자 동의가 전제
|
|
178
|
+
- **자동 fold 금지** — `status: pending` 기록까지만. DESIGN.md 반영은 omd:learn의 평소 임계/게이트가 결정
|
|
179
|
+
- run당 질문 2개 이상 금지, 거절된 후보의 같은 세션 재제안 금지
|
|
180
|
+
- 이 phase가 verdict나 round cap에 영향을 주면 안 됨 (rubric 8 items 고정 유지)
|
|
181
|
+
|
|
182
|
+
> **수동 검증**: KR/EN 두 artifact에서 rubric [3] Voice register가 모두 FAIL인 QA를 돌리면, verdict 출력 후 "이 패턴, 취향으로 기록할까요?" 질문이 정확히 1회 뜨고, 동의 시 `.omd/preferences.md`에 `scope: voice` / `signal: review` / `confidence: inferred` / `status: pending` 엔트리가 append되어야 한다 (artifact·DESIGN.md는 변경 없음).
|