@prisma-next/target-postgres 0.12.0-dev.9 → 0.13.0-dev.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{codec-ids-DliyCWPY.d.mts → codec-ids-B1vOchLE.d.mts} +3 -2
- package/dist/codec-ids-B1vOchLE.d.mts.map +1 -0
- package/dist/{codec-ids-C5qzBqus.mjs → codec-ids-CTikp1if.mjs} +3 -2
- package/dist/codec-ids-CTikp1if.mjs.map +1 -0
- package/dist/codec-ids.d.mts +2 -2
- package/dist/codec-ids.mjs +2 -2
- package/dist/{codec-types-BF1DXTPs.d.mts → codec-types-CnFiNML4.d.mts} +8 -9
- package/dist/codec-types-CnFiNML4.d.mts.map +1 -0
- package/dist/codec-types.d.mts +2 -2
- package/dist/{codecs-DuP3d9Au.d.mts → codecs-CBpEv4s5.d.mts} +33 -35
- package/dist/codecs-CBpEv4s5.d.mts.map +1 -0
- package/dist/codecs.d.mts +1 -1
- package/dist/codecs.mjs +37 -2
- package/dist/codecs.mjs.map +1 -1
- package/dist/contract-free.d.mts +80 -0
- package/dist/contract-free.d.mts.map +1 -0
- package/dist/contract-free.mjs +117 -0
- package/dist/contract-free.mjs.map +1 -0
- package/dist/control.d.mts +1 -1
- package/dist/control.d.mts.map +1 -1
- package/dist/control.mjs +66 -40
- package/dist/control.mjs.map +1 -1
- package/dist/{data-transform-CRkv2T_U.mjs → data-transform-D25tLeYU.mjs} +1 -1
- package/dist/{data-transform-CRkv2T_U.mjs.map → data-transform-D25tLeYU.mjs.map} +1 -1
- package/dist/{data-transform-CAPmAdxS.d.mts → data-transform-DGOqcLrf.d.mts} +2 -2
- package/dist/{data-transform-CAPmAdxS.d.mts.map → data-transform-DGOqcLrf.d.mts.map} +1 -1
- package/dist/data-transform.d.mts +1 -1
- package/dist/data-transform.mjs +1 -1
- package/dist/ddl-77SyXgFt.mjs +30 -0
- package/dist/ddl-77SyXgFt.mjs.map +1 -0
- package/dist/ddl.d.mts +2 -0
- package/dist/ddl.mjs +2 -0
- package/dist/{default-normalizer-DaAhmzBV.mjs → default-normalizer-DyyCHQWs.mjs} +1 -1
- package/dist/{default-normalizer-DaAhmzBV.mjs.map → default-normalizer-DyyCHQWs.mjs.map} +1 -1
- package/dist/default-normalizer.mjs +1 -1
- package/dist/{descriptor-meta-Uu8QaClr.mjs → descriptor-meta-DKmj-IMN.mjs} +3 -2
- package/dist/descriptor-meta-DKmj-IMN.mjs.map +1 -0
- package/dist/{descriptor-meta-runtime-DMRX39kp.mjs → descriptor-meta-runtime-My8_s4cs.mjs} +2 -2
- package/dist/{descriptor-meta-runtime-DMRX39kp.mjs.map → descriptor-meta-runtime-My8_s4cs.mjs.map} +1 -1
- package/dist/{enum-planning-DRA9LaMU.mjs → enum-planning-BCyvlFHk.mjs} +0 -0
- package/dist/{enum-planning-DRA9LaMU.mjs.map → enum-planning-BCyvlFHk.mjs.map} +1 -1
- package/dist/enum-planning.d.mts +1 -1
- package/dist/enum-planning.mjs +1 -1
- package/dist/{errors-BbnITmAD.mjs → errors-CUk87ByX.mjs} +1 -1
- package/dist/{errors-BbnITmAD.mjs.map → errors-CUk87ByX.mjs.map} +1 -1
- package/dist/errors.d.mts.map +1 -1
- package/dist/errors.mjs +1 -1
- package/dist/{issue-planner-BtaL6OfW.mjs → issue-planner-Br0pt1Ea.mjs} +130 -28
- package/dist/issue-planner-Br0pt1Ea.mjs.map +1 -0
- package/dist/issue-planner.d.mts +1 -1
- package/dist/issue-planner.d.mts.map +1 -1
- package/dist/issue-planner.mjs +1 -1
- package/dist/migration.d.mts +7 -8
- package/dist/migration.d.mts.map +1 -1
- package/dist/migration.mjs +5 -4
- package/dist/migration.mjs.map +1 -1
- package/dist/{native-type-normalizer-BNEQ9VDs.mjs → native-type-normalizer-Bc9XJzWC.mjs} +1 -1
- package/dist/{native-type-normalizer-BNEQ9VDs.mjs.map → native-type-normalizer-Bc9XJzWC.mjs.map} +1 -1
- package/dist/native-type-normalizer.mjs +1 -1
- package/dist/nodes-779hmCfL.d.mts +40 -0
- package/dist/nodes-779hmCfL.d.mts.map +1 -0
- package/dist/nodes-DZk2JZG3.mjs +47 -0
- package/dist/nodes-DZk2JZG3.mjs.map +1 -0
- package/dist/op-factory-call-D2aAUhmS.mjs +1307 -0
- package/dist/op-factory-call-D2aAUhmS.mjs.map +1 -0
- package/dist/{op-factory-call-CDlImOF_.d.mts → op-factory-call-DMA86_2D.d.mts} +39 -14
- package/dist/op-factory-call-DMA86_2D.d.mts.map +1 -0
- package/dist/op-factory-call.d.mts +2 -2
- package/dist/op-factory-call.mjs +2 -2
- package/dist/pack.d.mts +5 -6
- package/dist/pack.d.mts.map +1 -1
- package/dist/pack.mjs +1 -1
- package/dist/planner-CAYPJObw.mjs +344 -0
- package/dist/planner-CAYPJObw.mjs.map +1 -0
- package/dist/{planner-ddl-builders-BNMfSE9r.mjs → planner-ddl-builders-Cw2n2llW.mjs} +7 -30
- package/dist/planner-ddl-builders-Cw2n2llW.mjs.map +1 -0
- package/dist/planner-ddl-builders.d.mts +6 -7
- package/dist/planner-ddl-builders.d.mts.map +1 -1
- package/dist/planner-ddl-builders.mjs +2 -2
- package/dist/{planner-identity-values-nhVj0hp-.mjs → planner-identity-values-BIpa5p2I.mjs} +1 -1
- package/dist/{planner-identity-values-nhVj0hp-.mjs.map → planner-identity-values-BIpa5p2I.mjs.map} +1 -1
- package/dist/planner-identity-values.mjs +1 -1
- package/dist/{planner-produced-postgres-migration-D02NOhVQ.d.mts → planner-produced-postgres-migration-B4EDvLdz.d.mts} +5 -4
- package/dist/planner-produced-postgres-migration-B4EDvLdz.d.mts.map +1 -0
- package/dist/{planner-produced-postgres-migration-D_nsXbhl.mjs → planner-produced-postgres-migration-NSEhWL0L.mjs} +8 -6
- package/dist/planner-produced-postgres-migration-NSEhWL0L.mjs.map +1 -0
- package/dist/planner-produced-postgres-migration.d.mts +1 -1
- package/dist/planner-produced-postgres-migration.mjs +1 -1
- package/dist/{planner-schema-lookup-CGxxYfnD.mjs → planner-schema-lookup-CiVaAQP-.mjs} +1 -1
- package/dist/{planner-schema-lookup-CGxxYfnD.mjs.map → planner-schema-lookup-CiVaAQP-.mjs.map} +1 -1
- package/dist/planner-schema-lookup.mjs +1 -1
- package/dist/{planner-sql-checks-CfEiTXoQ.mjs → planner-sql-checks-DAdhnI2c.mjs} +41 -30
- package/dist/planner-sql-checks-DAdhnI2c.mjs.map +1 -0
- package/dist/planner-sql-checks.d.mts.map +1 -1
- package/dist/planner-sql-checks.mjs +1 -1
- package/dist/{planner-target-details-a_wuOiYf.d.mts → planner-target-details-CIY6tLeo.d.mts} +2 -2
- package/dist/planner-target-details-CIY6tLeo.d.mts.map +1 -0
- package/dist/planner-target-details.d.mts +2 -2
- package/dist/planner-type-resolution-836DExFN.mjs +20 -0
- package/dist/planner-type-resolution-836DExFN.mjs.map +1 -0
- package/dist/planner.d.mts +7 -3
- package/dist/planner.d.mts.map +1 -1
- package/dist/planner.mjs +1 -1
- package/dist/{postgres-contract-serializer-BnOboPWs.mjs → postgres-contract-serializer-DYTyXjPf.mjs} +33 -24
- package/dist/postgres-contract-serializer-DYTyXjPf.mjs.map +1 -0
- package/dist/{postgres-enum-type-CSzsvXrO.d.mts → postgres-enum-type-BVn63a89.d.mts} +1 -1
- package/dist/{postgres-enum-type-CSzsvXrO.d.mts.map → postgres-enum-type-BVn63a89.d.mts.map} +1 -1
- package/dist/{postgres-enum-type-BMgyxNyy.mjs → postgres-enum-type-DPKqCBem.mjs} +1 -1
- package/dist/{postgres-enum-type-BMgyxNyy.mjs.map → postgres-enum-type-DPKqCBem.mjs.map} +1 -1
- package/dist/{postgres-migration-BatbEvU6.mjs → postgres-migration-COore9Mz.mjs} +23 -3
- package/dist/postgres-migration-COore9Mz.mjs.map +1 -0
- package/dist/{postgres-migration-DRY8V-bQ.d.mts → postgres-migration-DZ_gLUOW.d.mts} +25 -3
- package/dist/postgres-migration-DZ_gLUOW.d.mts.map +1 -0
- package/dist/{postgres-schema-BxAuNFX0.mjs → postgres-schema-BuxCxbvB.mjs} +29 -14
- package/dist/postgres-schema-BuxCxbvB.mjs.map +1 -0
- package/dist/{render-ops-XhICjX_P.mjs → render-ops-BpjstrKQ.mjs} +4 -3
- package/dist/{render-ops-XhICjX_P.mjs.map → render-ops-BpjstrKQ.mjs.map} +1 -1
- package/dist/render-ops.d.mts +3 -2
- package/dist/render-ops.d.mts.map +1 -1
- package/dist/render-ops.mjs +1 -1
- package/dist/{render-typescript-K125n-RZ.mjs → render-typescript-KMgosran.mjs} +5 -2
- package/dist/render-typescript-KMgosran.mjs.map +1 -0
- package/dist/render-typescript.mjs +1 -1
- package/dist/runtime.d.mts.map +1 -1
- package/dist/runtime.mjs +2 -2
- package/dist/{shared-Do_a5ymU.d.mts → shared-DarONYBZ.d.mts} +5 -5
- package/dist/{shared-Do_a5ymU.d.mts.map → shared-DarONYBZ.d.mts.map} +1 -1
- package/dist/{sql-utils-CggjWNij.mjs → sql-utils-DcfMz4MQ.mjs} +1 -1
- package/dist/{sql-utils-CggjWNij.mjs.map → sql-utils-DcfMz4MQ.mjs.map} +1 -1
- package/dist/sql-utils.mjs +1 -1
- package/dist/{types-O40IcFV9.d.mts → types-BDKkx8MA.d.mts} +1 -1
- package/dist/types-BDKkx8MA.d.mts.map +1 -0
- package/dist/types.d.mts +16 -11
- package/dist/types.d.mts.map +1 -1
- package/dist/types.mjs +2 -2
- package/package.json +21 -20
- package/src/contract-free/columns.ts +49 -0
- package/src/contract-free/control-bootstrap.ts +55 -0
- package/src/contract-free/ddl.ts +37 -0
- package/src/core/ast/table-source.ts +23 -0
- package/src/core/codec-ids.ts +1 -0
- package/src/core/codecs.ts +44 -0
- package/src/core/ddl/nodes.ts +72 -0
- package/src/core/descriptor-meta.ts +1 -0
- package/src/core/migrations/control-policy.ts +234 -0
- package/src/core/migrations/issue-planner.ts +81 -13
- package/src/core/migrations/op-factory-call.ts +289 -46
- package/src/core/migrations/operations/constraints.ts +79 -10
- package/src/core/migrations/operations/dependencies.ts +0 -17
- package/src/core/migrations/operations/shared.ts +3 -3
- package/src/core/migrations/operations/tables.ts +1 -39
- package/src/core/migrations/planner-ddl-builders.ts +7 -48
- package/src/core/migrations/planner-produced-postgres-migration.ts +11 -6
- package/src/core/migrations/planner-sql-checks.ts +9 -9
- package/src/core/migrations/planner-strategies.ts +149 -11
- package/src/core/migrations/planner-target-details.ts +2 -1
- package/src/core/migrations/planner.ts +66 -8
- package/src/core/migrations/postgres-migration.ts +41 -0
- package/src/core/migrations/render-ops.ts +7 -2
- package/src/core/migrations/render-typescript.ts +5 -1
- package/src/core/migrations/runner.ts +78 -50
- package/src/core/postgres-contract-serializer.ts +52 -46
- package/src/core/postgres-schema.ts +43 -25
- package/src/exports/contract-free.ts +7 -0
- package/src/exports/control.ts +6 -8
- package/src/exports/ddl.ts +7 -0
- package/src/exports/migration.ts +11 -2
- package/src/exports/op-factory-call.ts +2 -0
- package/src/exports/planner-ddl-builders.ts +0 -1
- package/dist/codec-ids-C5qzBqus.mjs.map +0 -1
- package/dist/codec-ids-DliyCWPY.d.mts.map +0 -1
- package/dist/codec-types-BF1DXTPs.d.mts.map +0 -1
- package/dist/codecs-DuP3d9Au.d.mts.map +0 -1
- package/dist/descriptor-meta-Uu8QaClr.mjs.map +0 -1
- package/dist/issue-planner-BtaL6OfW.mjs.map +0 -1
- package/dist/op-factory-call-CDlImOF_.d.mts.map +0 -1
- package/dist/op-factory-call-ewOd5q6L.mjs +0 -625
- package/dist/op-factory-call-ewOd5q6L.mjs.map +0 -1
- package/dist/planner-Bjz5pnLa.mjs +0 -177
- package/dist/planner-Bjz5pnLa.mjs.map +0 -1
- package/dist/planner-ddl-builders-BNMfSE9r.mjs.map +0 -1
- package/dist/planner-produced-postgres-migration-D02NOhVQ.d.mts.map +0 -1
- package/dist/planner-produced-postgres-migration-D_nsXbhl.mjs.map +0 -1
- package/dist/planner-sql-checks-CfEiTXoQ.mjs.map +0 -1
- package/dist/planner-target-details-a_wuOiYf.d.mts.map +0 -1
- package/dist/postgres-contract-serializer-BnOboPWs.mjs.map +0 -1
- package/dist/postgres-migration-BatbEvU6.mjs.map +0 -1
- package/dist/postgres-migration-DRY8V-bQ.d.mts.map +0 -1
- package/dist/postgres-schema-BxAuNFX0.mjs.map +0 -1
- package/dist/render-typescript-K125n-RZ.mjs.map +0 -1
- package/dist/statement-builders-DVI5IVAa.mjs +0 -131
- package/dist/statement-builders-DVI5IVAa.mjs.map +0 -1
- package/dist/statement-builders.d.mts +0 -51
- package/dist/statement-builders.d.mts.map +0 -1
- package/dist/statement-builders.mjs +0 -2
- package/dist/tables-DoA39Yqo.mjs +0 -516
- package/dist/tables-DoA39Yqo.mjs.map +0 -1
- package/dist/types-O40IcFV9.d.mts.map +0 -1
- package/src/core/migrations/statement-builders.ts +0 -183
- package/src/exports/statement-builders.ts +0 -8
|
@@ -22,11 +22,23 @@
|
|
|
22
22
|
|
|
23
23
|
import { errorUnfilledPlaceholder } from '@prisma-next/errors/migration';
|
|
24
24
|
import type { SqlMigrationPlanOperation } from '@prisma-next/family-sql/control';
|
|
25
|
+
import type { Lowerer } from '@prisma-next/family-sql/control-adapter';
|
|
25
26
|
import type {
|
|
26
27
|
OpFactoryCall as FrameworkOpFactoryCall,
|
|
27
28
|
MigrationOperationClass,
|
|
28
29
|
} from '@prisma-next/framework-components/control';
|
|
30
|
+
import { UNBOUND_NAMESPACE_ID } from '@prisma-next/framework-components/ir';
|
|
31
|
+
import type {
|
|
32
|
+
AnyDdlColumnDefault,
|
|
33
|
+
DdlColumn,
|
|
34
|
+
DdlTableConstraint,
|
|
35
|
+
} from '@prisma-next/sql-relational-core/ast';
|
|
36
|
+
import { FunctionColumnDefault, LiteralColumnDefault } from '@prisma-next/sql-relational-core/ast';
|
|
29
37
|
import { type ImportRequirement, jsonToTsSource, TsExpression } from '@prisma-next/ts-render';
|
|
38
|
+
import { blindCast } from '@prisma-next/utils/casts';
|
|
39
|
+
import * as contractFreeDdl from '../../contract-free/ddl';
|
|
40
|
+
import { escapeLiteral, quoteIdentifier } from '../sql-utils';
|
|
41
|
+
import type { PostgresColumnDefault } from '../types';
|
|
30
42
|
import {
|
|
31
43
|
addColumn,
|
|
32
44
|
alterColumnType,
|
|
@@ -36,26 +48,47 @@ import {
|
|
|
36
48
|
setDefault,
|
|
37
49
|
setNotNull,
|
|
38
50
|
} from './operations/columns';
|
|
39
|
-
import {
|
|
40
|
-
|
|
51
|
+
import {
|
|
52
|
+
addCheckConstraint,
|
|
53
|
+
addForeignKey,
|
|
54
|
+
addPrimaryKey,
|
|
55
|
+
addUnique,
|
|
56
|
+
dropCheckConstraint,
|
|
57
|
+
dropConstraint,
|
|
58
|
+
} from './operations/constraints';
|
|
59
|
+
import { createExtension } from './operations/dependencies';
|
|
41
60
|
import { addEnumValues, createEnumType, dropEnumType, renameType } from './operations/enums';
|
|
42
61
|
import { createIndex, dropIndex } from './operations/indexes';
|
|
43
62
|
import type { ColumnSpec, ForeignKeySpec } from './operations/shared';
|
|
44
|
-
import {
|
|
63
|
+
import { step, targetDetails } from './operations/shared';
|
|
64
|
+
import { dropTable } from './operations/tables';
|
|
65
|
+
import { toRegclassLiteral } from './planner-sql-checks';
|
|
45
66
|
import type { PostgresPlanTargetDetails } from './planner-target-details';
|
|
46
67
|
|
|
47
68
|
type Op = SqlMigrationPlanOperation<PostgresPlanTargetDetails>;
|
|
48
69
|
|
|
49
|
-
|
|
70
|
+
// Single module specifier emitted in user-edited `migration.ts` imports. The
|
|
71
|
+
// Postgres migration facade re-exports both the `*Call` factory names
|
|
72
|
+
// (createTable / addColumn / …) and the contract-free DDL builders
|
|
73
|
+
// (col / lit / fn / primaryKey / foreignKey / unique) from
|
|
74
|
+
// sql-relational-core/contract-free. We emit imports against the facade,
|
|
75
|
+
// not against the underlying sql-relational-core subpath, because user
|
|
76
|
+
// projects depend on `@prisma-next/postgres` (a runtime dep of every
|
|
77
|
+
// init-scaffolded project) — they do not depend on the internal
|
|
78
|
+
// `@prisma-next/sql-relational-core` package, so an emitted
|
|
79
|
+
// `import … from '@prisma-next/sql-relational-core/contract-free'` fails
|
|
80
|
+
// ESM resolution at runtime in user migrations even though pnpm has the
|
|
81
|
+
// transitive package on disk.
|
|
82
|
+
const POSTGRES_MIGRATION_FACADE = '@prisma-next/postgres/migration';
|
|
50
83
|
|
|
51
84
|
abstract class PostgresOpFactoryCallNode extends TsExpression implements FrameworkOpFactoryCall {
|
|
52
85
|
abstract readonly factoryName: string;
|
|
53
86
|
abstract readonly operationClass: MigrationOperationClass;
|
|
54
87
|
abstract readonly label: string;
|
|
55
|
-
abstract toOp(): Op;
|
|
88
|
+
abstract toOp(lowerer?: Lowerer): Op;
|
|
56
89
|
|
|
57
90
|
importRequirements(): readonly ImportRequirement[] {
|
|
58
|
-
return [{ moduleSpecifier:
|
|
91
|
+
return [{ moduleSpecifier: POSTGRES_MIGRATION_FACADE, symbol: this.factoryName }];
|
|
59
92
|
}
|
|
60
93
|
|
|
61
94
|
protected freeze(): void {
|
|
@@ -67,8 +100,93 @@ abstract class PostgresOpFactoryCallNode extends TsExpression implements Framewo
|
|
|
67
100
|
// Table
|
|
68
101
|
// ============================================================================
|
|
69
102
|
|
|
70
|
-
export
|
|
71
|
-
|
|
103
|
+
export function postgresDefaultToDdlColumnDefault(
|
|
104
|
+
columnDefault: PostgresColumnDefault | undefined,
|
|
105
|
+
): DdlColumn['default'] {
|
|
106
|
+
if (!columnDefault) return undefined;
|
|
107
|
+
switch (columnDefault.kind) {
|
|
108
|
+
case 'literal':
|
|
109
|
+
return new LiteralColumnDefault(columnDefault.value);
|
|
110
|
+
case 'function':
|
|
111
|
+
if (columnDefault.expression === 'autoincrement()') return undefined;
|
|
112
|
+
return new FunctionColumnDefault(columnDefault.expression);
|
|
113
|
+
case 'sequence':
|
|
114
|
+
return new FunctionColumnDefault(
|
|
115
|
+
`nextval('${escapeLiteral(quoteIdentifier(columnDefault.name))}'::regclass)`,
|
|
116
|
+
);
|
|
117
|
+
default: {
|
|
118
|
+
const exhaustive: never = columnDefault;
|
|
119
|
+
throw new Error(
|
|
120
|
+
`postgresDefaultToDdlColumnDefault: unhandled kind "${blindCast<{ kind: string }, 'exhaustiveness: surface the unhandled default kind'>(exhaustive).kind}"`,
|
|
121
|
+
);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
// ---------------------------------------------------------------------------
|
|
127
|
+
// TypeScript rendering helpers for DdlColumn / DdlTableConstraint
|
|
128
|
+
// ---------------------------------------------------------------------------
|
|
129
|
+
|
|
130
|
+
function renderDdlColumnDefault(def: AnyDdlColumnDefault | undefined): string {
|
|
131
|
+
if (!def) return '';
|
|
132
|
+
if (def.kind === 'literal') {
|
|
133
|
+
return `lit(${jsonToTsSource(def.value)})`;
|
|
134
|
+
}
|
|
135
|
+
return `fn(${jsonToTsSource(def.expression)})`;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
function renderDdlColumnAsTsCall(col: DdlColumn): string {
|
|
139
|
+
const opts: string[] = [];
|
|
140
|
+
if (col.notNull) opts.push('notNull: true');
|
|
141
|
+
if (col.primaryKey) opts.push('primaryKey: true');
|
|
142
|
+
if (col.default) opts.push(`default: ${renderDdlColumnDefault(col.default)}`);
|
|
143
|
+
const optsStr = opts.length > 0 ? `, { ${opts.join(', ')} }` : '';
|
|
144
|
+
return `col(${jsonToTsSource(col.name)}, ${jsonToTsSource(col.type)}${optsStr})`;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
function renderDdlConstraintAsTsCall(constraint: DdlTableConstraint): string {
|
|
148
|
+
switch (constraint.kind) {
|
|
149
|
+
case 'primary-key': {
|
|
150
|
+
const nameOpt = constraint.name ? `, { name: ${jsonToTsSource(constraint.name)} }` : '';
|
|
151
|
+
return `primaryKey(${jsonToTsSource(constraint.columns)}${nameOpt})`;
|
|
152
|
+
}
|
|
153
|
+
case 'foreign-key': {
|
|
154
|
+
const opts: string[] = [];
|
|
155
|
+
if (constraint.name) opts.push(`name: ${jsonToTsSource(constraint.name)}`);
|
|
156
|
+
if (constraint.onDelete) opts.push(`onDelete: ${jsonToTsSource(constraint.onDelete)}`);
|
|
157
|
+
if (constraint.onUpdate) opts.push(`onUpdate: ${jsonToTsSource(constraint.onUpdate)}`);
|
|
158
|
+
const optsStr = opts.length > 0 ? `, { ${opts.join(', ')} }` : '';
|
|
159
|
+
return `foreignKey(${jsonToTsSource(constraint.columns)}, ${jsonToTsSource(constraint.refTable)}, ${jsonToTsSource(constraint.refColumns)}${optsStr})`;
|
|
160
|
+
}
|
|
161
|
+
case 'unique': {
|
|
162
|
+
const nameOpt = constraint.name ? `, { name: ${jsonToTsSource(constraint.name)} }` : '';
|
|
163
|
+
return `unique(${jsonToTsSource(constraint.columns)}${nameOpt})`;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
function needsColOrConstraintImport(columns: readonly DdlColumn[]): boolean {
|
|
169
|
+
return columns.length > 0;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
function constraintImportSymbols(constraints: readonly DdlTableConstraint[] | undefined): string[] {
|
|
173
|
+
if (!constraints || constraints.length === 0) return [];
|
|
174
|
+
const symbols = new Set<string>();
|
|
175
|
+
for (const c of constraints) {
|
|
176
|
+
if (c.kind === 'primary-key') symbols.add('primaryKey');
|
|
177
|
+
else if (c.kind === 'foreign-key') symbols.add('foreignKey');
|
|
178
|
+
else if (c.kind === 'unique') symbols.add('unique');
|
|
179
|
+
}
|
|
180
|
+
return [...symbols];
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
function defaultImportSymbols(columns: readonly DdlColumn[]): string[] {
|
|
184
|
+
const symbols = new Set<string>();
|
|
185
|
+
for (const col of columns) {
|
|
186
|
+
if (col.default?.kind === 'literal') symbols.add('lit');
|
|
187
|
+
else if (col.default?.kind === 'function') symbols.add('fn');
|
|
188
|
+
}
|
|
189
|
+
return [...symbols];
|
|
72
190
|
}
|
|
73
191
|
|
|
74
192
|
export class CreateTableCall extends PostgresOpFactoryCallNode {
|
|
@@ -76,37 +194,91 @@ export class CreateTableCall extends PostgresOpFactoryCallNode {
|
|
|
76
194
|
readonly operationClass = 'additive' as const;
|
|
77
195
|
readonly schemaName: string;
|
|
78
196
|
readonly tableName: string;
|
|
79
|
-
readonly columns: readonly
|
|
80
|
-
readonly
|
|
197
|
+
readonly columns: readonly DdlColumn[];
|
|
198
|
+
readonly constraints: readonly DdlTableConstraint[] | undefined;
|
|
81
199
|
readonly label: string;
|
|
82
200
|
|
|
83
201
|
constructor(
|
|
84
202
|
schemaName: string,
|
|
85
203
|
tableName: string,
|
|
86
|
-
columns: readonly
|
|
87
|
-
|
|
204
|
+
columns: readonly DdlColumn[],
|
|
205
|
+
constraints?: readonly DdlTableConstraint[],
|
|
88
206
|
) {
|
|
89
207
|
super();
|
|
90
208
|
this.schemaName = schemaName;
|
|
91
209
|
this.tableName = tableName;
|
|
92
|
-
this.columns = columns;
|
|
93
|
-
this.
|
|
210
|
+
this.columns = Object.freeze([...columns]);
|
|
211
|
+
this.constraints = constraints ? Object.freeze([...constraints]) : undefined;
|
|
94
212
|
this.label = `Create table "${tableName}"`;
|
|
95
213
|
this.freeze();
|
|
96
214
|
}
|
|
97
215
|
|
|
98
|
-
toOp(): Op {
|
|
99
|
-
|
|
216
|
+
toOp(lowerer?: Lowerer): Op {
|
|
217
|
+
if (lowerer === undefined) {
|
|
218
|
+
throw new Error(
|
|
219
|
+
`CreateTableCall.toOp: a DDL lowerer is required on the Postgres planner path (table "${this.tableName}"). Pass the control adapter to createPostgresMigrationPlanner.`,
|
|
220
|
+
);
|
|
221
|
+
}
|
|
222
|
+
const ddlNode = contractFreeDdl.createTable({
|
|
223
|
+
...(this.schemaName !== UNBOUND_NAMESPACE_ID ? { schema: this.schemaName } : {}),
|
|
224
|
+
table: this.tableName,
|
|
225
|
+
columns: this.columns,
|
|
226
|
+
...(this.constraints ? { constraints: this.constraints } : {}),
|
|
227
|
+
});
|
|
228
|
+
const { sql } = lowerer.lower(ddlNode, { contract: {} });
|
|
229
|
+
const schemaName = this.schemaName;
|
|
230
|
+
const tableName = this.tableName;
|
|
231
|
+
return {
|
|
232
|
+
id: `table.${tableName}`,
|
|
233
|
+
label: `Create table "${tableName}"`,
|
|
234
|
+
summary: `Creates table "${tableName}"`,
|
|
235
|
+
operationClass: 'additive',
|
|
236
|
+
target: targetDetails('table', tableName, schemaName),
|
|
237
|
+
precheck: [
|
|
238
|
+
step(
|
|
239
|
+
`ensure table "${tableName}" does not exist`,
|
|
240
|
+
`SELECT to_regclass(${toRegclassLiteral(schemaName, tableName)}) IS NULL`,
|
|
241
|
+
),
|
|
242
|
+
],
|
|
243
|
+
execute: [step(`create table "${tableName}"`, sql)],
|
|
244
|
+
postcheck: [
|
|
245
|
+
step(
|
|
246
|
+
`verify table "${tableName}" exists`,
|
|
247
|
+
`SELECT to_regclass(${toRegclassLiteral(schemaName, tableName)}) IS NOT NULL`,
|
|
248
|
+
),
|
|
249
|
+
],
|
|
250
|
+
};
|
|
100
251
|
}
|
|
101
252
|
|
|
102
253
|
renderTypeScript(): string {
|
|
103
|
-
const
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
254
|
+
const columnsList = this.columns.map(renderDdlColumnAsTsCall).join(', ');
|
|
255
|
+
const constraintsList = this.constraints
|
|
256
|
+
? this.constraints.map(renderDdlConstraintAsTsCall).join(', ')
|
|
257
|
+
: undefined;
|
|
258
|
+
|
|
259
|
+
const opts: string[] = [];
|
|
260
|
+
if (this.schemaName !== UNBOUND_NAMESPACE_ID) {
|
|
261
|
+
opts.push(`schema: ${jsonToTsSource(this.schemaName)}`);
|
|
262
|
+
}
|
|
263
|
+
opts.push(`table: ${jsonToTsSource(this.tableName)}`);
|
|
264
|
+
opts.push(`columns: [${columnsList}]`);
|
|
265
|
+
if (constraintsList) opts.push(`constraints: [${constraintsList}]`);
|
|
266
|
+
|
|
267
|
+
return `this.createTable({ ${opts.join(', ')} })`;
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
override importRequirements(): readonly ImportRequirement[] {
|
|
271
|
+
const req: ImportRequirement[] = [];
|
|
272
|
+
if (needsColOrConstraintImport(this.columns)) {
|
|
273
|
+
req.push({ moduleSpecifier: POSTGRES_MIGRATION_FACADE, symbol: 'col' });
|
|
274
|
+
for (const sym of defaultImportSymbols(this.columns)) {
|
|
275
|
+
req.push({ moduleSpecifier: POSTGRES_MIGRATION_FACADE, symbol: sym });
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
for (const sym of constraintImportSymbols(this.constraints)) {
|
|
279
|
+
req.push({ moduleSpecifier: POSTGRES_MIGRATION_FACADE, symbol: sym });
|
|
280
|
+
}
|
|
281
|
+
return req;
|
|
110
282
|
}
|
|
111
283
|
}
|
|
112
284
|
|
|
@@ -495,6 +667,74 @@ export class DropConstraintCall extends PostgresOpFactoryCallNode {
|
|
|
495
667
|
}
|
|
496
668
|
}
|
|
497
669
|
|
|
670
|
+
export class AddCheckConstraintCall extends PostgresOpFactoryCallNode {
|
|
671
|
+
readonly factoryName = 'addCheckConstraint' as const;
|
|
672
|
+
readonly operationClass = 'additive' as const;
|
|
673
|
+
readonly schemaName: string;
|
|
674
|
+
readonly tableName: string;
|
|
675
|
+
readonly constraintName: string;
|
|
676
|
+
readonly column: string;
|
|
677
|
+
readonly values: readonly string[];
|
|
678
|
+
readonly label: string;
|
|
679
|
+
|
|
680
|
+
constructor(
|
|
681
|
+
schemaName: string,
|
|
682
|
+
tableName: string,
|
|
683
|
+
constraintName: string,
|
|
684
|
+
column: string,
|
|
685
|
+
values: readonly string[],
|
|
686
|
+
) {
|
|
687
|
+
super();
|
|
688
|
+
this.schemaName = schemaName;
|
|
689
|
+
this.tableName = tableName;
|
|
690
|
+
this.constraintName = constraintName;
|
|
691
|
+
this.column = column;
|
|
692
|
+
this.values = values;
|
|
693
|
+
this.label = `Add check constraint "${constraintName}" on "${tableName}"."${column}"`;
|
|
694
|
+
this.freeze();
|
|
695
|
+
}
|
|
696
|
+
|
|
697
|
+
toOp(): Op {
|
|
698
|
+
return addCheckConstraint(
|
|
699
|
+
this.schemaName,
|
|
700
|
+
this.tableName,
|
|
701
|
+
this.constraintName,
|
|
702
|
+
this.column,
|
|
703
|
+
this.values,
|
|
704
|
+
);
|
|
705
|
+
}
|
|
706
|
+
|
|
707
|
+
renderTypeScript(): string {
|
|
708
|
+
return `addCheckConstraint(${jsonToTsSource(this.schemaName)}, ${jsonToTsSource(this.tableName)}, ${jsonToTsSource(this.constraintName)}, ${jsonToTsSource(this.column)}, ${jsonToTsSource(this.values)})`;
|
|
709
|
+
}
|
|
710
|
+
}
|
|
711
|
+
|
|
712
|
+
export class DropCheckConstraintCall extends PostgresOpFactoryCallNode {
|
|
713
|
+
readonly factoryName = 'dropCheckConstraint' as const;
|
|
714
|
+
readonly operationClass = 'destructive' as const;
|
|
715
|
+
readonly schemaName: string;
|
|
716
|
+
readonly tableName: string;
|
|
717
|
+
readonly constraintName: string;
|
|
718
|
+
readonly label: string;
|
|
719
|
+
|
|
720
|
+
constructor(schemaName: string, tableName: string, constraintName: string) {
|
|
721
|
+
super();
|
|
722
|
+
this.schemaName = schemaName;
|
|
723
|
+
this.tableName = tableName;
|
|
724
|
+
this.constraintName = constraintName;
|
|
725
|
+
this.label = `Drop check constraint "${constraintName}" on "${tableName}"`;
|
|
726
|
+
this.freeze();
|
|
727
|
+
}
|
|
728
|
+
|
|
729
|
+
toOp(): Op {
|
|
730
|
+
return dropCheckConstraint(this.schemaName, this.tableName, this.constraintName);
|
|
731
|
+
}
|
|
732
|
+
|
|
733
|
+
renderTypeScript(): string {
|
|
734
|
+
return `dropCheckConstraint(${jsonToTsSource(this.schemaName)}, ${jsonToTsSource(this.tableName)}, ${jsonToTsSource(this.constraintName)})`;
|
|
735
|
+
}
|
|
736
|
+
}
|
|
737
|
+
|
|
498
738
|
// ============================================================================
|
|
499
739
|
// Indexes
|
|
500
740
|
// ============================================================================
|
|
@@ -776,12 +1016,32 @@ export class CreateSchemaCall extends PostgresOpFactoryCallNode {
|
|
|
776
1016
|
this.freeze();
|
|
777
1017
|
}
|
|
778
1018
|
|
|
779
|
-
toOp(): Op {
|
|
780
|
-
|
|
1019
|
+
toOp(lowerer?: Lowerer): Op {
|
|
1020
|
+
if (lowerer === undefined) {
|
|
1021
|
+
throw new Error(
|
|
1022
|
+
`CreateSchemaCall.toOp: a DDL lowerer is required on the Postgres planner path (schema "${this.schemaName}"). Pass the control adapter to createPostgresMigrationPlanner.`,
|
|
1023
|
+
);
|
|
1024
|
+
}
|
|
1025
|
+
const ddlNode = contractFreeDdl.createSchema({ schema: this.schemaName, ifNotExists: true });
|
|
1026
|
+
const { sql } = lowerer.lower(ddlNode, { contract: {} });
|
|
1027
|
+
const schemaName = this.schemaName;
|
|
1028
|
+
return {
|
|
1029
|
+
id: `schema.${schemaName}`,
|
|
1030
|
+
label: `Create schema "${schemaName}"`,
|
|
1031
|
+
operationClass: 'additive',
|
|
1032
|
+
target: { id: 'postgres' },
|
|
1033
|
+
precheck: [],
|
|
1034
|
+
execute: [step(`Create schema "${schemaName}"`, sql)],
|
|
1035
|
+
postcheck: [],
|
|
1036
|
+
};
|
|
781
1037
|
}
|
|
782
1038
|
|
|
783
1039
|
renderTypeScript(): string {
|
|
784
|
-
return `createSchema(${jsonToTsSource(this.schemaName)})`;
|
|
1040
|
+
return `this.createSchema({ schema: ${jsonToTsSource(this.schemaName)} })`;
|
|
1041
|
+
}
|
|
1042
|
+
|
|
1043
|
+
override importRequirements(): readonly ImportRequirement[] {
|
|
1044
|
+
return [];
|
|
785
1045
|
}
|
|
786
1046
|
}
|
|
787
1047
|
|
|
@@ -832,7 +1092,7 @@ export class DataTransformCall extends PostgresOpFactoryCallNode {
|
|
|
832
1092
|
|
|
833
1093
|
override importRequirements(): readonly ImportRequirement[] {
|
|
834
1094
|
return [
|
|
835
|
-
{ moduleSpecifier:
|
|
1095
|
+
{ moduleSpecifier: POSTGRES_MIGRATION_FACADE, symbol: 'placeholder' },
|
|
836
1096
|
{
|
|
837
1097
|
moduleSpecifier: './end-contract.json',
|
|
838
1098
|
symbol: 'endContract',
|
|
@@ -856,6 +1116,8 @@ export type PostgresOpFactoryCall =
|
|
|
856
1116
|
| AddPrimaryKeyCall
|
|
857
1117
|
| AddForeignKeyCall
|
|
858
1118
|
| AddUniqueCall
|
|
1119
|
+
| AddCheckConstraintCall
|
|
1120
|
+
| DropCheckConstraintCall
|
|
859
1121
|
| CreateIndexCall
|
|
860
1122
|
| DropIndexCall
|
|
861
1123
|
| DropConstraintCall
|
|
@@ -867,22 +1129,3 @@ export type PostgresOpFactoryCall =
|
|
|
867
1129
|
| CreateExtensionCall
|
|
868
1130
|
| CreateSchemaCall
|
|
869
1131
|
| DataTransformCall;
|
|
870
|
-
|
|
871
|
-
/**
|
|
872
|
-
* Stable identity key for reconciliation-level dedup.
|
|
873
|
-
*
|
|
874
|
-
* Two calls whose runtime ops would share the same `id` return the same
|
|
875
|
-
* key, so a `Set<string>` can collapse them before they're emitted. The
|
|
876
|
-
* current implementation delegates to `toOp().id`, which is the
|
|
877
|
-
* authoritative identity; isolating dedup behind this helper lets a future
|
|
878
|
-
* pass replace it with an allocation-free computation directly from the
|
|
879
|
-
* call's fields without touching call sites.
|
|
880
|
-
*
|
|
881
|
-
* `DataTransformCall` intentionally has no sensible identity today — it
|
|
882
|
-
* throws `PN-MIG-2001` on `toOp()`. Reconciliation never produces one; the
|
|
883
|
-
* helper is unspecified for that variant and only meant for
|
|
884
|
-
* reconciliation-emitted calls.
|
|
885
|
-
*/
|
|
886
|
-
export function identityKeyFor(call: PostgresOpFactoryCall): string {
|
|
887
|
-
return call.toOp().id;
|
|
888
|
-
}
|
|
@@ -1,16 +1,8 @@
|
|
|
1
|
-
import
|
|
2
|
-
import { quoteIdentifier } from '../../sql-utils';
|
|
1
|
+
import { REFERENTIAL_ACTION_SQL } from '@prisma-next/sql-contract/referential-action-sql';
|
|
2
|
+
import { escapeLiteral, quoteIdentifier } from '../../sql-utils';
|
|
3
3
|
import { constraintExistsCheck, qualifyTableName } from '../planner-sql-checks';
|
|
4
4
|
import { type ForeignKeySpec, type Op, step, targetDetails } from './shared';
|
|
5
5
|
|
|
6
|
-
const REFERENTIAL_ACTION_SQL: Record<ReferentialAction, string> = {
|
|
7
|
-
noAction: 'NO ACTION',
|
|
8
|
-
restrict: 'RESTRICT',
|
|
9
|
-
cascade: 'CASCADE',
|
|
10
|
-
setNull: 'SET NULL',
|
|
11
|
-
setDefault: 'SET DEFAULT',
|
|
12
|
-
};
|
|
13
|
-
|
|
14
6
|
function renderForeignKeySql(schemaName: string, tableName: string, fk: ForeignKeySpec): string {
|
|
15
7
|
let sql = `ALTER TABLE ${qualifyTableName(schemaName, tableName)}
|
|
16
8
|
ADD CONSTRAINT ${quoteIdentifier(fk.name)}
|
|
@@ -145,6 +137,83 @@ export function addForeignKey(schemaName: string, tableName: string, fk: Foreign
|
|
|
145
137
|
};
|
|
146
138
|
}
|
|
147
139
|
|
|
140
|
+
export function addCheckConstraint(
|
|
141
|
+
schemaName: string,
|
|
142
|
+
tableName: string,
|
|
143
|
+
constraintName: string,
|
|
144
|
+
column: string,
|
|
145
|
+
values: readonly string[],
|
|
146
|
+
): Op {
|
|
147
|
+
const qualified = qualifyTableName(schemaName, tableName);
|
|
148
|
+
const valueList = values.map((v) => `'${escapeLiteral(v)}'`).join(', ');
|
|
149
|
+
return {
|
|
150
|
+
id: `checkConstraint.${tableName}.${constraintName}`,
|
|
151
|
+
label: `Add check constraint "${constraintName}" on "${tableName}"."${column}"`,
|
|
152
|
+
operationClass: 'additive',
|
|
153
|
+
target: targetDetails('checkConstraint', constraintName, schemaName, tableName),
|
|
154
|
+
precheck: [
|
|
155
|
+
step(
|
|
156
|
+
`ensure constraint "${constraintName}" does not exist`,
|
|
157
|
+
constraintExistsCheck({
|
|
158
|
+
constraintName,
|
|
159
|
+
schema: schemaName,
|
|
160
|
+
table: tableName,
|
|
161
|
+
exists: false,
|
|
162
|
+
}),
|
|
163
|
+
),
|
|
164
|
+
],
|
|
165
|
+
execute: [
|
|
166
|
+
step(
|
|
167
|
+
`add check constraint "${constraintName}"`,
|
|
168
|
+
`ALTER TABLE ${qualified} ADD CONSTRAINT ${quoteIdentifier(constraintName)} CHECK (${quoteIdentifier(column)} IN (${valueList}))`,
|
|
169
|
+
),
|
|
170
|
+
],
|
|
171
|
+
postcheck: [
|
|
172
|
+
step(
|
|
173
|
+
`verify constraint "${constraintName}" exists`,
|
|
174
|
+
constraintExistsCheck({ constraintName, schema: schemaName, table: tableName }),
|
|
175
|
+
),
|
|
176
|
+
],
|
|
177
|
+
};
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
export function dropCheckConstraint(
|
|
181
|
+
schemaName: string,
|
|
182
|
+
tableName: string,
|
|
183
|
+
constraintName: string,
|
|
184
|
+
): Op {
|
|
185
|
+
const qualified = qualifyTableName(schemaName, tableName);
|
|
186
|
+
return {
|
|
187
|
+
id: `dropCheckConstraint.${tableName}.${constraintName}`,
|
|
188
|
+
label: `Drop check constraint "${constraintName}" on "${tableName}"`,
|
|
189
|
+
operationClass: 'destructive',
|
|
190
|
+
target: targetDetails('checkConstraint', constraintName, schemaName, tableName),
|
|
191
|
+
precheck: [
|
|
192
|
+
step(
|
|
193
|
+
`ensure constraint "${constraintName}" exists`,
|
|
194
|
+
constraintExistsCheck({ constraintName, schema: schemaName, table: tableName }),
|
|
195
|
+
),
|
|
196
|
+
],
|
|
197
|
+
execute: [
|
|
198
|
+
step(
|
|
199
|
+
`drop check constraint "${constraintName}"`,
|
|
200
|
+
`ALTER TABLE ${qualified} DROP CONSTRAINT ${quoteIdentifier(constraintName)}`,
|
|
201
|
+
),
|
|
202
|
+
],
|
|
203
|
+
postcheck: [
|
|
204
|
+
step(
|
|
205
|
+
`verify constraint "${constraintName}" does not exist`,
|
|
206
|
+
constraintExistsCheck({
|
|
207
|
+
constraintName,
|
|
208
|
+
schema: schemaName,
|
|
209
|
+
table: tableName,
|
|
210
|
+
exists: false,
|
|
211
|
+
}),
|
|
212
|
+
),
|
|
213
|
+
],
|
|
214
|
+
};
|
|
215
|
+
}
|
|
216
|
+
|
|
148
217
|
/**
|
|
149
218
|
* `kind` feeds the operation's `target.details.objectType`. Descriptor-flow
|
|
150
219
|
* does not carry kind information in its drop-constraint descriptor, so the
|
|
@@ -69,20 +69,3 @@ export function installExtension(options: {
|
|
|
69
69
|
],
|
|
70
70
|
};
|
|
71
71
|
}
|
|
72
|
-
|
|
73
|
-
export function createSchema(schemaName: string): Op {
|
|
74
|
-
return {
|
|
75
|
-
id: `schema.${schemaName}`,
|
|
76
|
-
label: `Create schema "${schemaName}"`,
|
|
77
|
-
operationClass: 'additive',
|
|
78
|
-
target: { id: 'postgres' },
|
|
79
|
-
precheck: [],
|
|
80
|
-
execute: [
|
|
81
|
-
step(
|
|
82
|
-
`Create schema "${schemaName}"`,
|
|
83
|
-
`CREATE SCHEMA IF NOT EXISTS ${quoteIdentifier(schemaName)}`,
|
|
84
|
-
),
|
|
85
|
-
],
|
|
86
|
-
postcheck: [],
|
|
87
|
-
};
|
|
88
|
-
}
|
|
@@ -7,9 +7,9 @@ import type { OperationClass, PostgresPlanTargetDetails } from '../planner-targe
|
|
|
7
7
|
export type Op = SqlMigrationPlanOperation<PostgresPlanTargetDetails>;
|
|
8
8
|
|
|
9
9
|
/**
|
|
10
|
-
* Literal-args shape for a column definition consumed by `
|
|
11
|
-
*
|
|
12
|
-
* already happened in the wrapper.
|
|
10
|
+
* Literal-args shape for a column definition consumed by `addColumn` and
|
|
11
|
+
* similar operations. Fully materialized: codec expansion and default
|
|
12
|
+
* rendering have already happened in the wrapper.
|
|
13
13
|
*
|
|
14
14
|
* - `typeSql` is the column's DDL type string (e.g. `"integer"`, `"SERIAL"`,
|
|
15
15
|
* `"varchar(100)"`), already produced by `buildColumnTypeSql` in the
|
|
@@ -1,43 +1,5 @@
|
|
|
1
|
-
import { quoteIdentifier } from '../../sql-utils';
|
|
2
1
|
import { qualifyTableName, toRegclassLiteral } from '../planner-sql-checks';
|
|
3
|
-
import { type
|
|
4
|
-
|
|
5
|
-
export function createTable(
|
|
6
|
-
schemaName: string,
|
|
7
|
-
tableName: string,
|
|
8
|
-
columns: ReadonlyArray<ColumnSpec>,
|
|
9
|
-
primaryKey?: { readonly columns: readonly string[] },
|
|
10
|
-
): Op {
|
|
11
|
-
const qualified = qualifyTableName(schemaName, tableName);
|
|
12
|
-
const columnDefs = columns.map(renderColumnDefinition);
|
|
13
|
-
const constraintDefs: string[] = [];
|
|
14
|
-
if (primaryKey) {
|
|
15
|
-
constraintDefs.push(`PRIMARY KEY (${primaryKey.columns.map(quoteIdentifier).join(', ')})`);
|
|
16
|
-
}
|
|
17
|
-
const allDefs = [...columnDefs, ...constraintDefs];
|
|
18
|
-
const createSql = `CREATE TABLE ${qualified} (\n ${allDefs.join(',\n ')}\n)`;
|
|
19
|
-
|
|
20
|
-
return {
|
|
21
|
-
id: `table.${tableName}`,
|
|
22
|
-
label: `Create table "${tableName}"`,
|
|
23
|
-
summary: `Creates table "${tableName}"`,
|
|
24
|
-
operationClass: 'additive',
|
|
25
|
-
target: targetDetails('table', tableName, schemaName),
|
|
26
|
-
precheck: [
|
|
27
|
-
step(
|
|
28
|
-
`ensure table "${tableName}" does not exist`,
|
|
29
|
-
`SELECT to_regclass(${toRegclassLiteral(schemaName, tableName)}) IS NULL`,
|
|
30
|
-
),
|
|
31
|
-
],
|
|
32
|
-
execute: [step(`create table "${tableName}"`, createSql)],
|
|
33
|
-
postcheck: [
|
|
34
|
-
step(
|
|
35
|
-
`verify table "${tableName}" exists`,
|
|
36
|
-
`SELECT to_regclass(${toRegclassLiteral(schemaName, tableName)}) IS NOT NULL`,
|
|
37
|
-
),
|
|
38
|
-
],
|
|
39
|
-
};
|
|
40
|
-
}
|
|
2
|
+
import { type Op, step, targetDetails } from './shared';
|
|
41
3
|
|
|
42
4
|
export function dropTable(schemaName: string, tableName: string): Op {
|
|
43
5
|
const qualified = qualifyTableName(schemaName, tableName);
|