mythik-cli 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (128) hide show
  1. package/LICENSE +201 -0
  2. package/NOTICE +4 -0
  3. package/README.md +71 -0
  4. package/dist/api.d.ts +22 -0
  5. package/dist/api.d.ts.map +1 -0
  6. package/dist/api.js +17 -0
  7. package/dist/api.js.map +1 -0
  8. package/dist/cli.d.ts +3 -0
  9. package/dist/cli.d.ts.map +1 -0
  10. package/dist/cli.js +519 -0
  11. package/dist/cli.js.map +1 -0
  12. package/dist/commands/contract.d.ts +12 -0
  13. package/dist/commands/contract.d.ts.map +1 -0
  14. package/dist/commands/contract.js +133 -0
  15. package/dist/commands/contract.js.map +1 -0
  16. package/dist/commands/delete.d.ts +9 -0
  17. package/dist/commands/delete.d.ts.map +1 -0
  18. package/dist/commands/delete.js +46 -0
  19. package/dist/commands/delete.js.map +1 -0
  20. package/dist/commands/diff.d.ts +11 -0
  21. package/dist/commands/diff.d.ts.map +1 -0
  22. package/dist/commands/diff.js +49 -0
  23. package/dist/commands/diff.js.map +1 -0
  24. package/dist/commands/docs.d.ts +15 -0
  25. package/dist/commands/docs.d.ts.map +1 -0
  26. package/dist/commands/docs.js +117 -0
  27. package/dist/commands/docs.js.map +1 -0
  28. package/dist/commands/elements.d.ts +9 -0
  29. package/dist/commands/elements.d.ts.map +1 -0
  30. package/dist/commands/elements.js +48 -0
  31. package/dist/commands/elements.js.map +1 -0
  32. package/dist/commands/envs.d.ts +11 -0
  33. package/dist/commands/envs.d.ts.map +1 -0
  34. package/dist/commands/envs.js +54 -0
  35. package/dist/commands/envs.js.map +1 -0
  36. package/dist/commands/history.d.ts +10 -0
  37. package/dist/commands/history.d.ts.map +1 -0
  38. package/dist/commands/history.js +67 -0
  39. package/dist/commands/history.js.map +1 -0
  40. package/dist/commands/init.d.ts +9 -0
  41. package/dist/commands/init.d.ts.map +1 -0
  42. package/dist/commands/init.js +120 -0
  43. package/dist/commands/init.js.map +1 -0
  44. package/dist/commands/lint.d.ts +9 -0
  45. package/dist/commands/lint.d.ts.map +1 -0
  46. package/dist/commands/lint.js +15 -0
  47. package/dist/commands/lint.js.map +1 -0
  48. package/dist/commands/manifest.d.ts +11 -0
  49. package/dist/commands/manifest.d.ts.map +1 -0
  50. package/dist/commands/manifest.js +48 -0
  51. package/dist/commands/manifest.js.map +1 -0
  52. package/dist/commands/patch.d.ts +12 -0
  53. package/dist/commands/patch.d.ts.map +1 -0
  54. package/dist/commands/patch.js +103 -0
  55. package/dist/commands/patch.js.map +1 -0
  56. package/dist/commands/promote.d.ts +15 -0
  57. package/dist/commands/promote.d.ts.map +1 -0
  58. package/dist/commands/promote.js +83 -0
  59. package/dist/commands/promote.js.map +1 -0
  60. package/dist/commands/pull.d.ts +9 -0
  61. package/dist/commands/pull.d.ts.map +1 -0
  62. package/dist/commands/pull.js +36 -0
  63. package/dist/commands/pull.js.map +1 -0
  64. package/dist/commands/push-bulk.d.ts +11 -0
  65. package/dist/commands/push-bulk.d.ts.map +1 -0
  66. package/dist/commands/push-bulk.js +112 -0
  67. package/dist/commands/push-bulk.js.map +1 -0
  68. package/dist/commands/push.d.ts +19 -0
  69. package/dist/commands/push.d.ts.map +1 -0
  70. package/dist/commands/push.js +102 -0
  71. package/dist/commands/push.js.map +1 -0
  72. package/dist/commands/rollback.d.ts +13 -0
  73. package/dist/commands/rollback.d.ts.map +1 -0
  74. package/dist/commands/rollback.js +78 -0
  75. package/dist/commands/rollback.js.map +1 -0
  76. package/dist/commands/tokens.d.ts +8 -0
  77. package/dist/commands/tokens.d.ts.map +1 -0
  78. package/dist/commands/tokens.js +63 -0
  79. package/dist/commands/tokens.js.map +1 -0
  80. package/dist/commands/validate.d.ts +8 -0
  81. package/dist/commands/validate.d.ts.map +1 -0
  82. package/dist/commands/validate.js +60 -0
  83. package/dist/commands/validate.js.map +1 -0
  84. package/dist/config.d.ts +27 -0
  85. package/dist/config.d.ts.map +1 -0
  86. package/dist/config.js +112 -0
  87. package/dist/config.js.map +1 -0
  88. package/dist/input-resolver.d.ts +22 -0
  89. package/dist/input-resolver.d.ts.map +1 -0
  90. package/dist/input-resolver.js +55 -0
  91. package/dist/input-resolver.js.map +1 -0
  92. package/dist/levenshtein.d.ts +2 -0
  93. package/dist/levenshtein.d.ts.map +1 -0
  94. package/dist/levenshtein.js +2 -0
  95. package/dist/levenshtein.js.map +1 -0
  96. package/dist/lint/code-rules.d.ts +30 -0
  97. package/dist/lint/code-rules.d.ts.map +1 -0
  98. package/dist/lint/code-rules.js +81 -0
  99. package/dist/lint/code-rules.js.map +1 -0
  100. package/dist/lint/format.d.ts +12 -0
  101. package/dist/lint/format.d.ts.map +1 -0
  102. package/dist/lint/format.js +68 -0
  103. package/dist/lint/format.js.map +1 -0
  104. package/dist/lint/orchestrator.d.ts +17 -0
  105. package/dist/lint/orchestrator.d.ts.map +1 -0
  106. package/dist/lint/orchestrator.js +124 -0
  107. package/dist/lint/orchestrator.js.map +1 -0
  108. package/dist/lint/spec-discovery.d.ts +14 -0
  109. package/dist/lint/spec-discovery.d.ts.map +1 -0
  110. package/dist/lint/spec-discovery.js +72 -0
  111. package/dist/lint/spec-discovery.js.map +1 -0
  112. package/dist/lint/types.d.ts +72 -0
  113. package/dist/lint/types.d.ts.map +1 -0
  114. package/dist/lint/types.js +28 -0
  115. package/dist/lint/types.js.map +1 -0
  116. package/dist/output.d.ts +16 -0
  117. package/dist/output.d.ts.map +1 -0
  118. package/dist/output.js +91 -0
  119. package/dist/output.js.map +1 -0
  120. package/dist/stores/resolver.d.ts +12 -0
  121. package/dist/stores/resolver.d.ts.map +1 -0
  122. package/dist/stores/resolver.js +54 -0
  123. package/dist/stores/resolver.js.map +1 -0
  124. package/dist/toon.d.ts +12 -0
  125. package/dist/toon.d.ts.map +1 -0
  126. package/dist/toon.js +27 -0
  127. package/dist/toon.js.map +1 -0
  128. package/package.json +65 -0
@@ -0,0 +1,67 @@
1
+ import { computeStructuralDiff } from 'mythik';
2
+ import { formatSuccess, formatError, formatJson } from '../output.js';
3
+ export async function runHistory(specId, options) {
4
+ try {
5
+ const entries = await options.store.listVersions(specId, options.limit);
6
+ if (entries.length === 0) {
7
+ return {
8
+ output: options.json
9
+ ? formatJson({ error: `No version history for "${specId}"` })
10
+ : formatError({ what: 'No version history', why: `Spec "${specId}" has no versions` }),
11
+ exitCode: 1,
12
+ };
13
+ }
14
+ if (options.json) {
15
+ return { output: formatJson(entries), exitCode: 0 };
16
+ }
17
+ // Build environment map: version → env names
18
+ const envs = await options.envStore.getEnvironments(specId);
19
+ const envByVersion = new Map();
20
+ for (const env of envs) {
21
+ if (!envByVersion.has(env.version))
22
+ envByVersion.set(env.version, []);
23
+ envByVersion.get(env.version).push(env.name.toUpperCase());
24
+ }
25
+ const total = await options.store.currentVersion(specId);
26
+ const lines = [];
27
+ lines.push(formatSuccess(`History for "${specId}" (${total} versions)`));
28
+ lines.push('');
29
+ for (const entry of entries) {
30
+ const envTags = envByVersion.get(entry.version)?.map(e => `[${e}]`).join(' ') ?? '';
31
+ const desc = entry.description ? `"${entry.description}"` : '';
32
+ lines.push(` v${entry.version} ${entry.timestamp.slice(0, 10)} ${entry.author.padEnd(10)} ${entry.source.padEnd(8)} ${desc} ${envTags}`);
33
+ // Show change details for non-first versions
34
+ if (entry.version > 1) {
35
+ try {
36
+ const prevSpec = await options.store.loadVersion(specId, entry.version - 1);
37
+ const currSpec = await options.store.loadVersion(specId, entry.version);
38
+ const diff = computeStructuralDiff(prevSpec, currSpec, entry.version - 1, entry.version);
39
+ if (diff.changes.length > 0) {
40
+ for (const change of diff.changes) {
41
+ const icon = change.kind.includes('added') ? '+' : change.kind.includes('removed') ? '-' : '~';
42
+ lines.push(` ${icon} ${change.detail}`);
43
+ }
44
+ }
45
+ else {
46
+ lines.push(' (no diff)');
47
+ }
48
+ }
49
+ catch {
50
+ // Can't compute diff — skip
51
+ }
52
+ }
53
+ else {
54
+ lines.push(' snapshot');
55
+ }
56
+ }
57
+ return { output: lines.join('\n'), exitCode: 0 };
58
+ }
59
+ catch (err) {
60
+ const message = err.message;
61
+ return {
62
+ output: options.json ? formatJson({ error: message }) : formatError({ what: 'History failed', why: message }),
63
+ exitCode: 1,
64
+ };
65
+ }
66
+ }
67
+ //# sourceMappingURL=history.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"history.js","sourceRoot":"","sources":["../../src/commands/history.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { CommandResult } from './manifest.js';
2
+ export interface InitFlags {
3
+ store?: string;
4
+ url?: string;
5
+ key?: string;
6
+ dir?: string;
7
+ }
8
+ export declare function runInit(flags: InitFlags, cwd: string): Promise<CommandResult>;
9
+ //# sourceMappingURL=init.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../../src/commands/init.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAGnD,MAAM,WAAW,SAAS;IACxB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,GAAG,CAAC,EAAE,MAAM,CAAC;CACd;AAED,wBAAsB,OAAO,CAAC,KAAK,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,aAAa,CAAC,CAqFnF"}
@@ -0,0 +1,120 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import { formatSuccess, formatError } from '../output.js';
4
+ export async function runInit(flags, cwd) {
5
+ const rcPath = path.join(cwd, '.mythikrc');
6
+ if (fs.existsSync(rcPath)) {
7
+ return {
8
+ output: formatError({
9
+ what: '.mythikrc already exists',
10
+ why: `Found at ${rcPath}`,
11
+ fix: 'Delete it first if you want to reconfigure',
12
+ }),
13
+ exitCode: 1,
14
+ };
15
+ }
16
+ let config;
17
+ if (flags.store) {
18
+ // Non-interactive mode
19
+ config = buildConfig(flags);
20
+ }
21
+ else if (process.stdin.isTTY) {
22
+ // Interactive mode
23
+ config = await interactiveInit();
24
+ }
25
+ else {
26
+ return {
27
+ output: formatError({
28
+ what: 'Cannot run init interactively',
29
+ why: 'No TTY detected and no --store flag provided',
30
+ fix: 'Pass --store, --url, --key flags for non-interactive init',
31
+ }),
32
+ exitCode: 1,
33
+ };
34
+ }
35
+ // Validate connection for supabase
36
+ if (config.store === 'supabase') {
37
+ const sub = config.supabase;
38
+ try {
39
+ const res = await fetch(`${sub.url}/rest/v1/screens?select=id&limit=100`, {
40
+ headers: { apikey: sub.apiKey, Authorization: `Bearer ${sub.apiKey}` },
41
+ });
42
+ if (!res.ok) {
43
+ return {
44
+ output: formatError({
45
+ what: 'Connection verification failed',
46
+ why: `${res.status} ${res.statusText}`,
47
+ fix: 'Check your Supabase URL and API key',
48
+ }),
49
+ exitCode: 2,
50
+ };
51
+ }
52
+ const rows = await res.json();
53
+ const screenCount = rows.length;
54
+ process.stdout.write(formatSuccess(`Connection verified — ${screenCount} screen${screenCount !== 1 ? 's' : ''} found`) + '\n');
55
+ }
56
+ catch (err) {
57
+ return {
58
+ output: formatError({
59
+ what: 'Connection verification failed',
60
+ why: err.message,
61
+ fix: 'Check your Supabase URL is correct and reachable',
62
+ }),
63
+ exitCode: 2,
64
+ };
65
+ }
66
+ }
67
+ // Write config (with $VAR placeholder for apiKey)
68
+ const writeConfig = structuredClone(config);
69
+ if (writeConfig.store === 'supabase') {
70
+ writeConfig.supabase.apiKey = '$MYTHIK_API_KEY';
71
+ }
72
+ fs.writeFileSync(rcPath, JSON.stringify(writeConfig, null, 2) + '\n');
73
+ process.stdout.write(formatSuccess(`.mythikrc created at ${rcPath}`) + '\n');
74
+ // Add to .gitignore
75
+ const gitignorePath = path.join(cwd, '.gitignore');
76
+ if (fs.existsSync(gitignorePath)) {
77
+ const content = fs.readFileSync(gitignorePath, 'utf-8');
78
+ if (!content.includes('.mythikrc')) {
79
+ fs.appendFileSync(gitignorePath, '\n.mythikrc\n');
80
+ process.stdout.write(formatSuccess('.mythikrc added to .gitignore') + '\n');
81
+ }
82
+ }
83
+ return { output: '', exitCode: 0 };
84
+ }
85
+ function buildConfig(flags) {
86
+ const config = { store: flags.store };
87
+ if (flags.store === 'supabase') {
88
+ if (!flags.url || !flags.key) {
89
+ throw new Error('Supabase store requires --url and --key flags');
90
+ }
91
+ config.supabase = { url: flags.url, apiKey: flags.key };
92
+ }
93
+ else if (flags.store === 'file') {
94
+ config.file = { dir: flags.dir ?? './specs' };
95
+ }
96
+ return config;
97
+ }
98
+ async function interactiveInit() {
99
+ const { select, input, password } = await import('@inquirer/prompts');
100
+ const store = await select({
101
+ message: 'Select store type:',
102
+ choices: [
103
+ { value: 'supabase', name: 'Supabase' },
104
+ { value: 'file', name: 'File (local JSON)' },
105
+ { value: 'memory', name: 'Memory (testing)' },
106
+ ],
107
+ });
108
+ const config = { store };
109
+ if (store === 'supabase') {
110
+ const url = await input({ message: 'Supabase URL:' });
111
+ const apiKey = await password({ message: 'API Key (service_role):' });
112
+ config.supabase = { url, apiKey };
113
+ }
114
+ else if (store === 'file') {
115
+ const dir = await input({ message: 'Specs directory:', default: './specs' });
116
+ config.file = { dir };
117
+ }
118
+ return config;
119
+ }
120
+ //# sourceMappingURL=init.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.js","sourceRoot":"","sources":["../../src/commands/init.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ /**
2
+ * CLI wrapper for `mythik lint` — translates commander options to runLint
3
+ * and renders the result via human/JSON formatters.
4
+ */
5
+ import type { LintOptions } from '../lint/types.js';
6
+ import type { CommandResult } from './manifest.js';
7
+ export type RunLintCommandOptions = LintOptions;
8
+ export declare function runLintCommand(opts: RunLintCommandOptions): Promise<CommandResult>;
9
+ //# sourceMappingURL=lint.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lint.d.ts","sourceRoot":"","sources":["../../src/commands/lint.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAIH,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,kBAAkB,CAAC;AACpD,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAEnD,MAAM,MAAM,qBAAqB,GAAG,WAAW,CAAC;AAEhD,wBAAsB,cAAc,CAAC,IAAI,EAAE,qBAAqB,GAAG,OAAO,CAAC,aAAa,CAAC,CAOxF"}
@@ -0,0 +1,15 @@
1
+ /**
2
+ * CLI wrapper for `mythik lint` — translates commander options to runLint
3
+ * and renders the result via human/JSON formatters.
4
+ */
5
+ import { runLint } from '../lint/orchestrator.js';
6
+ import { formatHuman, formatJson, computeExitCode } from '../lint/format.js';
7
+ export async function runLintCommand(opts) {
8
+ const result = await runLint(opts);
9
+ const output = opts.json ? formatJson(result) : formatHuman(result);
10
+ return {
11
+ output: output + '\n',
12
+ exitCode: computeExitCode(result),
13
+ };
14
+ }
15
+ //# sourceMappingURL=lint.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lint.js","sourceRoot":"","sources":["../../src/commands/lint.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,yBAAyB,CAAC;AAClD,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAM7E,MAAM,CAAC,KAAK,UAAU,cAAc,CAAC,IAA2B;IAC9D,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IACnC,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;IACpE,OAAO;QACL,MAAM,EAAE,MAAM,GAAG,IAAI;QACrB,QAAQ,EAAE,eAAe,CAAC,MAAM,CAAC;KAClC,CAAC;AACJ,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { SpecStore } from 'mythik';
2
+ export interface ManifestOptions {
3
+ store: SpecStore;
4
+ json: boolean;
5
+ }
6
+ export interface CommandResult {
7
+ output: string;
8
+ exitCode: number;
9
+ }
10
+ export declare function runManifest(screenId: string, options: ManifestOptions): Promise<CommandResult>;
11
+ //# sourceMappingURL=manifest.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"manifest.d.ts","sourceRoot":"","sources":["../../src/commands/manifest.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AAKxC,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,SAAS,CAAC;IACjB,IAAI,EAAE,OAAO,CAAC;CACf;AAED,MAAM,WAAW,aAAa;IAC5B,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,wBAAsB,WAAW,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,aAAa,CAAC,CA8CpG"}
@@ -0,0 +1,48 @@
1
+ import { createSpecEngine } from 'mythik';
2
+ import { formatSuccess, formatError, formatJson, colorizeManifest } from '../output.js';
3
+ import { suggest } from '../levenshtein.js';
4
+ export async function runManifest(screenId, options) {
5
+ const engine = createSpecEngine({ store: options.store });
6
+ try {
7
+ const manifest = await engine.getManifest(screenId);
8
+ const elementCount = parseInt(manifest.match(/\((\d+) elements\)/)?.[1] ?? '0', 10);
9
+ if (options.json) {
10
+ return {
11
+ output: formatJson({ screenId, elementCount, manifest }),
12
+ exitCode: 0,
13
+ };
14
+ }
15
+ return {
16
+ output: `${formatSuccess(`${screenId} (${elementCount} elements)`)}\n\n${colorizeManifest(manifest)}`,
17
+ exitCode: 0,
18
+ };
19
+ }
20
+ catch (err) {
21
+ const message = err.message;
22
+ let availableScreens = [];
23
+ try {
24
+ availableScreens = await options.store.list();
25
+ }
26
+ catch { /* ignore */ }
27
+ const suggestion = suggest(screenId, availableScreens);
28
+ const why = availableScreens.length > 0
29
+ ? `Available screens: ${availableScreens.join(', ')}`
30
+ : message;
31
+ const fix = suggestion ? `Did you mean: ${suggestion}?` : 'Check the screen ID and try again';
32
+ if (options.json) {
33
+ return {
34
+ output: formatJson({ error: message, availableScreens, suggestion }),
35
+ exitCode: 1,
36
+ };
37
+ }
38
+ return {
39
+ output: formatError({
40
+ what: `Screen "${screenId}" not found`,
41
+ why,
42
+ fix,
43
+ }),
44
+ exitCode: 1,
45
+ };
46
+ }
47
+ }
48
+ //# sourceMappingURL=manifest.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"manifest.js","sourceRoot":"","sources":["../../src/commands/manifest.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,gBAAgB,EAAE,MAAM,QAAQ,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,WAAW,EAAE,UAAU,EAAE,gBAAgB,EAAE,MAAM,cAAc,CAAC;AACxF,OAAO,EAAE,OAAO,EAAE,MAAM,mBAAmB,CAAC;AAY5C,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,QAAgB,EAAE,OAAwB;IAC1E,MAAM,MAAM,GAAG,gBAAgB,CAAC,EAAE,KAAK,EAAE,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC;IAE1D,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QACpD,MAAM,YAAY,GAAG,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,oBAAoB,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,GAAG,EAAE,EAAE,CAAC,CAAC;QAEpF,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO;gBACL,MAAM,EAAE,UAAU,CAAC,EAAE,QAAQ,EAAE,YAAY,EAAE,QAAQ,EAAE,CAAC;gBACxD,QAAQ,EAAE,CAAC;aACZ,CAAC;QACJ,CAAC;QAED,OAAO;YACL,MAAM,EAAE,GAAG,aAAa,CAAC,GAAG,QAAQ,KAAK,YAAY,YAAY,CAAC,OAAO,gBAAgB,CAAC,QAAQ,CAAC,EAAE;YACrG,QAAQ,EAAE,CAAC;SACZ,CAAC;IACJ,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,MAAM,OAAO,GAAI,GAAa,CAAC,OAAO,CAAC;QAEvC,IAAI,gBAAgB,GAAa,EAAE,CAAC;QACpC,IAAI,CAAC;YAAC,gBAAgB,GAAG,MAAM,OAAO,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;QAC7E,MAAM,UAAU,GAAG,OAAO,CAAC,QAAQ,EAAE,gBAAgB,CAAC,CAAC;QAEvD,MAAM,GAAG,GAAG,gBAAgB,CAAC,MAAM,GAAG,CAAC;YACrC,CAAC,CAAC,sBAAsB,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YACrD,CAAC,CAAC,OAAO,CAAC;QACZ,MAAM,GAAG,GAAG,UAAU,CAAC,CAAC,CAAC,iBAAiB,UAAU,GAAG,CAAC,CAAC,CAAC,mCAAmC,CAAC;QAE9F,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO;gBACL,MAAM,EAAE,UAAU,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,gBAAgB,EAAE,UAAU,EAAE,CAAC;gBACpE,QAAQ,EAAE,CAAC;aACZ,CAAC;QACJ,CAAC;QAED,OAAO;YACL,MAAM,EAAE,WAAW,CAAC;gBAClB,IAAI,EAAE,WAAW,QAAQ,aAAa;gBACtC,GAAG;gBACH,GAAG;aACJ,CAAC;YACF,QAAQ,EAAE,CAAC;SACZ,CAAC;IACJ,CAAC;AACH,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { SpecStore, JsonPatch } from 'mythik';
2
+ import type { CommandResult } from './manifest.js';
3
+ export interface PatchOptions {
4
+ store: SpecStore;
5
+ json: boolean;
6
+ toon?: boolean;
7
+ author?: string;
8
+ description?: string;
9
+ }
10
+ export declare function parsePatchInput(input: string): JsonPatch[];
11
+ export declare function runPatch(screenId: string, patches: JsonPatch[], options: PatchOptions): Promise<CommandResult>;
12
+ //# sourceMappingURL=patch.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"patch.d.ts","sourceRoot":"","sources":["../../src/commands/patch.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,SAAS,EAAsB,MAAM,QAAQ,CAAC;AAIvE,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAEnD,MAAM,WAAW,YAAY;IAC3B,KAAK,EAAE,SAAS,CAAC;IACjB,IAAI,EAAE,OAAO,CAAC;IACd,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAOD,wBAAgB,eAAe,CAAC,KAAK,EAAE,MAAM,GAAG,SAAS,EAAE,CAa1D;AAED,wBAAsB,QAAQ,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,SAAS,EAAE,EAAE,OAAO,EAAE,YAAY,GAAG,OAAO,CAAC,aAAa,CAAC,CA+FpH"}
@@ -0,0 +1,103 @@
1
+ import { createSpecEngine } from 'mythik';
2
+ import { formatSuccess, formatError, formatJson, colorizeManifest, formatSuggestedFixes } from '../output.js';
3
+ import { toToon, autoparse } from '../toon.js';
4
+ export function parsePatchInput(input) {
5
+ let parsed;
6
+ try {
7
+ parsed = autoparse(input);
8
+ }
9
+ catch {
10
+ throw new Error(`Invalid patch input: ${input.slice(0, 100)}...`);
11
+ }
12
+ if (!Array.isArray(parsed)) {
13
+ throw new Error('Patches must be an array of RFC 6902 operations');
14
+ }
15
+ return parsed;
16
+ }
17
+ export async function runPatch(screenId, patches, options) {
18
+ const isVersioned = 'saveVersion' in options.store && typeof options.store.saveVersion === 'function';
19
+ const shouldSaveVersion = isVersioned && Boolean(options.author);
20
+ let patchedDocument;
21
+ const patchStore = shouldSaveVersion
22
+ ? {
23
+ load: (id) => options.store.load(id),
24
+ save: async (_id, doc) => { patchedDocument = doc; },
25
+ list: () => options.store.list(),
26
+ delete: (id) => options.store.delete(id),
27
+ }
28
+ : options.store;
29
+ const engine = createSpecEngine({ store: patchStore });
30
+ try {
31
+ const result = await engine.patch(screenId, patches);
32
+ const versionMeta = { versioned: false };
33
+ if (result.success && shouldSaveVersion) {
34
+ if (patchedDocument === undefined) {
35
+ throw new Error('Patch succeeded but produced no document to version');
36
+ }
37
+ const version = await options.store.saveVersion(screenId, patchedDocument, {
38
+ author: options.author,
39
+ source: 'patch',
40
+ description: options.description,
41
+ });
42
+ versionMeta.versioned = true;
43
+ versionMeta.version = version;
44
+ }
45
+ const outputResult = result.success ? { ...result, ...versionMeta } : result;
46
+ if (options.toon) {
47
+ return { output: toToon(outputResult), exitCode: result.success ? 0 : 1 };
48
+ }
49
+ if (options.json) {
50
+ return { output: formatJson(outputResult), exitCode: result.success ? 0 : 1 };
51
+ }
52
+ if (result.success) {
53
+ const patchCount = patches.length;
54
+ const elemCount = result.patchedElements.length;
55
+ const elemList = result.patchedElements.join(', ');
56
+ const secCount = result.patchedSections?.length ?? 0;
57
+ const secList = result.patchedSections?.join(', ') ?? '';
58
+ const parts = [];
59
+ if (elemCount > 0)
60
+ parts.push(`${elemCount} element${elemCount !== 1 ? 's' : ''} modified (${elemList})`);
61
+ if (secCount > 0)
62
+ parts.push(`${secCount} section${secCount !== 1 ? 's' : ''} modified (${secList})`);
63
+ const modifiedSummary = parts.length > 0 ? parts.join(', ') : 'no elements modified';
64
+ const lines = [
65
+ formatSuccess('Patch applied successfully'),
66
+ '',
67
+ ` ${patchCount} patch${patchCount !== 1 ? 'es' : ''} → ${modifiedSummary}`,
68
+ ` ${result.elementCount} total elements`,
69
+ '',
70
+ 'Updated manifest:',
71
+ colorizeManifest(result.manifest),
72
+ ];
73
+ return { output: lines.join('\n'), exitCode: 0 };
74
+ }
75
+ const lines = [
76
+ formatError({
77
+ what: `Patch failed — ${result.errors.length} error${result.errors.length !== 1 ? 's' : ''}`,
78
+ why: formatSuggestedFixes(result.errors, screenId),
79
+ }),
80
+ '',
81
+ ' Nothing was persisted. Original spec is untouched.',
82
+ '',
83
+ 'Resulting manifest (invalid):',
84
+ colorizeManifest(result.manifest),
85
+ ];
86
+ return { output: lines.join('\n'), exitCode: 1 };
87
+ }
88
+ catch (err) {
89
+ const message = err.message;
90
+ if (options.json) {
91
+ return { output: formatJson({ error: message }), exitCode: 1 };
92
+ }
93
+ return {
94
+ output: formatError({
95
+ what: `Screen "${screenId}" not found`,
96
+ why: message,
97
+ fix: 'Check the screen ID and try again',
98
+ }),
99
+ exitCode: 1,
100
+ };
101
+ }
102
+ }
103
+ //# sourceMappingURL=patch.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"patch.js","sourceRoot":"","sources":["../../src/commands/patch.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ import type { VersionedSpecStore, EnvironmentStore } from 'mythik';
2
+ import type { CommandResult } from './manifest.js';
3
+ export interface PromoteCommandOptions {
4
+ store: VersionedSpecStore;
5
+ envStore: EnvironmentStore;
6
+ specIds: string[];
7
+ fromEnv: string;
8
+ toEnv: string;
9
+ apiIds?: string[];
10
+ confirm: boolean;
11
+ json: boolean;
12
+ author?: string;
13
+ }
14
+ export declare function runPromoteCommand(options: PromoteCommandOptions): Promise<CommandResult>;
15
+ //# sourceMappingURL=promote.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"promote.d.ts","sourceRoot":"","sources":["../../src/commands/promote.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAE,gBAAgB,EAAE,MAAM,QAAQ,CAAC;AAGnE,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAEnD,MAAM,WAAW,qBAAqB;IACpC,KAAK,EAAE,kBAAkB,CAAC;IAC1B,QAAQ,EAAE,gBAAgB,CAAC;IAC3B,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,EAAE,CAAC;IAClB,OAAO,EAAE,OAAO,CAAC;IACjB,IAAI,EAAE,OAAO,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAED,wBAAsB,iBAAiB,CAAC,OAAO,EAAE,qBAAqB,GAAG,OAAO,CAAC,aAAa,CAAC,CAuF9F"}
@@ -0,0 +1,83 @@
1
+ import { runPromoteGate } from 'mythik';
2
+ import { formatSuccess, formatError, formatJson } from '../output.js';
3
+ export async function runPromoteCommand(options) {
4
+ try {
5
+ const result = await runPromoteGate({
6
+ specIds: options.specIds,
7
+ fromEnv: options.fromEnv,
8
+ toEnv: options.toEnv,
9
+ store: options.store,
10
+ envStore: options.envStore,
11
+ apiIds: options.apiIds,
12
+ });
13
+ if (!result.success) {
14
+ if (options.json) {
15
+ return { output: formatJson({ ...result, promoted: false }), exitCode: 1 };
16
+ }
17
+ const lines = [];
18
+ lines.push(formatError({ what: 'Promotion blocked', why: `${options.fromEnv} → ${options.toEnv}` }));
19
+ lines.push('');
20
+ for (const e of result.validation.errors) {
21
+ lines.push(` \u2717 ${e.message}`);
22
+ }
23
+ for (const w of result.validation.warnings) {
24
+ lines.push(` \u26A0 ${w.message}`);
25
+ }
26
+ return { output: lines.join('\n'), exitCode: 1 };
27
+ }
28
+ if (!options.confirm) {
29
+ if (options.json) {
30
+ return { output: formatJson({ ...result, preview: true }), exitCode: 0 };
31
+ }
32
+ const lines = [];
33
+ lines.push(`Promote: ${options.fromEnv} → ${options.toEnv} (${options.specIds.length} specs)`);
34
+ lines.push('');
35
+ for (const spec of result.specs) {
36
+ const diffSummary = spec.diff.summary === 'No changes'
37
+ ? '(no changes)'
38
+ : spec.diff.summary;
39
+ lines.push(` ${spec.id}: v${spec.fromVersion || 'new'} → v${spec.toVersion}`);
40
+ lines.push(` ${diffSummary}`);
41
+ }
42
+ lines.push('');
43
+ lines.push(' Validation:');
44
+ lines.push(` \u2713 Spec validation: ${result.validation.specValid ? 'passed' : 'failed'}`);
45
+ lines.push(` ${result.validation.crossScreenValid ? '\u2713' : '\u26A0'} Cross-screen: ${result.validation.crossScreenValid ? 'all references resolved' : 'warnings (see above)'}`);
46
+ if (result.validation.contractSkipped) {
47
+ lines.push(` \u26A0 Contract validation: skipped (no api-specs)`);
48
+ }
49
+ else {
50
+ lines.push(` ${result.validation.contractValid ? '\u2713' : '\u2717'} Contract: ${result.validation.contractValid ? 'passed' : 'failed'}`);
51
+ }
52
+ if (result.validation.warnings.length > 0) {
53
+ lines.push('');
54
+ for (const w of result.validation.warnings) {
55
+ lines.push(` \u26A0 ${w.message}`);
56
+ }
57
+ }
58
+ lines.push('');
59
+ lines.push(' Preview only. Run with --confirm to execute.');
60
+ return { output: lines.join('\n'), exitCode: 0 };
61
+ }
62
+ // Execute: move environment pointers
63
+ const author = options.author ?? 'system';
64
+ for (const spec of result.specs) {
65
+ await options.envStore.setEnvironment(spec.id, options.toEnv, spec.toVersion, author);
66
+ }
67
+ if (options.json) {
68
+ return { output: formatJson({ ...result, promoted: true }), exitCode: 0 };
69
+ }
70
+ return {
71
+ output: formatSuccess(`Promoted ${options.specIds.length} spec(s) ${options.fromEnv} → ${options.toEnv}`),
72
+ exitCode: 0,
73
+ };
74
+ }
75
+ catch (err) {
76
+ const message = err.message;
77
+ return {
78
+ output: options.json ? formatJson({ error: message }) : formatError({ what: 'Promote failed', why: message }),
79
+ exitCode: 1,
80
+ };
81
+ }
82
+ }
83
+ //# sourceMappingURL=promote.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"promote.js","sourceRoot":"","sources":["../../src/commands/promote.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { SpecStore } from 'mythik';
2
+ import type { CommandResult } from './manifest.js';
3
+ export interface PullOptions {
4
+ store: SpecStore;
5
+ json: boolean;
6
+ toon: boolean;
7
+ }
8
+ export declare function runPull(screenId: string, options: PullOptions): Promise<CommandResult>;
9
+ //# sourceMappingURL=pull.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pull.d.ts","sourceRoot":"","sources":["../../src/commands/pull.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AAIxC,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAEnD,MAAM,WAAW,WAAW;IAC1B,KAAK,EAAE,SAAS,CAAC;IACjB,IAAI,EAAE,OAAO,CAAC;IACd,IAAI,EAAE,OAAO,CAAC;CACf;AAED,wBAAsB,OAAO,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,WAAW,GAAG,OAAO,CAAC,aAAa,CAAC,CAiC5F"}
@@ -0,0 +1,36 @@
1
+ import { formatError, formatJson, formatJsonPretty } from '../output.js';
2
+ import { suggest } from '../levenshtein.js';
3
+ import { toToon } from '../toon.js';
4
+ export async function runPull(screenId, options) {
5
+ try {
6
+ const spec = await options.store.load(screenId);
7
+ if (options.toon) {
8
+ return { output: toToon(spec), exitCode: 0 };
9
+ }
10
+ if (options.json) {
11
+ return { output: formatJson({ success: true, spec }), exitCode: 0 };
12
+ }
13
+ return { output: formatJsonPretty(spec), exitCode: 0 };
14
+ }
15
+ catch (err) {
16
+ const message = err.message;
17
+ let availableScreens = [];
18
+ try {
19
+ availableScreens = await options.store.list();
20
+ }
21
+ catch { /* ignore */ }
22
+ const suggestion = suggest(screenId, availableScreens);
23
+ if (options.json) {
24
+ return { output: formatJson({ error: message, suggestion }), exitCode: 1 };
25
+ }
26
+ return {
27
+ output: formatError({
28
+ what: `Screen "${screenId}" not found`,
29
+ why: availableScreens.length > 0 ? `Available screens: ${availableScreens.join(', ')}` : message,
30
+ fix: suggestion ? `Did you mean: ${suggestion}?` : 'Check the screen ID and try again',
31
+ }),
32
+ exitCode: 1,
33
+ };
34
+ }
35
+ }
36
+ //# sourceMappingURL=pull.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pull.js","sourceRoot":"","sources":["../../src/commands/pull.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,gBAAgB,EAAE,MAAM,cAAc,CAAC;AACzE,OAAO,EAAE,OAAO,EAAE,MAAM,mBAAmB,CAAC;AAC5C,OAAO,EAAE,MAAM,EAAE,MAAM,YAAY,CAAC;AASpC,MAAM,CAAC,KAAK,UAAU,OAAO,CAAC,QAAgB,EAAE,OAAoB;IAClE,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEhD,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,EAAE,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAC;QAC/C,CAAC;QAED,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAC;QACtE,CAAC;QAED,OAAO,EAAE,MAAM,EAAE,gBAAgB,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAC;IACzD,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,MAAM,OAAO,GAAI,GAAa,CAAC,OAAO,CAAC;QAEvC,IAAI,gBAAgB,GAAa,EAAE,CAAC;QACpC,IAAI,CAAC;YAAC,gBAAgB,GAAG,MAAM,OAAO,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;QAC7E,MAAM,UAAU,GAAG,OAAO,CAAC,QAAQ,EAAE,gBAAgB,CAAC,CAAC;QAEvD,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,UAAU,EAAE,CAAC,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAC;QAC7E,CAAC;QAED,OAAO;YACL,MAAM,EAAE,WAAW,CAAC;gBAClB,IAAI,EAAE,WAAW,QAAQ,aAAa;gBACtC,GAAG,EAAE,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,sBAAsB,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO;gBAChG,GAAG,EAAE,UAAU,CAAC,CAAC,CAAC,iBAAiB,UAAU,GAAG,CAAC,CAAC,CAAC,mCAAmC;aACvF,CAAC;YACF,QAAQ,EAAE,CAAC;SACZ,CAAC;IACJ,CAAC;AACH,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { SpecStore } from 'mythik';
2
+ import type { CommandResult } from './manifest.js';
3
+ export interface PushBulkOptions {
4
+ store: SpecStore;
5
+ json: boolean;
6
+ force: boolean;
7
+ author?: string;
8
+ description?: string;
9
+ }
10
+ export declare function runPushBulk(dirPath: string, options: PushBulkOptions): Promise<CommandResult>;
11
+ //# sourceMappingURL=push-bulk.d.ts.map