@prisma-next/migration-tools 0.5.0-dev.6 → 0.5.0-dev.60

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 (91) hide show
  1. package/README.md +34 -22
  2. package/dist/{constants-BRi0X7B_.mjs → constants-BQEHsaEx.mjs} +1 -1
  3. package/dist/{constants-BRi0X7B_.mjs.map → constants-BQEHsaEx.mjs.map} +1 -1
  4. package/dist/errors-CfmjBeK0.mjs +272 -0
  5. package/dist/errors-CfmjBeK0.mjs.map +1 -0
  6. package/dist/exports/constants.mjs +1 -1
  7. package/dist/exports/errors.d.mts +63 -0
  8. package/dist/exports/errors.d.mts.map +1 -0
  9. package/dist/exports/errors.mjs +3 -0
  10. package/dist/exports/graph.d.mts +2 -0
  11. package/dist/exports/graph.mjs +1 -0
  12. package/dist/exports/hash.d.mts +52 -0
  13. package/dist/exports/hash.d.mts.map +1 -0
  14. package/dist/exports/hash.mjs +3 -0
  15. package/dist/exports/invariants.d.mts +24 -0
  16. package/dist/exports/invariants.d.mts.map +1 -0
  17. package/dist/exports/invariants.mjs +4 -0
  18. package/dist/exports/io.d.mts +7 -6
  19. package/dist/exports/io.d.mts.map +1 -1
  20. package/dist/exports/io.mjs +162 -2
  21. package/dist/exports/io.mjs.map +1 -0
  22. package/dist/exports/metadata.d.mts +2 -0
  23. package/dist/exports/metadata.mjs +1 -0
  24. package/dist/exports/migration-graph.d.mts +124 -0
  25. package/dist/exports/migration-graph.d.mts.map +1 -0
  26. package/dist/exports/migration-graph.mjs +526 -0
  27. package/dist/exports/migration-graph.mjs.map +1 -0
  28. package/dist/exports/migration-ts.mjs +1 -1
  29. package/dist/exports/migration.d.mts +15 -14
  30. package/dist/exports/migration.d.mts.map +1 -1
  31. package/dist/exports/migration.mjs +69 -41
  32. package/dist/exports/migration.mjs.map +1 -1
  33. package/dist/exports/package.d.mts +2 -0
  34. package/dist/exports/package.mjs +1 -0
  35. package/dist/exports/refs.mjs +2 -2
  36. package/dist/graph-BHPv-9Gl.d.mts +28 -0
  37. package/dist/graph-BHPv-9Gl.d.mts.map +1 -0
  38. package/dist/hash-BARZdVgW.mjs +76 -0
  39. package/dist/hash-BARZdVgW.mjs.map +1 -0
  40. package/dist/invariants-30VA65sB.mjs +42 -0
  41. package/dist/invariants-30VA65sB.mjs.map +1 -0
  42. package/dist/metadata-BP1cmU7Z.d.mts +50 -0
  43. package/dist/metadata-BP1cmU7Z.d.mts.map +1 -0
  44. package/dist/op-schema-DZKFua46.mjs +14 -0
  45. package/dist/op-schema-DZKFua46.mjs.map +1 -0
  46. package/dist/package-5HCCg0z-.d.mts +21 -0
  47. package/dist/package-5HCCg0z-.d.mts.map +1 -0
  48. package/package.json +31 -14
  49. package/src/errors.ts +210 -17
  50. package/src/exports/errors.ts +7 -0
  51. package/src/exports/graph.ts +1 -0
  52. package/src/exports/hash.ts +2 -0
  53. package/src/exports/invariants.ts +1 -0
  54. package/src/exports/io.ts +1 -1
  55. package/src/exports/metadata.ts +1 -0
  56. package/src/exports/{dag.ts → migration-graph.ts} +3 -2
  57. package/src/exports/migration.ts +0 -1
  58. package/src/exports/package.ts +1 -0
  59. package/src/graph-ops.ts +57 -30
  60. package/src/graph.ts +25 -0
  61. package/src/hash.ts +91 -0
  62. package/src/invariants.ts +45 -0
  63. package/src/io.ts +57 -31
  64. package/src/metadata.ts +41 -0
  65. package/src/migration-base.ts +97 -56
  66. package/src/migration-graph.ts +676 -0
  67. package/src/op-schema.ts +11 -0
  68. package/src/package.ts +18 -0
  69. package/dist/attestation-BnzTb0Qp.mjs +0 -65
  70. package/dist/attestation-BnzTb0Qp.mjs.map +0 -1
  71. package/dist/errors-BmiSgz1j.mjs +0 -160
  72. package/dist/errors-BmiSgz1j.mjs.map +0 -1
  73. package/dist/exports/attestation.d.mts +0 -37
  74. package/dist/exports/attestation.d.mts.map +0 -1
  75. package/dist/exports/attestation.mjs +0 -4
  76. package/dist/exports/dag.d.mts +0 -51
  77. package/dist/exports/dag.d.mts.map +0 -1
  78. package/dist/exports/dag.mjs +0 -386
  79. package/dist/exports/dag.mjs.map +0 -1
  80. package/dist/exports/types.d.mts +0 -35
  81. package/dist/exports/types.d.mts.map +0 -1
  82. package/dist/exports/types.mjs +0 -3
  83. package/dist/io-Cd6GLyjK.mjs +0 -153
  84. package/dist/io-Cd6GLyjK.mjs.map +0 -1
  85. package/dist/types-DyGXcWWp.d.mts +0 -71
  86. package/dist/types-DyGXcWWp.d.mts.map +0 -1
  87. package/src/attestation.ts +0 -81
  88. package/src/dag.ts +0 -426
  89. package/src/exports/attestation.ts +0 -2
  90. package/src/exports/types.ts +0 -10
  91. package/src/types.ts +0 -66
@@ -1,386 +0,0 @@
1
- import { h as errorSameSourceAndTarget, i as errorDuplicateMigrationId, m as errorNoTarget, n as errorAmbiguousTarget, p as errorNoInitialMigration } from "../errors-BmiSgz1j.mjs";
2
- import { t as EMPTY_CONTRACT_HASH } from "../constants-BRi0X7B_.mjs";
3
- import { ifDefined } from "@prisma-next/utils/defined";
4
-
5
- //#region src/queue.ts
6
- /**
7
- * FIFO queue with amortised O(1) push and shift.
8
- *
9
- * Uses a head-index cursor over a backing array rather than
10
- * `Array.prototype.shift()`, which is O(n) on V8. Intended for BFS-shaped
11
- * traversals where the queue is drained in a single pass — it does not
12
- * reclaim memory for already-shifted items, so it is not suitable for
13
- * long-lived queues with many push/shift cycles.
14
- */
15
- var Queue = class {
16
- items;
17
- head = 0;
18
- constructor(initial = []) {
19
- this.items = [...initial];
20
- }
21
- push(item) {
22
- this.items.push(item);
23
- }
24
- /**
25
- * Remove and return the next item. Caller must check `isEmpty` first —
26
- * shifting an empty queue throws.
27
- */
28
- shift() {
29
- if (this.head >= this.items.length) throw new Error("Queue.shift called on empty queue");
30
- return this.items[this.head++];
31
- }
32
- get isEmpty() {
33
- return this.head >= this.items.length;
34
- }
35
- };
36
-
37
- //#endregion
38
- //#region src/graph-ops.ts
39
- /**
40
- * Generic breadth-first traversal.
41
- *
42
- * Direction (forward/reverse) is expressed by the caller's `neighbours`
43
- * closure: return `{ next, edge }` pairs where `next` is the node to visit
44
- * next and `edge` is the edge that connects them. Callers that don't need
45
- * path reconstruction can ignore the `parent`/`incomingEdge` fields of each
46
- * yielded step.
47
- *
48
- * Stops are intrinsic — callers `break` out of the `for..of` loop when
49
- * they've found what they're looking for.
50
- *
51
- * `ordering`, if provided, controls the order in which neighbours of each
52
- * node are enqueued. Only matters for path-finding: a deterministic ordering
53
- * makes BFS return a deterministic shortest path when multiple exist.
54
- */
55
- function* bfs(starts, neighbours, ordering) {
56
- const visited = /* @__PURE__ */ new Set();
57
- const parentMap = /* @__PURE__ */ new Map();
58
- const queue = new Queue();
59
- for (const start of starts) if (!visited.has(start)) {
60
- visited.add(start);
61
- queue.push(start);
62
- }
63
- while (!queue.isEmpty) {
64
- const current = queue.shift();
65
- const parentInfo = parentMap.get(current);
66
- yield {
67
- node: current,
68
- parent: parentInfo?.parent ?? null,
69
- incomingEdge: parentInfo?.edge ?? null
70
- };
71
- const items = neighbours(current);
72
- const toVisit = ordering ? ordering([...items]) : items;
73
- for (const { next, edge } of toVisit) if (!visited.has(next)) {
74
- visited.add(next);
75
- parentMap.set(next, {
76
- parent: current,
77
- edge
78
- });
79
- queue.push(next);
80
- }
81
- }
82
- }
83
-
84
- //#endregion
85
- //#region src/dag.ts
86
- /** Forward-edge neighbours for BFS: edge `e` from `n` visits `e.to` next. */
87
- function forwardNeighbours(graph, node) {
88
- return (graph.forwardChain.get(node) ?? []).map((edge) => ({
89
- next: edge.to,
90
- edge
91
- }));
92
- }
93
- /** Reverse-edge neighbours for BFS: edge `e` from `n` visits `e.from` next. */
94
- function reverseNeighbours(graph, node) {
95
- return (graph.reverseChain.get(node) ?? []).map((edge) => ({
96
- next: edge.from,
97
- edge
98
- }));
99
- }
100
- function appendEdge(map, key, entry) {
101
- const bucket = map.get(key);
102
- if (bucket) bucket.push(entry);
103
- else map.set(key, [entry]);
104
- }
105
- function reconstructGraph(packages) {
106
- const nodes = /* @__PURE__ */ new Set();
107
- const forwardChain = /* @__PURE__ */ new Map();
108
- const reverseChain = /* @__PURE__ */ new Map();
109
- const migrationById = /* @__PURE__ */ new Map();
110
- for (const pkg of packages) {
111
- const { from, to } = pkg.manifest;
112
- if (from === to) throw errorSameSourceAndTarget(pkg.dirName, from);
113
- nodes.add(from);
114
- nodes.add(to);
115
- const migration = {
116
- from,
117
- to,
118
- migrationId: pkg.manifest.migrationId,
119
- dirName: pkg.dirName,
120
- createdAt: pkg.manifest.createdAt,
121
- labels: pkg.manifest.labels
122
- };
123
- if (migrationById.has(migration.migrationId)) throw errorDuplicateMigrationId(migration.migrationId);
124
- migrationById.set(migration.migrationId, migration);
125
- appendEdge(forwardChain, from, migration);
126
- appendEdge(reverseChain, to, migration);
127
- }
128
- return {
129
- nodes,
130
- forwardChain,
131
- reverseChain,
132
- migrationById
133
- };
134
- }
135
- const LABEL_PRIORITY = {
136
- main: 0,
137
- default: 1,
138
- feature: 2
139
- };
140
- function labelPriority(labels) {
141
- let best = 3;
142
- for (const l of labels) {
143
- const p = LABEL_PRIORITY[l];
144
- if (p !== void 0 && p < best) best = p;
145
- }
146
- return best;
147
- }
148
- function compareTieBreak(a, b) {
149
- const lp = labelPriority(a.labels) - labelPriority(b.labels);
150
- if (lp !== 0) return lp;
151
- const ca = a.createdAt.localeCompare(b.createdAt);
152
- if (ca !== 0) return ca;
153
- const tc = a.to.localeCompare(b.to);
154
- if (tc !== 0) return tc;
155
- return a.migrationId.localeCompare(b.migrationId);
156
- }
157
- function sortedNeighbors(edges) {
158
- return [...edges].sort(compareTieBreak);
159
- }
160
- /** Ordering adapter for `bfs` — sorts `{next, edge}` pairs by tie-break. */
161
- function bfsOrdering(items) {
162
- return items.slice().sort((a, b) => compareTieBreak(a.edge, b.edge));
163
- }
164
- /**
165
- * Find the shortest path from `fromHash` to `toHash` using BFS over the
166
- * contract-hash graph. Returns the ordered list of edges, or null if no path
167
- * exists. Returns an empty array when `fromHash === toHash` (no-op).
168
- *
169
- * Neighbor ordering is deterministic via the tie-break sort key:
170
- * label priority → createdAt → to → migrationId.
171
- */
172
- function findPath(graph, fromHash, toHash) {
173
- if (fromHash === toHash) return [];
174
- const parents = /* @__PURE__ */ new Map();
175
- for (const step of bfs([fromHash], (n) => forwardNeighbours(graph, n), bfsOrdering)) {
176
- if (step.parent !== null && step.incomingEdge !== null) parents.set(step.node, {
177
- parent: step.parent,
178
- edge: step.incomingEdge
179
- });
180
- if (step.node === toHash) {
181
- const path = [];
182
- let cur = toHash;
183
- let p = parents.get(cur);
184
- while (p) {
185
- path.push(p.edge);
186
- cur = p.parent;
187
- p = parents.get(cur);
188
- }
189
- path.reverse();
190
- return path;
191
- }
192
- }
193
- return null;
194
- }
195
- /**
196
- * Reverse-BFS from `toHash` over `reverseChain` to collect every node from
197
- * which `toHash` is reachable (inclusive of `toHash` itself).
198
- */
199
- function collectNodesReachingTarget(graph, toHash) {
200
- const reached = /* @__PURE__ */ new Set();
201
- for (const step of bfs([toHash], (n) => reverseNeighbours(graph, n))) reached.add(step.node);
202
- return reached;
203
- }
204
- /**
205
- * Find the shortest path from `fromHash` to `toHash` and return structured
206
- * path-decision metadata for machine-readable output.
207
- */
208
- function findPathWithDecision(graph, fromHash, toHash, refName) {
209
- if (fromHash === toHash) return {
210
- selectedPath: [],
211
- fromHash,
212
- toHash,
213
- alternativeCount: 0,
214
- tieBreakReasons: [],
215
- ...ifDefined("refName", refName)
216
- };
217
- const path = findPath(graph, fromHash, toHash);
218
- if (!path) return null;
219
- const reachesTarget = collectNodesReachingTarget(graph, toHash);
220
- const tieBreakReasons = [];
221
- let alternativeCount = 0;
222
- for (const edge of path) {
223
- const outgoing = graph.forwardChain.get(edge.from);
224
- if (outgoing && outgoing.length > 1) {
225
- const reachable = outgoing.filter((e) => reachesTarget.has(e.to));
226
- if (reachable.length > 1) {
227
- alternativeCount += reachable.length - 1;
228
- const sorted = sortedNeighbors(reachable);
229
- if (sorted[0] && sorted[0].migrationId === edge.migrationId) {
230
- if (reachable.some((e) => e.migrationId !== edge.migrationId)) tieBreakReasons.push(`at ${edge.from}: ${reachable.length} candidates, selected by tie-break`);
231
- }
232
- }
233
- }
234
- }
235
- return {
236
- selectedPath: path,
237
- fromHash,
238
- toHash,
239
- alternativeCount,
240
- tieBreakReasons,
241
- ...ifDefined("refName", refName)
242
- };
243
- }
244
- /**
245
- * Walk ancestors of each branch tip back to find the last node
246
- * that appears on all paths. Returns `fromHash` if no shared ancestor is found.
247
- */
248
- function findDivergencePoint(graph, fromHash, leaves) {
249
- const ancestorSets = leaves.map((leaf) => {
250
- const ancestors = /* @__PURE__ */ new Set();
251
- for (const step of bfs([leaf], (n) => reverseNeighbours(graph, n))) ancestors.add(step.node);
252
- return ancestors;
253
- });
254
- const commonAncestors = [...ancestorSets[0] ?? []].filter((node) => ancestorSets.every((s) => s.has(node)));
255
- let deepest = fromHash;
256
- let deepestDepth = -1;
257
- for (const ancestor of commonAncestors) {
258
- const path = findPath(graph, fromHash, ancestor);
259
- const depth = path ? path.length : 0;
260
- if (depth > deepestDepth) {
261
- deepestDepth = depth;
262
- deepest = ancestor;
263
- }
264
- }
265
- return deepest;
266
- }
267
- /**
268
- * Find all branch tips (nodes with no outgoing edges) reachable from
269
- * `fromHash` via forward edges.
270
- */
271
- function findReachableLeaves(graph, fromHash) {
272
- const leaves = [];
273
- for (const step of bfs([fromHash], (n) => forwardNeighbours(graph, n))) if (!graph.forwardChain.get(step.node)?.length) leaves.push(step.node);
274
- return leaves;
275
- }
276
- /**
277
- * Find the target contract hash of the migration graph reachable from
278
- * EMPTY_CONTRACT_HASH. Returns `null` for a graph that has no target
279
- * state (either empty, or containing only the root with no outgoing
280
- * edges). Throws NO_INITIAL_MIGRATION if the graph has nodes but none
281
- * originate from the empty hash, and AMBIGUOUS_TARGET if multiple
282
- * branch tips exist.
283
- */
284
- function findLeaf(graph) {
285
- if (graph.nodes.size === 0) return null;
286
- if (!graph.nodes.has(EMPTY_CONTRACT_HASH)) throw errorNoInitialMigration([...graph.nodes]);
287
- const leaves = findReachableLeaves(graph, EMPTY_CONTRACT_HASH);
288
- if (leaves.length === 0) {
289
- const reachable = [...graph.nodes].filter((n) => n !== EMPTY_CONTRACT_HASH);
290
- if (reachable.length > 0) throw errorNoTarget(reachable);
291
- return null;
292
- }
293
- if (leaves.length > 1) {
294
- const divergencePoint = findDivergencePoint(graph, EMPTY_CONTRACT_HASH, leaves);
295
- throw errorAmbiguousTarget(leaves, {
296
- divergencePoint,
297
- branches: leaves.map((tip) => {
298
- return {
299
- tip,
300
- edges: (findPath(graph, divergencePoint, tip) ?? []).map((e) => ({
301
- dirName: e.dirName,
302
- from: e.from,
303
- to: e.to
304
- }))
305
- };
306
- })
307
- });
308
- }
309
- return leaves[0];
310
- }
311
- /**
312
- * Find the latest migration entry by traversing from EMPTY_CONTRACT_HASH
313
- * to the single target. Returns null for an empty graph.
314
- * Throws AMBIGUOUS_TARGET if the graph has multiple branch tips.
315
- */
316
- function findLatestMigration(graph) {
317
- const leafHash = findLeaf(graph);
318
- if (leafHash === null) return null;
319
- return findPath(graph, EMPTY_CONTRACT_HASH, leafHash)?.at(-1) ?? null;
320
- }
321
- function detectCycles(graph) {
322
- const WHITE = 0;
323
- const GRAY = 1;
324
- const BLACK = 2;
325
- const color = /* @__PURE__ */ new Map();
326
- const parentMap = /* @__PURE__ */ new Map();
327
- const cycles = [];
328
- for (const node of graph.nodes) color.set(node, WHITE);
329
- const stack = [];
330
- function pushFrame(u) {
331
- color.set(u, GRAY);
332
- stack.push({
333
- node: u,
334
- outgoing: graph.forwardChain.get(u) ?? [],
335
- index: 0
336
- });
337
- }
338
- for (const root of graph.nodes) {
339
- if (color.get(root) !== WHITE) continue;
340
- parentMap.set(root, null);
341
- pushFrame(root);
342
- while (stack.length > 0) {
343
- const frame = stack[stack.length - 1];
344
- if (frame.index >= frame.outgoing.length) {
345
- color.set(frame.node, BLACK);
346
- stack.pop();
347
- continue;
348
- }
349
- const v = frame.outgoing[frame.index++].to;
350
- const vColor = color.get(v);
351
- if (vColor === GRAY) {
352
- const cycle = [v];
353
- let cur = frame.node;
354
- while (cur !== v) {
355
- cycle.push(cur);
356
- cur = parentMap.get(cur) ?? v;
357
- }
358
- cycle.reverse();
359
- cycles.push(cycle);
360
- } else if (vColor === WHITE) {
361
- parentMap.set(v, frame.node);
362
- pushFrame(v);
363
- }
364
- }
365
- }
366
- return cycles;
367
- }
368
- function detectOrphans(graph) {
369
- if (graph.nodes.size === 0) return [];
370
- const reachable = /* @__PURE__ */ new Set();
371
- const startNodes = [];
372
- if (graph.forwardChain.has(EMPTY_CONTRACT_HASH)) startNodes.push(EMPTY_CONTRACT_HASH);
373
- else {
374
- const allTargets = /* @__PURE__ */ new Set();
375
- for (const edges of graph.forwardChain.values()) for (const edge of edges) allTargets.add(edge.to);
376
- for (const node of graph.nodes) if (!allTargets.has(node)) startNodes.push(node);
377
- }
378
- for (const step of bfs(startNodes, (n) => forwardNeighbours(graph, n))) reachable.add(step.node);
379
- const orphans = [];
380
- for (const [from, migrations] of graph.forwardChain) if (!reachable.has(from)) orphans.push(...migrations);
381
- return orphans;
382
- }
383
-
384
- //#endregion
385
- export { detectCycles, detectOrphans, findLatestMigration, findLeaf, findPath, findPathWithDecision, findReachableLeaves, reconstructGraph };
386
- //# sourceMappingURL=dag.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"dag.mjs","names":["migration: MigrationChainEntry","LABEL_PRIORITY: Record<string, number>","path: MigrationChainEntry[]","tieBreakReasons: string[]","leaves: string[]","cycles: string[][]","stack: Frame[]","cycle: string[]","startNodes: string[]","orphans: MigrationChainEntry[]"],"sources":["../../src/queue.ts","../../src/graph-ops.ts","../../src/dag.ts"],"sourcesContent":["/**\n * FIFO queue with amortised O(1) push and shift.\n *\n * Uses a head-index cursor over a backing array rather than\n * `Array.prototype.shift()`, which is O(n) on V8. Intended for BFS-shaped\n * traversals where the queue is drained in a single pass — it does not\n * reclaim memory for already-shifted items, so it is not suitable for\n * long-lived queues with many push/shift cycles.\n */\nexport class Queue<T> {\n private readonly items: T[];\n private head = 0;\n\n constructor(initial: Iterable<T> = []) {\n this.items = [...initial];\n }\n\n push(item: T): void {\n this.items.push(item);\n }\n\n /**\n * Remove and return the next item. Caller must check `isEmpty` first —\n * shifting an empty queue throws.\n */\n shift(): T {\n if (this.head >= this.items.length) {\n throw new Error('Queue.shift called on empty queue');\n }\n // biome-ignore lint/style/noNonNullAssertion: bounds-checked on the line above\n return this.items[this.head++]!;\n }\n\n get isEmpty(): boolean {\n return this.head >= this.items.length;\n }\n}\n","import { Queue } from './queue';\n\n/**\n * One step of a BFS traversal.\n *\n * `parent` and `incomingEdge` are `null` for start nodes — they were not\n * reached via any edge. For every other node they record the node and edge\n * by which this node was first reached.\n */\nexport interface BfsStep<E> {\n readonly node: string;\n readonly parent: string | null;\n readonly incomingEdge: E | null;\n}\n\n/**\n * Generic breadth-first traversal.\n *\n * Direction (forward/reverse) is expressed by the caller's `neighbours`\n * closure: return `{ next, edge }` pairs where `next` is the node to visit\n * next and `edge` is the edge that connects them. Callers that don't need\n * path reconstruction can ignore the `parent`/`incomingEdge` fields of each\n * yielded step.\n *\n * Stops are intrinsic — callers `break` out of the `for..of` loop when\n * they've found what they're looking for.\n *\n * `ordering`, if provided, controls the order in which neighbours of each\n * node are enqueued. Only matters for path-finding: a deterministic ordering\n * makes BFS return a deterministic shortest path when multiple exist.\n */\nexport function* bfs<E>(\n starts: Iterable<string>,\n neighbours: (node: string) => Iterable<{ next: string; edge: E }>,\n ordering?: (items: readonly { next: string; edge: E }[]) => readonly { next: string; edge: E }[],\n): Generator<BfsStep<E>> {\n const visited = new Set<string>();\n const parentMap = new Map<string, { parent: string; edge: E }>();\n const queue = new Queue<string>();\n for (const start of starts) {\n if (!visited.has(start)) {\n visited.add(start);\n queue.push(start);\n }\n }\n while (!queue.isEmpty) {\n const current = queue.shift();\n const parentInfo = parentMap.get(current);\n yield {\n node: current,\n parent: parentInfo?.parent ?? null,\n incomingEdge: parentInfo?.edge ?? null,\n };\n\n const items = neighbours(current);\n const toVisit = ordering ? ordering([...items]) : items;\n for (const { next, edge } of toVisit) {\n if (!visited.has(next)) {\n visited.add(next);\n parentMap.set(next, { parent: current, edge });\n queue.push(next);\n }\n }\n }\n}\n","import { ifDefined } from '@prisma-next/utils/defined';\nimport { EMPTY_CONTRACT_HASH } from './constants';\nimport {\n errorAmbiguousTarget,\n errorDuplicateMigrationId,\n errorNoInitialMigration,\n errorNoTarget,\n errorSameSourceAndTarget,\n} from './errors';\nimport { bfs } from './graph-ops';\nimport type { MigrationBundle, MigrationChainEntry, MigrationGraph } from './types';\n\n/** Forward-edge neighbours for BFS: edge `e` from `n` visits `e.to` next. */\nfunction forwardNeighbours(graph: MigrationGraph, node: string) {\n return (graph.forwardChain.get(node) ?? []).map((edge) => ({ next: edge.to, edge }));\n}\n\n/** Reverse-edge neighbours for BFS: edge `e` from `n` visits `e.from` next. */\nfunction reverseNeighbours(graph: MigrationGraph, node: string) {\n return (graph.reverseChain.get(node) ?? []).map((edge) => ({ next: edge.from, edge }));\n}\n\nfunction appendEdge(\n map: Map<string, MigrationChainEntry[]>,\n key: string,\n entry: MigrationChainEntry,\n): void {\n const bucket = map.get(key);\n if (bucket) bucket.push(entry);\n else map.set(key, [entry]);\n}\n\nexport function reconstructGraph(packages: readonly MigrationBundle[]): MigrationGraph {\n const nodes = new Set<string>();\n const forwardChain = new Map<string, MigrationChainEntry[]>();\n const reverseChain = new Map<string, MigrationChainEntry[]>();\n const migrationById = new Map<string, MigrationChainEntry>();\n\n for (const pkg of packages) {\n const { from, to } = pkg.manifest;\n\n if (from === to) {\n throw errorSameSourceAndTarget(pkg.dirName, from);\n }\n\n nodes.add(from);\n nodes.add(to);\n\n const migration: MigrationChainEntry = {\n from,\n to,\n migrationId: pkg.manifest.migrationId,\n dirName: pkg.dirName,\n createdAt: pkg.manifest.createdAt,\n labels: pkg.manifest.labels,\n };\n\n if (migrationById.has(migration.migrationId)) {\n throw errorDuplicateMigrationId(migration.migrationId);\n }\n migrationById.set(migration.migrationId, migration);\n\n appendEdge(forwardChain, from, migration);\n appendEdge(reverseChain, to, migration);\n }\n\n return { nodes, forwardChain, reverseChain, migrationById };\n}\n\n// ---------------------------------------------------------------------------\n// Deterministic tie-breaking for BFS neighbour order.\n// Used by `findPath` and `findPathWithDecision` only; not a general-purpose\n// utility. Ordering: label priority → createdAt → to → migrationId.\n// ---------------------------------------------------------------------------\n\nconst LABEL_PRIORITY: Record<string, number> = { main: 0, default: 1, feature: 2 };\n\nfunction labelPriority(labels: readonly string[]): number {\n let best = 3;\n for (const l of labels) {\n const p = LABEL_PRIORITY[l];\n if (p !== undefined && p < best) best = p;\n }\n return best;\n}\n\nfunction compareTieBreak(a: MigrationChainEntry, b: MigrationChainEntry): number {\n const lp = labelPriority(a.labels) - labelPriority(b.labels);\n if (lp !== 0) return lp;\n const ca = a.createdAt.localeCompare(b.createdAt);\n if (ca !== 0) return ca;\n const tc = a.to.localeCompare(b.to);\n if (tc !== 0) return tc;\n return a.migrationId.localeCompare(b.migrationId);\n}\n\nfunction sortedNeighbors(edges: readonly MigrationChainEntry[]): readonly MigrationChainEntry[] {\n return [...edges].sort(compareTieBreak);\n}\n\n/** Ordering adapter for `bfs` — sorts `{next, edge}` pairs by tie-break. */\nfunction bfsOrdering(\n items: readonly { next: string; edge: MigrationChainEntry }[],\n): readonly { next: string; edge: MigrationChainEntry }[] {\n return items.slice().sort((a, b) => compareTieBreak(a.edge, b.edge));\n}\n\n/**\n * Find the shortest path from `fromHash` to `toHash` using BFS over the\n * contract-hash graph. Returns the ordered list of edges, or null if no path\n * exists. Returns an empty array when `fromHash === toHash` (no-op).\n *\n * Neighbor ordering is deterministic via the tie-break sort key:\n * label priority → createdAt → to → migrationId.\n */\nexport function findPath(\n graph: MigrationGraph,\n fromHash: string,\n toHash: string,\n): readonly MigrationChainEntry[] | null {\n if (fromHash === toHash) return [];\n\n const parents = new Map<string, { parent: string; edge: MigrationChainEntry }>();\n for (const step of bfs([fromHash], (n) => forwardNeighbours(graph, n), bfsOrdering)) {\n if (step.parent !== null && step.incomingEdge !== null) {\n parents.set(step.node, { parent: step.parent, edge: step.incomingEdge });\n }\n if (step.node === toHash) {\n const path: MigrationChainEntry[] = [];\n let cur = toHash;\n let p = parents.get(cur);\n while (p) {\n path.push(p.edge);\n cur = p.parent;\n p = parents.get(cur);\n }\n path.reverse();\n return path;\n }\n }\n\n return null;\n}\n\n/**\n * Reverse-BFS from `toHash` over `reverseChain` to collect every node from\n * which `toHash` is reachable (inclusive of `toHash` itself).\n */\nfunction collectNodesReachingTarget(graph: MigrationGraph, toHash: string): Set<string> {\n const reached = new Set<string>();\n for (const step of bfs([toHash], (n) => reverseNeighbours(graph, n))) {\n reached.add(step.node);\n }\n return reached;\n}\n\nexport interface PathDecision {\n readonly selectedPath: readonly MigrationChainEntry[];\n readonly fromHash: string;\n readonly toHash: string;\n readonly alternativeCount: number;\n readonly tieBreakReasons: readonly string[];\n readonly refName?: string;\n}\n\n/**\n * Find the shortest path from `fromHash` to `toHash` and return structured\n * path-decision metadata for machine-readable output.\n */\nexport function findPathWithDecision(\n graph: MigrationGraph,\n fromHash: string,\n toHash: string,\n refName?: string,\n): PathDecision | null {\n if (fromHash === toHash) {\n return {\n selectedPath: [],\n fromHash,\n toHash,\n alternativeCount: 0,\n tieBreakReasons: [],\n ...ifDefined('refName', refName),\n };\n }\n\n const path = findPath(graph, fromHash, toHash);\n if (!path) return null;\n\n // Single reverse BFS marks every node from which `toHash` is reachable.\n // Replaces a per-edge `findPath(e.to, toHash)` call inside the loop below,\n // which made the whole function O(|path| · (V + E)) instead of O(V + E).\n const reachesTarget = collectNodesReachingTarget(graph, toHash);\n\n const tieBreakReasons: string[] = [];\n let alternativeCount = 0;\n\n for (const edge of path) {\n const outgoing = graph.forwardChain.get(edge.from);\n if (outgoing && outgoing.length > 1) {\n const reachable = outgoing.filter((e) => reachesTarget.has(e.to));\n if (reachable.length > 1) {\n alternativeCount += reachable.length - 1;\n const sorted = sortedNeighbors(reachable);\n if (sorted[0] && sorted[0].migrationId === edge.migrationId) {\n if (reachable.some((e) => e.migrationId !== edge.migrationId)) {\n tieBreakReasons.push(\n `at ${edge.from}: ${reachable.length} candidates, selected by tie-break`,\n );\n }\n }\n }\n }\n }\n\n return {\n selectedPath: path,\n fromHash,\n toHash,\n alternativeCount,\n tieBreakReasons,\n ...ifDefined('refName', refName),\n };\n}\n\n/**\n * Walk ancestors of each branch tip back to find the last node\n * that appears on all paths. Returns `fromHash` if no shared ancestor is found.\n */\nfunction findDivergencePoint(\n graph: MigrationGraph,\n fromHash: string,\n leaves: readonly string[],\n): string {\n const ancestorSets = leaves.map((leaf) => {\n const ancestors = new Set<string>();\n for (const step of bfs([leaf], (n) => reverseNeighbours(graph, n))) {\n ancestors.add(step.node);\n }\n return ancestors;\n });\n\n const commonAncestors = [...(ancestorSets[0] ?? [])].filter((node) =>\n ancestorSets.every((s) => s.has(node)),\n );\n\n let deepest = fromHash;\n let deepestDepth = -1;\n for (const ancestor of commonAncestors) {\n const path = findPath(graph, fromHash, ancestor);\n const depth = path ? path.length : 0;\n if (depth > deepestDepth) {\n deepestDepth = depth;\n deepest = ancestor;\n }\n }\n return deepest;\n}\n\n/**\n * Find all branch tips (nodes with no outgoing edges) reachable from\n * `fromHash` via forward edges.\n */\nexport function findReachableLeaves(graph: MigrationGraph, fromHash: string): readonly string[] {\n const leaves: string[] = [];\n for (const step of bfs([fromHash], (n) => forwardNeighbours(graph, n))) {\n if (!graph.forwardChain.get(step.node)?.length) {\n leaves.push(step.node);\n }\n }\n return leaves;\n}\n\n/**\n * Find the target contract hash of the migration graph reachable from\n * EMPTY_CONTRACT_HASH. Returns `null` for a graph that has no target\n * state (either empty, or containing only the root with no outgoing\n * edges). Throws NO_INITIAL_MIGRATION if the graph has nodes but none\n * originate from the empty hash, and AMBIGUOUS_TARGET if multiple\n * branch tips exist.\n */\nexport function findLeaf(graph: MigrationGraph): string | null {\n if (graph.nodes.size === 0) {\n return null;\n }\n\n if (!graph.nodes.has(EMPTY_CONTRACT_HASH)) {\n throw errorNoInitialMigration([...graph.nodes]);\n }\n\n const leaves = findReachableLeaves(graph, EMPTY_CONTRACT_HASH);\n\n if (leaves.length === 0) {\n const reachable = [...graph.nodes].filter((n) => n !== EMPTY_CONTRACT_HASH);\n if (reachable.length > 0) {\n throw errorNoTarget(reachable);\n }\n return null;\n }\n\n if (leaves.length > 1) {\n const divergencePoint = findDivergencePoint(graph, EMPTY_CONTRACT_HASH, leaves);\n const branches = leaves.map((tip) => {\n const path = findPath(graph, divergencePoint, tip);\n return {\n tip,\n edges: (path ?? []).map((e) => ({ dirName: e.dirName, from: e.from, to: e.to })),\n };\n });\n throw errorAmbiguousTarget(leaves, { divergencePoint, branches });\n }\n\n // biome-ignore lint/style/noNonNullAssertion: leaves.length is neither 0 nor >1 per the branches above, so exactly one leaf remains\n return leaves[0]!;\n}\n\n/**\n * Find the latest migration entry by traversing from EMPTY_CONTRACT_HASH\n * to the single target. Returns null for an empty graph.\n * Throws AMBIGUOUS_TARGET if the graph has multiple branch tips.\n */\nexport function findLatestMigration(graph: MigrationGraph): MigrationChainEntry | null {\n const leafHash = findLeaf(graph);\n if (leafHash === null) return null;\n\n const path = findPath(graph, EMPTY_CONTRACT_HASH, leafHash);\n return path?.at(-1) ?? null;\n}\n\nexport function detectCycles(graph: MigrationGraph): readonly string[][] {\n const WHITE = 0;\n const GRAY = 1;\n const BLACK = 2;\n\n const color = new Map<string, number>();\n const parentMap = new Map<string, string | null>();\n const cycles: string[][] = [];\n\n for (const node of graph.nodes) {\n color.set(node, WHITE);\n }\n\n // Iterative three-color DFS. A frame is (node, outgoing edges, next-index).\n interface Frame {\n node: string;\n outgoing: readonly MigrationChainEntry[];\n index: number;\n }\n const stack: Frame[] = [];\n\n function pushFrame(u: string): void {\n color.set(u, GRAY);\n stack.push({ node: u, outgoing: graph.forwardChain.get(u) ?? [], index: 0 });\n }\n\n for (const root of graph.nodes) {\n if (color.get(root) !== WHITE) continue;\n parentMap.set(root, null);\n pushFrame(root);\n\n while (stack.length > 0) {\n // biome-ignore lint/style/noNonNullAssertion: stack.length > 0 should guarantee that this cannot be undefined\n const frame = stack[stack.length - 1]!;\n if (frame.index >= frame.outgoing.length) {\n color.set(frame.node, BLACK);\n stack.pop();\n continue;\n }\n // biome-ignore lint/style/noNonNullAssertion: the early-continue above guarantees frame.index < frame.outgoing.length here, so this is defined\n const edge = frame.outgoing[frame.index++]!;\n const v = edge.to;\n const vColor = color.get(v);\n if (vColor === GRAY) {\n const cycle: string[] = [v];\n let cur = frame.node;\n while (cur !== v) {\n cycle.push(cur);\n cur = parentMap.get(cur) ?? v;\n }\n cycle.reverse();\n cycles.push(cycle);\n } else if (vColor === WHITE) {\n parentMap.set(v, frame.node);\n pushFrame(v);\n }\n }\n }\n\n return cycles;\n}\n\nexport function detectOrphans(graph: MigrationGraph): readonly MigrationChainEntry[] {\n if (graph.nodes.size === 0) return [];\n\n const reachable = new Set<string>();\n const startNodes: string[] = [];\n\n if (graph.forwardChain.has(EMPTY_CONTRACT_HASH)) {\n startNodes.push(EMPTY_CONTRACT_HASH);\n } else {\n const allTargets = new Set<string>();\n for (const edges of graph.forwardChain.values()) {\n for (const edge of edges) {\n allTargets.add(edge.to);\n }\n }\n for (const node of graph.nodes) {\n if (!allTargets.has(node)) {\n startNodes.push(node);\n }\n }\n }\n\n for (const step of bfs(startNodes, (n) => forwardNeighbours(graph, n))) {\n reachable.add(step.node);\n }\n\n const orphans: MigrationChainEntry[] = [];\n for (const [from, migrations] of graph.forwardChain) {\n if (!reachable.has(from)) {\n orphans.push(...migrations);\n }\n }\n\n return orphans;\n}\n"],"mappings":"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"}
@@ -1,35 +0,0 @@
1
- import { a as MigrationManifest, i as MigrationHints, n as MigrationChainEntry, o as MigrationOps, r as MigrationGraph, t as MigrationBundle } from "../types-DyGXcWWp.mjs";
2
-
3
- //#region src/errors.d.ts
4
-
5
- /**
6
- * Structured error for migration tooling operations.
7
- *
8
- * Follows the NAMESPACE.SUBCODE convention from ADR 027. All codes live under
9
- * the MIGRATION namespace. These are tooling-time errors (file I/O, attestation,
10
- * migration history reconstruction), distinct from the runtime MIGRATION.* codes for apply-time
11
- * failures (PRECHECK_FAILED, POSTCHECK_FAILED, etc.).
12
- *
13
- * Fields:
14
- * - code: Stable machine-readable code (MIGRATION.SUBCODE)
15
- * - category: Always 'MIGRATION'
16
- * - why: Explains the cause in plain language
17
- * - fix: Actionable remediation step
18
- * - details: Machine-readable structured data for agents
19
- */
20
- declare class MigrationToolsError extends Error {
21
- readonly code: string;
22
- readonly category: "MIGRATION";
23
- readonly why: string;
24
- readonly fix: string;
25
- readonly details: Record<string, unknown> | undefined;
26
- constructor(code: string, summary: string, options: {
27
- readonly why: string;
28
- readonly fix: string;
29
- readonly details?: Record<string, unknown>;
30
- });
31
- static is(error: unknown): error is MigrationToolsError;
32
- }
33
- //#endregion
34
- export { type MigrationBundle, type MigrationBundle as MigrationPackage, type MigrationChainEntry, type MigrationGraph, type MigrationHints, type MigrationManifest, type MigrationOps, MigrationToolsError };
35
- //# sourceMappingURL=types.d.mts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"types.d.mts","names":[],"sources":["../../src/errors.ts"],"sourcesContent":[],"mappings":";;;;;;;AAeA;;;;;;;;;;;;cAAa,mBAAA,SAA4B,KAAA;;;;;oBAKrB;;;;uBAQK;;sCAWa"}
@@ -1,3 +0,0 @@
1
- import { t as MigrationToolsError } from "../errors-BmiSgz1j.mjs";
2
-
3
- export { MigrationToolsError };
@@ -1,153 +0,0 @@
1
- import { a as errorInvalidDestName, d as errorInvalidSlug, f as errorMissingFile, o as errorInvalidJson, r as errorDirectoryExists, s as errorInvalidManifest } from "./errors-BmiSgz1j.mjs";
2
- import { copyFile, mkdir, readFile, readdir, stat, writeFile } from "node:fs/promises";
3
- import { type } from "arktype";
4
- import { basename, dirname, join } from "pathe";
5
-
6
- //#region src/io.ts
7
- const MANIFEST_FILE = "migration.json";
8
- const OPS_FILE = "ops.json";
9
- const MAX_SLUG_LENGTH = 64;
10
- function hasErrnoCode(error, code) {
11
- return error instanceof Error && error.code === code;
12
- }
13
- const MigrationManifestSchema = type({
14
- from: "string",
15
- to: "string",
16
- migrationId: "string",
17
- kind: "'regular' | 'baseline'",
18
- fromContract: "object | null",
19
- toContract: "object",
20
- hints: type({
21
- used: "string[]",
22
- applied: "string[]",
23
- plannerVersion: "string"
24
- }),
25
- labels: "string[]",
26
- "authorship?": type({
27
- "author?": "string",
28
- "email?": "string"
29
- }),
30
- "signature?": type({
31
- keyId: "string",
32
- value: "string"
33
- }).or("null"),
34
- createdAt: "string"
35
- });
36
- const MigrationOpsSchema = type({
37
- id: "string",
38
- label: "string",
39
- operationClass: "'additive' | 'widening' | 'destructive' | 'data'"
40
- }).array();
41
- async function writeMigrationPackage(dir, manifest, ops) {
42
- await mkdir(dirname(dir), { recursive: true });
43
- try {
44
- await mkdir(dir);
45
- } catch (error) {
46
- if (hasErrnoCode(error, "EEXIST")) throw errorDirectoryExists(dir);
47
- throw error;
48
- }
49
- await writeFile(join(dir, MANIFEST_FILE), JSON.stringify(manifest, null, 2), { flag: "wx" });
50
- await writeFile(join(dir, OPS_FILE), JSON.stringify(ops, null, 2), { flag: "wx" });
51
- }
52
- /**
53
- * Copy a list of files into `destDir`, optionally renaming each one.
54
- *
55
- * The destination directory is created (with `recursive: true`) if it
56
- * does not already exist. Each source path is copied byte-for-byte into
57
- * `destDir/<destName>`; missing sources throw `ENOENT`. The helper is
58
- * intentionally generic: callers own the list of files (e.g. a contract
59
- * emitter's emitted output) and the naming convention (e.g. renaming
60
- * the destination contract to `end-contract.*` and the source contract
61
- * to `start-contract.*`).
62
- */
63
- async function copyFilesWithRename(destDir, files) {
64
- await mkdir(destDir, { recursive: true });
65
- for (const file of files) {
66
- if (basename(file.destName) !== file.destName) throw errorInvalidDestName(file.destName);
67
- await copyFile(file.sourcePath, join(destDir, file.destName));
68
- }
69
- }
70
- async function writeMigrationManifest(dir, manifest) {
71
- await writeFile(join(dir, MANIFEST_FILE), `${JSON.stringify(manifest, null, 2)}\n`);
72
- }
73
- async function writeMigrationOps(dir, ops) {
74
- await writeFile(join(dir, OPS_FILE), `${JSON.stringify(ops, null, 2)}\n`);
75
- }
76
- async function readMigrationPackage(dir) {
77
- const manifestPath = join(dir, MANIFEST_FILE);
78
- const opsPath = join(dir, OPS_FILE);
79
- let manifestRaw;
80
- try {
81
- manifestRaw = await readFile(manifestPath, "utf-8");
82
- } catch (error) {
83
- if (hasErrnoCode(error, "ENOENT")) throw errorMissingFile(MANIFEST_FILE, dir);
84
- throw error;
85
- }
86
- let opsRaw;
87
- try {
88
- opsRaw = await readFile(opsPath, "utf-8");
89
- } catch (error) {
90
- if (hasErrnoCode(error, "ENOENT")) throw errorMissingFile(OPS_FILE, dir);
91
- throw error;
92
- }
93
- let manifest;
94
- try {
95
- manifest = JSON.parse(manifestRaw);
96
- } catch (e) {
97
- throw errorInvalidJson(manifestPath, e instanceof Error ? e.message : String(e));
98
- }
99
- let ops;
100
- try {
101
- ops = JSON.parse(opsRaw);
102
- } catch (e) {
103
- throw errorInvalidJson(opsPath, e instanceof Error ? e.message : String(e));
104
- }
105
- validateManifest(manifest, manifestPath);
106
- validateOps(ops, opsPath);
107
- return {
108
- dirName: basename(dir),
109
- dirPath: dir,
110
- manifest,
111
- ops
112
- };
113
- }
114
- function validateManifest(manifest, filePath) {
115
- const result = MigrationManifestSchema(manifest);
116
- if (result instanceof type.errors) throw errorInvalidManifest(filePath, result.summary);
117
- }
118
- function validateOps(ops, filePath) {
119
- const result = MigrationOpsSchema(ops);
120
- if (result instanceof type.errors) throw errorInvalidManifest(filePath, result.summary);
121
- }
122
- async function readMigrationsDir(migrationsRoot) {
123
- let entries;
124
- try {
125
- entries = await readdir(migrationsRoot);
126
- } catch (error) {
127
- if (hasErrnoCode(error, "ENOENT")) return [];
128
- throw error;
129
- }
130
- const packages = [];
131
- for (const entry of entries.sort()) {
132
- const entryPath = join(migrationsRoot, entry);
133
- if (!(await stat(entryPath)).isDirectory()) continue;
134
- const manifestPath = join(entryPath, MANIFEST_FILE);
135
- try {
136
- await stat(manifestPath);
137
- } catch {
138
- continue;
139
- }
140
- packages.push(await readMigrationPackage(entryPath));
141
- }
142
- return packages;
143
- }
144
- function formatMigrationDirName(timestamp, slug) {
145
- const sanitized = slug.toLowerCase().replace(/[^a-z0-9]/g, "_").replace(/_+/g, "_").replace(/^_|_$/g, "");
146
- if (sanitized.length === 0) throw errorInvalidSlug(slug);
147
- const truncated = sanitized.slice(0, MAX_SLUG_LENGTH);
148
- return `${timestamp.getUTCFullYear()}${String(timestamp.getUTCMonth() + 1).padStart(2, "0")}${String(timestamp.getUTCDate()).padStart(2, "0")}T${String(timestamp.getUTCHours()).padStart(2, "0")}${String(timestamp.getUTCMinutes()).padStart(2, "0")}_${truncated}`;
149
- }
150
-
151
- //#endregion
152
- export { writeMigrationManifest as a, readMigrationsDir as i, formatMigrationDirName as n, writeMigrationOps as o, readMigrationPackage as r, writeMigrationPackage as s, copyFilesWithRename as t };
153
- //# sourceMappingURL=io-Cd6GLyjK.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"io-Cd6GLyjK.mjs","names":["manifestRaw: string","opsRaw: string","manifest: MigrationManifest","ops: MigrationOps","entries: string[]","packages: MigrationBundle[]"],"sources":["../src/io.ts"],"sourcesContent":["import { copyFile, mkdir, readdir, readFile, stat, writeFile } from 'node:fs/promises';\nimport { type } from 'arktype';\nimport { basename, dirname, join } from 'pathe';\nimport {\n errorDirectoryExists,\n errorInvalidDestName,\n errorInvalidJson,\n errorInvalidManifest,\n errorInvalidSlug,\n errorMissingFile,\n} from './errors';\nimport type { MigrationBundle, MigrationManifest, MigrationOps } from './types';\n\nconst MANIFEST_FILE = 'migration.json';\nconst OPS_FILE = 'ops.json';\nconst MAX_SLUG_LENGTH = 64;\n\nfunction hasErrnoCode(error: unknown, code: string): boolean {\n return error instanceof Error && (error as { code?: string }).code === code;\n}\n\nconst MigrationHintsSchema = type({\n used: 'string[]',\n applied: 'string[]',\n plannerVersion: 'string',\n});\n\nconst MigrationManifestSchema = type({\n from: 'string',\n to: 'string',\n migrationId: 'string',\n kind: \"'regular' | 'baseline'\",\n fromContract: 'object | null',\n toContract: 'object',\n hints: MigrationHintsSchema,\n labels: 'string[]',\n 'authorship?': type({\n 'author?': 'string',\n 'email?': 'string',\n }),\n 'signature?': type({\n keyId: 'string',\n value: 'string',\n }).or('null'),\n createdAt: 'string',\n});\n\nconst MigrationOpSchema = type({\n id: 'string',\n label: 'string',\n operationClass: \"'additive' | 'widening' | 'destructive' | 'data'\",\n});\n\n// Intentionally shallow: operation-specific payload validation is owned by planner/runner layers.\nconst MigrationOpsSchema = MigrationOpSchema.array();\n\nexport async function writeMigrationPackage(\n dir: string,\n manifest: MigrationManifest,\n ops: MigrationOps,\n): Promise<void> {\n await mkdir(dirname(dir), { recursive: true });\n\n try {\n await mkdir(dir);\n } catch (error) {\n if (hasErrnoCode(error, 'EEXIST')) {\n throw errorDirectoryExists(dir);\n }\n throw error;\n }\n\n await writeFile(join(dir, MANIFEST_FILE), JSON.stringify(manifest, null, 2), { flag: 'wx' });\n await writeFile(join(dir, OPS_FILE), JSON.stringify(ops, null, 2), { flag: 'wx' });\n}\n\n/**\n * Copy a list of files into `destDir`, optionally renaming each one.\n *\n * The destination directory is created (with `recursive: true`) if it\n * does not already exist. Each source path is copied byte-for-byte into\n * `destDir/<destName>`; missing sources throw `ENOENT`. The helper is\n * intentionally generic: callers own the list of files (e.g. a contract\n * emitter's emitted output) and the naming convention (e.g. renaming\n * the destination contract to `end-contract.*` and the source contract\n * to `start-contract.*`).\n */\nexport async function copyFilesWithRename(\n destDir: string,\n files: readonly { readonly sourcePath: string; readonly destName: string }[],\n): Promise<void> {\n await mkdir(destDir, { recursive: true });\n for (const file of files) {\n if (basename(file.destName) !== file.destName) {\n throw errorInvalidDestName(file.destName);\n }\n await copyFile(file.sourcePath, join(destDir, file.destName));\n }\n}\n\nexport async function writeMigrationManifest(\n dir: string,\n manifest: MigrationManifest,\n): Promise<void> {\n await writeFile(join(dir, MANIFEST_FILE), `${JSON.stringify(manifest, null, 2)}\\n`);\n}\n\nexport async function writeMigrationOps(dir: string, ops: MigrationOps): Promise<void> {\n await writeFile(join(dir, OPS_FILE), `${JSON.stringify(ops, null, 2)}\\n`);\n}\n\nexport async function readMigrationPackage(dir: string): Promise<MigrationBundle> {\n const manifestPath = join(dir, MANIFEST_FILE);\n const opsPath = join(dir, OPS_FILE);\n\n let manifestRaw: string;\n try {\n manifestRaw = await readFile(manifestPath, 'utf-8');\n } catch (error) {\n if (hasErrnoCode(error, 'ENOENT')) {\n throw errorMissingFile(MANIFEST_FILE, dir);\n }\n throw error;\n }\n\n let opsRaw: string;\n try {\n opsRaw = await readFile(opsPath, 'utf-8');\n } catch (error) {\n if (hasErrnoCode(error, 'ENOENT')) {\n throw errorMissingFile(OPS_FILE, dir);\n }\n throw error;\n }\n\n let manifest: MigrationManifest;\n try {\n manifest = JSON.parse(manifestRaw);\n } catch (e) {\n throw errorInvalidJson(manifestPath, e instanceof Error ? e.message : String(e));\n }\n\n let ops: MigrationOps;\n try {\n ops = JSON.parse(opsRaw);\n } catch (e) {\n throw errorInvalidJson(opsPath, e instanceof Error ? e.message : String(e));\n }\n\n validateManifest(manifest, manifestPath);\n validateOps(ops, opsPath);\n\n return {\n dirName: basename(dir),\n dirPath: dir,\n manifest,\n ops,\n };\n}\n\nfunction validateManifest(\n manifest: unknown,\n filePath: string,\n): asserts manifest is MigrationManifest {\n const result = MigrationManifestSchema(manifest);\n if (result instanceof type.errors) {\n throw errorInvalidManifest(filePath, result.summary);\n }\n}\n\nfunction validateOps(ops: unknown, filePath: string): asserts ops is MigrationOps {\n const result = MigrationOpsSchema(ops);\n if (result instanceof type.errors) {\n throw errorInvalidManifest(filePath, result.summary);\n }\n}\n\nexport async function readMigrationsDir(\n migrationsRoot: string,\n): Promise<readonly MigrationBundle[]> {\n let entries: string[];\n try {\n entries = await readdir(migrationsRoot);\n } catch (error) {\n if (hasErrnoCode(error, 'ENOENT')) {\n return [];\n }\n throw error;\n }\n\n const packages: MigrationBundle[] = [];\n\n for (const entry of entries.sort()) {\n const entryPath = join(migrationsRoot, entry);\n const entryStat = await stat(entryPath);\n if (!entryStat.isDirectory()) continue;\n\n const manifestPath = join(entryPath, MANIFEST_FILE);\n try {\n await stat(manifestPath);\n } catch {\n continue; // skip non-migration directories\n }\n\n packages.push(await readMigrationPackage(entryPath));\n }\n\n return packages;\n}\n\nexport function formatMigrationDirName(timestamp: Date, slug: string): string {\n const sanitized = slug\n .toLowerCase()\n .replace(/[^a-z0-9]/g, '_')\n .replace(/_+/g, '_')\n .replace(/^_|_$/g, '');\n\n if (sanitized.length === 0) {\n throw errorInvalidSlug(slug);\n }\n\n const truncated = sanitized.slice(0, MAX_SLUG_LENGTH);\n\n const y = timestamp.getUTCFullYear();\n const mo = String(timestamp.getUTCMonth() + 1).padStart(2, '0');\n const d = String(timestamp.getUTCDate()).padStart(2, '0');\n const h = String(timestamp.getUTCHours()).padStart(2, '0');\n const mi = String(timestamp.getUTCMinutes()).padStart(2, '0');\n\n return `${y}${mo}${d}T${h}${mi}_${truncated}`;\n}\n"],"mappings":"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"}