@prisma-next/cli 0.10.0 → 0.11.0-dev.2

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 (151) hide show
  1. package/README.md +1 -1
  2. package/dist/{cli-errors-CF60g2cG.mjs → cli-errors-Djtz98Vm.mjs} +3 -3
  3. package/dist/cli-errors-Djtz98Vm.mjs.map +1 -0
  4. package/dist/cli.mjs +400 -13
  5. package/dist/cli.mjs.map +1 -1
  6. package/dist/{client-Brv4qlfB.mjs → client-oXO2WCPD.mjs} +6 -5
  7. package/dist/client-oXO2WCPD.mjs.map +1 -0
  8. package/dist/{command-helpers-D3vL5yi8.mjs → command-helpers-DtavI0wJ.mjs} +109 -12
  9. package/dist/command-helpers-DtavI0wJ.mjs.map +1 -0
  10. package/dist/commands/contract-emit.mjs +1 -1
  11. package/dist/commands/contract-infer.mjs +1 -1
  12. package/dist/commands/db-init.d.mts.map +1 -1
  13. package/dist/commands/db-init.mjs +19 -20
  14. package/dist/commands/db-init.mjs.map +1 -1
  15. package/dist/commands/db-schema.mjs +6 -10
  16. package/dist/commands/db-schema.mjs.map +1 -1
  17. package/dist/commands/db-sign.mjs +7 -11
  18. package/dist/commands/db-sign.mjs.map +1 -1
  19. package/dist/commands/db-update.d.mts.map +1 -1
  20. package/dist/commands/db-update.mjs +16 -17
  21. package/dist/commands/db-update.mjs.map +1 -1
  22. package/dist/commands/db-verify.mjs +1 -1
  23. package/dist/commands/migrate.d.mts +1 -1
  24. package/dist/commands/migrate.mjs +7 -11
  25. package/dist/commands/migrate.mjs.map +1 -1
  26. package/dist/commands/migration-check.mjs +4 -7
  27. package/dist/commands/migration-check.mjs.map +1 -1
  28. package/dist/commands/migration-graph.d.mts +1 -1
  29. package/dist/commands/migration-graph.mjs +6 -10
  30. package/dist/commands/migration-graph.mjs.map +1 -1
  31. package/dist/commands/migration-list.mjs +5 -9
  32. package/dist/commands/migration-list.mjs.map +1 -1
  33. package/dist/commands/migration-log.d.mts.map +1 -1
  34. package/dist/commands/migration-log.mjs +7 -10
  35. package/dist/commands/migration-log.mjs.map +1 -1
  36. package/dist/commands/migration-new.mjs +6 -10
  37. package/dist/commands/migration-new.mjs.map +1 -1
  38. package/dist/commands/migration-plan.d.mts +1 -1
  39. package/dist/commands/migration-plan.mjs +1 -1
  40. package/dist/commands/migration-show.d.mts +1 -1
  41. package/dist/commands/migration-show.mjs +8 -12
  42. package/dist/commands/migration-show.mjs.map +1 -1
  43. package/dist/commands/migration-status.d.mts +1 -1
  44. package/dist/commands/migration-status.d.mts.map +1 -1
  45. package/dist/commands/migration-status.mjs +36 -14
  46. package/dist/commands/migration-status.mjs.map +1 -1
  47. package/dist/commands/ref.d.mts +1 -1
  48. package/dist/commands/ref.mjs +9 -19
  49. package/dist/commands/ref.mjs.map +1 -1
  50. package/dist/{contract-emit-iynA3BCA.mjs → contract-emit-bcrpT-wD.mjs} +3 -3
  51. package/dist/{contract-emit-iynA3BCA.mjs.map → contract-emit-bcrpT-wD.mjs.map} +1 -1
  52. package/dist/{contract-emit-C3STUIBg.mjs → contract-emit-uwT-Mj8-.mjs} +7 -12
  53. package/dist/contract-emit-uwT-Mj8-.mjs.map +1 -0
  54. package/dist/{contract-infer-Cnj8G1E2.mjs → contract-infer-pKkiCt7C.mjs} +9 -14
  55. package/dist/contract-infer-pKkiCt7C.mjs.map +1 -0
  56. package/dist/{contract-space-aggregate-loader-pAc8CDfY.mjs → contract-space-aggregate-loader-BmNQwlws.mjs} +2 -2
  57. package/dist/{contract-space-aggregate-loader-pAc8CDfY.mjs.map → contract-space-aggregate-loader-BmNQwlws.mjs.map} +1 -1
  58. package/dist/{db-verify-D7cyH_zz.mjs → db-verify-AoIUriL4.mjs} +9 -13
  59. package/dist/db-verify-AoIUriL4.mjs.map +1 -0
  60. package/dist/exports/control-api.d.mts +1 -1
  61. package/dist/exports/control-api.mjs +2 -2
  62. package/dist/exports/index.mjs +2 -2
  63. package/dist/exports/init-output.mjs +1 -1
  64. package/dist/{framework-components-xFLFpZUO.mjs → framework-components-65gOHkHB.mjs} +2 -2
  65. package/dist/{framework-components-xFLFpZUO.mjs.map → framework-components-65gOHkHB.mjs.map} +1 -1
  66. package/dist/{global-flags-DGmw6Kqg.d.mts → global-flags-CdE7M0d9.d.mts} +4 -1
  67. package/dist/global-flags-CdE7M0d9.d.mts.map +1 -0
  68. package/dist/{graph-render-eJDcLWny.mjs → graph-render-DJVv0_uf.mjs} +1 -1
  69. package/dist/{graph-render-eJDcLWny.mjs.map → graph-render-DJVv0_uf.mjs.map} +1 -1
  70. package/dist/{init-eh2z5Tl6.mjs → init-YX6lCJpG.mjs} +528 -627
  71. package/dist/init-YX6lCJpG.mjs.map +1 -0
  72. package/dist/{inspect-live-schema-CWLK_lgs.mjs → inspect-live-schema-LeWvkZVz.mjs} +4 -4
  73. package/dist/{inspect-live-schema-CWLK_lgs.mjs.map → inspect-live-schema-LeWvkZVz.mjs.map} +1 -1
  74. package/dist/{migration-command-scaffold-CmXXC1UZ.mjs → migration-command-scaffold-BtkunvFQ.mjs} +4 -4
  75. package/dist/{migration-command-scaffold-CmXXC1UZ.mjs.map → migration-command-scaffold-BtkunvFQ.mjs.map} +1 -1
  76. package/dist/{migration-plan-CHyUlBV0.mjs → migration-plan-C2jeH1J5.mjs} +8 -12
  77. package/dist/migration-plan-C2jeH1J5.mjs.map +1 -0
  78. package/dist/{migration-types-D2FW63pr.d.mts → migration-types-BXWvz12q.d.mts} +1 -1
  79. package/dist/{migration-types-D2FW63pr.d.mts.map → migration-types-BXWvz12q.d.mts.map} +1 -1
  80. package/dist/{migrations-DyUf5lTt.mjs → migrations-CwZMa1Ck.mjs} +2 -2
  81. package/dist/{migrations-DyUf5lTt.mjs.map → migrations-CwZMa1Ck.mjs.map} +1 -1
  82. package/dist/{output-B60Gw5fu.mjs → output-BlsrGMEF.mjs} +1 -1
  83. package/dist/{output-B60Gw5fu.mjs.map → output-BlsrGMEF.mjs.map} +1 -1
  84. package/dist/quick-reference-mongo.md +1 -1
  85. package/dist/quick-reference-postgres.md +1 -1
  86. package/dist/readme-mongo.md +35 -0
  87. package/dist/readme-postgres.md +34 -0
  88. package/dist/{terminal-ui-XtOQsqe9.mjs → terminal-ui-BiB_8KNo.mjs} +131 -24
  89. package/dist/terminal-ui-BiB_8KNo.mjs.map +1 -0
  90. package/dist/{types-0aS865QN.d.mts → types--CqjMdk0.d.mts} +2 -2
  91. package/dist/{types-0aS865QN.d.mts.map → types--CqjMdk0.d.mts.map} +1 -1
  92. package/dist/{verify-D7ypCCe6.mjs → verify-Bom75OYI.mjs} +2 -2
  93. package/dist/{verify-D7ypCCe6.mjs.map → verify-Bom75OYI.mjs.map} +1 -1
  94. package/package.json +19 -17
  95. package/src/cli.ts +42 -0
  96. package/src/commands/contract-emit.ts +4 -4
  97. package/src/commands/contract-infer.ts +7 -7
  98. package/src/commands/db-init.ts +13 -5
  99. package/src/commands/db-schema.ts +4 -4
  100. package/src/commands/db-sign.ts +4 -4
  101. package/src/commands/db-update.ts +13 -5
  102. package/src/commands/db-verify.ts +5 -5
  103. package/src/commands/init/detect-package-manager.ts +15 -0
  104. package/src/commands/init/errors.ts +33 -2
  105. package/src/commands/init/hygiene-gitattributes.ts +2 -2
  106. package/src/commands/init/index.ts +15 -6
  107. package/src/commands/init/init.ts +61 -32
  108. package/src/commands/init/inputs.ts +82 -5
  109. package/src/commands/init/output.ts +1 -1
  110. package/src/commands/init/{agent-skill-install.ts → skill-install.ts} +42 -31
  111. package/src/commands/init/templates/code-templates.ts +26 -24
  112. package/src/commands/init/templates/env.ts +8 -1
  113. package/src/commands/init/templates/quick-reference-mongo.md +1 -1
  114. package/src/commands/init/templates/quick-reference-postgres.md +1 -1
  115. package/src/commands/init/templates/readme-mongo.md +35 -0
  116. package/src/commands/init/templates/readme-postgres.md +34 -0
  117. package/src/commands/init/templates/readme.ts +62 -0
  118. package/src/commands/migrate.ts +4 -7
  119. package/src/commands/migration-check.ts +4 -4
  120. package/src/commands/migration-graph.ts +4 -4
  121. package/src/commands/migration-list.ts +4 -4
  122. package/src/commands/migration-log.ts +6 -5
  123. package/src/commands/migration-new.ts +4 -4
  124. package/src/commands/migration-plan.ts +4 -4
  125. package/src/commands/migration-show.ts +4 -4
  126. package/src/commands/migration-status.ts +49 -6
  127. package/src/commands/ref.ts +8 -8
  128. package/src/control-api/operations/apply-aggregate.ts +1 -0
  129. package/src/utils/cli-errors.ts +4 -0
  130. package/src/utils/command-helpers.ts +14 -6
  131. package/src/utils/global-flags.ts +105 -16
  132. package/src/utils/is-ci.ts +18 -0
  133. package/src/utils/telemetry.ts +141 -0
  134. package/src/utils/terminal-ui.ts +44 -23
  135. package/dist/cli-errors-CF60g2cG.mjs.map +0 -1
  136. package/dist/client-Brv4qlfB.mjs.map +0 -1
  137. package/dist/command-helpers-D3vL5yi8.mjs.map +0 -1
  138. package/dist/contract-emit-C3STUIBg.mjs.map +0 -1
  139. package/dist/contract-infer-Cnj8G1E2.mjs.map +0 -1
  140. package/dist/db-verify-D7cyH_zz.mjs.map +0 -1
  141. package/dist/errors-Cw6kyTyV.mjs +0 -56
  142. package/dist/errors-Cw6kyTyV.mjs.map +0 -1
  143. package/dist/global-flags-DGmw6Kqg.d.mts.map +0 -1
  144. package/dist/helpers-eqdN8tH6.mjs +0 -25
  145. package/dist/helpers-eqdN8tH6.mjs.map +0 -1
  146. package/dist/init-eh2z5Tl6.mjs.map +0 -1
  147. package/dist/migration-plan-CHyUlBV0.mjs.map +0 -1
  148. package/dist/result-handler-Bm_6dDYg.mjs +0 -25
  149. package/dist/result-handler-Bm_6dDYg.mjs.map +0 -1
  150. package/dist/terminal-ui-XtOQsqe9.mjs.map +0 -1
  151. /package/dist/{cli-errors-DdcjVLJV.d.mts → cli-errors-Czmx92Zy.d.mts} +0 -0
@@ -1 +1 @@
1
- {"version":3,"file":"migration-status.mjs","names":[],"sources":["../../src/commands/migration-status.ts"],"sourcesContent":["import {\n createControlStack,\n type MigrationPlanOperation,\n} from '@prisma-next/framework-components/control';\nimport {\n type ContractMarkerRecordLike,\n graphWalkStrategy,\n} from '@prisma-next/migration-tools/aggregate';\nimport { EMPTY_CONTRACT_HASH } from '@prisma-next/migration-tools/constants';\nimport {\n errorNoInvariantPath,\n errorUnknownInvariant,\n MigrationToolsError,\n} from '@prisma-next/migration-tools/errors';\nimport type { MigrationEdge, MigrationGraph } from '@prisma-next/migration-tools/graph';\nimport {\n findPath,\n findPathWithDecision,\n findReachableLeaves,\n} from '@prisma-next/migration-tools/migration-graph';\nimport type { OnDiskMigrationPackage } from '@prisma-next/migration-tools/package';\nimport { parseContractRef } from '@prisma-next/migration-tools/ref-resolution';\nimport type { RefEntry, Refs } from '@prisma-next/migration-tools/refs';\nimport { readRefs } from '@prisma-next/migration-tools/refs';\nimport { ifDefined } from '@prisma-next/utils/defined';\nimport { notOk, ok, type Result } from '@prisma-next/utils/result';\nimport { cyan, dim, magenta, yellow } from 'colorette';\nimport { Command } from 'commander';\n\nimport { loadConfig } from '../config-loader';\nimport { createControlClient } from '../control-api/client';\nimport {\n type CliStructuredError,\n errorRuntime,\n errorUnexpected,\n mapMigrationToolsError,\n mapRefResolutionError,\n} from '../utils/cli-errors';\nimport {\n addGlobalOptions,\n collectDeclaredInvariants,\n loadMigrationPackages,\n maskConnectionUrl,\n readContractEnvelope,\n resolveMigrationPaths,\n setCommandDescriptions,\n setCommandExamples,\n setCommandSeeAlso,\n toPathDecisionResult,\n toStructuralEdge,\n} from '../utils/command-helpers';\nimport {\n type BuildAggregateInputs,\n buildContractSpaceAggregate,\n} from '../utils/contract-space-aggregate-loader';\nimport {\n type EdgeStatus,\n type EdgeStatusKind,\n migrationGraphToRenderInput,\n} from '../utils/formatters/graph-migration-mapper';\nimport {\n extractRelevantSubgraph,\n graphRenderer,\n isLinearGraph,\n} from '../utils/formatters/graph-render';\nimport { formatStyledHeader } from '../utils/formatters/styled';\nimport type { CommonCommandOptions } from '../utils/global-flags';\nimport { type GlobalFlags, parseGlobalFlags } from '../utils/global-flags';\nimport type { StatusDiagnostic, StatusRef } from '../utils/migration-types';\nimport { handleResult } from '../utils/result-handler';\nimport { TerminalUI } from '../utils/terminal-ui';\n\ninterface MigrationStatusOptions extends CommonCommandOptions {\n readonly db?: string;\n readonly config?: string;\n readonly to?: string;\n readonly from?: string;\n}\n\nexport interface MigrationStatusEntry {\n readonly dirName: string;\n readonly from: string;\n readonly to: string;\n readonly migrationHash: string;\n readonly operationCount: number;\n readonly operationSummary: string;\n readonly hasDestructive: boolean;\n readonly status: EdgeStatusKind | 'unknown';\n}\n\n/**\n * Per-space status row in the aggregate-shaped status output.\n *\n * Surfaces, for each contract space:\n *\n * - `headHash`: the on-disk head ref's hash (where the space is going).\n * - `markerHash`: the live marker hash for the space, or null if no\n * marker has been written yet (greenfield, or pre-`migrate`).\n * - `pendingCount`: number of migration edges between marker and head.\n * Computed via {@link graphWalkStrategy}; 0 means the space is\n * already at head.\n * - `status`: convenience tag the formatter uses to pick a glyph.\n * `'never-planned'` is reserved for spaces with non-empty head but\n * no on-disk migrations — which shouldn't happen if the loader's\n * integrity check passes.\n *\n * Online-only fields (`markerHash`, `status`) are absent when the\n * command runs without a database connection.\n */\nexport interface MigrationStatusSpaceEntry {\n readonly spaceId: string;\n readonly kind: 'app' | 'extension';\n readonly headHash: string;\n readonly markerHash?: string | null;\n readonly pendingCount?: number;\n readonly status?: 'up-to-date' | 'pending' | 'no-marker' | 'never-planned' | 'unreachable';\n}\n\n/**\n * Sum per-space `pendingCount` into a cross-space total, but only when\n * every loaded space reports a defined `pendingCount`. Returns\n * `undefined` if any space is on the marker-unknown / offline path\n * (where `pendingCount` is intentionally absent), so JSON consumers can\n * distinguish \"no pending\" from \"unknown\".\n */\nexport function computeTotalPendingAcrossSpaces(\n spaces: readonly MigrationStatusSpaceEntry[],\n): number | undefined {\n if (spaces.length === 0) return undefined;\n let total = 0;\n for (const s of spaces) {\n if (s.pendingCount === undefined) return undefined;\n total += s.pendingCount;\n }\n return total;\n}\n\nexport type { StatusDiagnostic, StatusRef } from '../utils/migration-types';\n\nexport interface MigrationStatusResult {\n readonly ok: true;\n readonly mode: 'online' | 'offline';\n readonly migrations: readonly MigrationStatusEntry[];\n readonly markerHash?: string;\n readonly targetHash: string;\n readonly contractHash: string;\n readonly refs?: readonly StatusRef[];\n /** Required invariants from the active ref, sorted ascending. Always present (`[]` when no `--ref` or the ref declares none) — knowable offline. */\n readonly requiredInvariants: readonly string[];\n /**\n * Invariants the marker has applied at least once, intersected with\n * `requiredInvariants` for display relevance. JSON consumers see only the\n * subset overlapping the active ref's required set — the full unfiltered\n * marker invariant list lives on `marker.invariants` (control plane) and\n * is not surfaced here. Present only in `mode === 'online'`; absent when\n * offline (the marker is unknown, not empty).\n */\n readonly appliedInvariants?: readonly string[];\n /** required − applied. Present only in `mode === 'online'`; absent when offline. */\n readonly missingInvariants?: readonly string[];\n readonly pathDecision?: {\n readonly fromHash: string;\n readonly toHash: string;\n readonly alternativeCount: number;\n readonly tieBreakReasons: readonly string[];\n readonly refName?: string;\n readonly requiredInvariants: readonly string[];\n readonly satisfiedInvariants: readonly string[];\n readonly selectedPath: readonly {\n readonly dirName: string;\n readonly migrationHash: string;\n readonly from: string;\n readonly to: string;\n readonly invariants: readonly string[];\n }[];\n };\n readonly summary: string;\n readonly diagnostics: readonly StatusDiagnostic[];\n /**\n * Aggregate enumeration of every on-disk contract space (app +\n * extensions), in canonical schedule order (extensions\n * alphabetically, then app). Present whenever the aggregate loader\n * succeeded; absent in early-error returns (e.g. unreadable\n * migrations directory) where the existing diagnostics already\n * surface the failure.\n *\n * The legacy top-level fields (`migrations`, `markerHash`,\n * `targetHash`, `pathDecision`, …) describe the **app member**\n * specifically — back-compat with single-space callers. Per-space\n * detail for extension members lives only on this list.\n */\n readonly spaces?: readonly MigrationStatusSpaceEntry[];\n /** Cross-space pending-migration total (sum of `spaces[].pendingCount`). Present when `spaces` is. */\n readonly totalPendingAcrossSpaces?: number;\n readonly graph?: MigrationGraph;\n readonly bundles?: readonly OnDiskMigrationPackage[];\n readonly edgeStatuses?: readonly EdgeStatus[];\n readonly activeRefHash?: string;\n readonly activeRefName?: string;\n readonly diverged?: boolean;\n}\n\nfunction summarizeOps(ops: readonly MigrationPlanOperation[]): {\n summary: string;\n hasDestructive: boolean;\n} {\n if (ops.length === 0) return { summary: '0 ops', hasDestructive: false };\n\n const classes = new Map<string, number>();\n for (const op of ops) {\n classes.set(op.operationClass, (classes.get(op.operationClass) ?? 0) + 1);\n }\n\n const hasDestructive = classes.has('destructive');\n const count = ops.length;\n const noun = count === 1 ? 'op' : 'ops';\n\n if (classes.size === 1) {\n const cls = [...classes.keys()][0]!;\n return { summary: `${count} ${noun} (all ${cls})`, hasDestructive };\n }\n\n const destructiveCount = classes.get('destructive');\n if (destructiveCount) {\n return { summary: `${count} ${noun} (${destructiveCount} destructive)`, hasDestructive };\n }\n\n const parts = [...classes.entries()].map(([cls, n]) => `${n} ${cls}`);\n return { summary: `${count} ${noun} (${parts.join(', ')})`, hasDestructive };\n}\n\n/**\n * Derive per-edge status across the full graph using path analysis.\n *\n * - **applied**: edge is on the path from root to the DB marker\n * - **pending**: edge is on the path from the DB marker to the target\n * (and the marker is reachable from root, i.e. it's on the same branch)\n * - **unreachable**: edge is on the path from root to the target but the DB\n * marker is on a different branch — `apply` can't reach these edges\n * without the DB first moving to this branch\n *\n * Returns statuses only for edges that have a known status (skips offline\n * and edges not on any relevant path).\n *\n * @internal Exported for testing only.\n */\nexport function deriveEdgeStatuses(\n graph: MigrationGraph,\n targetHash: string,\n contractHash: string,\n markerHash: string | undefined,\n mode: 'online' | 'offline',\n): EdgeStatus[] {\n if (mode === 'offline') return [];\n\n const edgeKey = (e: MigrationEdge) => `${e.from}\\0${e.to}`;\n\n // No marker = empty DB — treat root as the marker (nothing applied, everything pending)\n const effectiveMarker = markerHash ?? EMPTY_CONTRACT_HASH;\n\n const appliedPath =\n markerHash !== undefined ? findPath(graph, EMPTY_CONTRACT_HASH, markerHash) : null;\n\n const pendingPath = findPath(graph, effectiveMarker, targetHash);\n const targetPath = findPath(graph, EMPTY_CONTRACT_HASH, targetHash);\n\n const statuses: EdgeStatus[] = [];\n const assignedKeys = new Set<string>();\n\n // Applied edges (root → marker)\n if (appliedPath) {\n for (const e of appliedPath) {\n assignedKeys.add(edgeKey(e));\n statuses.push({ dirName: e.dirName, status: 'applied' });\n }\n }\n\n // Pending edges (marker → target)\n if (pendingPath) {\n for (const e of pendingPath) {\n assignedKeys.add(edgeKey(e));\n statuses.push({ dirName: e.dirName, status: 'pending' });\n }\n }\n\n // Pending edges beyond the target: target → contract (when target is a ref\n // and the contract is reachable from it)\n if (\n contractHash !== EMPTY_CONTRACT_HASH &&\n contractHash !== targetHash &&\n graph.nodes.has(contractHash)\n ) {\n const beyondTarget = findPath(graph, targetHash, contractHash);\n if (beyondTarget) {\n for (const e of beyondTarget) {\n if (!assignedKeys.has(edgeKey(e))) {\n assignedKeys.add(edgeKey(e));\n statuses.push({ dirName: e.dirName, status: 'pending' });\n }\n }\n }\n }\n\n // Unreachable edges: on the path from root to the target but neither applied\n // nor pending. This covers two cases:\n // 1. Marker can't reach target at all (different branch entirely)\n // 2. Marker reaches target via a different route, leaving some root→target\n // edges orphaned (e.g. a fork where one branch was applied and apply\n // will continue through the other)\n if (targetPath) {\n for (const e of targetPath) {\n if (!assignedKeys.has(edgeKey(e))) {\n statuses.push({ dirName: e.dirName, status: 'unreachable' });\n }\n }\n }\n\n return statuses;\n}\n\n/**\n * @param mode — 'online' if we connected to the database, 'offline' otherwise\n * @param markerHash — the marker hash from the database, or undefined if no marker row / offline\n */\nfunction buildMigrationEntries(\n chain: readonly MigrationEdge[],\n packages: readonly OnDiskMigrationPackage[],\n mode: 'online' | 'offline',\n markerHash: string | undefined,\n edgeStatuses?: readonly EdgeStatus[],\n): MigrationStatusEntry[] {\n const pkgByDirName = new Map(packages.map((p) => [p.dirName, p]));\n const statusByDirName = edgeStatuses\n ? new Map(edgeStatuses.map((e) => [e.dirName, e.status]))\n : undefined;\n\n const markerInChain = markerHash === undefined || chain.some((e) => e.to === markerHash);\n\n const entries: MigrationStatusEntry[] = [];\n let reachedMarker = mode === 'online' && markerHash === undefined;\n\n for (const migration of chain) {\n const pkg = pkgByDirName.get(migration.dirName);\n const ops = (pkg?.ops ?? []) as readonly MigrationPlanOperation[];\n const { summary, hasDestructive } = summarizeOps(ops);\n\n let status: EdgeStatusKind | 'unknown';\n const edgeStatus = statusByDirName?.get(migration.dirName);\n if (edgeStatus) {\n status = edgeStatus;\n } else if (mode === 'offline' || !markerInChain) {\n status = 'unknown';\n } else if (reachedMarker) {\n status = 'pending';\n } else {\n status = 'applied';\n }\n\n entries.push({\n dirName: migration.dirName,\n from: migration.from,\n to: migration.to,\n migrationHash: migration.migrationHash,\n operationCount: ops.length,\n operationSummary: summary,\n hasDestructive,\n status,\n });\n\n if (!reachedMarker && migration.to === markerHash) {\n reachedMarker = true;\n }\n }\n\n return entries;\n}\n\n/**\n * Resolve the migration chain to display in status output.\n *\n * When offline or the marker is at EMPTY, the chain is simply the shortest\n * path from EMPTY to the target — all structural paths are equivalent per\n * the spec, so the deterministic shortest path is the canonical display.\n *\n * When online with a non-empty marker, the chain routes *through* the marker:\n * EMPTY→marker (applied history) + marker→target (pending edges). This ensures\n * the displayed chain includes the marker node so applied/pending status is\n * correct. Without this, BFS from EMPTY to target could pick a shortest path\n * that bypasses the marker entirely (e.g. in a diamond graph), causing the\n * marker to appear \"diverged\" when it isn't.\n */\nfunction resolveDisplayChain(\n graph: MigrationGraph,\n targetHash: string,\n markerHash: string | undefined,\n): readonly MigrationEdge[] | null {\n if (markerHash === undefined) {\n return findPath(graph, EMPTY_CONTRACT_HASH, targetHash);\n }\n\n const toMarker = findPath(graph, EMPTY_CONTRACT_HASH, markerHash);\n // Marker unreachable from EMPTY — show the target chain anyway.\n // The caller detects this via markerInChain and emits a divergence diagnostic.\n if (!toMarker) return findPath(graph, EMPTY_CONTRACT_HASH, targetHash);\n\n if (markerHash === targetHash) return toMarker;\n\n const fromMarker = findPath(graph, markerHash, targetHash);\n if (fromMarker) return [...toMarker, ...fromMarker];\n\n // Marker is ahead of target (or on a disconnected branch).\n // Try the inverse: target→marker. If it succeeds, the marker is ahead —\n // show the full chain from EMPTY through the target and on to the marker.\n const toTarget = findPath(graph, EMPTY_CONTRACT_HASH, targetHash);\n if (!toTarget) return null;\n\n const targetToMarker = findPath(graph, targetHash, markerHash);\n if (targetToMarker) return [...toTarget, ...targetToMarker];\n\n // Genuinely disconnected — show EMPTY→target; caller handles divergence diagnostic.\n return toTarget;\n}\n\n/**\n * Build the aggregate enumeration of contract spaces for the status\n * output. Loads the aggregate from disk (lossy on failure — extension\n * spaces are simply omitted, the existing single-space app behaviour\n * keeps working), reads per-space marker rows when online, and uses\n * {@link graphWalkStrategy} to compute each space's pending count.\n *\n * Sub-spec § `migration status` semantics — the aggregate-walking\n * version reports per-space marker + pending state alongside the\n * cross-space totals.\n */\nexport async function loadAggregateStatusSpaces(args: {\n readonly targetId: string;\n readonly migrationsDir: string;\n readonly appContractRaw: unknown;\n readonly extensionPacks: BuildAggregateInputs<string, string>['extensionPacks'];\n readonly deserializeContract: BuildAggregateInputs<string, string>['deserializeContract'];\n readonly markersBySpace: ReadonlyMap<string, ContractMarkerRecordLike> | null;\n}): Promise<readonly MigrationStatusSpaceEntry[]> {\n const loadInputs: BuildAggregateInputs<string, string> = {\n targetId: args.targetId,\n migrationsDir: args.migrationsDir,\n appContract: args.deserializeContract(args.appContractRaw),\n extensionPacks: args.extensionPacks,\n deserializeContract: args.deserializeContract,\n };\n\n const loaded = await buildContractSpaceAggregate(loadInputs);\n if (!loaded.ok) {\n // Loader failure (drift, layout violation, etc.) — surfacing it\n // as a status diagnostic would duplicate `migration plan`'s job.\n // The single-space app pipeline still runs; extensions are simply\n // not enumerated.\n return [];\n }\n const aggregate = loaded.value;\n\n const orderedMembers = [...aggregate.extensions, aggregate.app];\n const rows: MigrationStatusSpaceEntry[] = [];\n for (const member of orderedMembers) {\n const liveMarker = args.markersBySpace?.get(member.spaceId) ?? null;\n const isApp = member.spaceId === aggregate.app.spaceId;\n\n if (member.migrations.graph.nodes.size === 0) {\n rows.push({\n spaceId: member.spaceId,\n kind: isApp ? 'app' : 'extension',\n headHash: member.headRef.hash,\n ...(args.markersBySpace !== null\n ? {\n markerHash: liveMarker?.storageHash ?? null,\n status: member.headRef.hash === EMPTY_CONTRACT_HASH ? 'up-to-date' : 'never-planned',\n pendingCount: 0,\n }\n : {}),\n });\n continue;\n }\n\n if (args.markersBySpace === null) {\n rows.push({\n spaceId: member.spaceId,\n kind: isApp ? 'app' : 'extension',\n headHash: member.headRef.hash,\n });\n continue;\n }\n\n const walked = graphWalkStrategy({\n aggregateTargetId: aggregate.targetId,\n member,\n currentMarker: liveMarker,\n });\n let pendingCount = 0;\n let status: MigrationStatusSpaceEntry['status'];\n if (walked.kind === 'ok') {\n // Count pending *migrations* (graph edges), not operations: a\n // single authored migration that lowers to N ops or zero ops\n // both count as exactly one pending unit of work for the user.\n pendingCount = walked.result.migrationEdges?.length ?? 0;\n if (liveMarker === null) {\n status = pendingCount === 0 ? 'no-marker' : 'pending';\n } else {\n status = pendingCount === 0 ? 'up-to-date' : 'pending';\n }\n } else {\n status = 'unreachable';\n }\n\n rows.push({\n spaceId: member.spaceId,\n kind: isApp ? 'app' : 'extension',\n headHash: member.headRef.hash,\n markerHash: liveMarker?.storageHash ?? null,\n pendingCount,\n ...(status ? { status } : {}),\n });\n }\n return rows;\n}\n\n/**\n * Read the raw contract.json bytes from disk for the aggregate\n * loader. Returns `null` if the file is missing or unparseable —\n * the existing `readContractEnvelope` path will report the same\n * problem via a status diagnostic, no need to double-surface.\n */\nasync function loadContractRawSafely(config: {\n contract?: { output?: string };\n}): Promise<unknown | null> {\n try {\n const path = (await import('../utils/command-helpers')).resolveContractPath(config);\n const raw = await (await import('node:fs/promises')).readFile(path, 'utf-8');\n return JSON.parse(raw) as unknown;\n } catch {\n return null;\n }\n}\n\nasync function executeMigrationStatusCommand(\n options: MigrationStatusOptions,\n flags: GlobalFlags,\n ui: TerminalUI,\n): Promise<Result<MigrationStatusResult, CliStructuredError>> {\n const config = await loadConfig(options.config);\n const { configPath, appMigrationsDir, appMigrationsRelative, migrationsDir, refsDir } =\n resolveMigrationPaths(options.config, config);\n\n const dbConnection = options.db ?? config.db?.connection;\n const hasDriver = !!config.driver;\n\n let activeRefName: string | undefined;\n let activeRefHash: string | undefined;\n let activeRefEntry: RefEntry | undefined;\n let allRefs: Refs = {};\n try {\n allRefs = await readRefs(refsDir);\n } catch (error) {\n if (MigrationToolsError.is(error)) {\n return notOk(mapMigrationToolsError(error));\n }\n throw error;\n }\n\n let fromOverrideHash: string | undefined;\n\n if (options.to || options.from) {\n try {\n const { graph: earlyGraph } = await loadMigrationPackages(appMigrationsDir);\n\n if (options.to) {\n const refResult = parseContractRef(options.to, { graph: earlyGraph, refs: allRefs });\n if (!refResult.ok) {\n return notOk(mapRefResolutionError(refResult.failure));\n }\n activeRefHash = refResult.value.hash;\n if (refResult.value.provenance.kind === 'ref') {\n const resolvedRefName = refResult.value.provenance.refName;\n activeRefName = resolvedRefName;\n activeRefEntry = allRefs[resolvedRefName];\n }\n }\n\n if (options.from) {\n const fromResult = parseContractRef(options.from, { graph: earlyGraph, refs: allRefs });\n if (!fromResult.ok) {\n return notOk(mapRefResolutionError(fromResult.failure));\n }\n fromOverrideHash = fromResult.value.hash;\n }\n } catch (error) {\n if (MigrationToolsError.is(error)) {\n return notOk(mapMigrationToolsError(error));\n }\n throw error;\n }\n }\n\n const requiredInvariants: readonly string[] = [...(activeRefEntry?.invariants ?? [])].sort();\n\n const statusRefs: StatusRef[] = Object.entries(allRefs).map(([name, entry]) => ({\n name,\n hash: entry.hash,\n active: name === activeRefName,\n }));\n\n if (!flags.json && !flags.quiet) {\n const details: Array<{ label: string; value: string }> = [\n { label: 'config', value: configPath },\n { label: 'migrations', value: appMigrationsRelative },\n ];\n if (dbConnection && hasDriver) {\n details.push({ label: 'database', value: maskConnectionUrl(String(dbConnection)) });\n }\n if (activeRefName) {\n details.push({ label: 'ref', value: activeRefName });\n }\n if (options.from) {\n details.push({ label: 'from', value: options.from });\n }\n if (activeRefEntry && activeRefEntry.invariants.length > 0) {\n details.push({\n label: 'required',\n value: formatInvariantList(activeRefEntry.invariants),\n });\n }\n const header = formatStyledHeader({\n command: 'migration status',\n description: 'Show migration history and applied status',\n details,\n flags,\n });\n ui.stderr(header);\n }\n\n const diagnostics: StatusDiagnostic[] = [];\n let contractHash: string = EMPTY_CONTRACT_HASH;\n try {\n const envelope = await readContractEnvelope(config);\n contractHash = envelope.storageHash;\n } catch (error) {\n diagnostics.push({\n code: 'CONTRACT.UNREADABLE',\n severity: 'warn',\n message: `Could not read contract: ${error instanceof Error ? error.message : 'unknown error'}`,\n hints: [\"Run 'prisma-next contract emit' to generate a valid contract\"],\n });\n }\n\n let bundles: readonly OnDiskMigrationPackage[];\n let graph: MigrationGraph;\n try {\n ({ bundles, graph } = await loadMigrationPackages(appMigrationsDir));\n } catch (error) {\n if (MigrationToolsError.is(error)) {\n return notOk(mapMigrationToolsError(error));\n }\n return notOk(\n errorUnexpected(error instanceof Error ? error.message : String(error), {\n why: `Failed to read migrations directory: ${error instanceof Error ? error.message : String(error)}`,\n }),\n );\n }\n\n if (bundles.length === 0) {\n if (contractHash !== EMPTY_CONTRACT_HASH) {\n diagnostics.push({\n code: 'CONTRACT.AHEAD',\n severity: 'warn',\n message: 'No migration exists for the current contract',\n hints: [\n \"Run 'prisma-next migration plan' to generate a migration for the current contract\",\n ],\n });\n }\n return ok({\n ok: true,\n mode: dbConnection && hasDriver ? 'online' : 'offline',\n migrations: [],\n targetHash: EMPTY_CONTRACT_HASH,\n contractHash,\n summary: 'No migrations found',\n diagnostics,\n requiredInvariants,\n });\n }\n\n let targetHash: string | undefined;\n\n if (activeRefHash) {\n targetHash = activeRefHash;\n } else if (graph.nodes.has(contractHash)) {\n targetHash = contractHash;\n } else {\n const leaves = findReachableLeaves(graph, EMPTY_CONTRACT_HASH);\n if (leaves.length === 1) {\n targetHash = leaves[0];\n } else {\n diagnostics.push({\n code: 'MIGRATION.DIVERGED',\n severity: 'warn',\n message: 'There are multiple valid migration paths — you must select a target',\n hints: [\n \"Use '--to <contract>' to select a target\",\n \"Or 'prisma-next ref set <name> <hash>' to create one\",\n ],\n });\n }\n }\n\n let markerHash: string | undefined;\n let markerInvariants: readonly string[] = [];\n let mode: 'online' | 'offline' = 'offline';\n let allMarkers: ReadonlyMap<string, ContractMarkerRecordLike> | null = null;\n\n if (dbConnection && hasDriver) {\n const client = createControlClient({\n family: config.family,\n target: config.target,\n adapter: config.adapter,\n driver: config.driver,\n extensionPacks: config.extensionPacks ?? [],\n });\n try {\n await client.connect(dbConnection);\n const marker = await client.readMarker();\n markerHash = marker?.storageHash;\n markerInvariants = marker?.invariants ?? [];\n mode = 'online';\n // Read every space's marker so the aggregate enumeration can\n // surface per-space marker state. `readAllMarkers` mirrors what\n // `db init` / `db update` already use to drive the multi-space\n // planner; here it powers the aggregate status output.\n //\n // Probe for the method first so we only swallow the\n // unsupported-method case: older family instances may not\n // implement `readAllMarkers` (per-space enumeration then falls\n // back to \"marker unknown\"). Real query / runtime errors from\n // an instance that *does* expose the method must propagate up\n // — otherwise transient DB failures would silently degrade\n // status to \"markers unknown\".\n if (typeof client.readAllMarkers === 'function') {\n allMarkers = await client.readAllMarkers();\n } else {\n // Leaving `allMarkers` as `null` signals \"unknown\" to the\n // aggregate loader (an empty `Map` would instead mean \"every\n // space has no marker\", which is a different condition).\n allMarkers = null;\n }\n } catch {\n if (!flags.json && !flags.quiet) {\n ui.warn('Could not connect to database — showing offline status');\n }\n } finally {\n await client.close();\n }\n }\n\n if (fromOverrideHash !== undefined) {\n markerHash = fromOverrideHash;\n mode = 'offline';\n allMarkers = null;\n }\n\n // Build the aggregate enumeration of contract spaces. Lossy on\n // failure (extensions are simply omitted) so the existing\n // single-space app pipeline below still runs even if extensions\n // can't be loaded — a strict failure here would degrade the\n // load-bearing app-space output for unrelated reasons.\n const contractRawForAggregate = await loadContractRawSafely(config);\n let aggregateSpaces: readonly MigrationStatusSpaceEntry[] = [];\n if (contractRawForAggregate !== null) {\n // The aggregate loader needs a typed-Contract producer. Build a\n // real control stack so `deserializeContract` runs against a fully\n // composed family instance — descriptors that read stack members\n // during construction (e.g. codec lookups) get a consistent view.\n const stack = createControlStack(config);\n const familyInstance = config.family.create(stack);\n try {\n aggregateSpaces = await loadAggregateStatusSpaces({\n targetId: config.target.targetId,\n migrationsDir,\n appContractRaw: contractRawForAggregate,\n extensionPacks: config.extensionPacks ?? [],\n deserializeContract: (json: unknown) => familyInstance.deserializeContract(json),\n markersBySpace: allMarkers,\n });\n } catch {\n // Loader failure short-circuits silently — the existing\n // single-space app pipeline below still runs.\n aggregateSpaces = [];\n }\n }\n const totalPendingAcrossSpaces = computeTotalPendingAcrossSpaces(aggregateSpaces);\n\n // Pre-check unknown invariants. Online: union the graph's declared\n // invariants with the marker's recorded set so a retired-but-applied\n // invariant doesn't surface as MIGRATION.UNKNOWN_INVARIANT — apply would\n // route fine because marker subtraction empties `effectiveRequired`.\n // Offline: keep the check graph-strict (the marker is unknown, and a\n // missing declarer is the dominant signal we can offer).\n if (activeRefEntry && activeRefEntry.invariants.length > 0) {\n const declared = collectDeclaredInvariants(graph);\n const known = new Set<string>(declared);\n if (mode === 'online') {\n for (const id of markerInvariants) known.add(id);\n }\n const unknown = activeRefEntry.invariants.filter((id) => !known.has(id));\n if (unknown.length > 0) {\n return notOk(\n mapMigrationToolsError(\n errorUnknownInvariant({\n ...ifDefined('refName', activeRefName),\n unknown,\n declared: [...declared].sort(),\n }),\n ),\n );\n }\n }\n\n // Marker exists but is not in the migration graph and doesn't match the\n // contract hash. The DB is at an unknown state relative to the graph.\n // Bail out early with a clear diagnostic instead of rendering a confusing\n // graph with no statuses.\n //\n // When marker === contract (both off-graph), the DB matches the current\n // contract — proceed normally; the detached contract node will carry both\n // the db and contract markers.\n if (\n mode === 'online' &&\n markerHash !== undefined &&\n !graph.nodes.has(markerHash) &&\n markerHash !== contractHash\n ) {\n const hints: string[] = [];\n if (graph.nodes.has(contractHash)) {\n hints.push(\n \"Run 'prisma-next db sign' to overwrite the marker if the database already matches the contract\",\n \"Run 'prisma-next db update' to push the current contract to the database\",\n \"Run 'prisma-next contract infer' to make your contract match the database\",\n \"Run 'prisma-next db verify' to inspect the database state\",\n );\n } else {\n hints.push(\n \"Run 'prisma-next db update' to push the current contract to the database\",\n \"Run 'prisma-next contract infer' to make your contract match the database\",\n \"Run 'prisma-next db verify' to inspect the database state\",\n );\n }\n diagnostics.push({\n code: 'MIGRATION.MARKER_NOT_IN_HISTORY',\n severity: 'warn',\n message:\n 'Database was updated outside the migration system (marker does not match any migration)',\n hints,\n });\n return ok({\n ok: true,\n mode,\n migrations: [],\n targetHash: EMPTY_CONTRACT_HASH,\n contractHash,\n summary: `${bundles.length} migration(s) on disk`,\n diagnostics,\n markerHash,\n requiredInvariants,\n ...(statusRefs.length > 0 ? { refs: statusRefs } : {}),\n });\n }\n\n if (mode === 'online' && markerHash === undefined) {\n diagnostics.push({\n code: 'MIGRATION.NO_MARKER',\n severity: 'warn',\n message: 'Database has not been initialized — no migration marker found',\n hints: [\"Run 'prisma-next migrate' to apply pending migrations\"],\n });\n }\n\n // Contract diagnostic — fires when no migration produces the current contract hash.\n // Suppressed when: (a) graph is diverged (MIGRATION.DIVERGED already guides the user),\n // (b) marker === contract and both off-graph (marker-not-in-graph diagnostic covers it).\n if (\n targetHash &&\n contractHash !== EMPTY_CONTRACT_HASH &&\n !graph.nodes.has(contractHash) &&\n markerHash !== contractHash\n ) {\n diagnostics.push({\n code: 'CONTRACT.AHEAD',\n severity: 'warn',\n message: 'Contract has changed since the last migration was planned',\n hints: [\"Run 'prisma-next migration plan' to generate a migration for the current contract\"],\n });\n }\n\n if (!targetHash) {\n return ok({\n ok: true,\n mode,\n migrations: [],\n targetHash: EMPTY_CONTRACT_HASH,\n contractHash,\n summary: `${bundles.length} migration(s) on disk`,\n diagnostics,\n ...ifDefined('markerHash', markerHash),\n requiredInvariants,\n ...(statusRefs.length > 0 ? { refs: statusRefs } : {}),\n graph,\n bundles,\n diverged: true,\n });\n }\n\n const chain = resolveDisplayChain(graph, targetHash, markerHash);\n\n if (!chain) {\n return notOk(\n errorRuntime('Cannot reconstruct migration history', {\n why: `No path from ${EMPTY_CONTRACT_HASH} to target ${targetHash}`,\n fix: 'The migration history may have gaps. Check the migrations directory for missing or corrupted packages.',\n }),\n );\n }\n\n const edgeStatuses = deriveEdgeStatuses(graph, targetHash, contractHash, markerHash, mode);\n const entries = buildMigrationEntries(chain, bundles, mode, markerHash, edgeStatuses);\n\n const pendingCount = edgeStatuses.filter((e) => e.status === 'pending').length;\n const appliedCount = edgeStatuses.filter((e) => e.status === 'applied').length;\n\n let appliedInvariants: readonly string[] | undefined;\n let missingInvariants: readonly string[] | undefined;\n let effectiveRequired = new Set<string>();\n if (mode === 'online') {\n // Mirrors `migrate.ts`: compute `effectiveRequired = required −\n // marker.invariants` directly, then derive the display fields from it.\n // `appliedInvariants` is the intersection (`required ∩ marker`), which\n // is what JSON consumers see for the active ref; the unfiltered set\n // lives on `marker.invariants`.\n const markerSet = new Set(markerInvariants);\n effectiveRequired = new Set(requiredInvariants.filter((id) => !markerSet.has(id)));\n appliedInvariants = requiredInvariants.filter((id) => markerSet.has(id));\n missingInvariants = [...effectiveRequired].sort();\n }\n\n // The marker can match the structural target while still missing required\n // invariants — for example, a self-edge that provides X, applied via a ref\n // declaring X. `pendingCount` (structural) says zero in that case but\n // `effectiveRequired` is non-empty, so up-to-date messaging would mislead.\n const hasInvariantWork = effectiveRequired.size > 0;\n const missingList = [...effectiveRequired].sort().join(', ');\n\n let summary: string;\n if (mode === 'online') {\n if (markerHash !== undefined && !graph.nodes.has(markerHash) && markerHash === contractHash) {\n summary = `${bundles.length} migration(s) on disk`;\n } else if (activeRefHash && activeRefName && markerHash !== undefined) {\n const distance = summarizeRefDistance(graph, markerHash, activeRefHash, activeRefName);\n summary = hasInvariantWork ? `${distance} — missing invariant(s): ${missingList}` : distance;\n } else if (pendingCount === 0 && !hasInvariantWork) {\n summary = `Database is up to date (${appliedCount} migration${appliedCount !== 1 ? 's' : ''} applied)`;\n } else if (pendingCount === 0 && hasInvariantWork) {\n summary = `Missing invariant(s): ${missingList} — run 'prisma-next migrate --to ${activeRefName ?? '<ref>'}' to apply`;\n } else if (markerHash === undefined) {\n summary = `${pendingCount} pending migration(s) — database has no marker`;\n } else {\n summary = `${pendingCount} pending migration(s) — run 'prisma-next migrate' to apply`;\n }\n } else {\n summary = `${entries.length} migration(s) on disk`;\n }\n\n let pathDecision: MigrationStatusResult['pathDecision'];\n let routingUnreachable = false;\n if (mode === 'online') {\n const originHash = markerHash ?? EMPTY_CONTRACT_HASH;\n const outcome = findPathWithDecision(graph, originHash, targetHash, {\n ...ifDefined('refName', activeRefName),\n required: effectiveRequired,\n });\n if (outcome.kind === 'ok') {\n pathDecision = toPathDecisionResult(outcome.decision);\n } else if (outcome.kind === 'unsatisfiable') {\n return notOk(\n mapMigrationToolsError(\n errorNoInvariantPath({\n ...ifDefined('refName', activeRefName),\n required: [...effectiveRequired].sort(),\n missing: outcome.missing,\n structuralPath: outcome.structuralPath.map(toStructuralEdge),\n }),\n ),\n );\n } else {\n // outcome.kind === 'unreachable' — origin (marker) has no structural\n // path to the active target. `pendingCount` and `hasInvariantWork`\n // both report zero in this case, but emitting MIGRATION.UP_TO_DATE\n // would be wrong: the database simply cannot reach the requested\n // ref/contract from its current state. Suppress UP_TO_DATE below.\n routingUnreachable = true;\n }\n }\n\n if (mode === 'online') {\n if (markerHash !== undefined && !graph.nodes.has(markerHash) && markerHash === contractHash) {\n diagnostics.push({\n code: 'MIGRATION.MARKER_NOT_IN_HISTORY',\n severity: 'warn',\n message: 'Database matches the current contract but was updated directly (not via migrate)',\n hints: [\"Run 'prisma-next migration plan' to plan a migration to your current contract\"],\n });\n } else if (pendingCount > 0) {\n diagnostics.push({\n code: 'MIGRATION.DATABASE_BEHIND',\n severity: 'info',\n message: `${pendingCount} migration(s) pending`,\n hints: [\"Run 'prisma-next migrate' to apply pending migrations\"],\n });\n } else if (hasInvariantWork) {\n diagnostics.push({\n code: 'MIGRATION.INVARIANTS_PENDING',\n severity: 'info',\n message: `Missing required invariant(s): ${missingList}`,\n hints: [\n `Run 'prisma-next migrate --to ${activeRefName ?? '<ref>'}' to apply a path that covers the required invariants`,\n ],\n });\n } else if (!routingUnreachable) {\n diagnostics.push({\n code: 'MIGRATION.UP_TO_DATE',\n severity: 'info',\n message: 'Database is up to date',\n hints: [],\n });\n }\n }\n\n const result: MigrationStatusResult = {\n ok: true,\n mode,\n migrations: entries,\n targetHash,\n contractHash,\n summary,\n diagnostics,\n ...ifDefined('markerHash', markerHash),\n requiredInvariants,\n ...ifDefined('appliedInvariants', appliedInvariants),\n ...ifDefined('missingInvariants', missingInvariants),\n ...(statusRefs.length > 0 ? { refs: statusRefs } : {}),\n ...ifDefined('pathDecision', pathDecision),\n graph,\n bundles,\n edgeStatuses,\n ...ifDefined('activeRefHash', activeRefHash),\n ...ifDefined('activeRefName', activeRefName),\n spaces: aggregateSpaces,\n ...ifDefined('totalPendingAcrossSpaces', totalPendingAcrossSpaces),\n };\n return ok(result);\n}\n\nexport function createMigrationStatusCommand(): Command {\n const command = new Command('status');\n setCommandDescriptions(\n command,\n 'Show migration path and pending status',\n 'Shows which migrations are pending between the database marker and\\n' +\n 'the target contract. Requires a database connection for live status.\\n' +\n 'Use `migration graph` for topology, `migration log` for history,\\n' +\n 'and `migration list` for on-disk enumeration.',\n );\n setCommandExamples(command, [\n 'prisma-next migration status --db $DATABASE_URL',\n 'prisma-next migration status --to production --db $DATABASE_URL',\n ]);\n setCommandSeeAlso(command, [\n { verb: 'migration log', oneLiner: 'Show executed migration history' },\n { verb: 'migration list', oneLiner: 'List on-disk migrations' },\n { verb: 'migration graph', oneLiner: 'Show the migration graph topology' },\n { verb: 'migration show', oneLiner: 'Display migration package contents' },\n ]);\n addGlobalOptions(command)\n .option('--db <url>', 'Database connection string')\n .option('--config <path>', 'Path to prisma-next.config.ts')\n .option(\n '--to <contract>',\n 'Target contract reference (hash, prefix, ref name, migration dir name, <dir>^, or ./path)',\n )\n .option(\n '--from <contract>',\n 'Origin contract reference; same grammar as --to. Supplying --from switches to offline path computation.',\n )\n .action(async (options: MigrationStatusOptions) => {\n const flags = parseGlobalFlags(options);\n const ui = new TerminalUI({ color: flags.color, interactive: flags.interactive });\n\n const result = await executeMigrationStatusCommand(options, flags, ui);\n\n const exitCode = handleResult(result, flags, ui, (statusResult) => {\n if (flags.json) {\n const {\n graph: _graph,\n bundles: _bundles,\n edgeStatuses: _edgeStatuses,\n activeRefHash: _activeRefHash,\n activeRefName: _activeRefName,\n diverged: _diverged,\n ...jsonResult\n } = statusResult;\n ui.output(JSON.stringify(jsonResult, null, 2));\n } else if (!flags.quiet) {\n const colorize = flags.color !== false;\n\n if (statusResult.graph) {\n const renderInput = migrationGraphToRenderInput({\n graph: statusResult.graph,\n mode: statusResult.mode,\n markerHash: statusResult.markerHash,\n contractHash: statusResult.contractHash,\n refs: statusResult.refs,\n activeRefHash: statusResult.activeRefHash,\n activeRefName: statusResult.activeRefName,\n edgeStatuses: statusResult.edgeStatuses,\n });\n\n const graphToRender = statusResult.diverged\n ? renderInput.graph\n : extractRelevantSubgraph(renderInput.graph, renderInput.relevantPaths);\n const dagreOptions = isLinearGraph(graphToRender) ? { ranksep: 1 } : undefined;\n const renderOptions = {\n ...renderInput.options,\n colorize,\n ...ifDefined('dagreOptions', dagreOptions),\n };\n const graphOutput = graphRenderer.render(graphToRender, renderOptions);\n ui.log(graphOutput);\n if (statusResult.mode === 'online') {\n ui.log(formatLegend(colorize));\n }\n }\n ui.log('');\n ui.log(formatStatusSummary(statusResult, colorize));\n }\n });\n\n process.exit(exitCode);\n });\n\n return command;\n}\n\nfunction formatLegend(colorize: boolean): string {\n const c = (fn: (s: string) => string, s: string) => (colorize ? fn(s) : s);\n const parts = [\n `${c(cyan, '✓')} applied`,\n `${c(yellow, '⧗')} pending`,\n `${c(magenta, '✗')} unreachable`,\n ];\n return c(dim, parts.join(' '));\n}\n\nexport function formatStatusSummary(result: MigrationStatusResult, colorize: boolean): string {\n const c = (fn: (s: string) => string, s: string) => (colorize ? fn(s) : s);\n const lines: string[] = [];\n\n const hasUnknown = result.migrations.some((e) => e.status === 'unknown');\n const pendingCount = result.migrations.filter((e) => e.status === 'pending').length;\n\n const hasWarnings = result.diagnostics?.some((d) => d.severity === 'warn') ?? false;\n // INVARIANTS_PENDING is filed at severity 'info' (per ADR 208) so the\n // warn-severity check above doesn't see it. It still represents pending\n // work, so it must promote the summary off the success icon.\n const hasInvariantPending =\n result.diagnostics?.some((d) => d.code === 'MIGRATION.INVARIANTS_PENDING') ?? false;\n\n if (result.mode === 'online') {\n if (hasUnknown || hasWarnings) {\n lines.push(`${c(yellow, '⚠')} ${result.summary}`);\n } else if (pendingCount === 0 && !hasInvariantPending) {\n lines.push(`${c(cyan, '✔')} ${result.summary}`);\n } else {\n lines.push(`${c(yellow, '⧗')} ${result.summary}`);\n }\n } else {\n lines.push(result.summary);\n }\n\n if (result.requiredInvariants.length > 0) {\n if (result.appliedInvariants !== undefined && result.missingInvariants !== undefined) {\n lines.push(`${c(dim, 'applied ')}${formatInvariantList(result.appliedInvariants)}`);\n lines.push(`${c(dim, 'missing ')}${formatInvariantList(result.missingInvariants)}`);\n } else {\n lines.push(`${c(dim, 'applied ')}(unknown — connect a database to evaluate)`);\n }\n }\n\n const warnings = result.diagnostics?.filter((d) => d.severity === 'warn') ?? [];\n for (const diag of warnings) {\n lines.push(`${c(yellow, '⚠')} ${diag.message}`);\n for (const hint of diag.hints) {\n lines.push(` ${c(dim, hint)}`);\n }\n }\n\n // Per-space section. Suppressed when there's no extension space —\n // the legacy single-space output already covers the app member.\n // When extensions exist, render every space (including the app)\n // for consistency, plus a cross-space pending total + apply hint.\n if (result.spaces?.some((s) => s.kind === 'extension')) {\n const total = result.totalPendingAcrossSpaces ?? 0;\n lines.push('');\n lines.push(c(dim, 'spaces'));\n for (const space of result.spaces) {\n lines.push(formatSpaceLine(space, c));\n }\n if (total > 0) {\n lines.push('');\n lines.push(\n `${c(yellow, '⧗')} ${total} pending migration(s) across ${result.spaces.length} space(s) — run 'prisma-next migrate' to apply`,\n );\n }\n }\n\n return lines.join('\\n');\n}\n\nfunction formatSpaceLine(\n space: MigrationStatusSpaceEntry,\n c: (fn: (s: string) => string, s: string) => string,\n): string {\n const glyph = (() => {\n if (space.status === 'up-to-date' || space.status === 'no-marker') return c(cyan, '✓');\n if (space.status === 'pending') return c(yellow, '⧗');\n if (space.status === 'unreachable' || space.status === 'never-planned') return c(magenta, '✗');\n return ' ';\n })();\n const tag = space.kind === 'app' ? '[app]' : '[ext]';\n const head = space.headHash.slice(0, 8);\n const marker =\n space.markerHash === undefined\n ? '(unknown)'\n : space.markerHash === null\n ? '(no marker)'\n : space.markerHash.slice(0, 8);\n const pending =\n space.pendingCount === undefined\n ? ''\n : space.pendingCount === 0\n ? c(dim, ' (up to date)')\n : c(yellow, ` (${space.pendingCount} pending)`);\n return ` ${glyph} ${c(dim, tag)} ${space.spaceId} → head ${c(dim, head)}, marker ${c(dim, marker)}${pending}`;\n}\n\nfunction formatInvariantList(ids: readonly string[]): string {\n return ids.length === 0 ? '(none)' : ids.join(', ');\n}\n\nfunction summarizeRefDistance(\n graph: MigrationGraph,\n markerHash: string,\n refHash: string,\n refName: string,\n): string {\n if (markerHash === refHash) return `At ref \"${refName}\" target`;\n\n const pathToRef = findPath(graph, markerHash, refHash);\n if (pathToRef) return `${pathToRef.length} migration(s) behind ref \"${refName}\"`;\n\n const pathFromRef = findPath(graph, refHash, markerHash);\n if (pathFromRef) return `${pathFromRef.length} migration(s) ahead of ref \"${refName}\"`;\n\n return `No path between database marker and ref \"${refName}\" target`;\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"migration-status.mjs","names":[],"sources":["../../src/commands/migration-status.ts"],"sourcesContent":["import {\n createControlStack,\n type MigrationPlanOperation,\n} from '@prisma-next/framework-components/control';\nimport {\n type ContractMarkerRecordLike,\n graphWalkStrategy,\n} from '@prisma-next/migration-tools/aggregate';\nimport { EMPTY_CONTRACT_HASH } from '@prisma-next/migration-tools/constants';\nimport {\n errorNoInvariantPath,\n errorUnknownInvariant,\n MigrationToolsError,\n} from '@prisma-next/migration-tools/errors';\nimport type { MigrationEdge, MigrationGraph } from '@prisma-next/migration-tools/graph';\nimport {\n findPath,\n findPathWithDecision,\n findReachableLeaves,\n} from '@prisma-next/migration-tools/migration-graph';\nimport type { OnDiskMigrationPackage } from '@prisma-next/migration-tools/package';\nimport { parseContractRef } from '@prisma-next/migration-tools/ref-resolution';\nimport type { RefEntry, Refs } from '@prisma-next/migration-tools/refs';\nimport { readRefs } from '@prisma-next/migration-tools/refs';\nimport { ifDefined } from '@prisma-next/utils/defined';\nimport { notOk, ok, type Result } from '@prisma-next/utils/result';\nimport { cyan, dim, magenta, yellow } from 'colorette';\nimport { Command } from 'commander';\n\nimport { loadConfig } from '../config-loader';\nimport { createControlClient } from '../control-api/client';\nimport {\n CliStructuredError,\n errorRuntime,\n errorUnexpected,\n mapMigrationToolsError,\n mapRefResolutionError,\n} from '../utils/cli-errors';\nimport {\n addGlobalOptions,\n collectDeclaredInvariants,\n loadMigrationPackages,\n maskConnectionUrl,\n readContractEnvelope,\n resolveMigrationPaths,\n setCommandDescriptions,\n setCommandExamples,\n setCommandSeeAlso,\n toPathDecisionResult,\n toStructuralEdge,\n} from '../utils/command-helpers';\nimport {\n type BuildAggregateInputs,\n buildContractSpaceAggregate,\n} from '../utils/contract-space-aggregate-loader';\nimport {\n type EdgeStatus,\n type EdgeStatusKind,\n migrationGraphToRenderInput,\n} from '../utils/formatters/graph-migration-mapper';\nimport {\n extractRelevantSubgraph,\n graphRenderer,\n isLinearGraph,\n} from '../utils/formatters/graph-render';\nimport { formatStyledHeader } from '../utils/formatters/styled';\nimport type { CommonCommandOptions } from '../utils/global-flags';\nimport { type GlobalFlags, parseGlobalFlagsOrExit } from '../utils/global-flags';\nimport type { StatusDiagnostic, StatusRef } from '../utils/migration-types';\nimport { handleResult } from '../utils/result-handler';\nimport { createTerminalUI, type TerminalUI } from '../utils/terminal-ui';\n\ninterface MigrationStatusOptions extends CommonCommandOptions {\n readonly db?: string;\n readonly config?: string;\n readonly to?: string;\n readonly from?: string;\n}\n\nexport interface MigrationStatusEntry {\n readonly dirName: string;\n readonly from: string;\n readonly to: string;\n readonly migrationHash: string;\n readonly operationCount: number;\n readonly operationSummary: string;\n readonly hasDestructive: boolean;\n readonly status: EdgeStatusKind | 'unknown';\n}\n\n/**\n * Per-space status row in the aggregate-shaped status output.\n *\n * Surfaces, for each contract space:\n *\n * - `headHash`: the on-disk head ref's hash (where the space is going).\n * - `markerHash`: the live marker hash for the space, or null if no\n * marker has been written yet (greenfield, or pre-`migrate`).\n * - `pendingCount`: number of migration edges between marker and head.\n * Computed via {@link graphWalkStrategy}; 0 means the space is\n * already at head.\n * - `status`: convenience tag the formatter uses to pick a glyph.\n * `'never-planned'` is reserved for spaces with non-empty head but\n * no on-disk migrations — which shouldn't happen if the loader's\n * integrity check passes.\n *\n * Online-only fields (`markerHash`, `status`) are absent when the\n * command runs without a database connection.\n */\nexport interface MigrationStatusSpaceEntry {\n readonly spaceId: string;\n readonly kind: 'app' | 'extension';\n readonly headHash: string;\n readonly markerHash?: string | null;\n readonly pendingCount?: number;\n readonly status?: 'up-to-date' | 'pending' | 'no-marker' | 'never-planned' | 'unreachable';\n}\n\n/**\n * Sum per-space `pendingCount` into a cross-space total, but only when\n * every loaded space reports a defined `pendingCount`. Returns\n * `undefined` if any space is on the marker-unknown / offline path\n * (where `pendingCount` is intentionally absent), so JSON consumers can\n * distinguish \"no pending\" from \"unknown\".\n */\nexport function computeTotalPendingAcrossSpaces(\n spaces: readonly MigrationStatusSpaceEntry[],\n): number | undefined {\n if (spaces.length === 0) return undefined;\n let total = 0;\n for (const s of spaces) {\n if (s.pendingCount === undefined) return undefined;\n total += s.pendingCount;\n }\n return total;\n}\n\nexport type { StatusDiagnostic, StatusRef } from '../utils/migration-types';\n\nexport interface MigrationStatusResult {\n readonly ok: true;\n readonly mode: 'online' | 'offline';\n readonly migrations: readonly MigrationStatusEntry[];\n readonly markerHash?: string;\n readonly targetHash: string;\n readonly contractHash: string;\n readonly refs?: readonly StatusRef[];\n /** Required invariants from the active ref, sorted ascending. Always present (`[]` when no `--ref` or the ref declares none) — knowable offline. */\n readonly requiredInvariants: readonly string[];\n /**\n * Invariants the marker has applied at least once, intersected with\n * `requiredInvariants` for display relevance. JSON consumers see only the\n * subset overlapping the active ref's required set — the full unfiltered\n * marker invariant list lives on `marker.invariants` (control plane) and\n * is not surfaced here. Present only in `mode === 'online'`; absent when\n * offline (the marker is unknown, not empty).\n */\n readonly appliedInvariants?: readonly string[];\n /** required − applied. Present only in `mode === 'online'`; absent when offline. */\n readonly missingInvariants?: readonly string[];\n readonly pathDecision?: {\n readonly fromHash: string;\n readonly toHash: string;\n readonly alternativeCount: number;\n readonly tieBreakReasons: readonly string[];\n readonly refName?: string;\n readonly requiredInvariants: readonly string[];\n readonly satisfiedInvariants: readonly string[];\n readonly selectedPath: readonly {\n readonly dirName: string;\n readonly migrationHash: string;\n readonly from: string;\n readonly to: string;\n readonly invariants: readonly string[];\n }[];\n };\n readonly summary: string;\n readonly diagnostics: readonly StatusDiagnostic[];\n /**\n * Aggregate enumeration of every on-disk contract space (app +\n * extensions), in canonical schedule order (extensions\n * alphabetically, then app). Present whenever the aggregate loader\n * succeeded; absent in early-error returns (e.g. unreadable\n * migrations directory) where the existing diagnostics already\n * surface the failure.\n *\n * The legacy top-level fields (`migrations`, `markerHash`,\n * `targetHash`, `pathDecision`, …) describe the **app member**\n * specifically — back-compat with single-space callers. Per-space\n * detail for extension members lives only on this list.\n */\n readonly spaces?: readonly MigrationStatusSpaceEntry[];\n /** Cross-space pending-migration total (sum of `spaces[].pendingCount`). Present when `spaces` is. */\n readonly totalPendingAcrossSpaces?: number;\n readonly graph?: MigrationGraph;\n readonly bundles?: readonly OnDiskMigrationPackage[];\n readonly edgeStatuses?: readonly EdgeStatus[];\n readonly activeRefHash?: string;\n readonly activeRefName?: string;\n readonly diverged?: boolean;\n}\n\nfunction summarizeOps(ops: readonly MigrationPlanOperation[]): {\n summary: string;\n hasDestructive: boolean;\n} {\n if (ops.length === 0) return { summary: '0 ops', hasDestructive: false };\n\n const classes = new Map<string, number>();\n for (const op of ops) {\n classes.set(op.operationClass, (classes.get(op.operationClass) ?? 0) + 1);\n }\n\n const hasDestructive = classes.has('destructive');\n const count = ops.length;\n const noun = count === 1 ? 'op' : 'ops';\n\n if (classes.size === 1) {\n const cls = [...classes.keys()][0]!;\n return { summary: `${count} ${noun} (all ${cls})`, hasDestructive };\n }\n\n const destructiveCount = classes.get('destructive');\n if (destructiveCount) {\n return { summary: `${count} ${noun} (${destructiveCount} destructive)`, hasDestructive };\n }\n\n const parts = [...classes.entries()].map(([cls, n]) => `${n} ${cls}`);\n return { summary: `${count} ${noun} (${parts.join(', ')})`, hasDestructive };\n}\n\n/**\n * Derive per-edge status across the full graph using path analysis.\n *\n * - **applied**: edge is on the path from root to the DB marker\n * - **pending**: edge is on the path from the DB marker to the target\n * (and the marker is reachable from root, i.e. it's on the same branch)\n * - **unreachable**: edge is on the path from root to the target but the DB\n * marker is on a different branch — `apply` can't reach these edges\n * without the DB first moving to this branch\n *\n * Returns statuses only for edges that have a known status (skips offline\n * and edges not on any relevant path).\n *\n * @internal Exported for testing only.\n */\nexport function deriveEdgeStatuses(\n graph: MigrationGraph,\n targetHash: string,\n contractHash: string,\n markerHash: string | undefined,\n mode: 'online' | 'offline',\n): EdgeStatus[] {\n if (mode === 'offline') return [];\n\n const edgeKey = (e: MigrationEdge) => `${e.from}\\0${e.to}`;\n\n // No marker = empty DB — treat root as the marker (nothing applied, everything pending)\n const effectiveMarker = markerHash ?? EMPTY_CONTRACT_HASH;\n\n const appliedPath =\n markerHash !== undefined ? findPath(graph, EMPTY_CONTRACT_HASH, markerHash) : null;\n\n const pendingPath = findPath(graph, effectiveMarker, targetHash);\n const targetPath = findPath(graph, EMPTY_CONTRACT_HASH, targetHash);\n\n const statuses: EdgeStatus[] = [];\n const assignedKeys = new Set<string>();\n\n // Applied edges (root → marker)\n if (appliedPath) {\n for (const e of appliedPath) {\n assignedKeys.add(edgeKey(e));\n statuses.push({ dirName: e.dirName, status: 'applied' });\n }\n }\n\n // Pending edges (marker → target)\n if (pendingPath) {\n for (const e of pendingPath) {\n assignedKeys.add(edgeKey(e));\n statuses.push({ dirName: e.dirName, status: 'pending' });\n }\n }\n\n // Pending edges beyond the target: target → contract (when target is a ref\n // and the contract is reachable from it)\n if (\n contractHash !== EMPTY_CONTRACT_HASH &&\n contractHash !== targetHash &&\n graph.nodes.has(contractHash)\n ) {\n const beyondTarget = findPath(graph, targetHash, contractHash);\n if (beyondTarget) {\n for (const e of beyondTarget) {\n if (!assignedKeys.has(edgeKey(e))) {\n assignedKeys.add(edgeKey(e));\n statuses.push({ dirName: e.dirName, status: 'pending' });\n }\n }\n }\n }\n\n // Unreachable edges: on the path from root to the target but neither applied\n // nor pending. This covers two cases:\n // 1. Marker can't reach target at all (different branch entirely)\n // 2. Marker reaches target via a different route, leaving some root→target\n // edges orphaned (e.g. a fork where one branch was applied and apply\n // will continue through the other)\n if (targetPath) {\n for (const e of targetPath) {\n if (!assignedKeys.has(edgeKey(e))) {\n statuses.push({ dirName: e.dirName, status: 'unreachable' });\n }\n }\n }\n\n return statuses;\n}\n\n/**\n * @param mode — 'online' if we connected to the database, 'offline' otherwise\n * @param markerHash — the marker hash from the database, or undefined if no marker row / offline\n */\nfunction buildMigrationEntries(\n chain: readonly MigrationEdge[],\n packages: readonly OnDiskMigrationPackage[],\n mode: 'online' | 'offline',\n markerHash: string | undefined,\n edgeStatuses?: readonly EdgeStatus[],\n): MigrationStatusEntry[] {\n const pkgByDirName = new Map(packages.map((p) => [p.dirName, p]));\n const statusByDirName = edgeStatuses\n ? new Map(edgeStatuses.map((e) => [e.dirName, e.status]))\n : undefined;\n\n const markerInChain = markerHash === undefined || chain.some((e) => e.to === markerHash);\n\n const entries: MigrationStatusEntry[] = [];\n let reachedMarker = mode === 'online' && markerHash === undefined;\n\n for (const migration of chain) {\n const pkg = pkgByDirName.get(migration.dirName);\n const ops = (pkg?.ops ?? []) as readonly MigrationPlanOperation[];\n const { summary, hasDestructive } = summarizeOps(ops);\n\n let status: EdgeStatusKind | 'unknown';\n const edgeStatus = statusByDirName?.get(migration.dirName);\n if (edgeStatus) {\n status = edgeStatus;\n } else if (mode === 'offline' || !markerInChain) {\n status = 'unknown';\n } else if (reachedMarker) {\n status = 'pending';\n } else {\n status = 'applied';\n }\n\n entries.push({\n dirName: migration.dirName,\n from: migration.from,\n to: migration.to,\n migrationHash: migration.migrationHash,\n operationCount: ops.length,\n operationSummary: summary,\n hasDestructive,\n status,\n });\n\n if (!reachedMarker && migration.to === markerHash) {\n reachedMarker = true;\n }\n }\n\n return entries;\n}\n\n/**\n * Resolve the migration chain to display in status output.\n *\n * When offline or the marker is at EMPTY, the chain is simply the shortest\n * path from EMPTY to the target — all structural paths are equivalent per\n * the spec, so the deterministic shortest path is the canonical display.\n *\n * When online with a non-empty marker, the chain routes *through* the marker:\n * EMPTY→marker (applied history) + marker→target (pending edges). This ensures\n * the displayed chain includes the marker node so applied/pending status is\n * correct. Without this, BFS from EMPTY to target could pick a shortest path\n * that bypasses the marker entirely (e.g. in a diamond graph), causing the\n * marker to appear \"diverged\" when it isn't.\n */\nfunction resolveDisplayChain(\n graph: MigrationGraph,\n targetHash: string,\n markerHash: string | undefined,\n): readonly MigrationEdge[] | null {\n if (markerHash === undefined) {\n return findPath(graph, EMPTY_CONTRACT_HASH, targetHash);\n }\n\n const toMarker = findPath(graph, EMPTY_CONTRACT_HASH, markerHash);\n // Marker unreachable from EMPTY — show the target chain anyway.\n // The caller detects this via markerInChain and emits a divergence diagnostic.\n if (!toMarker) return findPath(graph, EMPTY_CONTRACT_HASH, targetHash);\n\n if (markerHash === targetHash) return toMarker;\n\n const fromMarker = findPath(graph, markerHash, targetHash);\n if (fromMarker) return [...toMarker, ...fromMarker];\n\n // Marker is ahead of target (or on a disconnected branch).\n // Try the inverse: target→marker. If it succeeds, the marker is ahead —\n // show the full chain from EMPTY through the target and on to the marker.\n const toTarget = findPath(graph, EMPTY_CONTRACT_HASH, targetHash);\n if (!toTarget) return null;\n\n const targetToMarker = findPath(graph, targetHash, markerHash);\n if (targetToMarker) return [...toTarget, ...targetToMarker];\n\n // Genuinely disconnected — show EMPTY→target; caller handles divergence diagnostic.\n return toTarget;\n}\n\n/**\n * Build the aggregate enumeration of contract spaces for the status\n * output. Loads the aggregate from disk (lossy on failure — extension\n * spaces are simply omitted, the existing single-space app behaviour\n * keeps working), reads per-space marker rows when online, and uses\n * {@link graphWalkStrategy} to compute each space's pending count.\n *\n * Sub-spec § `migration status` semantics — the aggregate-walking\n * version reports per-space marker + pending state alongside the\n * cross-space totals.\n */\nexport async function loadAggregateStatusSpaces(args: {\n readonly targetId: string;\n readonly migrationsDir: string;\n readonly appContractRaw: unknown;\n readonly extensionPacks: BuildAggregateInputs<string, string>['extensionPacks'];\n readonly deserializeContract: BuildAggregateInputs<string, string>['deserializeContract'];\n readonly markersBySpace: ReadonlyMap<string, ContractMarkerRecordLike> | null;\n}): Promise<readonly MigrationStatusSpaceEntry[]> {\n const loadInputs: BuildAggregateInputs<string, string> = {\n targetId: args.targetId,\n migrationsDir: args.migrationsDir,\n appContract: args.deserializeContract(args.appContractRaw),\n extensionPacks: args.extensionPacks,\n deserializeContract: args.deserializeContract,\n };\n\n const loaded = await buildContractSpaceAggregate(loadInputs);\n if (!loaded.ok) {\n // Loader failure (drift, layout violation, etc.) — surfacing it\n // as a status diagnostic would duplicate `migration plan`'s job.\n // The single-space app pipeline still runs; extensions are simply\n // not enumerated.\n return [];\n }\n const aggregate = loaded.value;\n\n const orderedMembers = [...aggregate.extensions, aggregate.app];\n const rows: MigrationStatusSpaceEntry[] = [];\n for (const member of orderedMembers) {\n const liveMarker = args.markersBySpace?.get(member.spaceId) ?? null;\n const isApp = member.spaceId === aggregate.app.spaceId;\n\n if (member.migrations.graph.nodes.size === 0) {\n rows.push({\n spaceId: member.spaceId,\n kind: isApp ? 'app' : 'extension',\n headHash: member.headRef.hash,\n ...(args.markersBySpace !== null\n ? {\n markerHash: liveMarker?.storageHash ?? null,\n status: member.headRef.hash === EMPTY_CONTRACT_HASH ? 'up-to-date' : 'never-planned',\n pendingCount: 0,\n }\n : {}),\n });\n continue;\n }\n\n if (args.markersBySpace === null) {\n rows.push({\n spaceId: member.spaceId,\n kind: isApp ? 'app' : 'extension',\n headHash: member.headRef.hash,\n });\n continue;\n }\n\n const walked = graphWalkStrategy({\n aggregateTargetId: aggregate.targetId,\n member,\n currentMarker: liveMarker,\n });\n let pendingCount = 0;\n let status: MigrationStatusSpaceEntry['status'];\n if (walked.kind === 'ok') {\n // Count pending *migrations* (graph edges), not operations: a\n // single authored migration that lowers to N ops or zero ops\n // both count as exactly one pending unit of work for the user.\n pendingCount = walked.result.migrationEdges?.length ?? 0;\n if (liveMarker === null) {\n status = pendingCount === 0 ? 'no-marker' : 'pending';\n } else {\n status = pendingCount === 0 ? 'up-to-date' : 'pending';\n }\n } else {\n status = 'unreachable';\n }\n\n rows.push({\n spaceId: member.spaceId,\n kind: isApp ? 'app' : 'extension',\n headHash: member.headRef.hash,\n markerHash: liveMarker?.storageHash ?? null,\n pendingCount,\n ...(status ? { status } : {}),\n });\n }\n return rows;\n}\n\n/**\n * Read the raw contract.json bytes from disk for the aggregate\n * loader. Returns `null` if the file is missing or unparseable —\n * the existing `readContractEnvelope` path will report the same\n * problem via a status diagnostic, no need to double-surface.\n */\nasync function loadContractRawSafely(config: {\n contract?: { output?: string };\n}): Promise<unknown | null> {\n try {\n const path = (await import('../utils/command-helpers')).resolveContractPath(config);\n const raw = await (await import('node:fs/promises')).readFile(path, 'utf-8');\n return JSON.parse(raw) as unknown;\n } catch {\n return null;\n }\n}\n\nasync function validateOnlineMarkerRead(\n config: Awaited<ReturnType<typeof loadConfig>>,\n dbConnection: unknown,\n): Promise<Result<void, CliStructuredError>> {\n const driver = config.driver;\n if (!driver) {\n return ok(undefined);\n }\n\n const client = createControlClient({\n family: config.family,\n target: config.target,\n adapter: config.adapter,\n driver,\n extensionPacks: config.extensionPacks ?? [],\n });\n try {\n await client.connect(dbConnection);\n await client.readMarker();\n return ok(undefined);\n } catch (error) {\n if (CliStructuredError.is(error)) {\n return notOk(error);\n }\n return notOk(\n errorUnexpected(error instanceof Error ? error.message : String(error), {\n why: `Failed to read database marker: ${error instanceof Error ? error.message : String(error)}`,\n }),\n );\n } finally {\n await client.close();\n }\n}\n\nasync function executeMigrationStatusCommand(\n options: MigrationStatusOptions,\n flags: GlobalFlags,\n ui: TerminalUI,\n): Promise<Result<MigrationStatusResult, CliStructuredError>> {\n const config = await loadConfig(options.config);\n const { configPath, appMigrationsDir, appMigrationsRelative, migrationsDir, refsDir } =\n resolveMigrationPaths(options.config, config);\n\n const dbConnection = options.db ?? config.db?.connection;\n const hasDriver = !!config.driver;\n\n let activeRefName: string | undefined;\n let activeRefHash: string | undefined;\n let activeRefEntry: RefEntry | undefined;\n let allRefs: Refs = {};\n try {\n allRefs = await readRefs(refsDir);\n } catch (error) {\n if (MigrationToolsError.is(error)) {\n return notOk(mapMigrationToolsError(error));\n }\n throw error;\n }\n\n let fromOverrideHash: string | undefined;\n\n if (options.to || options.from) {\n try {\n const { graph: earlyGraph } = await loadMigrationPackages(appMigrationsDir);\n\n if (options.to) {\n const refResult = parseContractRef(options.to, { graph: earlyGraph, refs: allRefs });\n if (!refResult.ok) {\n return notOk(mapRefResolutionError(refResult.failure));\n }\n activeRefHash = refResult.value.hash;\n if (refResult.value.provenance.kind === 'ref') {\n const resolvedRefName = refResult.value.provenance.refName;\n activeRefName = resolvedRefName;\n activeRefEntry = allRefs[resolvedRefName];\n }\n }\n\n if (options.from) {\n const fromResult = parseContractRef(options.from, { graph: earlyGraph, refs: allRefs });\n if (!fromResult.ok) {\n return notOk(mapRefResolutionError(fromResult.failure));\n }\n fromOverrideHash = fromResult.value.hash;\n }\n } catch (error) {\n if (MigrationToolsError.is(error)) {\n return notOk(mapMigrationToolsError(error));\n }\n throw error;\n }\n }\n\n const requiredInvariants: readonly string[] = [...(activeRefEntry?.invariants ?? [])].sort();\n\n const statusRefs: StatusRef[] = Object.entries(allRefs).map(([name, entry]) => ({\n name,\n hash: entry.hash,\n active: name === activeRefName,\n }));\n\n if (!flags.json && !flags.quiet) {\n const details: Array<{ label: string; value: string }> = [\n { label: 'config', value: configPath },\n { label: 'migrations', value: appMigrationsRelative },\n ];\n if (dbConnection && hasDriver) {\n details.push({ label: 'database', value: maskConnectionUrl(String(dbConnection)) });\n }\n if (activeRefName) {\n details.push({ label: 'ref', value: activeRefName });\n }\n if (options.from) {\n details.push({ label: 'from', value: options.from });\n }\n if (activeRefEntry && activeRefEntry.invariants.length > 0) {\n details.push({\n label: 'required',\n value: formatInvariantList(activeRefEntry.invariants),\n });\n }\n const header = formatStyledHeader({\n command: 'migration status',\n description: 'Show migration history and applied status',\n details,\n flags,\n });\n ui.stderr(header);\n }\n\n const diagnostics: StatusDiagnostic[] = [];\n let contractHash: string = EMPTY_CONTRACT_HASH;\n try {\n const envelope = await readContractEnvelope(config);\n contractHash = envelope.storageHash;\n } catch (error) {\n diagnostics.push({\n code: 'CONTRACT.UNREADABLE',\n severity: 'warn',\n message: `Could not read contract: ${error instanceof Error ? error.message : 'unknown error'}`,\n hints: [\"Run 'prisma-next contract emit' to generate a valid contract\"],\n });\n }\n\n let bundles: readonly OnDiskMigrationPackage[];\n let graph: MigrationGraph;\n try {\n ({ bundles, graph } = await loadMigrationPackages(appMigrationsDir));\n } catch (error) {\n if (MigrationToolsError.is(error)) {\n return notOk(mapMigrationToolsError(error));\n }\n return notOk(\n errorUnexpected(error instanceof Error ? error.message : String(error), {\n why: `Failed to read migrations directory: ${error instanceof Error ? error.message : String(error)}`,\n }),\n );\n }\n\n if (bundles.length === 0) {\n if (dbConnection && hasDriver) {\n const markerProbe = await validateOnlineMarkerRead(config, dbConnection);\n if (!markerProbe.ok) {\n return markerProbe;\n }\n }\n if (contractHash !== EMPTY_CONTRACT_HASH) {\n diagnostics.push({\n code: 'CONTRACT.AHEAD',\n severity: 'warn',\n message: 'No migration exists for the current contract',\n hints: [\n \"Run 'prisma-next migration plan' to generate a migration for the current contract\",\n ],\n });\n }\n return ok({\n ok: true,\n mode: dbConnection && hasDriver ? 'online' : 'offline',\n migrations: [],\n targetHash: EMPTY_CONTRACT_HASH,\n contractHash,\n summary: 'No migrations found',\n diagnostics,\n requiredInvariants,\n });\n }\n\n let targetHash: string | undefined;\n\n if (activeRefHash) {\n targetHash = activeRefHash;\n } else if (graph.nodes.has(contractHash)) {\n targetHash = contractHash;\n } else {\n const leaves = findReachableLeaves(graph, EMPTY_CONTRACT_HASH);\n if (leaves.length === 1) {\n targetHash = leaves[0];\n } else {\n diagnostics.push({\n code: 'MIGRATION.DIVERGED',\n severity: 'warn',\n message: 'There are multiple valid migration paths — you must select a target',\n hints: [\n \"Use '--to <contract>' to select a target\",\n \"Or 'prisma-next ref set <name> <hash>' to create one\",\n ],\n });\n }\n }\n\n let markerHash: string | undefined;\n let markerInvariants: readonly string[] = [];\n let mode: 'online' | 'offline' = 'offline';\n let allMarkers: ReadonlyMap<string, ContractMarkerRecordLike> | null = null;\n\n if (dbConnection && hasDriver) {\n const client = createControlClient({\n family: config.family,\n target: config.target,\n adapter: config.adapter,\n driver: config.driver,\n extensionPacks: config.extensionPacks ?? [],\n });\n try {\n await client.connect(dbConnection);\n const marker = await client.readMarker();\n markerHash = marker?.storageHash;\n markerInvariants = marker?.invariants ?? [];\n mode = 'online';\n // Read every space's marker so the aggregate enumeration can\n // surface per-space marker state. `readAllMarkers` mirrors what\n // `db init` / `db update` already use to drive the multi-space\n // planner; here it powers the aggregate status output.\n //\n // Probe for the method first so we only swallow the\n // unsupported-method case: older family instances may not\n // implement `readAllMarkers` (per-space enumeration then falls\n // back to \"marker unknown\"). Real query / runtime errors from\n // an instance that *does* expose the method must propagate up\n // — otherwise transient DB failures would silently degrade\n // status to \"markers unknown\".\n if (typeof client.readAllMarkers === 'function') {\n allMarkers = await client.readAllMarkers();\n } else {\n // Leaving `allMarkers` as `null` signals \"unknown\" to the\n // aggregate loader (an empty `Map` would instead mean \"every\n // space has no marker\", which is a different condition).\n allMarkers = null;\n }\n } catch (error) {\n if (CliStructuredError.is(error)) {\n return notOk(error);\n }\n if (!flags.json && !flags.quiet) {\n ui.warn('Could not connect to database — showing offline status');\n }\n } finally {\n await client.close();\n }\n }\n\n if (fromOverrideHash !== undefined) {\n markerHash = fromOverrideHash;\n mode = 'offline';\n allMarkers = null;\n }\n\n // Build the aggregate enumeration of contract spaces. Lossy on\n // failure (extensions are simply omitted) so the existing\n // single-space app pipeline below still runs even if extensions\n // can't be loaded — a strict failure here would degrade the\n // load-bearing app-space output for unrelated reasons.\n const contractRawForAggregate = await loadContractRawSafely(config);\n let aggregateSpaces: readonly MigrationStatusSpaceEntry[] = [];\n if (contractRawForAggregate !== null) {\n // The aggregate loader needs a typed-Contract producer. Build a\n // real control stack so `deserializeContract` runs against a fully\n // composed family instance — descriptors that read stack members\n // during construction (e.g. codec lookups) get a consistent view.\n const stack = createControlStack(config);\n const familyInstance = config.family.create(stack);\n try {\n aggregateSpaces = await loadAggregateStatusSpaces({\n targetId: config.target.targetId,\n migrationsDir,\n appContractRaw: contractRawForAggregate,\n extensionPacks: config.extensionPacks ?? [],\n deserializeContract: (json: unknown) => familyInstance.deserializeContract(json),\n markersBySpace: allMarkers,\n });\n } catch {\n // Loader failure short-circuits silently — the existing\n // single-space app pipeline below still runs.\n aggregateSpaces = [];\n }\n }\n const totalPendingAcrossSpaces = computeTotalPendingAcrossSpaces(aggregateSpaces);\n\n // Pre-check unknown invariants. Online: union the graph's declared\n // invariants with the marker's recorded set so a retired-but-applied\n // invariant doesn't surface as MIGRATION.UNKNOWN_INVARIANT — apply would\n // route fine because marker subtraction empties `effectiveRequired`.\n // Offline: keep the check graph-strict (the marker is unknown, and a\n // missing declarer is the dominant signal we can offer).\n if (activeRefEntry && activeRefEntry.invariants.length > 0) {\n const declared = collectDeclaredInvariants(graph);\n const known = new Set<string>(declared);\n if (mode === 'online') {\n for (const id of markerInvariants) known.add(id);\n }\n const unknown = activeRefEntry.invariants.filter((id) => !known.has(id));\n if (unknown.length > 0) {\n return notOk(\n mapMigrationToolsError(\n errorUnknownInvariant({\n ...ifDefined('refName', activeRefName),\n unknown,\n declared: [...declared].sort(),\n }),\n ),\n );\n }\n }\n\n // Marker exists but is not in the migration graph and doesn't match the\n // contract hash. The DB is at an unknown state relative to the graph.\n // Bail out early with a clear diagnostic instead of rendering a confusing\n // graph with no statuses.\n //\n // When marker === contract (both off-graph), the DB matches the current\n // contract — proceed normally; the detached contract node will carry both\n // the db and contract markers.\n if (\n mode === 'online' &&\n markerHash !== undefined &&\n !graph.nodes.has(markerHash) &&\n markerHash !== contractHash\n ) {\n const hints: string[] = [];\n if (graph.nodes.has(contractHash)) {\n hints.push(\n \"Run 'prisma-next db sign' to overwrite the marker if the database already matches the contract\",\n \"Run 'prisma-next db update' to push the current contract to the database\",\n \"Run 'prisma-next contract infer' to make your contract match the database\",\n \"Run 'prisma-next db verify' to inspect the database state\",\n );\n } else {\n hints.push(\n \"Run 'prisma-next db update' to push the current contract to the database\",\n \"Run 'prisma-next contract infer' to make your contract match the database\",\n \"Run 'prisma-next db verify' to inspect the database state\",\n );\n }\n diagnostics.push({\n code: 'MIGRATION.MARKER_NOT_IN_HISTORY',\n severity: 'warn',\n message:\n 'Database was updated outside the migration system (marker does not match any migration)',\n hints,\n });\n return ok({\n ok: true,\n mode,\n migrations: [],\n targetHash: EMPTY_CONTRACT_HASH,\n contractHash,\n summary: `${bundles.length} migration(s) on disk`,\n diagnostics,\n markerHash,\n requiredInvariants,\n ...(statusRefs.length > 0 ? { refs: statusRefs } : {}),\n });\n }\n\n if (mode === 'online' && markerHash === undefined) {\n diagnostics.push({\n code: 'MIGRATION.NO_MARKER',\n severity: 'warn',\n message: 'Database has not been initialized — no migration marker found',\n hints: [\"Run 'prisma-next migrate' to apply pending migrations\"],\n });\n }\n\n // Contract diagnostic — fires when no migration produces the current contract hash.\n // Suppressed when: (a) graph is diverged (MIGRATION.DIVERGED already guides the user),\n // (b) marker === contract and both off-graph (marker-not-in-graph diagnostic covers it).\n if (\n targetHash &&\n contractHash !== EMPTY_CONTRACT_HASH &&\n !graph.nodes.has(contractHash) &&\n markerHash !== contractHash\n ) {\n diagnostics.push({\n code: 'CONTRACT.AHEAD',\n severity: 'warn',\n message: 'Contract has changed since the last migration was planned',\n hints: [\"Run 'prisma-next migration plan' to generate a migration for the current contract\"],\n });\n }\n\n if (!targetHash) {\n return ok({\n ok: true,\n mode,\n migrations: [],\n targetHash: EMPTY_CONTRACT_HASH,\n contractHash,\n summary: `${bundles.length} migration(s) on disk`,\n diagnostics,\n ...ifDefined('markerHash', markerHash),\n requiredInvariants,\n ...(statusRefs.length > 0 ? { refs: statusRefs } : {}),\n graph,\n bundles,\n diverged: true,\n });\n }\n\n const chain = resolveDisplayChain(graph, targetHash, markerHash);\n\n if (!chain) {\n return notOk(\n errorRuntime('Cannot reconstruct migration history', {\n why: `No path from ${EMPTY_CONTRACT_HASH} to target ${targetHash}`,\n fix: 'The migration history may have gaps. Check the migrations directory for missing or corrupted packages.',\n }),\n );\n }\n\n const edgeStatuses = deriveEdgeStatuses(graph, targetHash, contractHash, markerHash, mode);\n const entries = buildMigrationEntries(chain, bundles, mode, markerHash, edgeStatuses);\n\n const pendingCount = edgeStatuses.filter((e) => e.status === 'pending').length;\n const appliedCount = edgeStatuses.filter((e) => e.status === 'applied').length;\n\n let appliedInvariants: readonly string[] | undefined;\n let missingInvariants: readonly string[] | undefined;\n let effectiveRequired = new Set<string>();\n if (mode === 'online') {\n // Mirrors `migrate.ts`: compute `effectiveRequired = required −\n // marker.invariants` directly, then derive the display fields from it.\n // `appliedInvariants` is the intersection (`required ∩ marker`), which\n // is what JSON consumers see for the active ref; the unfiltered set\n // lives on `marker.invariants`.\n const markerSet = new Set(markerInvariants);\n effectiveRequired = new Set(requiredInvariants.filter((id) => !markerSet.has(id)));\n appliedInvariants = requiredInvariants.filter((id) => markerSet.has(id));\n missingInvariants = [...effectiveRequired].sort();\n }\n\n // The marker can match the structural target while still missing required\n // invariants — for example, a self-edge that provides X, applied via a ref\n // declaring X. `pendingCount` (structural) says zero in that case but\n // `effectiveRequired` is non-empty, so up-to-date messaging would mislead.\n const hasInvariantWork = effectiveRequired.size > 0;\n const missingList = [...effectiveRequired].sort().join(', ');\n\n let summary: string;\n if (mode === 'online') {\n if (markerHash !== undefined && !graph.nodes.has(markerHash) && markerHash === contractHash) {\n summary = `${bundles.length} migration(s) on disk`;\n } else if (activeRefHash && activeRefName && markerHash !== undefined) {\n const distance = summarizeRefDistance(graph, markerHash, activeRefHash, activeRefName);\n summary = hasInvariantWork ? `${distance} — missing invariant(s): ${missingList}` : distance;\n } else if (pendingCount === 0 && !hasInvariantWork) {\n summary = `Database is up to date (${appliedCount} migration${appliedCount !== 1 ? 's' : ''} applied)`;\n } else if (pendingCount === 0 && hasInvariantWork) {\n summary = `Missing invariant(s): ${missingList} — run 'prisma-next migrate --to ${activeRefName ?? '<ref>'}' to apply`;\n } else if (markerHash === undefined) {\n summary = `${pendingCount} pending migration(s) — database has no marker`;\n } else {\n summary = `${pendingCount} pending migration(s) — run 'prisma-next migrate' to apply`;\n }\n } else {\n summary = `${entries.length} migration(s) on disk`;\n }\n\n let pathDecision: MigrationStatusResult['pathDecision'];\n let routingUnreachable = false;\n if (mode === 'online') {\n const originHash = markerHash ?? EMPTY_CONTRACT_HASH;\n const outcome = findPathWithDecision(graph, originHash, targetHash, {\n ...ifDefined('refName', activeRefName),\n required: effectiveRequired,\n });\n if (outcome.kind === 'ok') {\n pathDecision = toPathDecisionResult(outcome.decision);\n } else if (outcome.kind === 'unsatisfiable') {\n return notOk(\n mapMigrationToolsError(\n errorNoInvariantPath({\n ...ifDefined('refName', activeRefName),\n required: [...effectiveRequired].sort(),\n missing: outcome.missing,\n structuralPath: outcome.structuralPath.map(toStructuralEdge),\n }),\n ),\n );\n } else {\n // outcome.kind === 'unreachable' — origin (marker) has no structural\n // path to the active target. `pendingCount` and `hasInvariantWork`\n // both report zero in this case, but emitting MIGRATION.UP_TO_DATE\n // would be wrong: the database simply cannot reach the requested\n // ref/contract from its current state. Suppress UP_TO_DATE below.\n routingUnreachable = true;\n }\n }\n\n if (mode === 'online') {\n if (markerHash !== undefined && !graph.nodes.has(markerHash) && markerHash === contractHash) {\n diagnostics.push({\n code: 'MIGRATION.MARKER_NOT_IN_HISTORY',\n severity: 'warn',\n message: 'Database matches the current contract but was updated directly (not via migrate)',\n hints: [\"Run 'prisma-next migration plan' to plan a migration to your current contract\"],\n });\n } else if (pendingCount > 0) {\n diagnostics.push({\n code: 'MIGRATION.DATABASE_BEHIND',\n severity: 'info',\n message: `${pendingCount} migration(s) pending`,\n hints: [\"Run 'prisma-next migrate' to apply pending migrations\"],\n });\n } else if (hasInvariantWork) {\n diagnostics.push({\n code: 'MIGRATION.INVARIANTS_PENDING',\n severity: 'info',\n message: `Missing required invariant(s): ${missingList}`,\n hints: [\n `Run 'prisma-next migrate --to ${activeRefName ?? '<ref>'}' to apply a path that covers the required invariants`,\n ],\n });\n } else if (!routingUnreachable) {\n diagnostics.push({\n code: 'MIGRATION.UP_TO_DATE',\n severity: 'info',\n message: 'Database is up to date',\n hints: [],\n });\n }\n }\n\n const result: MigrationStatusResult = {\n ok: true,\n mode,\n migrations: entries,\n targetHash,\n contractHash,\n summary,\n diagnostics,\n ...ifDefined('markerHash', markerHash),\n requiredInvariants,\n ...ifDefined('appliedInvariants', appliedInvariants),\n ...ifDefined('missingInvariants', missingInvariants),\n ...(statusRefs.length > 0 ? { refs: statusRefs } : {}),\n ...ifDefined('pathDecision', pathDecision),\n graph,\n bundles,\n edgeStatuses,\n ...ifDefined('activeRefHash', activeRefHash),\n ...ifDefined('activeRefName', activeRefName),\n spaces: aggregateSpaces,\n ...ifDefined('totalPendingAcrossSpaces', totalPendingAcrossSpaces),\n };\n return ok(result);\n}\n\nexport function createMigrationStatusCommand(): Command {\n const command = new Command('status');\n setCommandDescriptions(\n command,\n 'Show migration path and pending status',\n 'Shows which migrations are pending between the database marker and\\n' +\n 'the target contract. Requires a database connection for live status.\\n' +\n 'Use `migration graph` for topology, `migration log` for history,\\n' +\n 'and `migration list` for on-disk enumeration.',\n );\n setCommandExamples(command, [\n 'prisma-next migration status --db $DATABASE_URL',\n 'prisma-next migration status --to production --db $DATABASE_URL',\n ]);\n setCommandSeeAlso(command, [\n { verb: 'migration log', oneLiner: 'Show executed migration history' },\n { verb: 'migration list', oneLiner: 'List on-disk migrations' },\n { verb: 'migration graph', oneLiner: 'Show the migration graph topology' },\n { verb: 'migration show', oneLiner: 'Display migration package contents' },\n ]);\n addGlobalOptions(command)\n .option('--db <url>', 'Database connection string')\n .option('--config <path>', 'Path to prisma-next.config.ts')\n .option(\n '--to <contract>',\n 'Target contract reference (hash, prefix, ref name, migration dir name, <dir>^, or ./path)',\n )\n .option(\n '--from <contract>',\n 'Origin contract reference; same grammar as --to. Supplying --from switches to offline path computation.',\n )\n .action(async (options: MigrationStatusOptions) => {\n const flags = parseGlobalFlagsOrExit(options);\n const ui = createTerminalUI(flags);\n\n const result = await executeMigrationStatusCommand(options, flags, ui);\n\n const exitCode = handleResult(result, flags, ui, (statusResult) => {\n if (flags.json) {\n const {\n graph: _graph,\n bundles: _bundles,\n edgeStatuses: _edgeStatuses,\n activeRefHash: _activeRefHash,\n activeRefName: _activeRefName,\n diverged: _diverged,\n ...jsonResult\n } = statusResult;\n ui.output(JSON.stringify(jsonResult, null, 2));\n } else if (!flags.quiet) {\n const colorize = flags.color !== false;\n\n if (statusResult.graph) {\n const renderInput = migrationGraphToRenderInput({\n graph: statusResult.graph,\n mode: statusResult.mode,\n markerHash: statusResult.markerHash,\n contractHash: statusResult.contractHash,\n refs: statusResult.refs,\n activeRefHash: statusResult.activeRefHash,\n activeRefName: statusResult.activeRefName,\n edgeStatuses: statusResult.edgeStatuses,\n });\n\n const graphToRender = statusResult.diverged\n ? renderInput.graph\n : extractRelevantSubgraph(renderInput.graph, renderInput.relevantPaths);\n const dagreOptions = isLinearGraph(graphToRender) ? { ranksep: 1 } : undefined;\n const renderOptions = {\n ...renderInput.options,\n colorize,\n ...ifDefined('dagreOptions', dagreOptions),\n };\n const graphOutput = graphRenderer.render(graphToRender, renderOptions);\n ui.log(graphOutput);\n if (statusResult.mode === 'online') {\n ui.log(formatLegend(colorize));\n }\n }\n ui.log('');\n ui.log(formatStatusSummary(statusResult, colorize));\n }\n });\n\n process.exit(exitCode);\n });\n\n return command;\n}\n\nfunction formatLegend(colorize: boolean): string {\n const c = (fn: (s: string) => string, s: string) => (colorize ? fn(s) : s);\n const parts = [\n `${c(cyan, '✓')} applied`,\n `${c(yellow, '⧗')} pending`,\n `${c(magenta, '✗')} unreachable`,\n ];\n return c(dim, parts.join(' '));\n}\n\nexport function formatStatusSummary(result: MigrationStatusResult, colorize: boolean): string {\n const c = (fn: (s: string) => string, s: string) => (colorize ? fn(s) : s);\n const lines: string[] = [];\n\n const hasUnknown = result.migrations.some((e) => e.status === 'unknown');\n const pendingCount = result.migrations.filter((e) => e.status === 'pending').length;\n\n const hasWarnings = result.diagnostics?.some((d) => d.severity === 'warn') ?? false;\n // INVARIANTS_PENDING is filed at severity 'info' (per ADR 208) so the\n // warn-severity check above doesn't see it. It still represents pending\n // work, so it must promote the summary off the success icon.\n const hasInvariantPending =\n result.diagnostics?.some((d) => d.code === 'MIGRATION.INVARIANTS_PENDING') ?? false;\n\n if (result.mode === 'online') {\n if (hasUnknown || hasWarnings) {\n lines.push(`${c(yellow, '⚠')} ${result.summary}`);\n } else if (pendingCount === 0 && !hasInvariantPending) {\n lines.push(`${c(cyan, '✔')} ${result.summary}`);\n } else {\n lines.push(`${c(yellow, '⧗')} ${result.summary}`);\n }\n } else {\n lines.push(result.summary);\n }\n\n if (result.requiredInvariants.length > 0) {\n if (result.appliedInvariants !== undefined && result.missingInvariants !== undefined) {\n lines.push(`${c(dim, 'applied ')}${formatInvariantList(result.appliedInvariants)}`);\n lines.push(`${c(dim, 'missing ')}${formatInvariantList(result.missingInvariants)}`);\n } else {\n lines.push(`${c(dim, 'applied ')}(unknown — connect a database to evaluate)`);\n }\n }\n\n const warnings = result.diagnostics?.filter((d) => d.severity === 'warn') ?? [];\n for (const diag of warnings) {\n lines.push(`${c(yellow, '⚠')} ${diag.message}`);\n for (const hint of diag.hints) {\n lines.push(` ${c(dim, hint)}`);\n }\n }\n\n // Per-space section. Suppressed when there's no extension space —\n // the legacy single-space output already covers the app member.\n // When extensions exist, render every space (including the app)\n // for consistency, plus a cross-space pending total + apply hint.\n if (result.spaces?.some((s) => s.kind === 'extension')) {\n const total = result.totalPendingAcrossSpaces ?? 0;\n lines.push('');\n lines.push(c(dim, 'spaces'));\n for (const space of result.spaces) {\n lines.push(formatSpaceLine(space, c));\n }\n if (total > 0) {\n lines.push('');\n lines.push(\n `${c(yellow, '⧗')} ${total} pending migration(s) across ${result.spaces.length} space(s) — run 'prisma-next migrate' to apply`,\n );\n }\n }\n\n return lines.join('\\n');\n}\n\nfunction formatSpaceLine(\n space: MigrationStatusSpaceEntry,\n c: (fn: (s: string) => string, s: string) => string,\n): string {\n const glyph = (() => {\n if (space.status === 'up-to-date' || space.status === 'no-marker') return c(cyan, '✓');\n if (space.status === 'pending') return c(yellow, '⧗');\n if (space.status === 'unreachable' || space.status === 'never-planned') return c(magenta, '✗');\n return ' ';\n })();\n const tag = space.kind === 'app' ? '[app]' : '[ext]';\n const head = space.headHash.slice(0, 8);\n const marker =\n space.markerHash === undefined\n ? '(unknown)'\n : space.markerHash === null\n ? '(no marker)'\n : space.markerHash.slice(0, 8);\n const pending =\n space.pendingCount === undefined\n ? ''\n : space.pendingCount === 0\n ? c(dim, ' (up to date)')\n : c(yellow, ` (${space.pendingCount} pending)`);\n return ` ${glyph} ${c(dim, tag)} ${space.spaceId} → head ${c(dim, head)}, marker ${c(dim, marker)}${pending}`;\n}\n\nfunction formatInvariantList(ids: readonly string[]): string {\n return ids.length === 0 ? '(none)' : ids.join(', ');\n}\n\nfunction summarizeRefDistance(\n graph: MigrationGraph,\n markerHash: string,\n refHash: string,\n refName: string,\n): string {\n if (markerHash === refHash) return `At ref \"${refName}\" target`;\n\n const pathToRef = findPath(graph, markerHash, refHash);\n if (pathToRef) return `${pathToRef.length} migration(s) behind ref \"${refName}\"`;\n\n const pathFromRef = findPath(graph, refHash, markerHash);\n if (pathFromRef) return `${pathFromRef.length} migration(s) ahead of ref \"${refName}\"`;\n\n return `No path between database marker and ref \"${refName}\" target`;\n}\n"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { t as CliStructuredError } from "../cli-errors-DdcjVLJV.mjs";
1
+ import { t as CliStructuredError } from "../cli-errors-Czmx92Zy.mjs";
2
2
  import { Command } from "commander";
3
3
  import { Result } from "@prisma-next/utils/result";
4
4
  import { RefEntry } from "@prisma-next/migration-tools/refs";
@@ -1,8 +1,7 @@
1
1
  import { t as loadConfig } from "../config-loader-B6sJjXTv.mjs";
2
- import { _ as errorUnexpected, m as errorRuntime, t as CliStructuredError, v as mapMigrationToolsError, y as mapRefResolutionError } from "../cli-errors-CF60g2cG.mjs";
3
- import { _ as parseGlobalFlags, a as loadMigrationPackages, d as setCommandDescriptions, l as resolveMigrationPaths, t as addGlobalOptions, v as formatCommandHelp } from "../command-helpers-D3vL5yi8.mjs";
4
- import { t as handleResult } from "../result-handler-Bm_6dDYg.mjs";
5
- import { t as TerminalUI } from "../terminal-ui-XtOQsqe9.mjs";
2
+ import { b as mapMigrationToolsError, g as errorRuntime, t as CliStructuredError, x as mapRefResolutionError, y as errorUnexpected } from "../cli-errors-Djtz98Vm.mjs";
3
+ import { t as createTerminalUI } from "../terminal-ui-BiB_8KNo.mjs";
4
+ import { _ as parseGlobalFlags, a as loadMigrationPackages, b as formatCommandHelp, d as setCommandDescriptions, l as resolveMigrationPaths, t as addGlobalOptions, v as parseGlobalFlagsOrExit, y as handleResult } from "../command-helpers-DtavI0wJ.mjs";
6
5
  import { Command } from "commander";
7
6
  import { notOk, ok } from "@prisma-next/utils/result";
8
7
  import { MigrationToolsError } from "@prisma-next/migration-tools/errors";
@@ -82,11 +81,8 @@ function createRefSetCommand() {
82
81
  const command = new Command("set");
83
82
  setCommandDescriptions(command, "Set a ref to a contract reference", "Sets a named ref to point to a resolved contract reference (hash, alias, or path) in migrations/refs/.");
84
83
  addGlobalOptions(command).argument("<name>", "Ref name (e.g., staging, production)").argument("<contract>", "Contract reference (hash, prefix, ref name, migration dir name, <dir>^, or ./path)").option("--config <path>", "Path to prisma-next.config.ts").action(async (name, hash, options) => {
85
- const flags = parseGlobalFlags(options);
86
- const ui = new TerminalUI({
87
- color: flags.color,
88
- interactive: flags.interactive
89
- });
84
+ const flags = parseGlobalFlagsOrExit(options);
85
+ const ui = createTerminalUI(flags);
90
86
  const exitCode = handleResult(await executeRefSetCommand(name, hash, options), flags, ui, (value) => {
91
87
  if (flags.json) ui.output(JSON.stringify(value));
92
88
  else if (!flags.quiet) ui.output(`Set ref "${value.ref}" → ${value.hash}`);
@@ -99,11 +95,8 @@ function createRefDeleteCommand() {
99
95
  const command = new Command("delete");
100
96
  setCommandDescriptions(command, "Delete a ref", "Removes a named ref from migrations/refs/.");
101
97
  addGlobalOptions(command).argument("<name>", "Ref name to delete").option("--config <path>", "Path to prisma-next.config.ts").action(async (name, options) => {
102
- const flags = parseGlobalFlags(options);
103
- const ui = new TerminalUI({
104
- color: flags.color,
105
- interactive: flags.interactive
106
- });
98
+ const flags = parseGlobalFlagsOrExit(options);
99
+ const ui = createTerminalUI(flags);
107
100
  const exitCode = handleResult(await executeRefDeleteCommand(name, options), flags, ui, (value) => {
108
101
  if (flags.json) ui.output(JSON.stringify(value));
109
102
  else if (!flags.quiet) ui.output(`Deleted ref "${value.ref}"`);
@@ -116,11 +109,8 @@ function createRefListCommand() {
116
109
  const command = new Command("list");
117
110
  setCommandDescriptions(command, "List all refs", "Lists all named refs from migrations/refs/.");
118
111
  addGlobalOptions(command).option("--config <path>", "Path to prisma-next.config.ts").action(async (options) => {
119
- const flags = parseGlobalFlags(options);
120
- const ui = new TerminalUI({
121
- color: flags.color,
122
- interactive: flags.interactive
123
- });
112
+ const flags = parseGlobalFlagsOrExit(options);
113
+ const ui = createTerminalUI(flags);
124
114
  const exitCode = handleResult(await executeRefListCommand(options), flags, ui, (value) => {
125
115
  if (flags.json) ui.output(JSON.stringify(value));
126
116
  else if (!flags.quiet) {
@@ -1 +1 @@
1
- {"version":3,"file":"ref.mjs","names":[],"sources":["../../src/commands/ref.ts"],"sourcesContent":["import { MigrationToolsError } from '@prisma-next/migration-tools/errors';\nimport { parseContractRef } from '@prisma-next/migration-tools/ref-resolution';\nimport type { RefEntry } from '@prisma-next/migration-tools/refs';\nimport {\n deleteRef,\n readRefs,\n validateRefName,\n validateRefValue,\n writeRef,\n} from '@prisma-next/migration-tools/refs';\nimport { notOk, ok, type Result } from '@prisma-next/utils/result';\nimport { Command } from 'commander';\nimport { loadConfig } from '../config-loader';\nimport {\n CliStructuredError,\n errorRuntime,\n errorUnexpected,\n mapMigrationToolsError,\n mapRefResolutionError,\n} from '../utils/cli-errors';\nimport {\n addGlobalOptions,\n loadMigrationPackages,\n resolveMigrationPaths,\n setCommandDescriptions,\n} from '../utils/command-helpers';\nimport { formatCommandHelp } from '../utils/formatters/help';\nimport { parseGlobalFlags } from '../utils/global-flags';\nimport { handleResult } from '../utils/result-handler';\nimport { TerminalUI } from '../utils/terminal-ui';\n\ninterface RefSetResult {\n readonly ok: true;\n readonly ref: string;\n readonly hash: string;\n readonly invariants: readonly string[];\n}\n\ninterface RefDeleteResult {\n readonly ok: true;\n readonly ref: string;\n readonly deleted: true;\n}\n\ninterface RefListResult {\n readonly ok: true;\n readonly refs: Record<string, RefEntry>;\n}\n\nfunction mapError(error: unknown): CliStructuredError {\n if (MigrationToolsError.is(error)) {\n return mapMigrationToolsError(error);\n }\n return errorUnexpected(error instanceof Error ? error.message : String(error));\n}\n\nfunction cliErrorInvalidRefName(name: string): CliStructuredError {\n return errorRuntime(`Invalid ref name \"${name}\"`, {\n why: `Ref name \"${name}\" does not match the required format`,\n fix: 'Ref names must be lowercase alphanumeric with hyphens or forward slashes, no `.` or `..` segments',\n });\n}\n\nexport async function executeRefSetCommand(\n name: string,\n contractInput: string,\n options: { config?: string },\n): Promise<Result<RefSetResult, CliStructuredError>> {\n if (!validateRefName(name)) {\n return notOk(cliErrorInvalidRefName(name));\n }\n\n try {\n const config = await loadConfig(options.config);\n const { appMigrationsDir, refsDir } = resolveMigrationPaths(options.config, config);\n\n let resolvedHash: string;\n if (validateRefValue(contractInput)) {\n resolvedHash = contractInput;\n } else {\n const { graph } = await loadMigrationPackages(appMigrationsDir);\n const refs = await readRefs(refsDir);\n const refResult = parseContractRef(contractInput, { graph, refs });\n if (!refResult.ok) {\n return notOk(mapRefResolutionError(refResult.failure));\n }\n resolvedHash = refResult.value.hash;\n }\n\n const entry: RefEntry = { hash: resolvedHash, invariants: [] };\n await writeRef(refsDir, name, entry);\n return ok({ ok: true as const, ref: name, hash: resolvedHash, invariants: [] });\n } catch (error) {\n if (error instanceof CliStructuredError) return notOk(error);\n return notOk(mapError(error));\n }\n}\n\nexport async function executeRefDeleteCommand(\n name: string,\n options: { config?: string },\n): Promise<Result<RefDeleteResult, CliStructuredError>> {\n try {\n const config = await loadConfig(options.config);\n const { refsDir } = resolveMigrationPaths(options.config, config);\n await deleteRef(refsDir, name);\n return ok({ ok: true as const, ref: name, deleted: true as const });\n } catch (error) {\n if (error instanceof CliStructuredError) return notOk(error);\n return notOk(mapError(error));\n }\n}\n\nexport async function executeRefListCommand(options: {\n config?: string;\n}): Promise<Result<RefListResult, CliStructuredError>> {\n try {\n const config = await loadConfig(options.config);\n const { refsDir } = resolveMigrationPaths(options.config, config);\n const refs = await readRefs(refsDir);\n return ok({ ok: true as const, refs });\n } catch (error) {\n if (error instanceof CliStructuredError) return notOk(error);\n return notOk(mapError(error));\n }\n}\n\nfunction createRefSetCommand(): Command {\n const command = new Command('set');\n setCommandDescriptions(\n command,\n 'Set a ref to a contract reference',\n 'Sets a named ref to point to a resolved contract reference (hash, alias, or path) in migrations/refs/.',\n );\n addGlobalOptions(command)\n .argument('<name>', 'Ref name (e.g., staging, production)')\n .argument(\n '<contract>',\n 'Contract reference (hash, prefix, ref name, migration dir name, <dir>^, or ./path)',\n )\n .option('--config <path>', 'Path to prisma-next.config.ts')\n .action(\n async (\n name: string,\n hash: string,\n options: { config?: string; json?: string | boolean; quiet?: boolean },\n ) => {\n const flags = parseGlobalFlags(options);\n const ui = new TerminalUI({ color: flags.color, interactive: flags.interactive });\n const result = await executeRefSetCommand(name, hash, options);\n const exitCode = handleResult(result, flags, ui, (value) => {\n if (flags.json) {\n ui.output(JSON.stringify(value));\n } else if (!flags.quiet) {\n ui.output(`Set ref \"${value.ref}\" → ${value.hash}`);\n }\n });\n process.exit(exitCode);\n },\n );\n return command;\n}\n\nfunction createRefDeleteCommand(): Command {\n const command = new Command('delete');\n setCommandDescriptions(command, 'Delete a ref', 'Removes a named ref from migrations/refs/.');\n addGlobalOptions(command)\n .argument('<name>', 'Ref name to delete')\n .option('--config <path>', 'Path to prisma-next.config.ts')\n .action(\n async (\n name: string,\n options: { config?: string; json?: string | boolean; quiet?: boolean },\n ) => {\n const flags = parseGlobalFlags(options);\n const ui = new TerminalUI({ color: flags.color, interactive: flags.interactive });\n const result = await executeRefDeleteCommand(name, options);\n const exitCode = handleResult(result, flags, ui, (value) => {\n if (flags.json) {\n ui.output(JSON.stringify(value));\n } else if (!flags.quiet) {\n ui.output(`Deleted ref \"${value.ref}\"`);\n }\n });\n process.exit(exitCode);\n },\n );\n return command;\n}\n\nfunction createRefListCommand(): Command {\n const command = new Command('list');\n setCommandDescriptions(command, 'List all refs', 'Lists all named refs from migrations/refs/.');\n addGlobalOptions(command)\n .option('--config <path>', 'Path to prisma-next.config.ts')\n .action(async (options: { config?: string; json?: string | boolean; quiet?: boolean }) => {\n const flags = parseGlobalFlags(options);\n const ui = new TerminalUI({ color: flags.color, interactive: flags.interactive });\n const result = await executeRefListCommand(options);\n const exitCode = handleResult(result, flags, ui, (value) => {\n if (flags.json) {\n ui.output(JSON.stringify(value));\n } else if (!flags.quiet) {\n const entries = Object.entries(value.refs);\n if (entries.length === 0) {\n ui.output('No refs defined');\n } else {\n for (const [refName, entry] of entries) {\n const invariantsSuffix =\n entry.invariants.length > 0 ? ` [invariants: ${entry.invariants.join(', ')}]` : '';\n ui.output(`${refName} → ${entry.hash}${invariantsSuffix}`);\n }\n }\n }\n });\n process.exit(exitCode);\n });\n return command;\n}\n\nexport function createRefCommand(): Command {\n const command = new Command('ref');\n setCommandDescriptions(\n command,\n 'Manage contract refs',\n 'Manage named refs in migrations/refs/. Refs map logical environment\\n' +\n 'names (e.g., staging, production) to contract hashes.',\n );\n addGlobalOptions(command).configureHelp({\n formatHelp: (cmd) => formatCommandHelp({ command: cmd, flags: parseGlobalFlags({}) }),\n subcommandDescription: () => '',\n });\n command.addCommand(createRefSetCommand());\n command.addCommand(createRefDeleteCommand());\n command.addCommand(createRefListCommand());\n return command;\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"ref.mjs","names":[],"sources":["../../src/commands/ref.ts"],"sourcesContent":["import { MigrationToolsError } from '@prisma-next/migration-tools/errors';\nimport { parseContractRef } from '@prisma-next/migration-tools/ref-resolution';\nimport type { RefEntry } from '@prisma-next/migration-tools/refs';\nimport {\n deleteRef,\n readRefs,\n validateRefName,\n validateRefValue,\n writeRef,\n} from '@prisma-next/migration-tools/refs';\nimport { notOk, ok, type Result } from '@prisma-next/utils/result';\nimport { Command } from 'commander';\nimport { loadConfig } from '../config-loader';\nimport {\n CliStructuredError,\n errorRuntime,\n errorUnexpected,\n mapMigrationToolsError,\n mapRefResolutionError,\n} from '../utils/cli-errors';\nimport {\n addGlobalOptions,\n loadMigrationPackages,\n resolveMigrationPaths,\n setCommandDescriptions,\n} from '../utils/command-helpers';\nimport { formatCommandHelp } from '../utils/formatters/help';\nimport { parseGlobalFlags, parseGlobalFlagsOrExit } from '../utils/global-flags';\nimport { handleResult } from '../utils/result-handler';\nimport { createTerminalUI } from '../utils/terminal-ui';\n\ninterface RefSetResult {\n readonly ok: true;\n readonly ref: string;\n readonly hash: string;\n readonly invariants: readonly string[];\n}\n\ninterface RefDeleteResult {\n readonly ok: true;\n readonly ref: string;\n readonly deleted: true;\n}\n\ninterface RefListResult {\n readonly ok: true;\n readonly refs: Record<string, RefEntry>;\n}\n\nfunction mapError(error: unknown): CliStructuredError {\n if (MigrationToolsError.is(error)) {\n return mapMigrationToolsError(error);\n }\n return errorUnexpected(error instanceof Error ? error.message : String(error));\n}\n\nfunction cliErrorInvalidRefName(name: string): CliStructuredError {\n return errorRuntime(`Invalid ref name \"${name}\"`, {\n why: `Ref name \"${name}\" does not match the required format`,\n fix: 'Ref names must be lowercase alphanumeric with hyphens or forward slashes, no `.` or `..` segments',\n });\n}\n\nexport async function executeRefSetCommand(\n name: string,\n contractInput: string,\n options: { config?: string },\n): Promise<Result<RefSetResult, CliStructuredError>> {\n if (!validateRefName(name)) {\n return notOk(cliErrorInvalidRefName(name));\n }\n\n try {\n const config = await loadConfig(options.config);\n const { appMigrationsDir, refsDir } = resolveMigrationPaths(options.config, config);\n\n let resolvedHash: string;\n if (validateRefValue(contractInput)) {\n resolvedHash = contractInput;\n } else {\n const { graph } = await loadMigrationPackages(appMigrationsDir);\n const refs = await readRefs(refsDir);\n const refResult = parseContractRef(contractInput, { graph, refs });\n if (!refResult.ok) {\n return notOk(mapRefResolutionError(refResult.failure));\n }\n resolvedHash = refResult.value.hash;\n }\n\n const entry: RefEntry = { hash: resolvedHash, invariants: [] };\n await writeRef(refsDir, name, entry);\n return ok({ ok: true as const, ref: name, hash: resolvedHash, invariants: [] });\n } catch (error) {\n if (error instanceof CliStructuredError) return notOk(error);\n return notOk(mapError(error));\n }\n}\n\nexport async function executeRefDeleteCommand(\n name: string,\n options: { config?: string },\n): Promise<Result<RefDeleteResult, CliStructuredError>> {\n try {\n const config = await loadConfig(options.config);\n const { refsDir } = resolveMigrationPaths(options.config, config);\n await deleteRef(refsDir, name);\n return ok({ ok: true as const, ref: name, deleted: true as const });\n } catch (error) {\n if (error instanceof CliStructuredError) return notOk(error);\n return notOk(mapError(error));\n }\n}\n\nexport async function executeRefListCommand(options: {\n config?: string;\n}): Promise<Result<RefListResult, CliStructuredError>> {\n try {\n const config = await loadConfig(options.config);\n const { refsDir } = resolveMigrationPaths(options.config, config);\n const refs = await readRefs(refsDir);\n return ok({ ok: true as const, refs });\n } catch (error) {\n if (error instanceof CliStructuredError) return notOk(error);\n return notOk(mapError(error));\n }\n}\n\nfunction createRefSetCommand(): Command {\n const command = new Command('set');\n setCommandDescriptions(\n command,\n 'Set a ref to a contract reference',\n 'Sets a named ref to point to a resolved contract reference (hash, alias, or path) in migrations/refs/.',\n );\n addGlobalOptions(command)\n .argument('<name>', 'Ref name (e.g., staging, production)')\n .argument(\n '<contract>',\n 'Contract reference (hash, prefix, ref name, migration dir name, <dir>^, or ./path)',\n )\n .option('--config <path>', 'Path to prisma-next.config.ts')\n .action(\n async (\n name: string,\n hash: string,\n options: { config?: string; json?: string | boolean; quiet?: boolean },\n ) => {\n const flags = parseGlobalFlagsOrExit(options);\n const ui = createTerminalUI(flags);\n const result = await executeRefSetCommand(name, hash, options);\n const exitCode = handleResult(result, flags, ui, (value) => {\n if (flags.json) {\n ui.output(JSON.stringify(value));\n } else if (!flags.quiet) {\n ui.output(`Set ref \"${value.ref}\" → ${value.hash}`);\n }\n });\n process.exit(exitCode);\n },\n );\n return command;\n}\n\nfunction createRefDeleteCommand(): Command {\n const command = new Command('delete');\n setCommandDescriptions(command, 'Delete a ref', 'Removes a named ref from migrations/refs/.');\n addGlobalOptions(command)\n .argument('<name>', 'Ref name to delete')\n .option('--config <path>', 'Path to prisma-next.config.ts')\n .action(\n async (\n name: string,\n options: { config?: string; json?: string | boolean; quiet?: boolean },\n ) => {\n const flags = parseGlobalFlagsOrExit(options);\n const ui = createTerminalUI(flags);\n const result = await executeRefDeleteCommand(name, options);\n const exitCode = handleResult(result, flags, ui, (value) => {\n if (flags.json) {\n ui.output(JSON.stringify(value));\n } else if (!flags.quiet) {\n ui.output(`Deleted ref \"${value.ref}\"`);\n }\n });\n process.exit(exitCode);\n },\n );\n return command;\n}\n\nfunction createRefListCommand(): Command {\n const command = new Command('list');\n setCommandDescriptions(command, 'List all refs', 'Lists all named refs from migrations/refs/.');\n addGlobalOptions(command)\n .option('--config <path>', 'Path to prisma-next.config.ts')\n .action(async (options: { config?: string; json?: string | boolean; quiet?: boolean }) => {\n const flags = parseGlobalFlagsOrExit(options);\n const ui = createTerminalUI(flags);\n const result = await executeRefListCommand(options);\n const exitCode = handleResult(result, flags, ui, (value) => {\n if (flags.json) {\n ui.output(JSON.stringify(value));\n } else if (!flags.quiet) {\n const entries = Object.entries(value.refs);\n if (entries.length === 0) {\n ui.output('No refs defined');\n } else {\n for (const [refName, entry] of entries) {\n const invariantsSuffix =\n entry.invariants.length > 0 ? ` [invariants: ${entry.invariants.join(', ')}]` : '';\n ui.output(`${refName} → ${entry.hash}${invariantsSuffix}`);\n }\n }\n }\n });\n process.exit(exitCode);\n });\n return command;\n}\n\nexport function createRefCommand(): Command {\n const command = new Command('ref');\n setCommandDescriptions(\n command,\n 'Manage contract refs',\n 'Manage named refs in migrations/refs/. Refs map logical environment\\n' +\n 'names (e.g., staging, production) to contract hashes.',\n );\n addGlobalOptions(command).configureHelp({\n formatHelp: (cmd) => formatCommandHelp({ command: cmd, flags: parseGlobalFlags({}) }),\n subcommandDescription: () => '',\n });\n command.addCommand(createRefSetCommand());\n command.addCommand(createRefDeleteCommand());\n command.addCommand(createRefListCommand());\n return command;\n}\n"],"mappings":"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"}
@@ -1,7 +1,7 @@
1
1
  import { t as __exportAll } from "./rolldown-runtime-twds-ZHy.mjs";
2
2
  import { t as loadConfig } from "./config-loader-B6sJjXTv.mjs";
3
- import { i as errorContractConfigMissing, m as errorRuntime } from "./cli-errors-CF60g2cG.mjs";
4
- import { t as assertFrameworkComponentsCompatible } from "./framework-components-xFLFpZUO.mjs";
3
+ import { g as errorRuntime, i as errorContractConfigMissing } from "./cli-errors-Djtz98Vm.mjs";
4
+ import { t as assertFrameworkComponentsCompatible } from "./framework-components-65gOHkHB.mjs";
5
5
  import { t as enrichContract } from "./contract-enrichment-Dani0mMW.mjs";
6
6
  import { createRequire } from "node:module";
7
7
  import { emit, getEmittedArtifactPaths } from "@prisma-next/emitter";
@@ -328,4 +328,4 @@ async function executeContractEmit(options) {
328
328
  //#endregion
329
329
  export { executeContractEmit as n, disposeEmitQueue as r, contract_emit_exports as t };
330
330
 
331
- //# sourceMappingURL=contract-emit-iynA3BCA.mjs.map
331
+ //# sourceMappingURL=contract-emit-bcrpT-wD.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"contract-emit-iynA3BCA.mjs","names":["isRecord"],"sources":["../src/utils/emit-queue.ts","../src/utils/publish-contract-artifact-pair.ts","../src/utils/validate-contract-deps.ts","../src/control-api/operations/contract-emit.ts"],"sourcesContent":["/**\n * Per-output FIFO queue for `executeContractEmit`.\n *\n * Ensures that at most one emit (load → resolve source → emit bytes → publish)\n * runs per output JSON path at a time. Concurrent calls for the same path\n * line up behind the in-flight one and run in submission order; the user-visible\n * outcome is \"last submission wins on disk\" without any supersession bookkeeping.\n *\n * Long-lived hosts (Vite dev server, watch CLIs) must call `disposeEmitQueue`\n * when they stop publishing to a path, otherwise the module-global `Map`\n * accumulates one entry per unique output path for the lifetime of the process.\n */\nconst emitQueues = new Map<string, Promise<unknown>>();\n\nexport function queueEmitByOutput<T>(outputJsonPath: string, action: () => Promise<T>): Promise<T> {\n const previous = emitQueues.get(outputJsonPath) ?? Promise.resolve();\n // Continue regardless of the previous task's outcome — a failed emit must not\n // block subsequent ones. The current task's outcome propagates via `next`.\n const next = previous.then(action, action);\n emitQueues.set(outputJsonPath, next);\n return next;\n}\n\nexport function disposeEmitQueue(outputJsonPath: string): void {\n emitQueues.delete(outputJsonPath);\n}\n","import { readFile, rename, rm, writeFile } from 'node:fs/promises';\nimport { basename, dirname, join } from 'pathe';\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === 'object' && value !== null;\n}\n\nfunction createTempArtifactPath(path: string, publicationToken: string, phase: string): string {\n return join(dirname(path), `.${basename(path)}.${process.pid}.${publicationToken}.${phase}.tmp`);\n}\n\ntype PreviousArtifact = { readonly content: string } | 'remove';\n\nasync function readExistingArtifact(path: string): Promise<PreviousArtifact> {\n try {\n return { content: await readFile(path, 'utf-8') };\n } catch (error) {\n if (isRecord(error) && error['code'] === 'ENOENT') {\n return 'remove';\n }\n throw error;\n }\n}\n\nasync function restoreArtifact(\n path: string,\n previous: PreviousArtifact,\n publicationToken: string,\n): Promise<void> {\n if (previous === 'remove') {\n await rm(path, { force: true });\n return;\n }\n\n const restorePath = createTempArtifactPath(path, publicationToken, 'rollback');\n await writeFile(restorePath, previous.content, 'utf-8');\n try {\n await rename(restorePath, path);\n } finally {\n await rm(restorePath, { force: true });\n }\n}\n\ninterface PublishEntry {\n readonly tempPath: string;\n readonly outputPath: string;\n readonly previous: PreviousArtifact;\n}\n\nfunction withRollbackFailureCause(error: unknown, rollbackFailures: readonly unknown[]): Error {\n const rollbackCause = new AggregateError(\n rollbackFailures,\n 'Failed to restore published artifacts',\n );\n\n if (error instanceof Error) {\n Object.defineProperty(error, 'cause', {\n value: rollbackCause,\n configurable: true,\n writable: true,\n });\n return error;\n }\n\n return new Error(String(error), { cause: rollbackCause });\n}\n\nasync function publishPairWithRollback(\n entries: readonly PublishEntry[],\n publicationToken: string,\n): Promise<void> {\n const replaced: PublishEntry[] = [];\n try {\n for (const entry of entries) {\n await rename(entry.tempPath, entry.outputPath);\n replaced.push(entry);\n }\n } catch (error) {\n const rollbackResults = await Promise.allSettled(\n replaced.map((entry) => restoreArtifact(entry.outputPath, entry.previous, publicationToken)),\n );\n const rollbackFailures = rollbackResults.flatMap((result) =>\n result.status === 'rejected' ? [result.reason] : [],\n );\n\n if (rollbackFailures.length > 0) {\n throw withRollbackFailureCause(error, rollbackFailures);\n }\n\n throw error;\n }\n}\n\nexport async function publishContractArtifactPair({\n outputJsonPath,\n outputDtsPath,\n contractJson,\n contractDts,\n publicationToken,\n beforePublish,\n}: {\n readonly outputJsonPath: string;\n readonly outputDtsPath: string;\n readonly contractJson: string;\n readonly contractDts: string;\n readonly publicationToken: string;\n readonly beforePublish?: () => Promise<boolean> | boolean;\n}): Promise<boolean> {\n const tempJsonPath = createTempArtifactPath(outputJsonPath, publicationToken, 'next');\n const tempDtsPath = createTempArtifactPath(outputDtsPath, publicationToken, 'next');\n\n try {\n await writeFile(tempJsonPath, contractJson, 'utf-8');\n await writeFile(tempDtsPath, contractDts, 'utf-8');\n\n if ((await beforePublish?.()) === false) {\n return false;\n }\n\n const previousJson = await readExistingArtifact(outputJsonPath);\n const previousDts = await readExistingArtifact(outputDtsPath);\n\n await publishPairWithRollback(\n [\n { tempPath: tempDtsPath, outputPath: outputDtsPath, previous: previousDts },\n { tempPath: tempJsonPath, outputPath: outputJsonPath, previous: previousJson },\n ],\n publicationToken,\n );\n return true;\n } finally {\n await Promise.allSettled([rm(tempJsonPath, { force: true }), rm(tempDtsPath, { force: true })]);\n }\n}\n","import { createRequire } from 'node:module';\n\nconst IMPORT_PATTERN = /import\\s+type\\s+.*?\\s+from\\s+['\"](@[^/]+\\/[^/'\"]+)/g;\n\nexport function extractPackageSpecifiers(dtsContent: string): string[] {\n const packages = new Set<string>();\n for (const match of dtsContent.matchAll(IMPORT_PATTERN)) {\n const pkg = match[1];\n if (pkg) packages.add(pkg);\n }\n return [...packages];\n}\n\nexport interface ContractDepsValidation {\n readonly missing: readonly string[];\n readonly warning?: string;\n}\n\nexport function validateContractDeps(\n dtsContent: string,\n projectRoot: string,\n): ContractDepsValidation {\n const packages = extractPackageSpecifiers(dtsContent);\n const resolve = createRequire(`${projectRoot}/package.json`);\n\n const missing: string[] = [];\n for (const pkg of packages) {\n try {\n resolve.resolve(`${pkg}/package.json`);\n } catch {\n missing.push(pkg);\n }\n }\n\n if (missing.length === 0) {\n return { missing };\n }\n\n const list = missing.map((p) => ` - ${p}`).join('\\n');\n const warning = [\n 'contract.d.ts imports types from packages that are not installed:',\n list,\n '',\n 'Install them with your package manager:',\n ...missing.map((p) => ` ${p}`),\n ].join('\\n');\n\n return { missing, warning };\n}\n","import { mkdir } from 'node:fs/promises';\nimport type { Contract } from '@prisma-next/contract/types';\nimport { emit, getEmittedArtifactPaths } from '@prisma-next/emitter';\nimport { createControlStack } from '@prisma-next/framework-components/control';\nimport { abortable } from '@prisma-next/utils/abortable';\nimport { ifDefined } from '@prisma-next/utils/defined';\nimport type { JsonObject } from '@prisma-next/utils/json';\nimport { dirname } from 'pathe';\nimport { loadConfig } from '../../config-loader';\nimport { errorContractConfigMissing, errorRuntime } from '../../utils/cli-errors';\nimport { queueEmitByOutput } from '../../utils/emit-queue';\nimport { assertFrameworkComponentsCompatible } from '../../utils/framework-components';\nimport { publishContractArtifactPair } from '../../utils/publish-contract-artifact-pair';\nimport { validateContractDeps } from '../../utils/validate-contract-deps';\nimport { enrichContract } from '../contract-enrichment';\nimport type {\n ContractEmitOptions,\n ContractEmitResult,\n ControlActionName,\n OnControlProgress,\n} from '../types';\n\nconst EMIT_ACTION: ControlActionName = 'emit';\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === 'object' && value !== null;\n}\n\nfunction startSpan(onProgress: OnControlProgress | undefined, spanId: string, label: string): void {\n onProgress?.({ action: EMIT_ACTION, kind: 'spanStart', spanId, label });\n}\n\nfunction endSpan(\n onProgress: OnControlProgress | undefined,\n spanId: string,\n outcome: 'ok' | 'error',\n): void {\n onProgress?.({ action: EMIT_ACTION, kind: 'spanEnd', spanId, outcome });\n}\n\nfunction failedToResolveContractSource(why: string, fix: string, meta?: Record<string, unknown>) {\n return errorRuntime('Failed to resolve contract source', {\n why,\n fix,\n ...ifDefined('meta', meta),\n });\n}\n\ntype ValidatedProviderResult =\n | { readonly ok: true; readonly value: unknown }\n | { readonly ok: false; readonly error: ReturnType<typeof errorRuntime> };\n\nfunction diagnosticLocationSuffix(diagnostic: Record<string, unknown>): string {\n const sourceId = typeof diagnostic['sourceId'] === 'string' ? diagnostic['sourceId'] : undefined;\n const span = isRecord(diagnostic['span']) ? diagnostic['span'] : undefined;\n const start = span && isRecord(span['start']) ? span['start'] : undefined;\n const line = start && typeof start['line'] === 'number' ? start['line'] : undefined;\n const column = start && typeof start['column'] === 'number' ? start['column'] : undefined;\n if (sourceId && line !== undefined && column !== undefined) {\n return ` (${sourceId}:${line}:${column})`;\n }\n if (sourceId) {\n return ` (${sourceId})`;\n }\n return '';\n}\n\nfunction mapDiagnosticsToIssues(\n diagnostics: readonly unknown[],\n): ReadonlyArray<{ readonly kind: string; readonly message: string }> {\n const issues: { readonly kind: string; readonly message: string }[] = [];\n for (const raw of diagnostics) {\n if (!isRecord(raw)) continue;\n const code = typeof raw['code'] === 'string' ? raw['code'] : 'diagnostic';\n const message = typeof raw['message'] === 'string' ? raw['message'] : '';\n issues.push({ kind: code, message: `${message}${diagnosticLocationSuffix(raw)}` });\n }\n return issues;\n}\n\nfunction validateProviderResult(providerResult: unknown): ValidatedProviderResult {\n if (!isRecord(providerResult) || typeof providerResult['ok'] !== 'boolean') {\n return {\n ok: false,\n error: failedToResolveContractSource(\n 'Contract source provider returned malformed result shape.',\n 'Ensure contract.source.load resolves to ok(Contract) or notOk({ summary, diagnostics }).',\n ),\n };\n }\n\n if (providerResult['ok']) {\n if (!('value' in providerResult)) {\n return {\n ok: false,\n error: failedToResolveContractSource(\n 'Contract source provider returned malformed success result: missing value.',\n 'Ensure contract.source.load success payload is ok(Contract).',\n ),\n };\n }\n return { ok: true, value: providerResult['value'] };\n }\n\n const failure = providerResult['failure'];\n if (\n !isRecord(failure) ||\n typeof failure['summary'] !== 'string' ||\n !Array.isArray(failure['diagnostics'])\n ) {\n return {\n ok: false,\n error: failedToResolveContractSource(\n 'Contract source provider returned malformed failure result: expected summary and diagnostics.',\n 'Ensure contract.source.load failure payload is notOk({ summary, diagnostics, meta? }).',\n ),\n };\n }\n return {\n ok: false,\n error: failedToResolveContractSource(\n String(failure['summary']),\n 'Fix contract source diagnostics and return ok(Contract).',\n {\n diagnostics: failure['diagnostics'],\n issues: mapDiagnosticsToIssues(failure['diagnostics']),\n ...ifDefined('providerMeta', failure['meta']),\n },\n ),\n };\n}\n\n/**\n * Canonical contract emit operation.\n *\n * This is the SINGLE publication path used by both the CLI command\n * (`prisma-next contract emit`) and the Vite plugin\n * (`@prisma-next/vite-plugin-contract-emit`). New callers must go through this\n * function rather than re-implementing load → emit → publish.\n *\n * The whole flow (load config → resolve source → emit bytes → atomic publish)\n * is serialized per output JSON path via `queueEmitByOutput`. Concurrent calls\n * for the same output line up FIFO; the user-visible outcome is \"last\n * submission wins on disk\" without any supersession bookkeeping. Within a\n * single emit, `publishContractArtifactPair` stages temp files, renames\n * `contract.d.ts` before `contract.json`, and attempts to restore the previous\n * pair if either rename fails — so type-only consumers never observe a\n * mismatched pair.\n *\n * @throws {CliStructuredError} on config/source/validation problems\n * @throws {DOMException} `AbortError` if cancelled via `signal`\n */\nexport async function executeContractEmit(\n options: ContractEmitOptions,\n): Promise<ContractEmitResult> {\n const { configPath, signal = new AbortController().signal, onProgress } = options;\n const unlessAborted = abortable(signal);\n\n const config = await unlessAborted(loadConfig(configPath));\n\n if (!config.contract) {\n throw errorContractConfigMissing({\n why: 'Config.contract is required for emit. Define it in your config: contract: { source: ..., output: ... }',\n });\n }\n\n const contractConfig = config.contract;\n\n if (!contractConfig.output) {\n throw errorContractConfigMissing({\n why: 'Contract config must have output path. This should not happen if defineConfig() was used.',\n });\n }\n\n if (typeof contractConfig.source?.load !== 'function') {\n throw errorContractConfigMissing({\n why: 'Contract config must include a valid source provider object',\n });\n }\n\n let outputPaths: ReturnType<typeof getEmittedArtifactPaths>;\n try {\n outputPaths = getEmittedArtifactPaths(contractConfig.output);\n } catch (error) {\n throw errorContractConfigMissing({\n why: error instanceof Error ? error.message : String(error),\n });\n }\n const { jsonPath: outputJsonPath, dtsPath: outputDtsPath } = outputPaths;\n\n return queueEmitByOutput(outputJsonPath, async () => {\n const stack = createControlStack(config);\n\n const sourceContext = {\n composedExtensionPacks: stack.extensionPacks.map((p) => p.id),\n scalarTypeDescriptors: stack.scalarTypeDescriptors,\n authoringContributions: stack.authoringContributions,\n codecLookup: stack.codecLookup,\n controlMutationDefaults: stack.controlMutationDefaults,\n resolvedInputs: contractConfig.source.inputs ?? [],\n };\n\n startSpan(onProgress, 'resolveSource', 'Resolving contract source...');\n let providerResult: Awaited<ReturnType<typeof contractConfig.source.load>>;\n try {\n providerResult = await unlessAborted(contractConfig.source.load(sourceContext));\n } catch (error) {\n endSpan(onProgress, 'resolveSource', 'error');\n if (signal.aborted || (isRecord(error) && error['name'] === 'AbortError')) {\n throw error;\n }\n throw failedToResolveContractSource(\n error instanceof Error ? error.message : String(error),\n 'Ensure contract.source.load resolves to ok(Contract) or returns structured diagnostics.',\n );\n }\n\n const validatedContract = validateProviderResult(providerResult);\n if (!validatedContract.ok) {\n endSpan(onProgress, 'resolveSource', 'error');\n throw validatedContract.error;\n }\n endSpan(onProgress, 'resolveSource', 'ok');\n\n startSpan(onProgress, 'emit', 'Emitting contract...');\n let emitResult: Awaited<ReturnType<typeof emit>>;\n try {\n const familyInstance = config.family.create(stack);\n const rawComponents = [config.target, config.adapter, ...(config.extensionPacks ?? [])];\n const frameworkComponents = assertFrameworkComponentsCompatible(\n config.family.familyId,\n config.target.targetId,\n rawComponents,\n );\n // Blind cast: `validateProviderResult` upstream has already\n // pinned `validatedContract.value` to the provider's loose\n // `Contract` envelope, but the local `Contract` type at this\n // call site is the precise structural interface. Re-narrowing\n // the envelope into the precise type is exactly what the\n // `familyInstance.deserializeContract(enrichedIR)` call on the\n // next line does — the cast just defers the structural check\n // by one statement so `enrichContract` can decorate first.\n const enrichedIR = enrichContract(\n validatedContract.value as unknown as Contract,\n frameworkComponents,\n );\n familyInstance.deserializeContract(enrichedIR);\n // Each target's descriptor ships a `contractSerializer` SPI; the\n // framework canonicalizer threads its `serializeContract` so the\n // on-disk JSON envelope is constructed by target-owned code\n // rather than by walking the in-memory contract with\n // `Object.entries` (which would leak runtime-only class API\n // fields into the persisted shape).\n const serializeContract = (c: Contract): JsonObject =>\n config.target.contractSerializer.serializeContract(c);\n emitResult = await unlessAborted(\n emit(enrichedIR, stack, config.family.emission, {\n outputJsonPath,\n serializeContract,\n }),\n );\n } catch (error) {\n endSpan(onProgress, 'emit', 'error');\n throw error;\n }\n endSpan(onProgress, 'emit', 'ok');\n\n await unlessAborted(mkdir(dirname(outputJsonPath), { recursive: true }));\n await publishContractArtifactPair({\n outputJsonPath,\n outputDtsPath,\n contractJson: emitResult.contractJson,\n contractDts: emitResult.contractDts,\n publicationToken: String(process.hrtime.bigint()),\n });\n\n const validationWarning = validateContractDeps(\n emitResult.contractDts,\n dirname(outputDtsPath),\n ).warning;\n\n return {\n storageHash: emitResult.storageHash,\n ...ifDefined('executionHash', emitResult.executionHash),\n profileHash: emitResult.profileHash,\n files: {\n json: outputJsonPath,\n dts: outputDtsPath,\n },\n ...ifDefined('validationWarning', validationWarning),\n };\n });\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"contract-emit-bcrpT-wD.mjs","names":["isRecord"],"sources":["../src/utils/emit-queue.ts","../src/utils/publish-contract-artifact-pair.ts","../src/utils/validate-contract-deps.ts","../src/control-api/operations/contract-emit.ts"],"sourcesContent":["/**\n * Per-output FIFO queue for `executeContractEmit`.\n *\n * Ensures that at most one emit (load → resolve source → emit bytes → publish)\n * runs per output JSON path at a time. Concurrent calls for the same path\n * line up behind the in-flight one and run in submission order; the user-visible\n * outcome is \"last submission wins on disk\" without any supersession bookkeeping.\n *\n * Long-lived hosts (Vite dev server, watch CLIs) must call `disposeEmitQueue`\n * when they stop publishing to a path, otherwise the module-global `Map`\n * accumulates one entry per unique output path for the lifetime of the process.\n */\nconst emitQueues = new Map<string, Promise<unknown>>();\n\nexport function queueEmitByOutput<T>(outputJsonPath: string, action: () => Promise<T>): Promise<T> {\n const previous = emitQueues.get(outputJsonPath) ?? Promise.resolve();\n // Continue regardless of the previous task's outcome — a failed emit must not\n // block subsequent ones. The current task's outcome propagates via `next`.\n const next = previous.then(action, action);\n emitQueues.set(outputJsonPath, next);\n return next;\n}\n\nexport function disposeEmitQueue(outputJsonPath: string): void {\n emitQueues.delete(outputJsonPath);\n}\n","import { readFile, rename, rm, writeFile } from 'node:fs/promises';\nimport { basename, dirname, join } from 'pathe';\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === 'object' && value !== null;\n}\n\nfunction createTempArtifactPath(path: string, publicationToken: string, phase: string): string {\n return join(dirname(path), `.${basename(path)}.${process.pid}.${publicationToken}.${phase}.tmp`);\n}\n\ntype PreviousArtifact = { readonly content: string } | 'remove';\n\nasync function readExistingArtifact(path: string): Promise<PreviousArtifact> {\n try {\n return { content: await readFile(path, 'utf-8') };\n } catch (error) {\n if (isRecord(error) && error['code'] === 'ENOENT') {\n return 'remove';\n }\n throw error;\n }\n}\n\nasync function restoreArtifact(\n path: string,\n previous: PreviousArtifact,\n publicationToken: string,\n): Promise<void> {\n if (previous === 'remove') {\n await rm(path, { force: true });\n return;\n }\n\n const restorePath = createTempArtifactPath(path, publicationToken, 'rollback');\n await writeFile(restorePath, previous.content, 'utf-8');\n try {\n await rename(restorePath, path);\n } finally {\n await rm(restorePath, { force: true });\n }\n}\n\ninterface PublishEntry {\n readonly tempPath: string;\n readonly outputPath: string;\n readonly previous: PreviousArtifact;\n}\n\nfunction withRollbackFailureCause(error: unknown, rollbackFailures: readonly unknown[]): Error {\n const rollbackCause = new AggregateError(\n rollbackFailures,\n 'Failed to restore published artifacts',\n );\n\n if (error instanceof Error) {\n Object.defineProperty(error, 'cause', {\n value: rollbackCause,\n configurable: true,\n writable: true,\n });\n return error;\n }\n\n return new Error(String(error), { cause: rollbackCause });\n}\n\nasync function publishPairWithRollback(\n entries: readonly PublishEntry[],\n publicationToken: string,\n): Promise<void> {\n const replaced: PublishEntry[] = [];\n try {\n for (const entry of entries) {\n await rename(entry.tempPath, entry.outputPath);\n replaced.push(entry);\n }\n } catch (error) {\n const rollbackResults = await Promise.allSettled(\n replaced.map((entry) => restoreArtifact(entry.outputPath, entry.previous, publicationToken)),\n );\n const rollbackFailures = rollbackResults.flatMap((result) =>\n result.status === 'rejected' ? [result.reason] : [],\n );\n\n if (rollbackFailures.length > 0) {\n throw withRollbackFailureCause(error, rollbackFailures);\n }\n\n throw error;\n }\n}\n\nexport async function publishContractArtifactPair({\n outputJsonPath,\n outputDtsPath,\n contractJson,\n contractDts,\n publicationToken,\n beforePublish,\n}: {\n readonly outputJsonPath: string;\n readonly outputDtsPath: string;\n readonly contractJson: string;\n readonly contractDts: string;\n readonly publicationToken: string;\n readonly beforePublish?: () => Promise<boolean> | boolean;\n}): Promise<boolean> {\n const tempJsonPath = createTempArtifactPath(outputJsonPath, publicationToken, 'next');\n const tempDtsPath = createTempArtifactPath(outputDtsPath, publicationToken, 'next');\n\n try {\n await writeFile(tempJsonPath, contractJson, 'utf-8');\n await writeFile(tempDtsPath, contractDts, 'utf-8');\n\n if ((await beforePublish?.()) === false) {\n return false;\n }\n\n const previousJson = await readExistingArtifact(outputJsonPath);\n const previousDts = await readExistingArtifact(outputDtsPath);\n\n await publishPairWithRollback(\n [\n { tempPath: tempDtsPath, outputPath: outputDtsPath, previous: previousDts },\n { tempPath: tempJsonPath, outputPath: outputJsonPath, previous: previousJson },\n ],\n publicationToken,\n );\n return true;\n } finally {\n await Promise.allSettled([rm(tempJsonPath, { force: true }), rm(tempDtsPath, { force: true })]);\n }\n}\n","import { createRequire } from 'node:module';\n\nconst IMPORT_PATTERN = /import\\s+type\\s+.*?\\s+from\\s+['\"](@[^/]+\\/[^/'\"]+)/g;\n\nexport function extractPackageSpecifiers(dtsContent: string): string[] {\n const packages = new Set<string>();\n for (const match of dtsContent.matchAll(IMPORT_PATTERN)) {\n const pkg = match[1];\n if (pkg) packages.add(pkg);\n }\n return [...packages];\n}\n\nexport interface ContractDepsValidation {\n readonly missing: readonly string[];\n readonly warning?: string;\n}\n\nexport function validateContractDeps(\n dtsContent: string,\n projectRoot: string,\n): ContractDepsValidation {\n const packages = extractPackageSpecifiers(dtsContent);\n const resolve = createRequire(`${projectRoot}/package.json`);\n\n const missing: string[] = [];\n for (const pkg of packages) {\n try {\n resolve.resolve(`${pkg}/package.json`);\n } catch {\n missing.push(pkg);\n }\n }\n\n if (missing.length === 0) {\n return { missing };\n }\n\n const list = missing.map((p) => ` - ${p}`).join('\\n');\n const warning = [\n 'contract.d.ts imports types from packages that are not installed:',\n list,\n '',\n 'Install them with your package manager:',\n ...missing.map((p) => ` ${p}`),\n ].join('\\n');\n\n return { missing, warning };\n}\n","import { mkdir } from 'node:fs/promises';\nimport type { Contract } from '@prisma-next/contract/types';\nimport { emit, getEmittedArtifactPaths } from '@prisma-next/emitter';\nimport { createControlStack } from '@prisma-next/framework-components/control';\nimport { abortable } from '@prisma-next/utils/abortable';\nimport { ifDefined } from '@prisma-next/utils/defined';\nimport type { JsonObject } from '@prisma-next/utils/json';\nimport { dirname } from 'pathe';\nimport { loadConfig } from '../../config-loader';\nimport { errorContractConfigMissing, errorRuntime } from '../../utils/cli-errors';\nimport { queueEmitByOutput } from '../../utils/emit-queue';\nimport { assertFrameworkComponentsCompatible } from '../../utils/framework-components';\nimport { publishContractArtifactPair } from '../../utils/publish-contract-artifact-pair';\nimport { validateContractDeps } from '../../utils/validate-contract-deps';\nimport { enrichContract } from '../contract-enrichment';\nimport type {\n ContractEmitOptions,\n ContractEmitResult,\n ControlActionName,\n OnControlProgress,\n} from '../types';\n\nconst EMIT_ACTION: ControlActionName = 'emit';\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === 'object' && value !== null;\n}\n\nfunction startSpan(onProgress: OnControlProgress | undefined, spanId: string, label: string): void {\n onProgress?.({ action: EMIT_ACTION, kind: 'spanStart', spanId, label });\n}\n\nfunction endSpan(\n onProgress: OnControlProgress | undefined,\n spanId: string,\n outcome: 'ok' | 'error',\n): void {\n onProgress?.({ action: EMIT_ACTION, kind: 'spanEnd', spanId, outcome });\n}\n\nfunction failedToResolveContractSource(why: string, fix: string, meta?: Record<string, unknown>) {\n return errorRuntime('Failed to resolve contract source', {\n why,\n fix,\n ...ifDefined('meta', meta),\n });\n}\n\ntype ValidatedProviderResult =\n | { readonly ok: true; readonly value: unknown }\n | { readonly ok: false; readonly error: ReturnType<typeof errorRuntime> };\n\nfunction diagnosticLocationSuffix(diagnostic: Record<string, unknown>): string {\n const sourceId = typeof diagnostic['sourceId'] === 'string' ? diagnostic['sourceId'] : undefined;\n const span = isRecord(diagnostic['span']) ? diagnostic['span'] : undefined;\n const start = span && isRecord(span['start']) ? span['start'] : undefined;\n const line = start && typeof start['line'] === 'number' ? start['line'] : undefined;\n const column = start && typeof start['column'] === 'number' ? start['column'] : undefined;\n if (sourceId && line !== undefined && column !== undefined) {\n return ` (${sourceId}:${line}:${column})`;\n }\n if (sourceId) {\n return ` (${sourceId})`;\n }\n return '';\n}\n\nfunction mapDiagnosticsToIssues(\n diagnostics: readonly unknown[],\n): ReadonlyArray<{ readonly kind: string; readonly message: string }> {\n const issues: { readonly kind: string; readonly message: string }[] = [];\n for (const raw of diagnostics) {\n if (!isRecord(raw)) continue;\n const code = typeof raw['code'] === 'string' ? raw['code'] : 'diagnostic';\n const message = typeof raw['message'] === 'string' ? raw['message'] : '';\n issues.push({ kind: code, message: `${message}${diagnosticLocationSuffix(raw)}` });\n }\n return issues;\n}\n\nfunction validateProviderResult(providerResult: unknown): ValidatedProviderResult {\n if (!isRecord(providerResult) || typeof providerResult['ok'] !== 'boolean') {\n return {\n ok: false,\n error: failedToResolveContractSource(\n 'Contract source provider returned malformed result shape.',\n 'Ensure contract.source.load resolves to ok(Contract) or notOk({ summary, diagnostics }).',\n ),\n };\n }\n\n if (providerResult['ok']) {\n if (!('value' in providerResult)) {\n return {\n ok: false,\n error: failedToResolveContractSource(\n 'Contract source provider returned malformed success result: missing value.',\n 'Ensure contract.source.load success payload is ok(Contract).',\n ),\n };\n }\n return { ok: true, value: providerResult['value'] };\n }\n\n const failure = providerResult['failure'];\n if (\n !isRecord(failure) ||\n typeof failure['summary'] !== 'string' ||\n !Array.isArray(failure['diagnostics'])\n ) {\n return {\n ok: false,\n error: failedToResolveContractSource(\n 'Contract source provider returned malformed failure result: expected summary and diagnostics.',\n 'Ensure contract.source.load failure payload is notOk({ summary, diagnostics, meta? }).',\n ),\n };\n }\n return {\n ok: false,\n error: failedToResolveContractSource(\n String(failure['summary']),\n 'Fix contract source diagnostics and return ok(Contract).',\n {\n diagnostics: failure['diagnostics'],\n issues: mapDiagnosticsToIssues(failure['diagnostics']),\n ...ifDefined('providerMeta', failure['meta']),\n },\n ),\n };\n}\n\n/**\n * Canonical contract emit operation.\n *\n * This is the SINGLE publication path used by both the CLI command\n * (`prisma-next contract emit`) and the Vite plugin\n * (`@prisma-next/vite-plugin-contract-emit`). New callers must go through this\n * function rather than re-implementing load → emit → publish.\n *\n * The whole flow (load config → resolve source → emit bytes → atomic publish)\n * is serialized per output JSON path via `queueEmitByOutput`. Concurrent calls\n * for the same output line up FIFO; the user-visible outcome is \"last\n * submission wins on disk\" without any supersession bookkeeping. Within a\n * single emit, `publishContractArtifactPair` stages temp files, renames\n * `contract.d.ts` before `contract.json`, and attempts to restore the previous\n * pair if either rename fails — so type-only consumers never observe a\n * mismatched pair.\n *\n * @throws {CliStructuredError} on config/source/validation problems\n * @throws {DOMException} `AbortError` if cancelled via `signal`\n */\nexport async function executeContractEmit(\n options: ContractEmitOptions,\n): Promise<ContractEmitResult> {\n const { configPath, signal = new AbortController().signal, onProgress } = options;\n const unlessAborted = abortable(signal);\n\n const config = await unlessAborted(loadConfig(configPath));\n\n if (!config.contract) {\n throw errorContractConfigMissing({\n why: 'Config.contract is required for emit. Define it in your config: contract: { source: ..., output: ... }',\n });\n }\n\n const contractConfig = config.contract;\n\n if (!contractConfig.output) {\n throw errorContractConfigMissing({\n why: 'Contract config must have output path. This should not happen if defineConfig() was used.',\n });\n }\n\n if (typeof contractConfig.source?.load !== 'function') {\n throw errorContractConfigMissing({\n why: 'Contract config must include a valid source provider object',\n });\n }\n\n let outputPaths: ReturnType<typeof getEmittedArtifactPaths>;\n try {\n outputPaths = getEmittedArtifactPaths(contractConfig.output);\n } catch (error) {\n throw errorContractConfigMissing({\n why: error instanceof Error ? error.message : String(error),\n });\n }\n const { jsonPath: outputJsonPath, dtsPath: outputDtsPath } = outputPaths;\n\n return queueEmitByOutput(outputJsonPath, async () => {\n const stack = createControlStack(config);\n\n const sourceContext = {\n composedExtensionPacks: stack.extensionPacks.map((p) => p.id),\n scalarTypeDescriptors: stack.scalarTypeDescriptors,\n authoringContributions: stack.authoringContributions,\n codecLookup: stack.codecLookup,\n controlMutationDefaults: stack.controlMutationDefaults,\n resolvedInputs: contractConfig.source.inputs ?? [],\n };\n\n startSpan(onProgress, 'resolveSource', 'Resolving contract source...');\n let providerResult: Awaited<ReturnType<typeof contractConfig.source.load>>;\n try {\n providerResult = await unlessAborted(contractConfig.source.load(sourceContext));\n } catch (error) {\n endSpan(onProgress, 'resolveSource', 'error');\n if (signal.aborted || (isRecord(error) && error['name'] === 'AbortError')) {\n throw error;\n }\n throw failedToResolveContractSource(\n error instanceof Error ? error.message : String(error),\n 'Ensure contract.source.load resolves to ok(Contract) or returns structured diagnostics.',\n );\n }\n\n const validatedContract = validateProviderResult(providerResult);\n if (!validatedContract.ok) {\n endSpan(onProgress, 'resolveSource', 'error');\n throw validatedContract.error;\n }\n endSpan(onProgress, 'resolveSource', 'ok');\n\n startSpan(onProgress, 'emit', 'Emitting contract...');\n let emitResult: Awaited<ReturnType<typeof emit>>;\n try {\n const familyInstance = config.family.create(stack);\n const rawComponents = [config.target, config.adapter, ...(config.extensionPacks ?? [])];\n const frameworkComponents = assertFrameworkComponentsCompatible(\n config.family.familyId,\n config.target.targetId,\n rawComponents,\n );\n // Blind cast: `validateProviderResult` upstream has already\n // pinned `validatedContract.value` to the provider's loose\n // `Contract` envelope, but the local `Contract` type at this\n // call site is the precise structural interface. Re-narrowing\n // the envelope into the precise type is exactly what the\n // `familyInstance.deserializeContract(enrichedIR)` call on the\n // next line does — the cast just defers the structural check\n // by one statement so `enrichContract` can decorate first.\n const enrichedIR = enrichContract(\n validatedContract.value as unknown as Contract,\n frameworkComponents,\n );\n familyInstance.deserializeContract(enrichedIR);\n // Each target's descriptor ships a `contractSerializer` SPI; the\n // framework canonicalizer threads its `serializeContract` so the\n // on-disk JSON envelope is constructed by target-owned code\n // rather than by walking the in-memory contract with\n // `Object.entries` (which would leak runtime-only class API\n // fields into the persisted shape).\n const serializeContract = (c: Contract): JsonObject =>\n config.target.contractSerializer.serializeContract(c);\n emitResult = await unlessAborted(\n emit(enrichedIR, stack, config.family.emission, {\n outputJsonPath,\n serializeContract,\n }),\n );\n } catch (error) {\n endSpan(onProgress, 'emit', 'error');\n throw error;\n }\n endSpan(onProgress, 'emit', 'ok');\n\n await unlessAborted(mkdir(dirname(outputJsonPath), { recursive: true }));\n await publishContractArtifactPair({\n outputJsonPath,\n outputDtsPath,\n contractJson: emitResult.contractJson,\n contractDts: emitResult.contractDts,\n publicationToken: String(process.hrtime.bigint()),\n });\n\n const validationWarning = validateContractDeps(\n emitResult.contractDts,\n dirname(outputDtsPath),\n ).warning;\n\n return {\n storageHash: emitResult.storageHash,\n ...ifDefined('executionHash', emitResult.executionHash),\n profileHash: emitResult.profileHash,\n files: {\n json: outputJsonPath,\n dts: outputDtsPath,\n },\n ...ifDefined('validationWarning', validationWarning),\n };\n });\n}\n"],"mappings":"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"}