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.
@@ -0,0 +1,253 @@
1
+ import * as cdk from 'aws-cdk-lib';
2
+ import * as ec2 from 'aws-cdk-lib/aws-ec2';
3
+ import * as iam from 'aws-cdk-lib/aws-iam';
4
+ import * as kms from 'aws-cdk-lib/aws-kms';
5
+ import * as logs from 'aws-cdk-lib/aws-logs';
6
+ import { Construct } from 'constructs';
7
+ import { DmsEndpointProps, IDmsEndpoint } from './endpoint';
8
+ import { MigrationType, ReplicationInstanceClass } from './enums';
9
+ import { DmsReplicationInstance } from './replication-instance';
10
+ import { DmsReplicationTask } from './replication-task';
11
+ /**
12
+ * Properties for the source endpoint of a {@link DmsMigrationPipeline}.
13
+ * Extends {@link DmsEndpointProps} but omits `endpointType` (always SOURCE).
14
+ */
15
+ export interface SourceEndpointOptions {
16
+ readonly engine: DmsEndpointProps['engine'];
17
+ readonly endpointIdentifier?: string;
18
+ readonly serverName?: string;
19
+ readonly port?: number;
20
+ readonly username?: string;
21
+ /**
22
+ * Database password. The resolved value is stored as **plaintext** in the
23
+ * CloudFormation template. Prefer `secretsManagerSecretId` in the
24
+ * engine-specific settings for production workloads.
25
+ */
26
+ readonly password?: cdk.SecretValue;
27
+ readonly databaseName?: string;
28
+ readonly extraConnectionAttributes?: string;
29
+ readonly certificateArn?: string;
30
+ readonly sslMode?: string;
31
+ readonly mySqlSettings?: DmsEndpointProps['mySqlSettings'];
32
+ readonly postgreSqlSettings?: DmsEndpointProps['postgreSqlSettings'];
33
+ readonly oracleSettings?: DmsEndpointProps['oracleSettings'];
34
+ readonly sqlServerSettings?: DmsEndpointProps['sqlServerSettings'];
35
+ readonly sapAseSettings?: DmsEndpointProps['sapAseSettings'];
36
+ readonly db2Settings?: DmsEndpointProps['db2Settings'];
37
+ readonly mongoDbSettings?: DmsEndpointProps['mongoDbSettings'];
38
+ readonly s3Settings?: DmsEndpointProps['s3Settings'];
39
+ }
40
+ /**
41
+ * Properties for the target endpoint of a {@link DmsMigrationPipeline}.
42
+ * Extends {@link DmsEndpointProps} but omits `endpointType` (always TARGET).
43
+ */
44
+ export interface TargetEndpointOptions {
45
+ readonly engine: DmsEndpointProps['engine'];
46
+ readonly endpointIdentifier?: string;
47
+ readonly serverName?: string;
48
+ readonly port?: number;
49
+ readonly username?: string;
50
+ /**
51
+ * Database password. The resolved value is stored as **plaintext** in the
52
+ * CloudFormation template. Prefer `secretsManagerSecretId` in the
53
+ * engine-specific settings for production workloads.
54
+ */
55
+ readonly password?: cdk.SecretValue;
56
+ readonly databaseName?: string;
57
+ readonly extraConnectionAttributes?: string;
58
+ readonly certificateArn?: string;
59
+ readonly sslMode?: string;
60
+ readonly mySqlSettings?: DmsEndpointProps['mySqlSettings'];
61
+ readonly postgreSqlSettings?: DmsEndpointProps['postgreSqlSettings'];
62
+ readonly oracleSettings?: DmsEndpointProps['oracleSettings'];
63
+ readonly sqlServerSettings?: DmsEndpointProps['sqlServerSettings'];
64
+ readonly sapAseSettings?: DmsEndpointProps['sapAseSettings'];
65
+ readonly db2Settings?: DmsEndpointProps['db2Settings'];
66
+ readonly mongoDbSettings?: DmsEndpointProps['mongoDbSettings'];
67
+ readonly s3Settings?: DmsEndpointProps['s3Settings'];
68
+ readonly dynamoDbSettings?: DmsEndpointProps['dynamoDbSettings'];
69
+ readonly redshiftSettings?: DmsEndpointProps['redshiftSettings'];
70
+ readonly kinesisSettings?: DmsEndpointProps['kinesisSettings'];
71
+ readonly kafkaSettings?: DmsEndpointProps['kafkaSettings'];
72
+ readonly openSearchSettings?: DmsEndpointProps['openSearchSettings'];
73
+ readonly neptuneSettings?: DmsEndpointProps['neptuneSettings'];
74
+ readonly redisSettings?: DmsEndpointProps['redisSettings'];
75
+ }
76
+ /** Top-level props for {@link DmsMigrationPipeline}. */
77
+ export interface DmsMigrationPipelineProps {
78
+ /**
79
+ * VPC in which to place the replication instance.
80
+ * The instance is placed in private subnets.
81
+ */
82
+ readonly vpc: ec2.IVpc;
83
+ /**
84
+ * Subnet selection for the replication instance.
85
+ * @default private subnets with egress
86
+ */
87
+ readonly vpcSubnets?: ec2.SubnetSelection;
88
+ /**
89
+ * Replication instance class.
90
+ * @default ReplicationInstanceClass.R6I_LARGE
91
+ */
92
+ readonly replicationInstanceClass?: ReplicationInstanceClass;
93
+ /**
94
+ * Allocated storage for the replication instance in GB.
95
+ * @default 100
96
+ */
97
+ readonly allocatedStorage?: number;
98
+ /**
99
+ * Whether the replication instance is Multi-AZ.
100
+ * @default false
101
+ */
102
+ readonly multiAz?: boolean;
103
+ /**
104
+ * Engine version for the replication instance.
105
+ * @default latest version available in the region (chosen by DMS)
106
+ */
107
+ readonly engineVersion?: string;
108
+ /**
109
+ * KMS key for encrypting the replication instance storage at rest.
110
+ * A new key is created if not provided.
111
+ */
112
+ readonly encryptionKey?: kms.IKey;
113
+ /**
114
+ * Source endpoint configuration.
115
+ * Provide this OR `existingSourceEndpoint` — not both.
116
+ */
117
+ readonly sourceEndpoint?: SourceEndpointOptions;
118
+ /**
119
+ * An existing {@link IDmsEndpoint} to use as the source.
120
+ * Provide this OR `sourceEndpoint` — not both.
121
+ */
122
+ readonly existingSourceEndpoint?: IDmsEndpoint;
123
+ /**
124
+ * Target endpoint configuration.
125
+ * Provide this OR `existingTargetEndpoint` — not both.
126
+ */
127
+ readonly targetEndpoint?: TargetEndpointOptions;
128
+ /**
129
+ * An existing {@link IDmsEndpoint} to use as the target.
130
+ * Provide this OR `targetEndpoint` — not both.
131
+ */
132
+ readonly existingTargetEndpoint?: IDmsEndpoint;
133
+ /** The type of migration to perform. */
134
+ readonly migrationType: MigrationType;
135
+ /**
136
+ * Table mappings JSON string.
137
+ * Use {@link TableMappings} to build this.
138
+ * Defaults to "include all tables in all schemas" if not provided.
139
+ */
140
+ readonly tableMappings?: string;
141
+ /**
142
+ * Task settings JSON string.
143
+ * Use {@link TaskSettings} to build this.
144
+ * Sensible defaults are applied if not provided.
145
+ */
146
+ readonly taskSettings?: string;
147
+ /**
148
+ * CDC start time (ISO-8601 string or Unix epoch seconds).
149
+ * Only used when migrationType includes CDC.
150
+ */
151
+ readonly cdcStartTime?: string;
152
+ /**
153
+ * CDC start position (LSN or equivalent).
154
+ * Only used when migrationType includes CDC.
155
+ */
156
+ readonly cdcStartPosition?: string;
157
+ /**
158
+ * CDC stop position.
159
+ * Only used when migrationType includes CDC.
160
+ */
161
+ readonly cdcStopPosition?: string;
162
+ /**
163
+ * Whether to create a CloudWatch log group for the task.
164
+ * @default true
165
+ */
166
+ readonly enableCloudWatchLogs?: boolean;
167
+ /**
168
+ * Retention period for the CloudWatch log group.
169
+ * @default logs.RetentionDays.ONE_MONTH
170
+ */
171
+ readonly logRetention?: logs.RetentionDays;
172
+ /**
173
+ * Whether to create the two account-level DMS service roles (`dms-vpc-role`
174
+ * and `dms-cloudwatch-logs-role`) required by DMS.
175
+ *
176
+ * Set this to `false` if the roles already exist in the AWS account — for
177
+ * example, because another CDK stack (or a manual deployment) already
178
+ * created them. Attempting to create roles with the same name twice in the
179
+ * same account causes a CloudFormation `EntityAlreadyExists` error.
180
+ *
181
+ * When `false`, the construct expects the roles to already be present and
182
+ * skips creating them. The `dmsVpcRole` and `dmsCloudWatchRole` properties
183
+ * will be `undefined`.
184
+ *
185
+ * @default true
186
+ */
187
+ readonly createDmsServiceRoles?: boolean;
188
+ /**
189
+ * Removal policy applied to all resources in the pipeline.
190
+ * @default cdk.RemovalPolicy.DESTROY
191
+ */
192
+ readonly removalPolicy?: cdk.RemovalPolicy;
193
+ }
194
+ /**
195
+ * An L3 CDK pattern construct that provisions a complete DMS migration pipeline:
196
+ *
197
+ * - **Replication instance** — placed in private subnets with KMS encryption
198
+ * and a dedicated security group.
199
+ * - **Source endpoint** — supports every engine DMS supports.
200
+ * - **Target endpoint** — supports every engine DMS supports.
201
+ * - **Replication task** — wires the instance, endpoints, table mappings and
202
+ * task settings together.
203
+ * - **IAM role** — grants DMS permission to write to CloudWatch Logs.
204
+ * - **CloudWatch log group** — (optional) retains task logs.
205
+ *
206
+ * @example
207
+ * // MySQL → Aurora PostgreSQL, full-load-and-CDC
208
+ * const pipeline = new DmsMigrationPipeline(this, 'MigrationPipeline', {
209
+ * vpc,
210
+ * migrationType: MigrationType.FULL_LOAD_AND_CDC,
211
+ * sourceEndpoint: {
212
+ * engine: EndpointEngine.MYSQL,
213
+ * serverName: 'mysql.example.com',
214
+ * port: 3306,
215
+ * username: 'dms_user',
216
+ * password: cdk.SecretValue.secretsManager('mysql-dms-secret'),
217
+ * databaseName: 'mydb',
218
+ * },
219
+ * targetEndpoint: {
220
+ * engine: EndpointEngine.AURORA_POSTGRESQL,
221
+ * serverName: cluster.clusterEndpoint.hostname,
222
+ * port: 5432,
223
+ * username: 'dms_user',
224
+ * password: cdk.SecretValue.secretsManager('aurora-dms-secret'),
225
+ * databaseName: 'mydb',
226
+ * },
227
+ * tableMappings: new TableMappings().includeSchema('public').toJson(),
228
+ * });
229
+ */
230
+ export declare class DmsMigrationPipeline extends Construct {
231
+ /** The replication instance provisioned by this pipeline. */
232
+ readonly replicationInstance: DmsReplicationInstance;
233
+ /** The source endpoint used by this pipeline. */
234
+ readonly source: IDmsEndpoint;
235
+ /** The target endpoint used by this pipeline. */
236
+ readonly target: IDmsEndpoint;
237
+ /** The replication task that drives the migration. */
238
+ readonly replicationTask: DmsReplicationTask;
239
+ /** CloudWatch log group for the task (if enableCloudWatchLogs is true). */
240
+ readonly logGroup?: logs.LogGroup;
241
+ /**
242
+ * IAM role that allows DMS to write to CloudWatch Logs.
243
+ * `undefined` when `createDmsServiceRoles` is `false`.
244
+ */
245
+ readonly dmsCloudWatchRole?: iam.Role;
246
+ /**
247
+ * IAM role that allows DMS to manage VPC resources (dms-vpc-role).
248
+ * `undefined` when `createDmsServiceRoles` is `false`.
249
+ */
250
+ readonly dmsVpcRole?: iam.Role;
251
+ constructor(scope: Construct, id: string, props: DmsMigrationPipelineProps);
252
+ private validateProps;
253
+ }
@@ -0,0 +1,218 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.DmsMigrationPipeline = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const cdk = require("aws-cdk-lib");
7
+ const logs = require("aws-cdk-lib/aws-logs");
8
+ const constructs_1 = require("constructs");
9
+ const dms_roles_1 = require("./dms-roles");
10
+ const endpoint_1 = require("./endpoint");
11
+ const enums_1 = require("./enums");
12
+ const replication_instance_1 = require("./replication-instance");
13
+ const replication_task_1 = require("./replication-task");
14
+ const table_mappings_1 = require("./table-mappings");
15
+ // ---------------------------------------------------------------------------
16
+ // Construct
17
+ // ---------------------------------------------------------------------------
18
+ /**
19
+ * An L3 CDK pattern construct that provisions a complete DMS migration pipeline:
20
+ *
21
+ * - **Replication instance** — placed in private subnets with KMS encryption
22
+ * and a dedicated security group.
23
+ * - **Source endpoint** — supports every engine DMS supports.
24
+ * - **Target endpoint** — supports every engine DMS supports.
25
+ * - **Replication task** — wires the instance, endpoints, table mappings and
26
+ * task settings together.
27
+ * - **IAM role** — grants DMS permission to write to CloudWatch Logs.
28
+ * - **CloudWatch log group** — (optional) retains task logs.
29
+ *
30
+ * @example
31
+ * // MySQL → Aurora PostgreSQL, full-load-and-CDC
32
+ * const pipeline = new DmsMigrationPipeline(this, 'MigrationPipeline', {
33
+ * vpc,
34
+ * migrationType: MigrationType.FULL_LOAD_AND_CDC,
35
+ * sourceEndpoint: {
36
+ * engine: EndpointEngine.MYSQL,
37
+ * serverName: 'mysql.example.com',
38
+ * port: 3306,
39
+ * username: 'dms_user',
40
+ * password: cdk.SecretValue.secretsManager('mysql-dms-secret'),
41
+ * databaseName: 'mydb',
42
+ * },
43
+ * targetEndpoint: {
44
+ * engine: EndpointEngine.AURORA_POSTGRESQL,
45
+ * serverName: cluster.clusterEndpoint.hostname,
46
+ * port: 5432,
47
+ * username: 'dms_user',
48
+ * password: cdk.SecretValue.secretsManager('aurora-dms-secret'),
49
+ * databaseName: 'mydb',
50
+ * },
51
+ * tableMappings: new TableMappings().includeSchema('public').toJson(),
52
+ * });
53
+ */
54
+ class DmsMigrationPipeline extends constructs_1.Construct {
55
+ constructor(scope, id, props) {
56
+ super(scope, id);
57
+ this.validateProps(props);
58
+ const removalPolicy = props.removalPolicy ?? cdk.RemovalPolicy.DESTROY;
59
+ const createRoles = props.createDmsServiceRoles ?? true;
60
+ // -----------------------------------------------------------------------
61
+ // dms-vpc-role — account-level singleton required by DMS to place
62
+ // replication instances inside a VPC. Must exist before the first
63
+ // CfnReplicationInstance is created in the account.
64
+ // Skip when createDmsServiceRoles is false (roles already exist).
65
+ // -----------------------------------------------------------------------
66
+ if (createRoles) {
67
+ this.dmsVpcRole = (0, dms_roles_1.ensureDmsVpcRole)(cdk.Stack.of(this));
68
+ }
69
+ // -----------------------------------------------------------------------
70
+ // CloudWatch log group & IAM role
71
+ // -----------------------------------------------------------------------
72
+ const enableLogs = props.enableCloudWatchLogs ?? true;
73
+ if (enableLogs) {
74
+ this.logGroup = new logs.LogGroup(this, 'TaskLogGroup', {
75
+ logGroupName: `/aws/dms/tasks/${id}`,
76
+ retention: props.logRetention ?? logs.RetentionDays.ONE_MONTH,
77
+ removalPolicy,
78
+ });
79
+ if (createRoles) {
80
+ // dms-cloudwatch-logs-role is an account-level singleton — DMS requires
81
+ // this exact name. Guard with a stack-level singleton so deploying multiple
82
+ // pipelines in the same stack doesn't create duplicate roles.
83
+ this.dmsCloudWatchRole = (0, dms_roles_1.ensureDmsCloudWatchRole)(cdk.Stack.of(this));
84
+ }
85
+ }
86
+ // -----------------------------------------------------------------------
87
+ // Replication instance
88
+ // -----------------------------------------------------------------------
89
+ this.replicationInstance = new replication_instance_1.DmsReplicationInstance(this, 'ReplicationInstance', {
90
+ vpc: props.vpc,
91
+ vpcSubnets: props.vpcSubnets,
92
+ replicationInstanceClass: props.replicationInstanceClass,
93
+ allocatedStorage: props.allocatedStorage,
94
+ multiAz: props.multiAz,
95
+ engineVersion: props.engineVersion,
96
+ kmsKey: props.encryptionKey,
97
+ removalPolicy,
98
+ });
99
+ // The replication instance cannot be placed in a VPC until dms-vpc-role exists.
100
+ // Only add the dependency when we created the role ourselves.
101
+ if (this.dmsVpcRole) {
102
+ this.replicationInstance.cfnReplicationInstance.addDependency(this.dmsVpcRole.node.defaultChild);
103
+ }
104
+ // -----------------------------------------------------------------------
105
+ // Source endpoint
106
+ // -----------------------------------------------------------------------
107
+ if (props.existingSourceEndpoint) {
108
+ this.source = props.existingSourceEndpoint;
109
+ }
110
+ else {
111
+ const sourceOpts = props.sourceEndpoint;
112
+ this.source = new endpoint_1.DmsEndpoint(this, 'SourceEndpoint', {
113
+ endpointType: enums_1.EndpointType.SOURCE,
114
+ engine: sourceOpts.engine,
115
+ endpointIdentifier: sourceOpts.endpointIdentifier,
116
+ serverName: sourceOpts.serverName,
117
+ port: sourceOpts.port,
118
+ username: sourceOpts.username,
119
+ password: sourceOpts.password,
120
+ databaseName: sourceOpts.databaseName,
121
+ extraConnectionAttributes: sourceOpts.extraConnectionAttributes,
122
+ certificateArn: sourceOpts.certificateArn,
123
+ sslMode: sourceOpts.sslMode,
124
+ mySqlSettings: sourceOpts.mySqlSettings,
125
+ postgreSqlSettings: sourceOpts.postgreSqlSettings,
126
+ oracleSettings: sourceOpts.oracleSettings,
127
+ sqlServerSettings: sourceOpts.sqlServerSettings,
128
+ sapAseSettings: sourceOpts.sapAseSettings,
129
+ db2Settings: sourceOpts.db2Settings,
130
+ mongoDbSettings: sourceOpts.mongoDbSettings,
131
+ s3Settings: sourceOpts.s3Settings,
132
+ removalPolicy,
133
+ });
134
+ }
135
+ // -----------------------------------------------------------------------
136
+ // Target endpoint
137
+ // -----------------------------------------------------------------------
138
+ if (props.existingTargetEndpoint) {
139
+ this.target = props.existingTargetEndpoint;
140
+ }
141
+ else {
142
+ const targetOpts = props.targetEndpoint;
143
+ this.target = new endpoint_1.DmsEndpoint(this, 'TargetEndpoint', {
144
+ endpointType: enums_1.EndpointType.TARGET,
145
+ engine: targetOpts.engine,
146
+ endpointIdentifier: targetOpts.endpointIdentifier,
147
+ serverName: targetOpts.serverName,
148
+ port: targetOpts.port,
149
+ username: targetOpts.username,
150
+ password: targetOpts.password,
151
+ databaseName: targetOpts.databaseName,
152
+ extraConnectionAttributes: targetOpts.extraConnectionAttributes,
153
+ certificateArn: targetOpts.certificateArn,
154
+ sslMode: targetOpts.sslMode,
155
+ mySqlSettings: targetOpts.mySqlSettings,
156
+ postgreSqlSettings: targetOpts.postgreSqlSettings,
157
+ oracleSettings: targetOpts.oracleSettings,
158
+ sqlServerSettings: targetOpts.sqlServerSettings,
159
+ sapAseSettings: targetOpts.sapAseSettings,
160
+ db2Settings: targetOpts.db2Settings,
161
+ mongoDbSettings: targetOpts.mongoDbSettings,
162
+ s3Settings: targetOpts.s3Settings,
163
+ dynamoDbSettings: targetOpts.dynamoDbSettings,
164
+ redshiftSettings: targetOpts.redshiftSettings,
165
+ kinesisSettings: targetOpts.kinesisSettings,
166
+ kafkaSettings: targetOpts.kafkaSettings,
167
+ openSearchSettings: targetOpts.openSearchSettings,
168
+ neptuneSettings: targetOpts.neptuneSettings,
169
+ redisSettings: targetOpts.redisSettings,
170
+ removalPolicy,
171
+ });
172
+ }
173
+ // -----------------------------------------------------------------------
174
+ // Table mappings — default to "include everything"
175
+ // -----------------------------------------------------------------------
176
+ const tableMappings = props.tableMappings ?? new table_mappings_1.TableMappings().includeSchema('%').toJson();
177
+ // -----------------------------------------------------------------------
178
+ // Replication task
179
+ // -----------------------------------------------------------------------
180
+ this.replicationTask = new replication_task_1.DmsReplicationTask(this, 'ReplicationTask', {
181
+ replicationInstanceArn: this.replicationInstance.replicationInstanceArn,
182
+ sourceEndpoint: this.source,
183
+ targetEndpoint: this.target,
184
+ migrationType: props.migrationType,
185
+ tableMappings,
186
+ taskSettings: props.taskSettings,
187
+ cdcStartTime: props.cdcStartTime,
188
+ cdcStartPosition: props.cdcStartPosition,
189
+ cdcStopPosition: props.cdcStopPosition,
190
+ removalPolicy,
191
+ });
192
+ }
193
+ // ---------------------------------------------------------------------------
194
+ // Private helpers
195
+ // ---------------------------------------------------------------------------
196
+ validateProps(props) {
197
+ const hasSource = !!props.sourceEndpoint;
198
+ const hasExistingSource = !!props.existingSourceEndpoint;
199
+ if (hasSource && hasExistingSource) {
200
+ throw new Error('Provide either `sourceEndpoint` or `existingSourceEndpoint`, not both.');
201
+ }
202
+ if (!hasSource && !hasExistingSource) {
203
+ throw new Error('One of `sourceEndpoint` or `existingSourceEndpoint` is required.');
204
+ }
205
+ const hasTarget = !!props.targetEndpoint;
206
+ const hasExistingTarget = !!props.existingTargetEndpoint;
207
+ if (hasTarget && hasExistingTarget) {
208
+ throw new Error('Provide either `targetEndpoint` or `existingTargetEndpoint`, not both.');
209
+ }
210
+ if (!hasTarget && !hasExistingTarget) {
211
+ throw new Error('One of `targetEndpoint` or `existingTargetEndpoint` is required.');
212
+ }
213
+ }
214
+ }
215
+ exports.DmsMigrationPipeline = DmsMigrationPipeline;
216
+ _a = JSII_RTTI_SYMBOL_1;
217
+ DmsMigrationPipeline[_a] = { fqn: "cdk-dms-replication.DmsMigrationPipeline", version: "0.0.0" };
218
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"migration-pipeline.js","sourceRoot":"","sources":["../src/migration-pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\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, DmsEndpointProps, IDmsEndpoint } from './endpoint';\nimport { EndpointType, MigrationType, ReplicationInstanceClass } from './enums';\nimport { DmsReplicationInstance } from './replication-instance';\nimport { DmsReplicationTask } from './replication-task';\nimport { TableMappings } from './table-mappings';\nimport { TaskSettings } from './task-settings';\n\n// ---------------------------------------------------------------------------\n// Props\n// ---------------------------------------------------------------------------\n\n/**\n * Properties for the source endpoint of a {@link DmsMigrationPipeline}.\n * Extends {@link DmsEndpointProps} but omits `endpointType` (always SOURCE).\n */\nexport interface SourceEndpointOptions {\n  readonly engine: DmsEndpointProps['engine'];\n  readonly endpointIdentifier?: string;\n  readonly serverName?: string;\n  readonly port?: number;\n  readonly username?: string;\n  /**\n   * Database password. The resolved value is stored as **plaintext** in the\n   * CloudFormation template. Prefer `secretsManagerSecretId` in the\n   * engine-specific settings for production workloads.\n   */\n  readonly password?: cdk.SecretValue;\n  readonly databaseName?: string;\n  readonly extraConnectionAttributes?: string;\n  readonly certificateArn?: string;\n  readonly sslMode?: string;\n  readonly mySqlSettings?: DmsEndpointProps['mySqlSettings'];\n  readonly postgreSqlSettings?: DmsEndpointProps['postgreSqlSettings'];\n  readonly oracleSettings?: DmsEndpointProps['oracleSettings'];\n  readonly sqlServerSettings?: DmsEndpointProps['sqlServerSettings'];\n  readonly sapAseSettings?: DmsEndpointProps['sapAseSettings'];\n  readonly db2Settings?: DmsEndpointProps['db2Settings'];\n  readonly mongoDbSettings?: DmsEndpointProps['mongoDbSettings'];\n  readonly s3Settings?: DmsEndpointProps['s3Settings'];\n}\n\n/**\n * Properties for the target endpoint of a {@link DmsMigrationPipeline}.\n * Extends {@link DmsEndpointProps} but omits `endpointType` (always TARGET).\n */\nexport interface TargetEndpointOptions {\n  readonly engine: DmsEndpointProps['engine'];\n  readonly endpointIdentifier?: string;\n  readonly serverName?: string;\n  readonly port?: number;\n  readonly username?: string;\n  /**\n   * Database password. The resolved value is stored as **plaintext** in the\n   * CloudFormation template. Prefer `secretsManagerSecretId` in the\n   * engine-specific settings for production workloads.\n   */\n  readonly password?: cdk.SecretValue;\n  readonly databaseName?: string;\n  readonly extraConnectionAttributes?: string;\n  readonly certificateArn?: string;\n  readonly sslMode?: string;\n  readonly mySqlSettings?: DmsEndpointProps['mySqlSettings'];\n  readonly postgreSqlSettings?: DmsEndpointProps['postgreSqlSettings'];\n  readonly oracleSettings?: DmsEndpointProps['oracleSettings'];\n  readonly sqlServerSettings?: DmsEndpointProps['sqlServerSettings'];\n  readonly sapAseSettings?: DmsEndpointProps['sapAseSettings'];\n  readonly db2Settings?: DmsEndpointProps['db2Settings'];\n  readonly mongoDbSettings?: DmsEndpointProps['mongoDbSettings'];\n  readonly s3Settings?: DmsEndpointProps['s3Settings'];\n  readonly dynamoDbSettings?: DmsEndpointProps['dynamoDbSettings'];\n  readonly redshiftSettings?: DmsEndpointProps['redshiftSettings'];\n  readonly kinesisSettings?: DmsEndpointProps['kinesisSettings'];\n  readonly kafkaSettings?: DmsEndpointProps['kafkaSettings'];\n  readonly openSearchSettings?: DmsEndpointProps['openSearchSettings'];\n  readonly neptuneSettings?: DmsEndpointProps['neptuneSettings'];\n  readonly redisSettings?: DmsEndpointProps['redisSettings'];\n}\n\n/** Top-level props for {@link DmsMigrationPipeline}. */\nexport interface DmsMigrationPipelineProps {\n  // -------------------------------------------------------------------------\n  // Replication instance\n  // -------------------------------------------------------------------------\n\n  /**\n   * VPC in which to place the replication instance.\n   * The instance is placed in private subnets.\n   */\n  readonly vpc: ec2.IVpc;\n\n  /**\n   * Subnet selection for the replication instance.\n   * @default private subnets with egress\n   */\n  readonly vpcSubnets?: ec2.SubnetSelection;\n\n  /**\n   * Replication instance class.\n   * @default ReplicationInstanceClass.R6I_LARGE\n   */\n  readonly replicationInstanceClass?: ReplicationInstanceClass;\n\n  /**\n   * Allocated storage for the replication instance in GB.\n   * @default 100\n   */\n  readonly allocatedStorage?: number;\n\n  /**\n   * Whether the replication instance is Multi-AZ.\n   * @default false\n   */\n  readonly multiAz?: boolean;\n\n  /**\n   * Engine version for the replication instance.\n   * @default latest version available in the region (chosen by DMS)\n   */\n  readonly engineVersion?: string;\n\n  /**\n   * KMS key for encrypting the replication instance storage at rest.\n   * A new key is created if not provided.\n   */\n  readonly encryptionKey?: 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  // Replication task\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   * Task settings JSON string.\n   * Use {@link TaskSettings} to build this.\n   * Sensible defaults are applied if not provided.\n   */\n  readonly taskSettings?: string;\n\n  /**\n   * CDC start time (ISO-8601 string or Unix epoch seconds).\n   * Only used when migrationType includes CDC.\n   */\n  readonly cdcStartTime?: string;\n\n  /**\n   * CDC start position (LSN or equivalent).\n   * Only used when migrationType includes CDC.\n   */\n  readonly cdcStartPosition?: string;\n\n  /**\n   * CDC stop position.\n   * Only used when migrationType includes CDC.\n   */\n  readonly cdcStopPosition?: string;\n\n  // -------------------------------------------------------------------------\n  // Observability\n  // -------------------------------------------------------------------------\n\n  /**\n   * Whether to create a CloudWatch log group for the task.\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 this to `false` if the roles already exist in the AWS account — for\n   * example, because another CDK stack (or a manual deployment) already\n   * created them. Attempting to create roles with the same name twice in the\n   * same account causes a CloudFormation `EntityAlreadyExists` error.\n   *\n   * When `false`, the construct expects the roles to already be present and\n   * skips creating them. The `dmsVpcRole` and `dmsCloudWatchRole` properties\n   * will be `undefined`.\n   *\n   * @default true\n   */\n  readonly createDmsServiceRoles?: boolean;\n\n  // -------------------------------------------------------------------------\n  // Lifecycle\n  // -------------------------------------------------------------------------\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 complete DMS migration pipeline:\n *\n * - **Replication instance** — placed in private subnets with KMS encryption\n *   and a dedicated security group.\n * - **Source endpoint** — supports every engine DMS supports.\n * - **Target endpoint** — supports every engine DMS supports.\n * - **Replication task** — wires the instance, endpoints, table mappings and\n *   task settings together.\n * - **IAM role** — grants DMS permission to write to CloudWatch Logs.\n * - **CloudWatch log group** — (optional) retains task logs.\n *\n * @example\n * // MySQL → Aurora PostgreSQL, full-load-and-CDC\n * const pipeline = new DmsMigrationPipeline(this, 'MigrationPipeline', {\n *   vpc,\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 *     databaseName: 'mydb',\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 *     databaseName: 'mydb',\n *   },\n *   tableMappings: new TableMappings().includeSchema('public').toJson(),\n * });\n */\nexport class DmsMigrationPipeline extends Construct {\n  // ---------------------------------------------------------------------------\n  // Instance fields\n  // ---------------------------------------------------------------------------\n\n  /** The replication instance provisioned by this pipeline. */\n  readonly replicationInstance: DmsReplicationInstance;\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 task that drives the migration. */\n  readonly replicationTask: DmsReplicationTask;\n\n  /** CloudWatch log group for the task (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: DmsMigrationPipelineProps) {\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 — account-level singleton required by DMS to place\n    // replication instances inside a VPC. Must exist before the first\n    // CfnReplicationInstance is created in the account.\n    // Skip when createDmsServiceRoles is false (roles already exist).\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, 'TaskLogGroup', {\n        logGroupName: `/aws/dms/tasks/${id}`,\n        retention: props.logRetention ?? logs.RetentionDays.ONE_MONTH,\n        removalPolicy,\n      });\n\n      if (createRoles) {\n        // dms-cloudwatch-logs-role is an account-level singleton — DMS requires\n        // this exact name. Guard with a stack-level singleton so deploying multiple\n        // pipelines in the same stack doesn't create duplicate roles.\n        this.dmsCloudWatchRole = ensureDmsCloudWatchRole(cdk.Stack.of(this));\n      }\n    }\n\n    // -----------------------------------------------------------------------\n    // Replication instance\n    // -----------------------------------------------------------------------\n    this.replicationInstance = new DmsReplicationInstance(this, 'ReplicationInstance', {\n      vpc: props.vpc,\n      vpcSubnets: props.vpcSubnets,\n      replicationInstanceClass: props.replicationInstanceClass,\n      allocatedStorage: props.allocatedStorage,\n      multiAz: props.multiAz,\n      engineVersion: props.engineVersion,\n      kmsKey: props.encryptionKey,\n      removalPolicy,\n    });\n    // The replication instance cannot be placed in a VPC until dms-vpc-role exists.\n    // Only add the dependency when we created the role ourselves.\n    if (this.dmsVpcRole) {\n      this.replicationInstance.cfnReplicationInstance.addDependency(\n        this.dmsVpcRole.node.defaultChild as cdk.CfnResource,\n      );\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 tableMappings =\n      props.tableMappings ?? new TableMappings().includeSchema('%').toJson();\n\n    // -----------------------------------------------------------------------\n    // Replication task\n    // -----------------------------------------------------------------------\n    this.replicationTask = new DmsReplicationTask(this, 'ReplicationTask', {\n      replicationInstanceArn: this.replicationInstance.replicationInstanceArn,\n      sourceEndpoint: this.source,\n      targetEndpoint: this.target,\n      migrationType: props.migrationType,\n      tableMappings,\n      taskSettings: props.taskSettings,\n      cdcStartTime: props.cdcStartTime,\n      cdcStartPosition: props.cdcStartPosition,\n      cdcStopPosition: props.cdcStopPosition,\n      removalPolicy,\n    });\n\n  }\n\n  // ---------------------------------------------------------------------------\n  // Private helpers\n  // ---------------------------------------------------------------------------\n\n  private validateProps(props: DmsMigrationPipelineProps): void {\n    const hasSource = !!props.sourceEndpoint;\n    const hasExistingSource = !!props.existingSourceEndpoint;\n    if (hasSource && hasExistingSource) {\n      throw new Error(\n        'Provide either `sourceEndpoint` or `existingSourceEndpoint`, not both.',\n      );\n    }\n    if (!hasSource && !hasExistingSource) {\n      throw new Error(\n        'One of `sourceEndpoint` or `existingSourceEndpoint` is required.',\n      );\n    }\n\n    const hasTarget = !!props.targetEndpoint;\n    const hasExistingTarget = !!props.existingTargetEndpoint;\n    if (hasTarget && hasExistingTarget) {\n      throw new Error(\n        'Provide either `targetEndpoint` or `existingTargetEndpoint`, not both.',\n      );\n    }\n    if (!hasTarget && !hasExistingTarget) {\n      throw new Error(\n        'One of `targetEndpoint` or `existingTargetEndpoint` is required.',\n      );\n    }\n  }\n}\n"]}
@@ -0,0 +1,99 @@
1
+ import * as cdk from 'aws-cdk-lib';
2
+ import * as dms from 'aws-cdk-lib/aws-dms';
3
+ import * as ec2 from 'aws-cdk-lib/aws-ec2';
4
+ import * as kms from 'aws-cdk-lib/aws-kms';
5
+ import { Construct } from 'constructs';
6
+ import { ReplicationInstanceClass } from './enums';
7
+ /** Props for {@link DmsReplicationInstance}. */
8
+ export interface DmsReplicationInstanceProps {
9
+ /**
10
+ * Instance class for the replication instance.
11
+ * @default ReplicationInstanceClass.R6I_LARGE
12
+ */
13
+ readonly replicationInstanceClass?: ReplicationInstanceClass;
14
+ /**
15
+ * VPC in which to place the replication instance.
16
+ * The construct creates a dedicated subnet group from the private subnets.
17
+ */
18
+ readonly vpc: ec2.IVpc;
19
+ /**
20
+ * Specific subnets to use for the replication subnet group.
21
+ * Defaults to all private subnets in the VPC.
22
+ */
23
+ readonly vpcSubnets?: ec2.SubnetSelection;
24
+ /**
25
+ * Security groups to attach to the replication instance.
26
+ * A new security group is created if none are provided.
27
+ */
28
+ readonly securityGroups?: ec2.ISecurityGroup[];
29
+ /**
30
+ * KMS key used to encrypt the replication instance storage at rest.
31
+ * A new key is created if not provided.
32
+ */
33
+ readonly kmsKey?: kms.IKey;
34
+ /**
35
+ * Allocated storage in GB.
36
+ * @default 100
37
+ */
38
+ readonly allocatedStorage?: number;
39
+ /**
40
+ * Whether the replication instance is Multi-AZ.
41
+ * @default false
42
+ */
43
+ readonly multiAz?: boolean;
44
+ /**
45
+ * Whether the replication instance is publicly accessible.
46
+ * Setting this to true is strongly discouraged for production workloads.
47
+ * @default false
48
+ */
49
+ readonly publiclyAccessible?: boolean;
50
+ /**
51
+ * Whether to allow minor version upgrades to be applied automatically during maintenance.
52
+ * @default true
53
+ */
54
+ readonly autoMinorVersionUpgrade?: boolean;
55
+ /**
56
+ * Preferred maintenance window, e.g. "sun:04:00-sun:04:30".
57
+ */
58
+ readonly preferredMaintenanceWindow?: string;
59
+ /**
60
+ * Replication engine version.
61
+ * @default latest version available in the region (chosen by DMS)
62
+ */
63
+ readonly engineVersion?: string;
64
+ /**
65
+ * Logical name of the replication instance.
66
+ * Used in the replication instance identifier and resource naming.
67
+ * @default id of the construct
68
+ */
69
+ readonly replicationInstanceIdentifier?: string;
70
+ /**
71
+ * Removal policy applied to the replication instance.
72
+ * @default cdk.RemovalPolicy.DESTROY
73
+ */
74
+ readonly removalPolicy?: cdk.RemovalPolicy;
75
+ }
76
+ /**
77
+ * An L2-style construct that provisions a DMS replication instance with:
78
+ * - Private subnet placement via a dedicated subnet group
79
+ * - KMS encryption at rest (key created if not provided)
80
+ * - A dedicated security group (created if not provided)
81
+ */
82
+ export declare class DmsReplicationInstance extends Construct {
83
+ /** The underlying CloudFormation replication instance resource. */
84
+ readonly cfnReplicationInstance: dms.CfnReplicationInstance;
85
+ /** The replication subnet group created for this instance. */
86
+ readonly subnetGroup: dms.CfnReplicationSubnetGroup;
87
+ /** The KMS key used for storage encryption. */
88
+ readonly encryptionKey: kms.IKey;
89
+ /** The security group attached to this instance. */
90
+ readonly securityGroup: ec2.ISecurityGroup;
91
+ /** The replication instance ARN. */
92
+ readonly replicationInstanceArn: string;
93
+ constructor(scope: Construct, id: string, props: DmsReplicationInstanceProps);
94
+ /**
95
+ * Allow inbound access to the replication instance on a given port from a peer.
96
+ * Useful when the source or target database security group needs to trust the instance.
97
+ */
98
+ allowInboundFrom(peer: ec2.IPeer, port: ec2.Port, description?: string): void;
99
+ }