cdk-dms-replication 0.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +13491 -0
- package/API.md +8606 -0
- package/LICENSE +202 -0
- package/README.md +883 -0
- package/cdk.context.json +1 -0
- package/integ/sample-app.ts +229 -0
- package/lib/dms-roles.d.ts +14 -0
- package/lib/dms-roles.js +42 -0
- package/lib/endpoint-settings.d.ts +419 -0
- package/lib/endpoint-settings.js +3 -0
- package/lib/endpoint.d.ts +143 -0
- package/lib/endpoint.js +402 -0
- package/lib/enums.d.ts +231 -0
- package/lib/enums.js +266 -0
- package/lib/index.d.ts +9 -0
- package/lib/index.js +49 -0
- package/lib/migration-pipeline.d.ts +253 -0
- package/lib/migration-pipeline.js +218 -0
- package/lib/replication-instance.d.ts +99 -0
- package/lib/replication-instance.js +93 -0
- package/lib/replication-task.d.ts +72 -0
- package/lib/replication-task.js +46 -0
- package/lib/serverless-pipeline.d.ts +196 -0
- package/lib/serverless-pipeline.js +271 -0
- package/lib/table-mappings.d.ts +178 -0
- package/lib/table-mappings.js +283 -0
- package/lib/task-settings.d.ts +228 -0
- package/lib/task-settings.js +291 -0
- package/package.json +170 -0
- package/scripts/sync-instance-classes.js +213 -0
|
@@ -0,0 +1,271 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.DmsServerlessPipeline = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const cdk = require("aws-cdk-lib");
|
|
7
|
+
const dms = require("aws-cdk-lib/aws-dms");
|
|
8
|
+
const ec2 = require("aws-cdk-lib/aws-ec2");
|
|
9
|
+
const kms = require("aws-cdk-lib/aws-kms");
|
|
10
|
+
const logs = require("aws-cdk-lib/aws-logs");
|
|
11
|
+
const constructs_1 = require("constructs");
|
|
12
|
+
const dms_roles_1 = require("./dms-roles");
|
|
13
|
+
const endpoint_1 = require("./endpoint");
|
|
14
|
+
const enums_1 = require("./enums");
|
|
15
|
+
const table_mappings_1 = require("./table-mappings");
|
|
16
|
+
// ---------------------------------------------------------------------------
|
|
17
|
+
// Valid capacity unit values — documented in the DMS API reference
|
|
18
|
+
// ---------------------------------------------------------------------------
|
|
19
|
+
const VALID_CAPACITY_UNITS = new Set([1, 2, 4, 8, 16, 32, 64, 128, 192, 256, 384]);
|
|
20
|
+
// ---------------------------------------------------------------------------
|
|
21
|
+
// Construct
|
|
22
|
+
// ---------------------------------------------------------------------------
|
|
23
|
+
/**
|
|
24
|
+
* An L3 CDK pattern construct that provisions a DMS Serverless replication
|
|
25
|
+
* pipeline:
|
|
26
|
+
*
|
|
27
|
+
* - **Replication config** — backed by `CfnReplicationConfig`; DMS auto-scales
|
|
28
|
+
* capacity between `minCapacityUnits` and `maxCapacityUnits`.
|
|
29
|
+
* - **Source endpoint** — supports every engine DMS supports.
|
|
30
|
+
* - **Target endpoint** — supports every engine DMS supports.
|
|
31
|
+
* - **Subnet group** — private subnet placement.
|
|
32
|
+
* - **KMS key** — storage encryption at rest (created if not provided).
|
|
33
|
+
* - **Security group** — dedicated group (created if not provided).
|
|
34
|
+
* - **IAM roles** — `dms-vpc-role` and `dms-cloudwatch-logs-role`.
|
|
35
|
+
* - **CloudWatch log group** — (optional) retains replication logs.
|
|
36
|
+
*
|
|
37
|
+
* > **CDC start/stop position limitation**: `CfnReplicationConfig` does not
|
|
38
|
+
* > expose `cdcStartTime` / `cdcStartPosition` / `cdcStopPosition`. To start
|
|
39
|
+
* > from a specific position, call the `StartReplication` API directly after
|
|
40
|
+
* > the config is created (e.g. from a Lambda custom resource or CLI).
|
|
41
|
+
*
|
|
42
|
+
* @example
|
|
43
|
+
* const pipeline = new DmsServerlessPipeline(this, 'ServerlessPipeline', {
|
|
44
|
+
* vpc,
|
|
45
|
+
* maxCapacityUnits: 8,
|
|
46
|
+
* migrationType: MigrationType.FULL_LOAD_AND_CDC,
|
|
47
|
+
* sourceEndpoint: {
|
|
48
|
+
* engine: EndpointEngine.MYSQL,
|
|
49
|
+
* serverName: 'mysql.example.com',
|
|
50
|
+
* port: 3306,
|
|
51
|
+
* username: 'dms_user',
|
|
52
|
+
* password: cdk.SecretValue.secretsManager('mysql-dms-secret'),
|
|
53
|
+
* },
|
|
54
|
+
* targetEndpoint: {
|
|
55
|
+
* engine: EndpointEngine.AURORA_POSTGRESQL,
|
|
56
|
+
* serverName: cluster.clusterEndpoint.hostname,
|
|
57
|
+
* port: 5432,
|
|
58
|
+
* username: 'dms_user',
|
|
59
|
+
* password: cdk.SecretValue.secretsManager('aurora-dms-secret'),
|
|
60
|
+
* },
|
|
61
|
+
* });
|
|
62
|
+
*/
|
|
63
|
+
class DmsServerlessPipeline extends constructs_1.Construct {
|
|
64
|
+
constructor(scope, id, props) {
|
|
65
|
+
super(scope, id);
|
|
66
|
+
this.validateProps(props);
|
|
67
|
+
const removalPolicy = props.removalPolicy ?? cdk.RemovalPolicy.DESTROY;
|
|
68
|
+
const createRoles = props.createDmsServiceRoles ?? true;
|
|
69
|
+
// -----------------------------------------------------------------------
|
|
70
|
+
// dms-vpc-role — required by DMS to place the config inside a VPC.
|
|
71
|
+
// -----------------------------------------------------------------------
|
|
72
|
+
if (createRoles) {
|
|
73
|
+
this.dmsVpcRole = (0, dms_roles_1.ensureDmsVpcRole)(cdk.Stack.of(this));
|
|
74
|
+
}
|
|
75
|
+
// -----------------------------------------------------------------------
|
|
76
|
+
// CloudWatch log group & IAM role
|
|
77
|
+
// -----------------------------------------------------------------------
|
|
78
|
+
const enableLogs = props.enableCloudWatchLogs ?? true;
|
|
79
|
+
if (enableLogs) {
|
|
80
|
+
this.logGroup = new logs.LogGroup(this, 'LogGroup', {
|
|
81
|
+
logGroupName: `/aws/dms/serverless/${id}`,
|
|
82
|
+
retention: props.logRetention ?? logs.RetentionDays.ONE_MONTH,
|
|
83
|
+
removalPolicy,
|
|
84
|
+
});
|
|
85
|
+
if (createRoles) {
|
|
86
|
+
this.dmsCloudWatchRole = (0, dms_roles_1.ensureDmsCloudWatchRole)(cdk.Stack.of(this));
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
// -----------------------------------------------------------------------
|
|
90
|
+
// KMS key
|
|
91
|
+
// -----------------------------------------------------------------------
|
|
92
|
+
this.encryptionKey =
|
|
93
|
+
props.kmsKey ??
|
|
94
|
+
new kms.Key(this, 'EncryptionKey', {
|
|
95
|
+
description: `DMS serverless replication encryption key for ${id}`,
|
|
96
|
+
enableKeyRotation: true,
|
|
97
|
+
removalPolicy,
|
|
98
|
+
});
|
|
99
|
+
// -----------------------------------------------------------------------
|
|
100
|
+
// Security group
|
|
101
|
+
// -----------------------------------------------------------------------
|
|
102
|
+
const providedSgs = props.securityGroups ?? [];
|
|
103
|
+
if (providedSgs.length === 0) {
|
|
104
|
+
this.securityGroup = new ec2.SecurityGroup(this, 'SecurityGroup', {
|
|
105
|
+
vpc: props.vpc,
|
|
106
|
+
description: `DMS serverless replication security group for ${id}`,
|
|
107
|
+
allowAllOutbound: true,
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
else {
|
|
111
|
+
this.securityGroup = providedSgs[0];
|
|
112
|
+
}
|
|
113
|
+
const securityGroupIds = providedSgs.length > 0
|
|
114
|
+
? providedSgs.map((sg) => sg.securityGroupId)
|
|
115
|
+
: [this.securityGroup.securityGroupId];
|
|
116
|
+
// -----------------------------------------------------------------------
|
|
117
|
+
// Subnet group
|
|
118
|
+
// -----------------------------------------------------------------------
|
|
119
|
+
const subnetSelection = props.vpcSubnets ?? {
|
|
120
|
+
subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS,
|
|
121
|
+
};
|
|
122
|
+
const subnets = props.vpc.selectSubnets(subnetSelection);
|
|
123
|
+
this.subnetGroup = new dms.CfnReplicationSubnetGroup(this, 'SubnetGroup', {
|
|
124
|
+
replicationSubnetGroupDescription: `DMS serverless subnet group for ${id}`,
|
|
125
|
+
subnetIds: subnets.subnetIds,
|
|
126
|
+
});
|
|
127
|
+
// -----------------------------------------------------------------------
|
|
128
|
+
// Source endpoint
|
|
129
|
+
// -----------------------------------------------------------------------
|
|
130
|
+
if (props.existingSourceEndpoint) {
|
|
131
|
+
this.source = props.existingSourceEndpoint;
|
|
132
|
+
}
|
|
133
|
+
else {
|
|
134
|
+
const sourceOpts = props.sourceEndpoint;
|
|
135
|
+
this.source = new endpoint_1.DmsEndpoint(this, 'SourceEndpoint', {
|
|
136
|
+
endpointType: enums_1.EndpointType.SOURCE,
|
|
137
|
+
engine: sourceOpts.engine,
|
|
138
|
+
endpointIdentifier: sourceOpts.endpointIdentifier,
|
|
139
|
+
serverName: sourceOpts.serverName,
|
|
140
|
+
port: sourceOpts.port,
|
|
141
|
+
username: sourceOpts.username,
|
|
142
|
+
password: sourceOpts.password,
|
|
143
|
+
databaseName: sourceOpts.databaseName,
|
|
144
|
+
extraConnectionAttributes: sourceOpts.extraConnectionAttributes,
|
|
145
|
+
certificateArn: sourceOpts.certificateArn,
|
|
146
|
+
sslMode: sourceOpts.sslMode,
|
|
147
|
+
mySqlSettings: sourceOpts.mySqlSettings,
|
|
148
|
+
postgreSqlSettings: sourceOpts.postgreSqlSettings,
|
|
149
|
+
oracleSettings: sourceOpts.oracleSettings,
|
|
150
|
+
sqlServerSettings: sourceOpts.sqlServerSettings,
|
|
151
|
+
sapAseSettings: sourceOpts.sapAseSettings,
|
|
152
|
+
db2Settings: sourceOpts.db2Settings,
|
|
153
|
+
mongoDbSettings: sourceOpts.mongoDbSettings,
|
|
154
|
+
s3Settings: sourceOpts.s3Settings,
|
|
155
|
+
removalPolicy,
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
// -----------------------------------------------------------------------
|
|
159
|
+
// Target endpoint
|
|
160
|
+
// -----------------------------------------------------------------------
|
|
161
|
+
if (props.existingTargetEndpoint) {
|
|
162
|
+
this.target = props.existingTargetEndpoint;
|
|
163
|
+
}
|
|
164
|
+
else {
|
|
165
|
+
const targetOpts = props.targetEndpoint;
|
|
166
|
+
this.target = new endpoint_1.DmsEndpoint(this, 'TargetEndpoint', {
|
|
167
|
+
endpointType: enums_1.EndpointType.TARGET,
|
|
168
|
+
engine: targetOpts.engine,
|
|
169
|
+
endpointIdentifier: targetOpts.endpointIdentifier,
|
|
170
|
+
serverName: targetOpts.serverName,
|
|
171
|
+
port: targetOpts.port,
|
|
172
|
+
username: targetOpts.username,
|
|
173
|
+
password: targetOpts.password,
|
|
174
|
+
databaseName: targetOpts.databaseName,
|
|
175
|
+
extraConnectionAttributes: targetOpts.extraConnectionAttributes,
|
|
176
|
+
certificateArn: targetOpts.certificateArn,
|
|
177
|
+
sslMode: targetOpts.sslMode,
|
|
178
|
+
mySqlSettings: targetOpts.mySqlSettings,
|
|
179
|
+
postgreSqlSettings: targetOpts.postgreSqlSettings,
|
|
180
|
+
oracleSettings: targetOpts.oracleSettings,
|
|
181
|
+
sqlServerSettings: targetOpts.sqlServerSettings,
|
|
182
|
+
sapAseSettings: targetOpts.sapAseSettings,
|
|
183
|
+
db2Settings: targetOpts.db2Settings,
|
|
184
|
+
mongoDbSettings: targetOpts.mongoDbSettings,
|
|
185
|
+
s3Settings: targetOpts.s3Settings,
|
|
186
|
+
dynamoDbSettings: targetOpts.dynamoDbSettings,
|
|
187
|
+
redshiftSettings: targetOpts.redshiftSettings,
|
|
188
|
+
kinesisSettings: targetOpts.kinesisSettings,
|
|
189
|
+
kafkaSettings: targetOpts.kafkaSettings,
|
|
190
|
+
openSearchSettings: targetOpts.openSearchSettings,
|
|
191
|
+
neptuneSettings: targetOpts.neptuneSettings,
|
|
192
|
+
redisSettings: targetOpts.redisSettings,
|
|
193
|
+
removalPolicy,
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
// -----------------------------------------------------------------------
|
|
197
|
+
// Table mappings — default to "include everything"
|
|
198
|
+
// -----------------------------------------------------------------------
|
|
199
|
+
const tableMappingsJson = props.tableMappings ?? new table_mappings_1.TableMappings().includeSchema('%').toJson();
|
|
200
|
+
// CfnReplicationConfig.tableMappings expects a JSON object, not a string.
|
|
201
|
+
const tableMappingsObj = JSON.parse(tableMappingsJson);
|
|
202
|
+
const replicationSettingsObj = props.taskSettings !== undefined ? JSON.parse(props.taskSettings) : undefined;
|
|
203
|
+
// -----------------------------------------------------------------------
|
|
204
|
+
// Replication config identifier
|
|
205
|
+
// -----------------------------------------------------------------------
|
|
206
|
+
const configIdentifier = props.replicationConfigIdentifier ??
|
|
207
|
+
cdk.Names.uniqueResourceName(this, { maxLength: 63 }).toLowerCase();
|
|
208
|
+
// -----------------------------------------------------------------------
|
|
209
|
+
// CfnReplicationConfig
|
|
210
|
+
// -----------------------------------------------------------------------
|
|
211
|
+
this.cfnReplicationConfig = new dms.CfnReplicationConfig(this, 'Resource', {
|
|
212
|
+
replicationConfigIdentifier: configIdentifier,
|
|
213
|
+
replicationType: props.migrationType,
|
|
214
|
+
sourceEndpointArn: this.source.endpointArn,
|
|
215
|
+
targetEndpointArn: this.target.endpointArn,
|
|
216
|
+
tableMappings: tableMappingsObj,
|
|
217
|
+
replicationSettings: replicationSettingsObj,
|
|
218
|
+
computeConfig: {
|
|
219
|
+
maxCapacityUnits: props.maxCapacityUnits,
|
|
220
|
+
minCapacityUnits: props.minCapacityUnits,
|
|
221
|
+
multiAz: props.multiAz ?? false,
|
|
222
|
+
kmsKeyId: this.encryptionKey.keyArn,
|
|
223
|
+
replicationSubnetGroupId: this.subnetGroup.ref,
|
|
224
|
+
vpcSecurityGroupIds: securityGroupIds,
|
|
225
|
+
preferredMaintenanceWindow: props.preferredMaintenanceWindow,
|
|
226
|
+
},
|
|
227
|
+
});
|
|
228
|
+
this.cfnReplicationConfig.applyRemovalPolicy(removalPolicy);
|
|
229
|
+
// The replication config cannot be placed in a VPC until dms-vpc-role exists.
|
|
230
|
+
if (this.dmsVpcRole) {
|
|
231
|
+
this.cfnReplicationConfig.addDependency(this.dmsVpcRole.node.defaultChild);
|
|
232
|
+
}
|
|
233
|
+
this.replicationConfigArn = this.cfnReplicationConfig.ref;
|
|
234
|
+
}
|
|
235
|
+
// ---------------------------------------------------------------------------
|
|
236
|
+
// Private helpers
|
|
237
|
+
// ---------------------------------------------------------------------------
|
|
238
|
+
validateProps(props) {
|
|
239
|
+
if (!VALID_CAPACITY_UNITS.has(props.maxCapacityUnits)) {
|
|
240
|
+
throw new Error(`maxCapacityUnits must be one of ${[...VALID_CAPACITY_UNITS].join(', ')}. Got: ${props.maxCapacityUnits}`);
|
|
241
|
+
}
|
|
242
|
+
if (props.minCapacityUnits !== undefined &&
|
|
243
|
+
!VALID_CAPACITY_UNITS.has(props.minCapacityUnits)) {
|
|
244
|
+
throw new Error(`minCapacityUnits must be one of ${[...VALID_CAPACITY_UNITS].join(', ')}. Got: ${props.minCapacityUnits}`);
|
|
245
|
+
}
|
|
246
|
+
if (props.minCapacityUnits !== undefined &&
|
|
247
|
+
props.minCapacityUnits > props.maxCapacityUnits) {
|
|
248
|
+
throw new Error(`minCapacityUnits (${props.minCapacityUnits}) must be <= maxCapacityUnits (${props.maxCapacityUnits}).`);
|
|
249
|
+
}
|
|
250
|
+
const hasSource = !!props.sourceEndpoint;
|
|
251
|
+
const hasExistingSource = !!props.existingSourceEndpoint;
|
|
252
|
+
if (hasSource && hasExistingSource) {
|
|
253
|
+
throw new Error('Provide either `sourceEndpoint` or `existingSourceEndpoint`, not both.');
|
|
254
|
+
}
|
|
255
|
+
if (!hasSource && !hasExistingSource) {
|
|
256
|
+
throw new Error('One of `sourceEndpoint` or `existingSourceEndpoint` is required.');
|
|
257
|
+
}
|
|
258
|
+
const hasTarget = !!props.targetEndpoint;
|
|
259
|
+
const hasExistingTarget = !!props.existingTargetEndpoint;
|
|
260
|
+
if (hasTarget && hasExistingTarget) {
|
|
261
|
+
throw new Error('Provide either `targetEndpoint` or `existingTargetEndpoint`, not both.');
|
|
262
|
+
}
|
|
263
|
+
if (!hasTarget && !hasExistingTarget) {
|
|
264
|
+
throw new Error('One of `targetEndpoint` or `existingTargetEndpoint` is required.');
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
exports.DmsServerlessPipeline = DmsServerlessPipeline;
|
|
269
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
270
|
+
DmsServerlessPipeline[_a] = { fqn: "cdk-dms-replication.DmsServerlessPipeline", version: "0.0.0" };
|
|
271
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"serverless-pipeline.js","sourceRoot":"","sources":["../src/serverless-pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport * as dms from 'aws-cdk-lib/aws-dms';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as kms from 'aws-cdk-lib/aws-kms';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport { Construct } from 'constructs';\nimport { ensureDmsCloudWatchRole, ensureDmsVpcRole } from './dms-roles';\nimport { DmsEndpoint, IDmsEndpoint } from './endpoint';\nimport { EndpointType, MigrationType } from './enums';\nimport { SourceEndpointOptions, TargetEndpointOptions } from './migration-pipeline';\nimport { TableMappings } from './table-mappings';\n\n// ---------------------------------------------------------------------------\n// Valid capacity unit values — documented in the DMS API reference\n// ---------------------------------------------------------------------------\nconst VALID_CAPACITY_UNITS = new Set([1, 2, 4, 8, 16, 32, 64, 128, 192, 256, 384]);\n\n// ---------------------------------------------------------------------------\n// Props\n// ---------------------------------------------------------------------------\n\n/** Props for {@link DmsServerlessPipeline}. */\nexport interface DmsServerlessPipelineProps {\n  // -------------------------------------------------------------------------\n  // Networking\n  // -------------------------------------------------------------------------\n\n  /**\n   * VPC in which to place the serverless replication config.\n   * The config is placed in private subnets.\n   */\n  readonly vpc: ec2.IVpc;\n\n  /**\n   * Subnet selection for the replication subnet group.\n   * @default private subnets with egress\n   */\n  readonly vpcSubnets?: ec2.SubnetSelection;\n\n  /**\n   * Security groups to attach to the serverless replication config.\n   * A new security group is created if none are provided.\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  // -------------------------------------------------------------------------\n  // Compute config\n  // -------------------------------------------------------------------------\n\n  /**\n   * Maximum number of DMS Capacity Units (DCUs) the serverless replication may\n   * scale up to.\n   *\n   * Valid values: 1, 2, 4, 8, 16, 32, 64, 128, 192, 256, 384.\n   */\n  readonly maxCapacityUnits: number;\n\n  /**\n   * Minimum number of DCUs DMS will provision at start-up.\n   * @default DMS auto-determines based on workload\n   */\n  readonly minCapacityUnits?: number;\n\n  /**\n   * Whether the serverless replication is Multi-AZ.\n   * @default false\n   */\n  readonly multiAz?: boolean;\n\n  /**\n   * Preferred maintenance window, e.g. \"sun:04:00-sun:04:30\".\n   */\n  readonly preferredMaintenanceWindow?: string;\n\n  // -------------------------------------------------------------------------\n  // Encryption\n  // -------------------------------------------------------------------------\n\n  /**\n   * KMS key used to encrypt replication storage at rest.\n   * A new key is created if not provided.\n   */\n  readonly kmsKey?: kms.IKey;\n\n  // -------------------------------------------------------------------------\n  // Endpoints\n  // -------------------------------------------------------------------------\n\n  /**\n   * Source endpoint configuration.\n   * Provide this OR `existingSourceEndpoint` — not both.\n   */\n  readonly sourceEndpoint?: SourceEndpointOptions;\n\n  /**\n   * An existing {@link IDmsEndpoint} to use as the source.\n   * Provide this OR `sourceEndpoint` — not both.\n   */\n  readonly existingSourceEndpoint?: IDmsEndpoint;\n\n  /**\n   * Target endpoint configuration.\n   * Provide this OR `existingTargetEndpoint` — not both.\n   */\n  readonly targetEndpoint?: TargetEndpointOptions;\n\n  /**\n   * An existing {@link IDmsEndpoint} to use as the target.\n   * Provide this OR `targetEndpoint` — not both.\n   */\n  readonly existingTargetEndpoint?: IDmsEndpoint;\n\n  // -------------------------------------------------------------------------\n  // Migration config\n  // -------------------------------------------------------------------------\n\n  /** The type of migration to perform. */\n  readonly migrationType: MigrationType;\n\n  /**\n   * Table mappings JSON string.\n   * Use {@link TableMappings} to build this.\n   * Defaults to \"include all tables in all schemas\" if not provided.\n   */\n  readonly tableMappings?: string;\n\n  /**\n   * Replication settings JSON string (equivalent to task settings for classic DMS).\n   * Use {@link TaskSettings} to build this.\n   * Sensible defaults are applied by DMS if not provided.\n   *\n   * @remarks DMS Serverless does not support setting CDC start/stop positions\n   * via CloudFormation. To start replication from a specific LSN or timestamp\n   * after the replication config has been created, call the\n   * `StartReplication` API with `StartReplicationTaskType` set to\n   * `start-replication` and supply `CdcStartPosition` or `CdcStartTime`.\n   */\n  readonly taskSettings?: string;\n\n  // -------------------------------------------------------------------------\n  // Observability\n  // -------------------------------------------------------------------------\n\n  /**\n   * Whether to create a CloudWatch log group for the replication config.\n   * @default true\n   */\n  readonly enableCloudWatchLogs?: boolean;\n\n  /**\n   * Retention period for the CloudWatch log group.\n   * @default logs.RetentionDays.ONE_MONTH\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  // -------------------------------------------------------------------------\n  // IAM / service roles\n  // -------------------------------------------------------------------------\n\n  /**\n   * Whether to create the two account-level DMS service roles (`dms-vpc-role`\n   * and `dms-cloudwatch-logs-role`) required by DMS.\n   *\n   * Set to `false` if the roles already exist — for example, because a\n   * `DmsMigrationPipeline` or a prior manual deployment already created them.\n   * Attempting to create roles with the same name twice causes a CloudFormation\n   * `EntityAlreadyExists` error.\n   *\n   * @default true\n   */\n  readonly createDmsServiceRoles?: boolean;\n\n  // -------------------------------------------------------------------------\n  // Naming / lifecycle\n  // -------------------------------------------------------------------------\n\n  /**\n   * Logical identifier for the replication config resource.\n   * @default unique name derived from the construct id\n   */\n  readonly replicationConfigIdentifier?: string;\n\n  /**\n   * Removal policy applied to all resources in the pipeline.\n   * @default cdk.RemovalPolicy.DESTROY\n   */\n  readonly removalPolicy?: cdk.RemovalPolicy;\n}\n\n// ---------------------------------------------------------------------------\n// Construct\n// ---------------------------------------------------------------------------\n\n/**\n * An L3 CDK pattern construct that provisions a DMS Serverless replication\n * pipeline:\n *\n * - **Replication config** — backed by `CfnReplicationConfig`; DMS auto-scales\n *   capacity between `minCapacityUnits` and `maxCapacityUnits`.\n * - **Source endpoint** — supports every engine DMS supports.\n * - **Target endpoint** — supports every engine DMS supports.\n * - **Subnet group** — private subnet placement.\n * - **KMS key** — storage encryption at rest (created if not provided).\n * - **Security group** — dedicated group (created if not provided).\n * - **IAM roles** — `dms-vpc-role` and `dms-cloudwatch-logs-role`.\n * - **CloudWatch log group** — (optional) retains replication logs.\n *\n * > **CDC start/stop position limitation**: `CfnReplicationConfig` does not\n * > expose `cdcStartTime` / `cdcStartPosition` / `cdcStopPosition`. To start\n * > from a specific position, call the `StartReplication` API directly after\n * > the config is created (e.g. from a Lambda custom resource or CLI).\n *\n * @example\n * const pipeline = new DmsServerlessPipeline(this, 'ServerlessPipeline', {\n *   vpc,\n *   maxCapacityUnits: 8,\n *   migrationType: MigrationType.FULL_LOAD_AND_CDC,\n *   sourceEndpoint: {\n *     engine: EndpointEngine.MYSQL,\n *     serverName: 'mysql.example.com',\n *     port: 3306,\n *     username: 'dms_user',\n *     password: cdk.SecretValue.secretsManager('mysql-dms-secret'),\n *   },\n *   targetEndpoint: {\n *     engine: EndpointEngine.AURORA_POSTGRESQL,\n *     serverName: cluster.clusterEndpoint.hostname,\n *     port: 5432,\n *     username: 'dms_user',\n *     password: cdk.SecretValue.secretsManager('aurora-dms-secret'),\n *   },\n * });\n */\nexport class DmsServerlessPipeline extends Construct {\n  /** The underlying `CfnReplicationConfig` resource. */\n  readonly cfnReplicationConfig: dms.CfnReplicationConfig;\n\n  /** The ARN of the replication config. */\n  readonly replicationConfigArn: string;\n\n  /** The source endpoint used by this pipeline. */\n  readonly source: IDmsEndpoint;\n\n  /** The target endpoint used by this pipeline. */\n  readonly target: IDmsEndpoint;\n\n  /** The replication subnet group created for this pipeline. */\n  readonly subnetGroup: dms.CfnReplicationSubnetGroup;\n\n  /** The KMS key used for storage encryption. */\n  readonly encryptionKey: kms.IKey;\n\n  /** The security group attached to this pipeline. */\n  readonly securityGroup: ec2.ISecurityGroup;\n\n  /** CloudWatch log group for the replication config (if enableCloudWatchLogs is true). */\n  readonly logGroup?: logs.LogGroup;\n\n  /**\n   * IAM role that allows DMS to write to CloudWatch Logs.\n   * `undefined` when `createDmsServiceRoles` is `false`.\n   */\n  readonly dmsCloudWatchRole?: iam.Role;\n\n  /**\n   * IAM role that allows DMS to manage VPC resources (dms-vpc-role).\n   * `undefined` when `createDmsServiceRoles` is `false`.\n   */\n  readonly dmsVpcRole?: iam.Role;\n\n  constructor(scope: Construct, id: string, props: DmsServerlessPipelineProps) {\n    super(scope, id);\n\n    this.validateProps(props);\n\n    const removalPolicy = props.removalPolicy ?? cdk.RemovalPolicy.DESTROY;\n    const createRoles = props.createDmsServiceRoles ?? true;\n\n    // -----------------------------------------------------------------------\n    // dms-vpc-role — required by DMS to place the config inside a VPC.\n    // -----------------------------------------------------------------------\n    if (createRoles) {\n      this.dmsVpcRole = ensureDmsVpcRole(cdk.Stack.of(this));\n    }\n\n    // -----------------------------------------------------------------------\n    // CloudWatch log group & IAM role\n    // -----------------------------------------------------------------------\n    const enableLogs = props.enableCloudWatchLogs ?? true;\n\n    if (enableLogs) {\n      this.logGroup = new logs.LogGroup(this, 'LogGroup', {\n        logGroupName: `/aws/dms/serverless/${id}`,\n        retention: props.logRetention ?? logs.RetentionDays.ONE_MONTH,\n        removalPolicy,\n      });\n\n      if (createRoles) {\n        this.dmsCloudWatchRole = ensureDmsCloudWatchRole(cdk.Stack.of(this));\n      }\n    }\n\n    // -----------------------------------------------------------------------\n    // KMS key\n    // -----------------------------------------------------------------------\n    this.encryptionKey =\n      props.kmsKey ??\n      new kms.Key(this, 'EncryptionKey', {\n        description: `DMS serverless replication encryption key for ${id}`,\n        enableKeyRotation: true,\n        removalPolicy,\n      });\n\n    // -----------------------------------------------------------------------\n    // Security group\n    // -----------------------------------------------------------------------\n    const providedSgs = props.securityGroups ?? [];\n    if (providedSgs.length === 0) {\n      this.securityGroup = new ec2.SecurityGroup(this, 'SecurityGroup', {\n        vpc: props.vpc,\n        description: `DMS serverless replication security group for ${id}`,\n        allowAllOutbound: true,\n      });\n    } else {\n      this.securityGroup = providedSgs[0];\n    }\n\n    const securityGroupIds =\n      providedSgs.length > 0\n        ? providedSgs.map((sg) => sg.securityGroupId)\n        : [this.securityGroup.securityGroupId];\n\n    // -----------------------------------------------------------------------\n    // Subnet group\n    // -----------------------------------------------------------------------\n    const subnetSelection = props.vpcSubnets ?? {\n      subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS,\n    };\n    const subnets = props.vpc.selectSubnets(subnetSelection);\n\n    this.subnetGroup = new dms.CfnReplicationSubnetGroup(this, 'SubnetGroup', {\n      replicationSubnetGroupDescription: `DMS serverless subnet group for ${id}`,\n      subnetIds: subnets.subnetIds,\n    });\n\n    // -----------------------------------------------------------------------\n    // Source endpoint\n    // -----------------------------------------------------------------------\n    if (props.existingSourceEndpoint) {\n      this.source = props.existingSourceEndpoint;\n    } else {\n      const sourceOpts = props.sourceEndpoint!;\n      this.source = new DmsEndpoint(this, 'SourceEndpoint', {\n        endpointType: EndpointType.SOURCE,\n        engine: sourceOpts.engine,\n        endpointIdentifier: sourceOpts.endpointIdentifier,\n        serverName: sourceOpts.serverName,\n        port: sourceOpts.port,\n        username: sourceOpts.username,\n        password: sourceOpts.password,\n        databaseName: sourceOpts.databaseName,\n        extraConnectionAttributes: sourceOpts.extraConnectionAttributes,\n        certificateArn: sourceOpts.certificateArn,\n        sslMode: sourceOpts.sslMode,\n        mySqlSettings: sourceOpts.mySqlSettings,\n        postgreSqlSettings: sourceOpts.postgreSqlSettings,\n        oracleSettings: sourceOpts.oracleSettings,\n        sqlServerSettings: sourceOpts.sqlServerSettings,\n        sapAseSettings: sourceOpts.sapAseSettings,\n        db2Settings: sourceOpts.db2Settings,\n        mongoDbSettings: sourceOpts.mongoDbSettings,\n        s3Settings: sourceOpts.s3Settings,\n        removalPolicy,\n      });\n    }\n\n    // -----------------------------------------------------------------------\n    // Target endpoint\n    // -----------------------------------------------------------------------\n    if (props.existingTargetEndpoint) {\n      this.target = props.existingTargetEndpoint;\n    } else {\n      const targetOpts = props.targetEndpoint!;\n      this.target = new DmsEndpoint(this, 'TargetEndpoint', {\n        endpointType: EndpointType.TARGET,\n        engine: targetOpts.engine,\n        endpointIdentifier: targetOpts.endpointIdentifier,\n        serverName: targetOpts.serverName,\n        port: targetOpts.port,\n        username: targetOpts.username,\n        password: targetOpts.password,\n        databaseName: targetOpts.databaseName,\n        extraConnectionAttributes: targetOpts.extraConnectionAttributes,\n        certificateArn: targetOpts.certificateArn,\n        sslMode: targetOpts.sslMode,\n        mySqlSettings: targetOpts.mySqlSettings,\n        postgreSqlSettings: targetOpts.postgreSqlSettings,\n        oracleSettings: targetOpts.oracleSettings,\n        sqlServerSettings: targetOpts.sqlServerSettings,\n        sapAseSettings: targetOpts.sapAseSettings,\n        db2Settings: targetOpts.db2Settings,\n        mongoDbSettings: targetOpts.mongoDbSettings,\n        s3Settings: targetOpts.s3Settings,\n        dynamoDbSettings: targetOpts.dynamoDbSettings,\n        redshiftSettings: targetOpts.redshiftSettings,\n        kinesisSettings: targetOpts.kinesisSettings,\n        kafkaSettings: targetOpts.kafkaSettings,\n        openSearchSettings: targetOpts.openSearchSettings,\n        neptuneSettings: targetOpts.neptuneSettings,\n        redisSettings: targetOpts.redisSettings,\n        removalPolicy,\n      });\n    }\n\n    // -----------------------------------------------------------------------\n    // Table mappings — default to \"include everything\"\n    // -----------------------------------------------------------------------\n    const tableMappingsJson =\n      props.tableMappings ?? new TableMappings().includeSchema('%').toJson();\n    // CfnReplicationConfig.tableMappings expects a JSON object, not a string.\n    const tableMappingsObj = JSON.parse(tableMappingsJson);\n\n    const replicationSettingsObj =\n      props.taskSettings !== undefined ? JSON.parse(props.taskSettings) : undefined;\n\n    // -----------------------------------------------------------------------\n    // Replication config identifier\n    // -----------------------------------------------------------------------\n    const configIdentifier =\n      props.replicationConfigIdentifier ??\n      cdk.Names.uniqueResourceName(this, { maxLength: 63 }).toLowerCase();\n\n    // -----------------------------------------------------------------------\n    // CfnReplicationConfig\n    // -----------------------------------------------------------------------\n    this.cfnReplicationConfig = new dms.CfnReplicationConfig(this, 'Resource', {\n      replicationConfigIdentifier: configIdentifier,\n      replicationType: props.migrationType,\n      sourceEndpointArn: this.source.endpointArn,\n      targetEndpointArn: this.target.endpointArn,\n      tableMappings: tableMappingsObj,\n      replicationSettings: replicationSettingsObj,\n      computeConfig: {\n        maxCapacityUnits: props.maxCapacityUnits,\n        minCapacityUnits: props.minCapacityUnits,\n        multiAz: props.multiAz ?? false,\n        kmsKeyId: this.encryptionKey.keyArn,\n        replicationSubnetGroupId: this.subnetGroup.ref,\n        vpcSecurityGroupIds: securityGroupIds,\n        preferredMaintenanceWindow: props.preferredMaintenanceWindow,\n      },\n    });\n    this.cfnReplicationConfig.applyRemovalPolicy(removalPolicy);\n\n    // The replication config cannot be placed in a VPC until dms-vpc-role exists.\n    if (this.dmsVpcRole) {\n      this.cfnReplicationConfig.addDependency(\n        this.dmsVpcRole.node.defaultChild as cdk.CfnResource,\n      );\n    }\n\n    this.replicationConfigArn = this.cfnReplicationConfig.ref;\n  }\n\n  // ---------------------------------------------------------------------------\n  // Private helpers\n  // ---------------------------------------------------------------------------\n\n  private validateProps(props: DmsServerlessPipelineProps): void {\n    if (!VALID_CAPACITY_UNITS.has(props.maxCapacityUnits)) {\n      throw new Error(\n        `maxCapacityUnits must be one of ${[...VALID_CAPACITY_UNITS].join(', ')}. Got: ${props.maxCapacityUnits}`,\n      );\n    }\n    if (\n      props.minCapacityUnits !== undefined &&\n      !VALID_CAPACITY_UNITS.has(props.minCapacityUnits)\n    ) {\n      throw new Error(\n        `minCapacityUnits must be one of ${[...VALID_CAPACITY_UNITS].join(', ')}. Got: ${props.minCapacityUnits}`,\n      );\n    }\n    if (\n      props.minCapacityUnits !== undefined &&\n      props.minCapacityUnits > props.maxCapacityUnits\n    ) {\n      throw new Error(\n        `minCapacityUnits (${props.minCapacityUnits}) must be <= maxCapacityUnits (${props.maxCapacityUnits}).`,\n      );\n    }\n\n    const hasSource = !!props.sourceEndpoint;\n    const hasExistingSource = !!props.existingSourceEndpoint;\n    if (hasSource && hasExistingSource) {\n      throw new Error('Provide either `sourceEndpoint` or `existingSourceEndpoint`, not both.');\n    }\n    if (!hasSource && !hasExistingSource) {\n      throw new Error('One of `sourceEndpoint` or `existingSourceEndpoint` is required.');\n    }\n\n    const hasTarget = !!props.targetEndpoint;\n    const hasExistingTarget = !!props.existingTargetEndpoint;\n    if (hasTarget && hasExistingTarget) {\n      throw new Error('Provide either `targetEndpoint` or `existingTargetEndpoint`, not both.');\n    }\n    if (!hasTarget && !hasExistingTarget) {\n      throw new Error('One of `targetEndpoint` or `existingTargetEndpoint` is required.');\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,178 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Table mappings define which schemas/tables DMS migrates and how they are
|
|
3
|
+
* transformed. This builder produces the JSON string that DMS expects for
|
|
4
|
+
* the `tableMappings` property of a replication task.
|
|
5
|
+
*
|
|
6
|
+
* @see https://docs.aws.amazon.com/dms/latest/userguide/CHAP_Tasks.CustomizingTasks.TableMapping.html
|
|
7
|
+
*/
|
|
8
|
+
/** Whether the rule selects or excludes objects. */
|
|
9
|
+
export declare enum SelectionAction {
|
|
10
|
+
INCLUDE = "include",
|
|
11
|
+
EXCLUDE = "exclude",
|
|
12
|
+
EXPLICIT = "explicit"
|
|
13
|
+
}
|
|
14
|
+
/** Table-level transformation actions. */
|
|
15
|
+
export declare enum TransformationAction {
|
|
16
|
+
/** Convert the name to lowercase. */
|
|
17
|
+
CONVERT_LOWERCASE = "convert-lowercase",
|
|
18
|
+
/** Convert the name to uppercase. */
|
|
19
|
+
CONVERT_UPPERCASE = "convert-uppercase",
|
|
20
|
+
/** Add a prefix to the name. */
|
|
21
|
+
ADD_PREFIX = "add-prefix",
|
|
22
|
+
/** Remove a prefix from the name. */
|
|
23
|
+
REMOVE_PREFIX = "remove-prefix",
|
|
24
|
+
/** Add a suffix to the name. */
|
|
25
|
+
ADD_SUFFIX = "add-suffix",
|
|
26
|
+
/** Remove a suffix from the name. */
|
|
27
|
+
REMOVE_SUFFIX = "remove-suffix",
|
|
28
|
+
/** Rename the object. */
|
|
29
|
+
RENAME = "rename",
|
|
30
|
+
/** Remove the column. */
|
|
31
|
+
REMOVE_COLUMN = "remove-column",
|
|
32
|
+
/** Add a column. */
|
|
33
|
+
ADD_COLUMN = "add-column",
|
|
34
|
+
/** Include the column. */
|
|
35
|
+
INCLUDE_COLUMN = "include-column"
|
|
36
|
+
}
|
|
37
|
+
/** The object type a rule targets. */
|
|
38
|
+
export declare enum RuleObjectLocator {
|
|
39
|
+
SCHEMA = "schema",
|
|
40
|
+
TABLE = "table",
|
|
41
|
+
COLUMN = "column",
|
|
42
|
+
TABLE_TABLESPACE = "table-tablespace",
|
|
43
|
+
INDEX_TABLESPACE = "index-tablespace"
|
|
44
|
+
}
|
|
45
|
+
/** Data type for added columns. */
|
|
46
|
+
export declare enum ColumnDataType {
|
|
47
|
+
STRING = "string",
|
|
48
|
+
INT4 = "int4",
|
|
49
|
+
INT8 = "int8",
|
|
50
|
+
FLOAT4 = "float4",
|
|
51
|
+
FLOAT8 = "float8",
|
|
52
|
+
NUMERIC = "numeric",
|
|
53
|
+
DATETIME = "datetime",
|
|
54
|
+
BYTES = "bytes",
|
|
55
|
+
BLOB = "blob"
|
|
56
|
+
}
|
|
57
|
+
/** A column added via an ADD_COLUMN transformation. */
|
|
58
|
+
export interface AddColumnDefinition {
|
|
59
|
+
/** Name of the new column. */
|
|
60
|
+
readonly columnName: string;
|
|
61
|
+
/** Data type of the new column. */
|
|
62
|
+
readonly columnType: ColumnDataType;
|
|
63
|
+
/** Character length for string columns. */
|
|
64
|
+
readonly columnLength?: number;
|
|
65
|
+
/** Precision for numeric columns. */
|
|
66
|
+
readonly columnPrecision?: number;
|
|
67
|
+
/** Scale for numeric columns. */
|
|
68
|
+
readonly columnScale?: number;
|
|
69
|
+
/**
|
|
70
|
+
* Constant value to populate the column with.
|
|
71
|
+
* Exactly one of `columnValue` or `expression` must be set.
|
|
72
|
+
*/
|
|
73
|
+
readonly columnValue?: string;
|
|
74
|
+
/**
|
|
75
|
+
* Expression (e.g. `$timestamp`) to populate the column.
|
|
76
|
+
* Exactly one of `columnValue` or `expression` must be set.
|
|
77
|
+
*/
|
|
78
|
+
readonly expression?: string;
|
|
79
|
+
}
|
|
80
|
+
/** Object locator identifying the schema, table, and optional column a rule targets. */
|
|
81
|
+
export interface RuleObjectLocatorValue {
|
|
82
|
+
readonly schemaName: string;
|
|
83
|
+
readonly tableName?: string;
|
|
84
|
+
readonly columnName?: string;
|
|
85
|
+
}
|
|
86
|
+
/** Represents a single, fully built rule in the table-mappings JSON. */
|
|
87
|
+
export interface TableMappingRule {
|
|
88
|
+
readonly ruleType: string;
|
|
89
|
+
readonly ruleId: string;
|
|
90
|
+
readonly ruleName: string;
|
|
91
|
+
readonly objectLocator: RuleObjectLocatorValue;
|
|
92
|
+
readonly ruleAction: string;
|
|
93
|
+
readonly value?: string;
|
|
94
|
+
readonly oldValue?: string;
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Fluent builder for DMS table mappings.
|
|
98
|
+
*
|
|
99
|
+
* @example
|
|
100
|
+
* const mappings = new TableMappings()
|
|
101
|
+
* .includeSchema('public')
|
|
102
|
+
* .includeTable('public', 'orders')
|
|
103
|
+
* .excludeTable('public', 'audit_log')
|
|
104
|
+
* .renameSchema('public', 'prod')
|
|
105
|
+
* .toLowerCaseTable('public', '%')
|
|
106
|
+
* .toJson();
|
|
107
|
+
*/
|
|
108
|
+
export declare class TableMappings {
|
|
109
|
+
private readonly rules;
|
|
110
|
+
private nextId;
|
|
111
|
+
/**
|
|
112
|
+
* Include all tables in a schema.
|
|
113
|
+
* Use `%` as a wildcard for `schemaName` to include all schemas.
|
|
114
|
+
*/
|
|
115
|
+
includeSchema(schemaName: string): TableMappings;
|
|
116
|
+
/** Exclude all tables in a schema. */
|
|
117
|
+
excludeSchema(schemaName: string): TableMappings;
|
|
118
|
+
/**
|
|
119
|
+
* Include a specific table (or a wildcard pattern) within a schema.
|
|
120
|
+
* Use `%` for `tableName` to match all tables in the schema.
|
|
121
|
+
*/
|
|
122
|
+
includeTable(schemaName: string, tableName: string): TableMappings;
|
|
123
|
+
/** Exclude a specific table (or wildcard) within a schema. */
|
|
124
|
+
excludeTable(schemaName: string, tableName: string): TableMappings;
|
|
125
|
+
/**
|
|
126
|
+
* Explicitly include a single table. Unlike `include`, `explicit` means DMS
|
|
127
|
+
* only migrates this one table regardless of other `include` rules.
|
|
128
|
+
*/
|
|
129
|
+
explicitTable(schemaName: string, tableName: string): TableMappings;
|
|
130
|
+
private addSelectionRule;
|
|
131
|
+
/** Rename a schema. */
|
|
132
|
+
renameSchema(schemaName: string, newName: string): TableMappings;
|
|
133
|
+
/** Convert all schema names to lowercase. */
|
|
134
|
+
toLowerCaseSchema(schemaName: string): TableMappings;
|
|
135
|
+
/** Convert all schema names to uppercase. */
|
|
136
|
+
toUpperCaseSchema(schemaName: string): TableMappings;
|
|
137
|
+
/** Add a prefix to schema names. */
|
|
138
|
+
addPrefixToSchema(schemaName: string, prefix: string): TableMappings;
|
|
139
|
+
/** Add a suffix to schema names. */
|
|
140
|
+
addSuffixToSchema(schemaName: string, suffix: string): TableMappings;
|
|
141
|
+
/** Rename a table. */
|
|
142
|
+
renameTable(schemaName: string, tableName: string, newName: string): TableMappings;
|
|
143
|
+
/** Convert matching table names to lowercase. Use `%` to match all tables. */
|
|
144
|
+
toLowerCaseTable(schemaName: string, tableName: string): TableMappings;
|
|
145
|
+
/** Convert matching table names to uppercase. */
|
|
146
|
+
toUpperCaseTable(schemaName: string, tableName: string): TableMappings;
|
|
147
|
+
/** Add a prefix to table names. */
|
|
148
|
+
addPrefixToTable(schemaName: string, tableName: string, prefix: string): TableMappings;
|
|
149
|
+
/** Add a suffix to table names. */
|
|
150
|
+
addSuffixToTable(schemaName: string, tableName: string, suffix: string): TableMappings;
|
|
151
|
+
/** Rename a column in a table. */
|
|
152
|
+
renameColumn(schemaName: string, tableName: string, columnName: string, newName: string): TableMappings;
|
|
153
|
+
/** Convert matching column names to lowercase. */
|
|
154
|
+
toLowerCaseColumn(schemaName: string, tableName: string, columnName: string): TableMappings;
|
|
155
|
+
/** Convert matching column names to uppercase. */
|
|
156
|
+
toUpperCaseColumn(schemaName: string, tableName: string, columnName: string): TableMappings;
|
|
157
|
+
/** Remove a column from a table. */
|
|
158
|
+
removeColumn(schemaName: string, tableName: string, columnName: string): TableMappings;
|
|
159
|
+
/**
|
|
160
|
+
* Add a new column to a table.
|
|
161
|
+
*
|
|
162
|
+
* @example
|
|
163
|
+
* mappings.addColumn('public', 'orders', {
|
|
164
|
+
* columnName: 'migration_ts',
|
|
165
|
+
* columnType: ColumnDataType.DATETIME,
|
|
166
|
+
* expression: '$timestamp',
|
|
167
|
+
* });
|
|
168
|
+
*/
|
|
169
|
+
addColumn(schemaName: string, tableName: string, column: AddColumnDefinition): TableMappings;
|
|
170
|
+
private buildDataType;
|
|
171
|
+
private addTransformationRule;
|
|
172
|
+
/**
|
|
173
|
+
* Serialise the accumulated rules to the JSON string expected by DMS.
|
|
174
|
+
* Passes the result directly to `replicationTaskSettings` or
|
|
175
|
+
* `DmsReplicationTask.tableMappings`.
|
|
176
|
+
*/
|
|
177
|
+
toJson(): string;
|
|
178
|
+
}
|