@aws-mdaa/dataops-job-l3-construct 1.4.0 → 1.6.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 +117 -121
- package/README.md +5 -0
- package/lib/dataops-job-l3-construct.d.ts +37 -240
- package/lib/dataops-job-l3-construct.js +61 -34
- package/node_modules/@aws-mdaa/config/.npmignore +34 -0
- package/node_modules/@aws-mdaa/config/README.md +3 -0
- package/node_modules/@aws-mdaa/config/jest.config.js +5 -0
- package/node_modules/@aws-mdaa/config/lib/blueprint-value-transformer.d.ts +20 -0
- package/node_modules/@aws-mdaa/config/lib/blueprint-value-transformer.js +70 -0
- package/node_modules/@aws-mdaa/config/lib/blueprint-value-transformer.ts +88 -0
- package/node_modules/@aws-mdaa/config/lib/config.d.ts +87 -0
- package/node_modules/@aws-mdaa/config/lib/config.js +7 -0
- package/node_modules/@aws-mdaa/config/lib/config.ts +92 -0
- package/node_modules/@aws-mdaa/config/lib/index.d.ts +11 -0
- package/node_modules/@aws-mdaa/config/lib/index.js +28 -0
- package/node_modules/@aws-mdaa/config/lib/index.ts +12 -0
- package/node_modules/@aws-mdaa/config/lib/param-transformer.d.ts +49 -0
- package/node_modules/@aws-mdaa/config/lib/param-transformer.js +160 -0
- package/node_modules/@aws-mdaa/config/lib/param-transformer.ts +159 -0
- package/node_modules/@aws-mdaa/config/lib/path-value-transformer.d.ts +10 -0
- package/node_modules/@aws-mdaa/config/lib/path-value-transformer.js +30 -0
- package/node_modules/@aws-mdaa/config/lib/path-value-transformer.ts +27 -0
- package/node_modules/@aws-mdaa/config/lib/ref-value-transformer.d.ts +44 -0
- package/node_modules/@aws-mdaa/config/lib/ref-value-transformer.js +243 -0
- package/node_modules/@aws-mdaa/config/lib/ref-value-transformer.ts +302 -0
- package/node_modules/@aws-mdaa/config/lib/ssm-ref-transformer.d.ts +8 -0
- package/node_modules/@aws-mdaa/config/lib/ssm-ref-transformer.js +22 -0
- package/node_modules/@aws-mdaa/config/lib/ssm-ref-transformer.ts +21 -0
- package/node_modules/@aws-mdaa/config/lib/transformer.d.ts +35 -0
- package/node_modules/@aws-mdaa/config/lib/transformer.js +66 -0
- package/node_modules/@aws-mdaa/config/lib/transformer.ts +74 -0
- package/node_modules/@aws-mdaa/{s3-bucketpolicy-helper → config}/package.json +17 -17
- package/node_modules/@aws-mdaa/config/test/blueprint-value-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/blueprint-value-transformer.test.js +224 -0
- package/node_modules/@aws-mdaa/config/test/blueprint-value-transformer.test.ts +259 -0
- package/node_modules/@aws-mdaa/config/test/config-nt.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/config-nt.test.js +129 -0
- package/node_modules/@aws-mdaa/config/test/config-nt.test.ts +163 -0
- package/node_modules/@aws-mdaa/config/test/config.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/config.test.js +409 -0
- package/node_modules/@aws-mdaa/config/test/config.test.ts +517 -0
- package/node_modules/@aws-mdaa/config/test/param-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/param-transformer.test.js +216 -0
- package/node_modules/@aws-mdaa/config/test/param-transformer.test.ts +234 -0
- package/node_modules/@aws-mdaa/config/test/path-value-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/path-value-transformer.test.js +59 -0
- package/node_modules/@aws-mdaa/config/test/path-value-transformer.test.ts +68 -0
- package/node_modules/@aws-mdaa/config/test/ref-value-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/ref-value-transformer.test.js +254 -0
- package/node_modules/@aws-mdaa/config/test/ref-value-transformer.test.ts +304 -0
- package/node_modules/@aws-mdaa/config/test/ssm-ref-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/ssm-ref-transformer.test.js +66 -0
- package/node_modules/@aws-mdaa/config/test/ssm-ref-transformer.test.ts +79 -0
- package/node_modules/@aws-mdaa/config/tsconfig.json +40 -0
- package/node_modules/@aws-mdaa/config/tsconfig.tsbuildinfo +1 -0
- package/node_modules/@aws-mdaa/config/typedoc.json +7 -0
- package/node_modules/lodash/README.md +2 -2
- package/node_modules/lodash/_baseOrderBy.js +1 -1
- package/node_modules/lodash/_baseUnset.js +7 -20
- package/node_modules/lodash/_setCacheHas.js +1 -1
- package/node_modules/lodash/compact.js +1 -1
- package/node_modules/lodash/core.js +3 -3
- package/node_modules/lodash/core.min.js +26 -25
- package/node_modules/lodash/fromPairs.js +3 -1
- package/node_modules/lodash/lodash.js +38 -27
- package/node_modules/lodash/lodash.min.js +125 -129
- package/node_modules/lodash/package.json +4 -2
- package/node_modules/lodash/random.js +9 -0
- package/node_modules/lodash/template.js +16 -4
- package/node_modules/lodash/templateSettings.js +4 -0
- package/package.json +27 -30
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/README.md +0 -185
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/lib/index.d.ts +0 -57
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/lib/index.js +0 -198
- package/node_modules/@aws-mdaa/s3-inventory-helper/README.md +0 -3
- package/node_modules/@aws-mdaa/s3-inventory-helper/lib/index.d.ts +0 -66
- package/node_modules/@aws-mdaa/s3-inventory-helper/lib/index.js +0 -222
- package/node_modules/@aws-mdaa/s3-inventory-helper/package.json +0 -42
|
@@ -31,43 +31,51 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
31
31
|
throw new Error('Deployment role ARN is required for job configuration');
|
|
32
32
|
}
|
|
33
33
|
const deploymentRole = iam_constructs_1.MdaaRole.fromRoleArn(this.scope, `deployment-role`, this.props.deploymentRoleArn);
|
|
34
|
-
if (!this.props.
|
|
34
|
+
if (!this.props.bucketName) {
|
|
35
35
|
throw new Error('Project bucket name is required for job configuration');
|
|
36
36
|
}
|
|
37
|
-
const
|
|
38
|
-
if (!this.props.
|
|
37
|
+
const bucket = s3_constructs_1.MdaaBucket.fromBucketName(this.scope, `project-bucket`, this.props.bucketName);
|
|
38
|
+
if (!this.props.kmsArn) {
|
|
39
39
|
throw new Error('Project KMS Key is required for job configuration');
|
|
40
40
|
}
|
|
41
|
-
this.
|
|
41
|
+
this.kmsKey = aws_kms_1.Key.fromKeyArn(this, this.props.projectName ?? 'kms-key', this.props.kmsArn);
|
|
42
42
|
// Build our jobs!
|
|
43
43
|
const allJobs = this.props.jobConfigs;
|
|
44
|
-
Object.keys(allJobs)
|
|
44
|
+
for (const jobName of Object.keys(allJobs)) {
|
|
45
45
|
const jobConfig = allJobs[jobName];
|
|
46
|
-
this.createJob(jobName, jobConfig, deploymentRole,
|
|
47
|
-
}
|
|
48
|
-
//
|
|
46
|
+
this.createJob(jobName, jobConfig, deploymentRole, bucket);
|
|
47
|
+
}
|
|
48
|
+
// BucketDeployment adds an inline policy to the imported deployment role,
|
|
49
|
+
// which CDK synthesizes as a standalone AWS::IAM::Policy resource.
|
|
50
|
+
// Because the policy name is derived from the construct path (not the stack name),
|
|
51
|
+
// multiple stacks importing the same role produce colliding physical names.
|
|
52
|
+
// Remove the inline policy node — the deployment role already has the necessary
|
|
53
|
+
// S3 permissions granted in the dataops-project construct that owns it.
|
|
54
|
+
for (const child of deploymentRole.node.children) {
|
|
55
|
+
if (child.node.id === 'Policy') {
|
|
56
|
+
deploymentRole.node.tryRemoveChild(child.node.id);
|
|
57
|
+
break;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
// Suppress nag warnings on the imported deployment role node itself
|
|
61
|
+
// (remaining after inline policy removal, e.g. wildcard resource warnings)
|
|
49
62
|
this.scope.node.children.forEach(child => {
|
|
50
63
|
if (child.node.id.startsWith('deployment-role')) {
|
|
51
|
-
construct_1.MdaaNagSuppressions.addCodeResourceSuppressions(child, [
|
|
52
|
-
{ id: 'AwsSolutions-IAM5', reason: 'Inline policy used only for deployment.' },
|
|
53
|
-
{ id: 'NIST.800.53.R5-IAMNoInlinePolicy', reason: 'Policy used only for deployment.' },
|
|
54
|
-
{ id: 'HIPAA.Security-IAMNoInlinePolicy', reason: 'Policy used only for deployment.' },
|
|
55
|
-
{ id: 'PCI.DSS.321-IAMNoInlinePolicy', reason: 'Policy used only for deployment.' },
|
|
56
|
-
], true);
|
|
64
|
+
construct_1.MdaaNagSuppressions.addCodeResourceSuppressions(child, [{ id: 'AwsSolutions-IAM5', reason: 'Permissions granted on deployment role in owning stack.' }], true);
|
|
57
65
|
}
|
|
58
66
|
});
|
|
59
67
|
}
|
|
60
|
-
deployAdditionalFiles(additionalFilesSources,
|
|
68
|
+
deployAdditionalFiles(additionalFilesSources, bucket, deploymentRole, deploymentId, deploymentPath, extract) {
|
|
61
69
|
// Deploy Source asset(s) to /deployment/libs/<job> location.
|
|
62
70
|
return new aws_s3_deployment_1.BucketDeployment(this.scope, deploymentId, {
|
|
63
71
|
sources: additionalFilesSources,
|
|
64
|
-
destinationBucket:
|
|
72
|
+
destinationBucket: bucket,
|
|
65
73
|
destinationKeyPrefix: deploymentPath,
|
|
66
74
|
role: deploymentRole,
|
|
67
75
|
extract: extract,
|
|
68
76
|
});
|
|
69
77
|
}
|
|
70
|
-
addAdditionalScripts(jobName, jobConfig,
|
|
78
|
+
addAdditionalScripts(jobName, jobConfig, bucket, deploymentRole, defaultArguments) {
|
|
71
79
|
if (jobConfig.additionalScripts) {
|
|
72
80
|
/**
|
|
73
81
|
* Group all files at parent directory level. This will allow creating zip lib assests at various directory levels
|
|
@@ -88,7 +96,7 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
88
96
|
return aws_s3_deployment_1.Source.asset(filePath, { exclude: ['**', ...fileNames] });
|
|
89
97
|
});
|
|
90
98
|
const deploymentPath = `deployment/libs/${jobName}`;
|
|
91
|
-
const additionalFileDeployment = this.deployAdditionalFiles(additionalFilesSources,
|
|
99
|
+
const additionalFileDeployment = this.deployAdditionalFiles(additionalFilesSources, bucket, deploymentRole, `job-deployment-${jobName}-additional-script`, deploymentPath, false);
|
|
92
100
|
// Extract zip name(s) for each source and create comma separated list of s3 locations
|
|
93
101
|
const libraryZipNames = [];
|
|
94
102
|
for (let i = 0; i < additionalFilesSources.length; i++) {
|
|
@@ -104,7 +112,7 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
104
112
|
}
|
|
105
113
|
}
|
|
106
114
|
}
|
|
107
|
-
addAdditionalJars(jobName, jobConfig,
|
|
115
|
+
addAdditionalJars(jobName, jobConfig, bucket, deploymentRole, defaultArguments) {
|
|
108
116
|
if (jobConfig.additionalJars) {
|
|
109
117
|
// Create Source asset for each directory
|
|
110
118
|
const additionalFilesSources = jobConfig.additionalJars.map(fullFileName => {
|
|
@@ -113,7 +121,7 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
113
121
|
return aws_s3_deployment_1.Source.asset(filePath, { exclude: ['**', `!${fileName}`] });
|
|
114
122
|
});
|
|
115
123
|
const deploymentPath = `deployment/libs/${jobName}`;
|
|
116
|
-
const additionalFileDeployment = this.deployAdditionalFiles(additionalFilesSources,
|
|
124
|
+
const additionalFileDeployment = this.deployAdditionalFiles(additionalFilesSources, bucket, deploymentRole, `job-deployment-${jobName}-additional-jar`, deploymentPath, true);
|
|
117
125
|
const extraJarNames = jobConfig.additionalJars.map(fullFileName => {
|
|
118
126
|
const fileName = path.basename(fullFileName.trim());
|
|
119
127
|
return `s3://${additionalFileDeployment.deployedBucket.bucketName}/${deploymentPath}/${fileName}`;
|
|
@@ -127,7 +135,7 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
127
135
|
}
|
|
128
136
|
}
|
|
129
137
|
}
|
|
130
|
-
addAdditionalFiles(jobName, jobConfig,
|
|
138
|
+
addAdditionalFiles(jobName, jobConfig, bucket, deploymentRole, defaultArguments) {
|
|
131
139
|
if (jobConfig.additionalFiles) {
|
|
132
140
|
// Create Source asset for each directory
|
|
133
141
|
const additionalFilesSources = jobConfig.additionalFiles.map(fullFileName => {
|
|
@@ -136,7 +144,7 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
136
144
|
return aws_s3_deployment_1.Source.asset(filePath, { exclude: ['**', `!${fileName}`] });
|
|
137
145
|
});
|
|
138
146
|
const deploymentPath = `deployment/files/${jobName}`;
|
|
139
|
-
const additionalFileDeployment = this.deployAdditionalFiles(additionalFilesSources,
|
|
147
|
+
const additionalFileDeployment = this.deployAdditionalFiles(additionalFilesSources, bucket, deploymentRole, `job-deployment-${jobName}-additional-file`, deploymentPath, true);
|
|
140
148
|
const extraFileNames = jobConfig.additionalFiles.map(fullFileName => {
|
|
141
149
|
const fileName = path.basename(fullFileName.trim());
|
|
142
150
|
return `s3://${additionalFileDeployment.deployedBucket.bucketName}/${deploymentPath}/${fileName}`;
|
|
@@ -150,22 +158,41 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
150
158
|
}
|
|
151
159
|
}
|
|
152
160
|
}
|
|
153
|
-
|
|
161
|
+
resolveAssetPath(jobName, location) {
|
|
162
|
+
if (location.startsWith('asset:')) {
|
|
163
|
+
const assetName = location.substring('asset:'.length);
|
|
164
|
+
if (!this.props.assetBasePath) {
|
|
165
|
+
throw new Error(`Job '${jobName}' uses asset: prefix but no assetBasePath is configured. ` +
|
|
166
|
+
`Set assetBasePath in the construct props.`);
|
|
167
|
+
}
|
|
168
|
+
return path.join(this.props.assetBasePath, assetName);
|
|
169
|
+
}
|
|
170
|
+
return location;
|
|
171
|
+
}
|
|
172
|
+
createJob(jobName, jobConfig, deploymentRole, bucket) {
|
|
154
173
|
const defaultArguments = jobConfig.defaultArguments ? jobConfig.defaultArguments : {};
|
|
155
|
-
|
|
156
|
-
const
|
|
174
|
+
let scriptLocation = this.resolveAssetPath(jobName, jobConfig.command.scriptLocation.trim());
|
|
175
|
+
const scriptPath = path.dirname(scriptLocation);
|
|
176
|
+
const scriptName = path.basename(scriptLocation);
|
|
157
177
|
const scriptSource = aws_s3_deployment_1.Source.asset(scriptPath, { exclude: ['**', `!${scriptName}`] });
|
|
158
178
|
const scriptDeploymentPath = `deployment/jobs/${jobName}`;
|
|
159
179
|
const scriptDeployment = new aws_s3_deployment_1.BucketDeployment(this.scope, `job-deployment-${jobName}`, {
|
|
160
180
|
sources: [scriptSource],
|
|
161
|
-
destinationBucket:
|
|
181
|
+
destinationBucket: bucket,
|
|
162
182
|
destinationKeyPrefix: scriptDeploymentPath,
|
|
163
183
|
role: deploymentRole,
|
|
164
184
|
extract: true,
|
|
165
185
|
});
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
186
|
+
// Resolve asset: prefixes in additionalScripts before processing
|
|
187
|
+
if (jobConfig.additionalScripts) {
|
|
188
|
+
jobConfig = {
|
|
189
|
+
...jobConfig,
|
|
190
|
+
additionalScripts: jobConfig.additionalScripts.map(s => this.resolveAssetPath(jobName, s.trim())),
|
|
191
|
+
};
|
|
192
|
+
}
|
|
193
|
+
this.addAdditionalScripts(jobName, jobConfig, bucket, deploymentRole, defaultArguments);
|
|
194
|
+
this.addAdditionalJars(jobName, jobConfig, bucket, deploymentRole, defaultArguments);
|
|
195
|
+
this.addAdditionalFiles(jobName, jobConfig, bucket, deploymentRole, defaultArguments);
|
|
169
196
|
construct_1.MdaaNagSuppressions.addCodeResourceSuppressions(this.scope, [
|
|
170
197
|
{ id: 'AwsSolutions-L1', reason: 'Function is used only as custom resource during CDK deployment.' },
|
|
171
198
|
{
|
|
@@ -212,7 +239,7 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
212
239
|
connections: jobConfig.connections,
|
|
213
240
|
};
|
|
214
241
|
}
|
|
215
|
-
defaultArguments['--TempDir'] = `s3://${this.props.
|
|
242
|
+
defaultArguments['--TempDir'] = `s3://${this.props.bucketName}/temp/jobs/${jobName}`;
|
|
216
243
|
// add continuous logging unless explicitly disabled
|
|
217
244
|
if (jobConfig.continuousLogging) {
|
|
218
245
|
const logGroupName = jobName;
|
|
@@ -252,7 +279,7 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
252
279
|
workerType: jobConfig.workerType,
|
|
253
280
|
naming: this.props.naming,
|
|
254
281
|
});
|
|
255
|
-
if (job.name) {
|
|
282
|
+
if (job.name && this.props.projectName) {
|
|
256
283
|
dataops_project_l3_construct_1.DataOpsProjectUtils.createProjectSSMParam(this.scope, this.props.naming, this.props.projectName, `job/name/${jobName}`, job.name);
|
|
257
284
|
const eventRule = this.createJobMonitoringEventRule(`${jobName}-monitor`, [job.name]);
|
|
258
285
|
if (!this.props.notificationTopicArn) {
|
|
@@ -279,7 +306,7 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
279
306
|
naming: this.props.naming,
|
|
280
307
|
logGroupName: logGroupName,
|
|
281
308
|
logGroupNamePathPrefix: logGroupNamePathPrefix,
|
|
282
|
-
encryptionKey: this.
|
|
309
|
+
encryptionKey: this.kmsKey,
|
|
283
310
|
retention: logGroupRetentionDays,
|
|
284
311
|
};
|
|
285
312
|
new cloudwatch_constructs_1.MdaaLogGroup(this, logGroupName, logProps);
|
|
@@ -289,5 +316,5 @@ class GlueJobL3Construct extends l3_construct_1.MdaaL3Construct {
|
|
|
289
316
|
}
|
|
290
317
|
exports.GlueJobL3Construct = GlueJobL3Construct;
|
|
291
318
|
_a = JSII_RTTI_SYMBOL_1;
|
|
292
|
-
GlueJobL3Construct[_a] = { fqn: "@aws-mdaa/dataops-job-l3-construct.GlueJobL3Construct", version: "1.
|
|
293
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dataops-job-l3-construct.js","sourceRoot":"","sources":["dataops-job-l3-construct.ts"],"names":[],"mappings":"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","sourcesContent":["/*!\n * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport { DataOpsProjectUtils } from '@aws-mdaa/dataops-project-l3-construct';\nimport { EventBridgeHelper } from '@aws-mdaa/eventbridge-helper';\nimport { MdaaCfnJob } from '@aws-mdaa/glue-constructs';\nimport { MdaaRole } from '@aws-mdaa/iam-constructs';\nimport { MdaaBucket } from '@aws-mdaa/s3-constructs';\nimport { MdaaL3Construct, MdaaL3ConstructProps } from '@aws-mdaa/l3-construct';\nimport { CfnJob } from 'aws-cdk-lib/aws-glue';\nimport { BucketDeployment, ISource, Source } from 'aws-cdk-lib/aws-s3-deployment';\nimport { MdaaNagSuppressions } from '@aws-mdaa/construct'; //NOSONAR\nimport { Construct } from 'constructs';\nimport * as path from 'path';\nimport { SnsTopic } from 'aws-cdk-lib/aws-events-targets';\nimport { MdaaSnsTopic } from '@aws-mdaa/sns-constructs';\nimport { Rule } from 'aws-cdk-lib/aws-events';\nimport { Fn } from 'aws-cdk-lib';\nimport { ConfigurationElement } from '@aws-mdaa/config';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { MdaaLogGroup } from '@aws-mdaa/cloudwatch-constructs';\nimport { IKey, Key } from 'aws-cdk-lib/aws-kms';\nimport { updateProps } from '@aws-mdaa/cloudwatch-constructs/lib/loggroup-utils';\nimport { IBucket } from 'aws-cdk-lib/aws-s3';\nimport { IRole } from 'aws-cdk-lib/aws-iam';\n\nexport type JobCommandPythonVersion = '2' | '3' | undefined;\nexport type JobCommandName = 'glueetl' | 'pythonshell';\n\n/**\n * Q-ENHANCED-INTERFACE\n * Configuration interface for Glue job logging with CloudWatch log retention management and compliance requirements. Defines log retention settings for Glue job execution logs ensuring proper log management, compliance, and cost optimization for DataOps job monitoring and audit trails.\n *\n * Use cases: Log retention management; Compliance requirements; Cost optimization; Audit trail management\n *\n * AWS: CloudWatch log group retention for Glue job execution logs and monitoring\n *\n * Validation: logGroupRetentionDays is required with specific allowed values for retention period\n */\nexport interface LoggingConfig {\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Required CloudWatch log group retention period in days for Glue job log management and compliance requirements. Defines how long job execution logs are retained in CloudWatch with specific allowed values for log retention, compliance management, and storage cost optimization.\n   *\n   * Use cases: Log retention management; Compliance requirements; Storage cost optimization; Audit trail management\n   *\n   * AWS: CloudWatch log group retention for Glue job execution logs and audit trail management\n   *\n   * Validation: Must be 1,3,5,7,14,30,60,90,120,150,180,365,400,545,731,1827,3653, or 0; required for log retention configuration\n   **/\n  readonly logGroupRetentionDays: number;\n}\n\n/**\n * Q-ENHANCED-INTERFACE\n * Configuration interface for Glue job command specification with script execution and runtime environment settings. Defines job command configuration including job type, Python version, and script location for Glue ETL and Python shell job execution within DataOps workflows.\n *\n * Use cases: Job command configuration; Script execution; Runtime environment; ETL job setup\n *\n * AWS: Glue job command configuration for script execution and runtime environment\n *\n * Validation: name and scriptLocation are required; pythonVersion is optional with specific constraints\n */\nexport interface JobCommand {\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Required job command name specification controlling Glue job execution type and runtime environment. Defines whether to use glueetl for ETL jobs with Spark runtime or pythonshell for Python script execution with different resource allocation and capabilities.\n   *\n   * Use cases: Job type selection; Runtime environment; Execution model; Resource allocation\n   *\n   * AWS: Glue job command name for execution type and runtime environment selection\n   *\n   * Validation: Must be glueetl or pythonshell; required for job command type and execution environment\n   *   **/\n  readonly name: JobCommandName;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional Python version specification for Glue job runtime environment enabling version-specific script execution and compatibility. When specified, controls the Python runtime version for job execution ensuring script compatibility and feature availability.\n   *\n   * Use cases: Python version control; Runtime compatibility; Script execution; Version management\n   *\n   * AWS: Glue job Python version for runtime environment and script compatibility\n   *\n   * Validation: Must be 2 or 3 if provided; controls Python runtime version for job execution\n   *   **/\n  readonly pythonVersion?: JobCommandPythonVersion;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Required relative path to Glue script for job execution enabling script location specification and code deployment. Provides the S3 path or relative location of the Glue script that will be executed by the job for ETL processing and data transformation.\n   *\n   * Use cases: Script location; Code deployment; Job execution; ETL processing\n   *\n   * AWS: Glue job script location for code execution and ETL processing\n   *\n   * Validation: Must be valid script path; required for job script location and code execution\n   **/\n  readonly scriptLocation: string;\n}\n\nexport type JobWorkerType = 'Standard' | 'G.1X' | 'G.2X';\n\n/**\n * Q-ENHANCED-INTERFACE\n * Configuration interface for Glue job properties providing ETL job configuration and resource management capabilities. Defines complete Glue job setup including execution roles, commands, capacity allocation, retry policies, and monitoring for DataOps ETL processing and data transformation workflows.\n *\n * Use cases: ETL job configuration; Data transformation; Job resource management; DataOps processing\n *\n * AWS: Glue job configuration for ETL processing and data transformation workflows\n *\n * Validation: executionRoleArn, command, and description are required; other properties are optional with specific constraints\n */\nexport interface JobConfig {\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Required execution role ARN for Glue job permissions enabling secure job execution and AWS service access. Provides the IAM role that the Glue job will assume for executing ETL operations and accessing AWS services during data processing workflows.\n   *\n   * Use cases: Job permissions; Service access; Secure execution; IAM role management\n   *\n   * AWS: IAM role ARN for Glue job execution permissions and service access\n   *\n   * Validation: Must be valid IAM role ARN; required for job execution permissions and service access\n   **/\n  readonly executionRoleArn: string;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Required job command configuration defining script execution and runtime environment for Glue job processing. Provides the command specification including job type, Python version, and script location for ETL job execution and data transformation.\n   *\n   * Use cases: Script execution; Runtime configuration; Job command setup; ETL processing\n   *\n   * AWS: Glue job command for script execution and runtime environment configuration\n   *\n   * Validation: Must be valid JobCommand; required for job command configuration and script execution\n   *   **/\n  readonly command: JobCommand;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional template reference for job configuration inheritance enabling reusable job configurations and standardized setups. When specified, inherits configuration from a template defined elsewhere in the configuration for consistent job setup and management.\n   *\n   * Use cases: Configuration inheritance; Template reuse; Standardized setup; Configuration management\n   *\n   * AWS: Job template reference for configuration inheritance and standardized job setup\n   *\n   * Validation: Must be valid template name if provided; enables configuration inheritance from defined templates\n   **/\n  readonly template?: string;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional allocated capacity specification for Glue job resource allocation controlling job performance and cost management. Defines the number of capacity units allocated to the job for processing performance and resource utilization optimization.\n   *\n   * Use cases: Resource allocation; Performance tuning; Cost management; Capacity planning\n   *\n   * AWS: Glue job allocated capacity for resource allocation and performance optimization\n   *\n   * Validation: Must be positive integer if provided; controls job resource allocation and performance\n   **/\n  readonly allocatedCapacity?: number;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional array of connection names for Glue job network connectivity enabling database and external system access. Provides network connections for accessing databases, VPC resources, and external systems during ETL processing and data integration.\n   *\n   * Use cases: Database connectivity; VPC access; External system integration; Network configuration\n   *\n   * AWS: Glue job connections for database and external system connectivity\n   *\n   * Validation: Must be array of valid connection names if provided; enables database and external system access\n   **/\n  readonly connections?: string[];\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional default arguments map for Glue job parameter configuration enabling job customization and runtime behavior control. Provides default parameters and arguments that will be passed to the job for runtime configuration and behavior customization.\n   *\n   * Use cases: Job parameters; Runtime configuration; Behavior customization; Parameter management\n   *\n   * AWS: Glue job default arguments for runtime configuration and parameter management\n   *\n   * Validation: Must be valid ConfigurationElement if provided; enables job parameter configuration and customization\n   **/\n  readonly defaultArguments?: ConfigurationElement;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Required job description for documentation and operational management enabling job identification and purpose documentation. Provides descriptive information about the job's purpose, functionality, and operational characteristics for management and documentation.\n   *\n   * Use cases: Job documentation; Operational management; Purpose identification; Management information\n   *\n   * AWS: Glue job description for documentation and operational management\n   *\n   * Validation: Must be descriptive text; required for job documentation and operational management\n   **/\n  readonly description: string;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional execution property configuration for Glue job concurrency control enabling parallel execution management and resource optimization. Defines execution properties including maximum concurrent executions for job scheduling and resource management.\n   *\n   * Use cases: Concurrency control; Parallel execution; Resource management; Job scheduling\n   *\n   * AWS: Glue job execution properties for concurrency control and parallel execution management\n   *\n   * Validation: Must be valid ExecutionPropertyProperty if provided; controls job concurrency and parallel execution\n   *   **/\n  readonly executionProperty?: CfnJob.ExecutionPropertyProperty;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional Glue version specification for runtime environment control enabling version-specific features and compatibility. Defines the Glue runtime version for job execution ensuring feature availability and compatibility with job requirements.\n   *\n   * Use cases: Runtime version control; Feature availability; Compatibility management; Version specification\n   *\n   * AWS: Glue version for runtime environment and feature availability\n   *\n   * Validation: Must be valid Glue version if provided; controls runtime environment and feature availability\n   **/\n  readonly glueVersion?: string;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional maximum capacity specification for Glue job resource limits controlling maximum resource allocation and cost management. Defines the maximum number of DPUs (Data Processing Units) that can be allocated to the job for resource control and cost optimization.\n   *\n   * Use cases: Resource limits; Cost control; Maximum allocation; Resource management\n   *\n   * AWS: Glue job maximum capacity for resource limits and cost control\n   *\n   * Validation: Must be positive number if provided; controls maximum DPU allocation and resource limits\n   **/\n  readonly maxCapacity?: number;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional maximum retry count for Glue job failure handling enabling automatic retry and error recovery. Defines the maximum number of retries before job failure occurs for automatic error recovery and job reliability.\n   *\n   * Use cases: Error recovery; Automatic retry; Job reliability; Failure handling\n   *\n   * AWS: Glue job maximum retries for automatic error recovery and job reliability\n   *\n   * Validation: Must be non-negative integer if provided; controls automatic retry behavior and error recovery\n   **/\n  readonly maxRetries?: number;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional notification property configuration for Glue job monitoring and alerting enabling job status notifications and operational awareness. Defines notification settings including notification delay for job monitoring and operational alerting.\n   *\n   * Use cases: Job monitoring; Status notifications; Operational alerting; Monitoring configuration\n   *\n   * AWS: Glue job notification properties for monitoring and alerting configuration\n   *\n   * Validation: Must be valid NotificationPropertyProperty if provided; enables job monitoring and alerting\n   *   **/\n  readonly notificationProperty?: CfnJob.NotificationPropertyProperty;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional number of workers specification for Glue job parallel processing enabling distributed processing and performance optimization. Defines the number of workers assigned to the job for parallel processing and distributed data transformation.\n   *\n   * Use cases: Parallel processing; Distributed processing; Performance optimization; Worker allocation\n   *\n   * AWS: Glue job number of workers for parallel processing and performance optimization\n   *\n   * Validation: Must be positive integer if provided; controls parallel processing and worker allocation\n   **/\n  readonly numberOfWorkers?: number;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional timeout specification for Glue job execution limits enabling job runtime control and resource management. Defines the maximum execution time for the job in minutes for runtime control and resource optimization.\n   *\n   * Use cases: Runtime control; Execution limits; Resource management; Timeout configuration\n   *\n   * AWS: Glue job timeout for execution time limits and resource management\n   *\n   * Validation: Must be positive integer in minutes if provided; controls job execution time limits\n   **/\n  readonly timeout?: number;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional worker type specification for Glue job compute resource selection enabling performance and cost optimization. Defines the type of workers (Standard, G.1X, G.2X) for compute resource allocation and performance characteristics.\n   *\n   * Use cases: Compute resource selection; Performance optimization; Cost management; Worker type configuration\n   *\n   * AWS: Glue job worker type for compute resource allocation and performance optimization\n   *\n   * Validation: Must be Standard, G.1X, or G.2X if provided; controls compute resource type and performance\n   *   **/\n  readonly workerType?: JobWorkerType;\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional array of additional script paths for Glue job dependency management enabling modular ETL development and code reuse. Provides relative paths to additional Glue scripts referenced by the main ETL script for modular development and code organization.\n   *\n   * Use cases: Script dependencies; Modular development; Code reuse; Dependency management\n   *\n   * AWS: Additional Glue scripts for job dependencies and modular ETL development\n   *\n   * Validation: Must be array of valid script paths if provided; enables script dependencies and modular development\n   **/\n  readonly additionalScripts?: string[];\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional array of additional JAR file paths for Glue job library dependencies enabling Java library integration and extended functionality. Provides relative paths to JAR files referenced by the ETL script for Java library integration and extended processing capabilities.\n   *\n   * Use cases: Java library integration; Extended functionality; Library dependencies; JAR file management\n   *\n   * AWS: Additional JAR files for Glue job library dependencies and Java integration\n   *\n   * Validation: Must be array of valid JAR file paths if provided; enables Java library integration and extended functionality\n   **/\n  readonly additionalJars?: string[];\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional array of additional file paths for Glue job resource dependencies enabling external file access and resource management. Provides relative paths to additional files referenced by the ETL script for external resource access and file dependencies.\n   *\n   * Use cases: File dependencies; External resources; Resource management; File access\n   *\n   * AWS: Additional files for Glue job resource dependencies and external file access\n   *\n   * Validation: Must be array of valid file paths if provided; enables external file access and resource dependencies\n   **/\n  readonly additionalFiles?: string[];\n  /**\n   * Q-ENHANCED-PROPERTY\n   * Optional continuous logging configuration for Glue job real-time monitoring enabling live log streaming and troubleshooting support. When enabled, streams job logs to CloudWatch Logs in real-time for live monitoring and troubleshooting during job execution.\n   *\n   * Use cases: Real-time monitoring; Live log streaming; Troubleshooting support; Operational visibility\n   *\n   * AWS: Glue job continuous logging for real-time monitoring and troubleshooting\n   *\n   * Validation: Must be valid LoggingConfig if provided; enables real-time log streaming and monitoring\n   **/\n  readonly continuousLogging?: LoggingConfig;\n}\n\nexport interface GlueJobL3ConstructProps extends MdaaL3ConstructProps {\n  /**\n   * Role which will be used to deploy the Job code. Should be obtained from the DataOps Project\n   */\n  readonly deploymentRoleArn?: string;\n  /**\n   * The name of the Data Ops project bucket where job resources will be deployed and which will be used as a temporary job location\n   */\n  readonly projectBucketName?: string;\n  /**\n   * Map of job names to job configurations\n   */\n  readonly jobConfigs: { [key: string]: JobConfig };\n  /**\n   * Name of the Glue Security configuration to be used for all jobs. Likely supplied by the DataOps Project.\n   */\n  readonly securityConfigurationName?: string;\n  /**\n   * Name of the dataops project to which the job will be associated.\n   */\n  readonly projectName: string;\n\n  /**\n   * Notification topic Arn\n   */\n  readonly notificationTopicArn?: string;\n\n  /**\n   * Dataops project KMS key ARN.\n   */\n  readonly projectKMSArn?: string;\n}\n\nexport class GlueJobL3Construct extends MdaaL3Construct {\n  protected readonly props: GlueJobL3ConstructProps;\n  private readonly projectKmsKey: IKey;\n\n  constructor(scope: Construct, id: string, props: GlueJobL3ConstructProps) {\n    super(scope, id, props);\n    this.props = props;\n\n    if (!this.props.deploymentRoleArn) {\n      throw new Error('Deployment role ARN is required for job configuration');\n    }\n    const deploymentRole = MdaaRole.fromRoleArn(this.scope, `deployment-role`, this.props.deploymentRoleArn);\n    if (!this.props.projectBucketName) {\n      throw new Error('Project bucket name is required for job configuration');\n    }\n    const projectBucket = MdaaBucket.fromBucketName(this.scope, `project-bucket`, this.props.projectBucketName);\n    if (!this.props.projectKMSArn) {\n      throw new Error('Project KMS Key is required for job configuration');\n    }\n    this.projectKmsKey = Key.fromKeyArn(this, this.props.projectName, this.props.projectKMSArn);\n\n    // Build our jobs!\n    const allJobs = this.props.jobConfigs;\n    Object.keys(allJobs).forEach(jobName => {\n      const jobConfig = allJobs[jobName];\n      this.createJob(jobName, jobConfig, deploymentRole, projectBucket);\n    });\n    //CDK S3 Deployment automatically adds inline policy to project deployment role.\n    this.scope.node.children.forEach(child => {\n      if (child.node.id.startsWith('deployment-role')) {\n        MdaaNagSuppressions.addCodeResourceSuppressions(\n          child,\n          [\n            { id: 'AwsSolutions-IAM5', reason: 'Inline policy used only for deployment.' },\n            { id: 'NIST.800.53.R5-IAMNoInlinePolicy', reason: 'Policy used only for deployment.' },\n            { id: 'HIPAA.Security-IAMNoInlinePolicy', reason: 'Policy used only for deployment.' },\n            { id: 'PCI.DSS.321-IAMNoInlinePolicy', reason: 'Policy used only for deployment.' },\n          ],\n          true,\n        );\n      }\n    });\n  }\n\n  private deployAdditionalFiles(\n    additionalFilesSources: ISource[],\n    projectBucket: IBucket,\n    deploymentRole: IRole,\n    deploymentId: string,\n    deploymentPath: string,\n    extract: boolean,\n  ): BucketDeployment {\n    // Deploy Source asset(s) to /deployment/libs/<job> location.\n    return new BucketDeployment(this.scope, deploymentId, {\n      sources: additionalFilesSources,\n      destinationBucket: projectBucket,\n      destinationKeyPrefix: deploymentPath,\n      role: deploymentRole,\n      extract: extract,\n    });\n  }\n\n  private addAdditionalScripts(\n    jobName: string,\n    jobConfig: JobConfig,\n    projectBucket: IBucket,\n    deploymentRole: IRole,\n    defaultArguments: ConfigurationElement,\n  ) {\n    if (jobConfig.additionalScripts) {\n      /**\n       * Group all files at parent directory level. This will allow creating zip lib assests at various directory levels\n       * ex. '/main/script1.py' , '/util/script2.py' , '/util/script3.py' will create 2 zip files representing 'main' and 'utils'\n       *  */\n      const directoryToFile: { [filePath: string]: string[] } = {};\n      jobConfig.additionalScripts.forEach(fileLocation => {\n        const filePath = path.dirname(fileLocation.trim());\n        if (filePath in directoryToFile) {\n          directoryToFile[filePath].push(`!${path.basename(fileLocation.trim())}`);\n        } else {\n          directoryToFile[filePath] = [`!${path.basename(fileLocation.trim())}`];\n        }\n      });\n\n      // Create Source asset for each directory\n      const additionalFilesSources = Object.entries(directoryToFile).map(([filePath, fileNames]) => {\n        return Source.asset(filePath, { exclude: ['**', ...fileNames] });\n      });\n      const deploymentPath = `deployment/libs/${jobName}`;\n      const additionalFileDeployment = this.deployAdditionalFiles(\n        additionalFilesSources,\n        projectBucket,\n        deploymentRole,\n        `job-deployment-${jobName}-additional-script`,\n        deploymentPath,\n        false, // Glue expects zip of additional scripts, hence disabling the extraction\n      );\n\n      // Extract zip name(s) for each source and create comma separated list of s3 locations\n      const libraryZipNames: string[] = [];\n\n      for (let i = 0; i < additionalFilesSources.length; i++) {\n        const libName = Fn.select(i, additionalFileDeployment.objectKeys); // Extract file name of zip containing additional scripts\n        libraryZipNames.push(`s3://${additionalFileDeployment.deployedBucket.bucketName}/${deploymentPath}/${libName}`);\n      }\n\n      // Add comma separated list of zip file names to default arguments.\n      if (defaultArguments['--extra-py-files']) {\n        defaultArguments['--extra-py-files'] += ',' + libraryZipNames.join(',');\n      } else {\n        defaultArguments['--extra-py-files'] = libraryZipNames.join(',');\n      }\n    }\n  }\n\n  private addAdditionalJars(\n    jobName: string,\n    jobConfig: JobConfig,\n    projectBucket: IBucket,\n    deploymentRole: IRole,\n    defaultArguments: ConfigurationElement,\n  ) {\n    if (jobConfig.additionalJars) {\n      // Create Source asset for each directory\n      const additionalFilesSources = jobConfig.additionalJars.map(fullFileName => {\n        const filePath = path.dirname(fullFileName.trim());\n        const fileName = path.basename(fullFileName.trim());\n        return Source.asset(filePath, { exclude: ['**', `!${fileName}`] });\n      });\n      const deploymentPath = `deployment/libs/${jobName}`;\n\n      const additionalFileDeployment = this.deployAdditionalFiles(\n        additionalFilesSources,\n        projectBucket,\n        deploymentRole,\n        `job-deployment-${jobName}-additional-jar`,\n        deploymentPath,\n        true,\n      );\n\n      const extraJarNames = jobConfig.additionalJars.map(fullFileName => {\n        const fileName = path.basename(fullFileName.trim());\n        return `s3://${additionalFileDeployment.deployedBucket.bucketName}/${deploymentPath}/${fileName}`;\n      });\n\n      // Add comma separated list of zip file names to default arguments.\n      if (defaultArguments['--extra-jars']) {\n        defaultArguments['--extra-jars'] += ',' + extraJarNames.join(',');\n      } else {\n        defaultArguments['--extra-jars'] = extraJarNames.join(',');\n      }\n    }\n  }\n\n  private addAdditionalFiles(\n    jobName: string,\n    jobConfig: JobConfig,\n    projectBucket: IBucket,\n    deploymentRole: IRole,\n    defaultArguments: ConfigurationElement,\n  ) {\n    if (jobConfig.additionalFiles) {\n      // Create Source asset for each directory\n      const additionalFilesSources = jobConfig.additionalFiles.map(fullFileName => {\n        const filePath = path.dirname(fullFileName.trim());\n        const fileName = path.basename(fullFileName.trim());\n        return Source.asset(filePath, { exclude: ['**', `!${fileName}`] });\n      });\n      const deploymentPath = `deployment/files/${jobName}`;\n      const additionalFileDeployment = this.deployAdditionalFiles(\n        additionalFilesSources,\n        projectBucket,\n        deploymentRole,\n        `job-deployment-${jobName}-additional-file`,\n        deploymentPath,\n        true,\n      );\n      const extraFileNames = jobConfig.additionalFiles.map(fullFileName => {\n        const fileName = path.basename(fullFileName.trim());\n        return `s3://${additionalFileDeployment.deployedBucket.bucketName}/${deploymentPath}/${fileName}`;\n      });\n\n      // Add comma separated list of zip file names to default arguments.\n      if (defaultArguments['--extra-files']) {\n        defaultArguments['--extra-files'] += ',' + extraFileNames.join(',');\n      } else {\n        defaultArguments['--extra-files'] = extraFileNames.join(',');\n      }\n    }\n  }\n\n  private createJob(jobName: string, jobConfig: JobConfig, deploymentRole: IRole, projectBucket: IBucket) {\n    const defaultArguments = jobConfig.defaultArguments ? jobConfig.defaultArguments : {};\n    const scriptPath = path.dirname(jobConfig.command.scriptLocation.trim());\n    const scriptName = path.basename(jobConfig.command.scriptLocation.trim());\n    const scriptSource = Source.asset(scriptPath, { exclude: ['**', `!${scriptName}`] });\n\n    const scriptDeploymentPath = `deployment/jobs/${jobName}`;\n    const scriptDeployment = new BucketDeployment(this.scope, `job-deployment-${jobName}`, {\n      sources: [scriptSource],\n      destinationBucket: projectBucket,\n      destinationKeyPrefix: scriptDeploymentPath,\n      role: deploymentRole,\n      extract: true,\n    });\n\n    this.addAdditionalScripts(jobName, jobConfig, projectBucket, deploymentRole, defaultArguments);\n    this.addAdditionalJars(jobName, jobConfig, projectBucket, deploymentRole, defaultArguments);\n    this.addAdditionalFiles(jobName, jobConfig, projectBucket, deploymentRole, defaultArguments);\n    MdaaNagSuppressions.addCodeResourceSuppressions(\n      this.scope,\n      [\n        { id: 'AwsSolutions-L1', reason: 'Function is used only as custom resource during CDK deployment.' },\n        {\n          id: 'NIST.800.53.R5-LambdaConcurrency',\n          reason: 'Function is used only as custom resource during CDK deployment.',\n        },\n        {\n          id: 'NIST.800.53.R5-LambdaInsideVPC',\n          reason: 'Function is used only as custom resource during CDK deployment and interacts only with S3.',\n        },\n        {\n          id: 'NIST.800.53.R5-LambdaDLQ',\n          reason:\n            'Function is used only as custom resource during CDK deployment. Errors will be handled by CloudFormation.',\n        },\n        {\n          id: 'HIPAA.Security-LambdaConcurrency',\n          reason: 'Function is used only as custom resource during CDK deployment.',\n        },\n        {\n          id: 'PCI.DSS.321-LambdaConcurrency',\n          reason: 'Function is used only as custom resource during CDK deployment.',\n        },\n        {\n          id: 'HIPAA.Security-LambdaInsideVPC',\n          reason: 'Function is used only as custom resource during CDK deployment and interacts only with S3.',\n        },\n        {\n          id: 'PCI.DSS.321-LambdaInsideVPC',\n          reason: 'Function is used only as custom resource during CDK deployment and interacts only with S3.',\n        },\n        {\n          id: 'HIPAA.Security-LambdaDLQ',\n          reason:\n            'Function is used only as custom resource during CDK deployment. Errors will be handled by CloudFormation.',\n        },\n        {\n          id: 'PCI.DSS.321-LambdaDLQ',\n          reason:\n            'Function is used only as custom resource during CDK deployment. Errors will be handled by CloudFormation.',\n        },\n      ],\n      true,\n    );\n    // Connections will require an array of references where they are defined\n    let connectionsConfigured: ConfigurationElement | undefined;\n    if (jobConfig.connections) {\n      connectionsConfigured = {\n        connections: jobConfig.connections,\n      };\n    }\n\n    defaultArguments['--TempDir'] = `s3://${this.props.projectBucketName}/temp/jobs/${jobName}`;\n\n    // add continuous logging unless explicitly disabled\n    if (jobConfig.continuousLogging) {\n      const logGroupName = jobName;\n      const logGroupNamePathPrefix = '/aws/glue';\n      defaultArguments['--continuous-log-logGroup'] = this.createLogGroup(\n        logGroupNamePathPrefix,\n        logGroupName,\n        jobConfig.continuousLogging,\n      );\n    } else {\n      console.log(`Continuous logging not enabled for job: ${jobName}`);\n    }\n\n    const inputParams = defaultArguments['--input_params'];\n    if (inputParams) {\n      defaultArguments['--input_params'] = JSON.stringify(inputParams);\n    }\n    if (!this.props.securityConfigurationName) {\n      throw new Error('Security configuration name is required for job monitoring event rule');\n    }\n    const job = new MdaaCfnJob(this.scope, `${jobName}-job`, {\n      command: {\n        name: jobConfig.command.name,\n        pythonVersion: jobConfig.command.pythonVersion,\n        scriptLocation: `s3://${scriptDeployment.deployedBucket.bucketName}/${scriptDeploymentPath}/${scriptName}`,\n      },\n      role: jobConfig.executionRoleArn,\n      allocatedCapacity: jobConfig.allocatedCapacity,\n      connections: connectionsConfigured,\n      defaultArguments: defaultArguments,\n      description: jobConfig.description,\n      executionProperty: jobConfig.executionProperty,\n      glueVersion: jobConfig.glueVersion,\n      maxCapacity: jobConfig.maxCapacity,\n      maxRetries: jobConfig.maxRetries,\n      name: jobName,\n      notificationProperty: jobConfig.notificationProperty,\n      numberOfWorkers: jobConfig.numberOfWorkers,\n      securityConfiguration: this.props.securityConfigurationName,\n      timeout: jobConfig.timeout,\n      workerType: jobConfig.workerType,\n      naming: this.props.naming,\n    });\n    if (job.name) {\n      DataOpsProjectUtils.createProjectSSMParam(\n        this.scope,\n        this.props.naming,\n        this.props.projectName,\n        `job/name/${jobName}`,\n        job.name,\n      );\n\n      const eventRule = this.createJobMonitoringEventRule(`${jobName}-monitor`, [job.name]);\n      if (!this.props.notificationTopicArn) {\n        throw new Error('Notification topic ARN is required for job monitoring event rule');\n      }\n      eventRule.addTarget(\n        new SnsTopic(MdaaSnsTopic.fromTopicArn(this.scope, `${jobName}-topic`, this.props.notificationTopicArn)),\n      );\n    }\n  }\n\n  private createJobMonitoringEventRule(ruleName: string, jobNames: string[]): Rule {\n    return EventBridgeHelper.createGlueMonitoringEventRule(\n      this.scope,\n      this.props.naming,\n      ruleName,\n      'Workflow Job failure events',\n      {\n        jobName: jobNames,\n        state: ['FAILED', 'TIMEOUT', 'STOPPED'],\n      },\n    );\n  }\n\n  private createLogGroup(logGroupNamePathPrefix: string, logGroupName: string, loggingConfig: LoggingConfig): string {\n    let logGroupRetentionDays: RetentionDays;\n\n    if (loggingConfig.logGroupRetentionDays != 0) {\n      logGroupRetentionDays = loggingConfig.logGroupRetentionDays;\n    } else {\n      logGroupRetentionDays = RetentionDays.INFINITE;\n    }\n\n    const logProps = {\n      naming: this.props.naming,\n      logGroupName: logGroupName,\n      logGroupNamePathPrefix: logGroupNamePathPrefix,\n      encryptionKey: this.projectKmsKey,\n      retention: logGroupRetentionDays,\n    };\n    new MdaaLogGroup(this, logGroupName, logProps);\n    // `updateProps` always returns a prop with a logGroupName\n    return updateProps(logProps).logGroupName!;\n  }\n}\n"]}
|
|
319
|
+
GlueJobL3Construct[_a] = { fqn: "@aws-mdaa/dataops-job-l3-construct.GlueJobL3Construct", version: "1.6.0" };
|
|
320
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dataops-job-l3-construct.js","sourceRoot":"","sources":["dataops-job-l3-construct.ts"],"names":[],"mappings":";;;;;AAAA;;;GAGG;AAEH,yFAA6E;AAC7E,qEAAiE;AACjE,+DAAuD;AACvD,6DAAoD;AACpD,2DAAqD;AACrD,yDAA+E;AAE/E,qEAAkF;AAClF,mDAA0D,CAAC,SAAS;AAEpE,6BAA6B;AAC7B,uEAA0D;AAC1D,6DAAwD;AAExD,6CAAiC;AAEjC,mDAAqD;AACrD,2EAA+D;AAC/D,iDAAgD;AAChD,uFAAiF;AAmIjF,MAAa,kBAAmB,SAAQ,8BAAe;IAIrD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA8B;QACtE,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC;QACxB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QAEnB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,iBAAiB,EAAE,CAAC;YAClC,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;QAC3E,CAAC;QACD,MAAM,cAAc,GAAG,yBAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,EAAE,iBAAiB,EAAE,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC;QACzG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,EAAE,CAAC;YAC3B,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;QAC3E,CAAC;QACD,MAAM,MAAM,GAAG,0BAAU,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,EAAE,gBAAgB,EAAE,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAC9F,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACvB,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;QACvE,CAAC;QACD,IAAI,CAAC,MAAM,GAAG,aAAG,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAE3F,kBAAkB;QAClB,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC;QACtC,KAAK,MAAM,OAAO,IAAI,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YAC3C,MAAM,SAAS,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;YACnC,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,SAAS,EAAE,cAAc,EAAE,MAAM,CAAC,CAAC;QAC7D,CAAC;QAED,0EAA0E;QAC1E,mEAAmE;QACnE,mFAAmF;QACnF,4EAA4E;QAC5E,gFAAgF;QAChF,wEAAwE;QACxE,KAAK,MAAM,KAAK,IAAI,cAAc,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACjD,IAAI,KAAK,CAAC,IAAI,CAAC,EAAE,KAAK,QAAQ,EAAE,CAAC;gBAC/B,cAAc,CAAC,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBAClD,MAAM;YACR,CAAC;QACH,CAAC;QAED,oEAAoE;QACpE,2EAA2E;QAC3E,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YACvC,IAAI,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,iBAAiB,CAAC,EAAE,CAAC;gBAChD,+BAAmB,CAAC,2BAA2B,CAC7C,KAAK,EACL,CAAC,EAAE,EAAE,EAAE,mBAAmB,EAAE,MAAM,EAAE,yDAAyD,EAAE,CAAC,EAChG,IAAI,CACL,CAAC;YACJ,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,qBAAqB,CAC3B,sBAAiC,EACjC,MAAe,EACf,cAAqB,EACrB,YAAoB,EACpB,cAAsB,EACtB,OAAgB;QAEhB,6DAA6D;QAC7D,OAAO,IAAI,oCAAgB,CAAC,IAAI,CAAC,KAAK,EAAE,YAAY,EAAE;YACpD,OAAO,EAAE,sBAAsB;YAC/B,iBAAiB,EAAE,MAAM;YACzB,oBAAoB,EAAE,cAAc;YACpC,IAAI,EAAE,cAAc;YACpB,OAAO,EAAE,OAAO;SACjB,CAAC,CAAC;IACL,CAAC;IAEO,oBAAoB,CAC1B,OAAe,EACf,SAAoB,EACpB,MAAe,EACf,cAAqB,EACrB,gBAAsC;QAEtC,IAAI,SAAS,CAAC,iBAAiB,EAAE,CAAC;YAChC;;;kBAGM;YACN,MAAM,eAAe,GAAqC,EAAE,CAAC;YAC7D,SAAS,CAAC,iBAAiB,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE;gBACjD,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;gBACnD,IAAI,QAAQ,IAAI,eAAe,EAAE,CAAC;oBAChC,eAAe,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC;gBAC3E,CAAC;qBAAM,CAAC;oBACN,eAAe,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC;gBACzE,CAAC;YACH,CAAC,CAAC,CAAC;YAEH,yCAAyC;YACzC,MAAM,sBAAsB,GAAG,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,EAAE,SAAS,CAAC,EAAE,EAAE;gBAC3F,OAAO,0BAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,OAAO,EAAE,CAAC,IAAI,EAAE,GAAG,SAAS,CAAC,EAAE,CAAC,CAAC;YACnE,CAAC,CAAC,CAAC;YACH,MAAM,cAAc,GAAG,mBAAmB,OAAO,EAAE,CAAC;YACpD,MAAM,wBAAwB,GAAG,IAAI,CAAC,qBAAqB,CACzD,sBAAsB,EACtB,MAAM,EACN,cAAc,EACd,kBAAkB,OAAO,oBAAoB,EAC7C,cAAc,EACd,KAAK,CACN,CAAC;YAEF,sFAAsF;YACtF,MAAM,eAAe,GAAa,EAAE,CAAC;YAErC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,sBAAsB,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACvD,MAAM,OAAO,GAAG,gBAAE,CAAC,MAAM,CAAC,CAAC,EAAE,wBAAwB,CAAC,UAAU,CAAC,CAAC,CAAC,yDAAyD;gBAC5H,eAAe,CAAC,IAAI,CAAC,QAAQ,wBAAwB,CAAC,cAAc,CAAC,UAAU,IAAI,cAAc,IAAI,OAAO,EAAE,CAAC,CAAC;YAClH,CAAC;YAED,mEAAmE;YACnE,IAAI,gBAAgB,CAAC,kBAAkB,CAAC,EAAE,CAAC;gBACzC,gBAAgB,CAAC,kBAAkB,CAAC,IAAI,GAAG,GAAG,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAC1E,CAAC;iBAAM,CAAC;gBACN,gBAAgB,CAAC,kBAAkB,CAAC,GAAG,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACnE,CAAC;QACH,CAAC;IACH,CAAC;IAEO,iBAAiB,CACvB,OAAe,EACf,SAAoB,EACpB,MAAe,EACf,cAAqB,EACrB,gBAAsC;QAEtC,IAAI,SAAS,CAAC,cAAc,EAAE,CAAC;YAC7B,yCAAyC;YACzC,MAAM,sBAAsB,GAAG,SAAS,CAAC,cAAc,CAAC,GAAG,CAAC,YAAY,CAAC,EAAE;gBACzE,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;gBACnD,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;gBACpD,OAAO,0BAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,OAAO,EAAE,CAAC,IAAI,EAAE,IAAI,QAAQ,EAAE,CAAC,EAAE,CAAC,CAAC;YACrE,CAAC,CAAC,CAAC;YACH,MAAM,cAAc,GAAG,mBAAmB,OAAO,EAAE,CAAC;YAEpD,MAAM,wBAAwB,GAAG,IAAI,CAAC,qBAAqB,CACzD,sBAAsB,EACtB,MAAM,EACN,cAAc,EACd,kBAAkB,OAAO,iBAAiB,EAC1C,cAAc,EACd,IAAI,CACL,CAAC;YAEF,MAAM,aAAa,GAAG,SAAS,CAAC,cAAc,CAAC,GAAG,CAAC,YAAY,CAAC,EAAE;gBAChE,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;gBACpD,OAAO,QAAQ,wBAAwB,CAAC,cAAc,CAAC,UAAU,IAAI,cAAc,IAAI,QAAQ,EAAE,CAAC;YACpG,CAAC,CAAC,CAAC;YAEH,mEAAmE;YACnE,IAAI,gBAAgB,CAAC,cAAc,CAAC,EAAE,CAAC;gBACrC,gBAAgB,CAAC,cAAc,CAAC,IAAI,GAAG,GAAG,aAAa,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACpE,CAAC;iBAAM,CAAC;gBACN,gBAAgB,CAAC,cAAc,CAAC,GAAG,aAAa,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAC7D,CAAC;QACH,CAAC;IACH,CAAC;IAEO,kBAAkB,CACxB,OAAe,EACf,SAAoB,EACpB,MAAe,EACf,cAAqB,EACrB,gBAAsC;QAEtC,IAAI,SAAS,CAAC,eAAe,EAAE,CAAC;YAC9B,yCAAyC;YACzC,MAAM,sBAAsB,GAAG,SAAS,CAAC,eAAe,CAAC,GAAG,CAAC,YAAY,CAAC,EAAE;gBAC1E,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;gBACnD,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;gBACpD,OAAO,0BAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,OAAO,EAAE,CAAC,IAAI,EAAE,IAAI,QAAQ,EAAE,CAAC,EAAE,CAAC,CAAC;YACrE,CAAC,CAAC,CAAC;YACH,MAAM,cAAc,GAAG,oBAAoB,OAAO,EAAE,CAAC;YACrD,MAAM,wBAAwB,GAAG,IAAI,CAAC,qBAAqB,CACzD,sBAAsB,EACtB,MAAM,EACN,cAAc,EACd,kBAAkB,OAAO,kBAAkB,EAC3C,cAAc,EACd,IAAI,CACL,CAAC;YACF,MAAM,cAAc,GAAG,SAAS,CAAC,eAAe,CAAC,GAAG,CAAC,YAAY,CAAC,EAAE;gBAClE,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC;gBACpD,OAAO,QAAQ,wBAAwB,CAAC,cAAc,CAAC,UAAU,IAAI,cAAc,IAAI,QAAQ,EAAE,CAAC;YACpG,CAAC,CAAC,CAAC;YAEH,mEAAmE;YACnE,IAAI,gBAAgB,CAAC,eAAe,CAAC,EAAE,CAAC;gBACtC,gBAAgB,CAAC,eAAe,CAAC,IAAI,GAAG,GAAG,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACtE,CAAC;iBAAM,CAAC;gBACN,gBAAgB,CAAC,eAAe,CAAC,GAAG,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAC/D,CAAC;QACH,CAAC;IACH,CAAC;IAEO,gBAAgB,CAAC,OAAe,EAAE,QAAgB;QACxD,IAAI,QAAQ,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAClC,MAAM,SAAS,GAAG,QAAQ,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;YACtD,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;gBAC9B,MAAM,IAAI,KAAK,CACb,QAAQ,OAAO,2DAA2D;oBACxE,2CAA2C,CAC9C,CAAC;YACJ,CAAC;YACD,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,SAAS,CAAC,CAAC;QACxD,CAAC;QACD,OAAO,QAAQ,CAAC;IAClB,CAAC;IAEO,SAAS,CAAC,OAAe,EAAE,SAAoB,EAAE,cAAqB,EAAE,MAAe;QAC7F,MAAM,gBAAgB,GAAG,SAAS,CAAC,gBAAgB,CAAC,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC,CAAC,EAAE,CAAC;QACtF,IAAI,cAAc,GAAG,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,SAAS,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC,CAAC;QAE7F,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QAChD,MAAM,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;QACjD,MAAM,YAAY,GAAG,0BAAM,CAAC,KAAK,CAAC,UAAU,EAAE,EAAE,OAAO,EAAE,CAAC,IAAI,EAAE,IAAI,UAAU,EAAE,CAAC,EAAE,CAAC,CAAC;QAErF,MAAM,oBAAoB,GAAG,mBAAmB,OAAO,EAAE,CAAC;QAC1D,MAAM,gBAAgB,GAAG,IAAI,oCAAgB,CAAC,IAAI,CAAC,KAAK,EAAE,kBAAkB,OAAO,EAAE,EAAE;YACrF,OAAO,EAAE,CAAC,YAAY,CAAC;YACvB,iBAAiB,EAAE,MAAM;YACzB,oBAAoB,EAAE,oBAAoB;YAC1C,IAAI,EAAE,cAAc;YACpB,OAAO,EAAE,IAAI;SACd,CAAC,CAAC;QAEH,iEAAiE;QACjE,IAAI,SAAS,CAAC,iBAAiB,EAAE,CAAC;YAChC,SAAS,GAAG;gBACV,GAAG,SAAS;gBACZ,iBAAiB,EAAE,SAAS,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;aAClG,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,oBAAoB,CAAC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,cAAc,EAAE,gBAAgB,CAAC,CAAC;QACxF,IAAI,CAAC,iBAAiB,CAAC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,cAAc,EAAE,gBAAgB,CAAC,CAAC;QACrF,IAAI,CAAC,kBAAkB,CAAC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,cAAc,EAAE,gBAAgB,CAAC,CAAC;QACtF,+BAAmB,CAAC,2BAA2B,CAC7C,IAAI,CAAC,KAAK,EACV;YACE,EAAE,EAAE,EAAE,iBAAiB,EAAE,MAAM,EAAE,iEAAiE,EAAE;YACpG;gBACE,EAAE,EAAE,kCAAkC;gBACtC,MAAM,EAAE,iEAAiE;aAC1E;YACD;gBACE,EAAE,EAAE,gCAAgC;gBACpC,MAAM,EAAE,4FAA4F;aACrG;YACD;gBACE,EAAE,EAAE,0BAA0B;gBAC9B,MAAM,EACJ,2GAA2G;aAC9G;YACD;gBACE,EAAE,EAAE,kCAAkC;gBACtC,MAAM,EAAE,iEAAiE;aAC1E;YACD;gBACE,EAAE,EAAE,+BAA+B;gBACnC,MAAM,EAAE,iEAAiE;aAC1E;YACD;gBACE,EAAE,EAAE,gCAAgC;gBACpC,MAAM,EAAE,4FAA4F;aACrG;YACD;gBACE,EAAE,EAAE,6BAA6B;gBACjC,MAAM,EAAE,4FAA4F;aACrG;YACD;gBACE,EAAE,EAAE,0BAA0B;gBAC9B,MAAM,EACJ,2GAA2G;aAC9G;YACD;gBACE,EAAE,EAAE,uBAAuB;gBAC3B,MAAM,EACJ,2GAA2G;aAC9G;SACF,EACD,IAAI,CACL,CAAC;QACF,yEAAyE;QACzE,IAAI,qBAAuD,CAAC;QAC5D,IAAI,SAAS,CAAC,WAAW,EAAE,CAAC;YAC1B,qBAAqB,GAAG;gBACtB,WAAW,EAAE,SAAS,CAAC,WAAW;aACnC,CAAC;QACJ,CAAC;QAED,gBAAgB,CAAC,WAAW,CAAC,GAAG,QAAQ,IAAI,CAAC,KAAK,CAAC,UAAU,cAAc,OAAO,EAAE,CAAC;QAErF,oDAAoD;QACpD,IAAI,SAAS,CAAC,iBAAiB,EAAE,CAAC;YAChC,MAAM,YAAY,GAAG,OAAO,CAAC;YAC7B,MAAM,sBAAsB,GAAG,WAAW,CAAC;YAC3C,gBAAgB,CAAC,2BAA2B,CAAC,GAAG,IAAI,CAAC,cAAc,CACjE,sBAAsB,EACtB,YAAY,EACZ,SAAS,CAAC,iBAAiB,CAC5B,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,2CAA2C,OAAO,EAAE,CAAC,CAAC;QACpE,CAAC;QAED,MAAM,WAAW,GAAG,gBAAgB,CAAC,gBAAgB,CAAC,CAAC;QACvD,IAAI,WAAW,EAAE,CAAC;YAChB,gBAAgB,CAAC,gBAAgB,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC;QACnE,CAAC;QACD,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,yBAAyB,EAAE,CAAC;YAC1C,MAAM,IAAI,KAAK,CAAC,uEAAuE,CAAC,CAAC;QAC3F,CAAC;QACD,MAAM,GAAG,GAAG,IAAI,4BAAU,CAAC,IAAI,CAAC,KAAK,EAAE,GAAG,OAAO,MAAM,EAAE;YACvD,OAAO,EAAE;gBACP,IAAI,EAAE,SAAS,CAAC,OAAO,CAAC,IAAI;gBAC5B,aAAa,EAAE,SAAS,CAAC,OAAO,CAAC,aAAa;gBAC9C,cAAc,EAAE,QAAQ,gBAAgB,CAAC,cAAc,CAAC,UAAU,IAAI,oBAAoB,IAAI,UAAU,EAAE;aAC3G;YACD,IAAI,EAAE,SAAS,CAAC,gBAAgB;YAChC,iBAAiB,EAAE,SAAS,CAAC,iBAAiB;YAC9C,WAAW,EAAE,qBAAqB;YAClC,gBAAgB,EAAE,gBAAgB;YAClC,WAAW,EAAE,SAAS,CAAC,WAAW;YAClC,iBAAiB,EAAE,SAAS,CAAC,iBAAiB;YAC9C,WAAW,EAAE,SAAS,CAAC,WAAW;YAClC,WAAW,EAAE,SAAS,CAAC,WAAW;YAClC,UAAU,EAAE,SAAS,CAAC,UAAU;YAChC,IAAI,EAAE,OAAO;YACb,oBAAoB,EAAE,SAAS,CAAC,oBAAoB;YACpD,eAAe,EAAE,SAAS,CAAC,eAAe;YAC1C,qBAAqB,EAAE,IAAI,CAAC,KAAK,CAAC,yBAAyB;YAC3D,OAAO,EAAE,SAAS,CAAC,OAAO;YAC1B,UAAU,EAAE,SAAS,CAAC,UAAU;YAChC,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM;SAC1B,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,IAAI,IAAI,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;YACvC,kDAAmB,CAAC,qBAAqB,CACvC,IAAI,CAAC,KAAK,EACV,IAAI,CAAC,KAAK,CAAC,MAAM,EACjB,IAAI,CAAC,KAAK,CAAC,WAAW,EACtB,YAAY,OAAO,EAAE,EACrB,GAAG,CAAC,IAAI,CACT,CAAC;YAEF,MAAM,SAAS,GAAG,IAAI,CAAC,4BAA4B,CAAC,GAAG,OAAO,UAAU,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;YACtF,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,oBAAoB,EAAE,CAAC;gBACrC,MAAM,IAAI,KAAK,CAAC,kEAAkE,CAAC,CAAC;YACtF,CAAC;YACD,SAAS,CAAC,SAAS,CACjB,IAAI,6BAAQ,CAAC,6BAAY,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,EAAE,GAAG,OAAO,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC,CACzG,CAAC;QACJ,CAAC;IACH,CAAC;IAEO,4BAA4B,CAAC,QAAgB,EAAE,QAAkB;QACvE,OAAO,sCAAiB,CAAC,6BAA6B,CACpD,IAAI,CAAC,KAAK,EACV,IAAI,CAAC,KAAK,CAAC,MAAM,EACjB,QAAQ,EACR,6BAA6B,EAC7B;YACE,OAAO,EAAE,QAAQ;YACjB,KAAK,EAAE,CAAC,QAAQ,EAAE,SAAS,EAAE,SAAS,CAAC;SACxC,CACF,CAAC;IACJ,CAAC;IAEO,cAAc,CAAC,sBAA8B,EAAE,YAAoB,EAAE,aAA4B;QACvG,IAAI,qBAAoC,CAAC;QAEzC,IAAI,aAAa,CAAC,qBAAqB,IAAI,CAAC,EAAE,CAAC;YAC7C,qBAAqB,GAAG,aAAa,CAAC,qBAAqB,CAAC;QAC9D,CAAC;aAAM,CAAC;YACN,qBAAqB,GAAG,wBAAa,CAAC,QAAQ,CAAC;QACjD,CAAC;QAED,MAAM,QAAQ,GAAG;YACf,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM;YACzB,YAAY,EAAE,YAAY;YAC1B,sBAAsB,EAAE,sBAAsB;YAC9C,aAAa,EAAE,IAAI,CAAC,MAAM;YAC1B,SAAS,EAAE,qBAAqB;SACjC,CAAC;QACF,IAAI,oCAAY,CAAC,IAAI,EAAE,YAAY,EAAE,QAAQ,CAAC,CAAC;QAC/C,0DAA0D;QAC1D,OAAO,IAAA,4BAAW,EAAC,QAAQ,CAAC,CAAC,YAAa,CAAC;IAC7C,CAAC;;AA1YH,gDA2YC","sourcesContent":["/*!\n * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport { DataOpsProjectUtils } from '@aws-mdaa/dataops-project-l3-construct';\nimport { EventBridgeHelper } from '@aws-mdaa/eventbridge-helper';\nimport { MdaaCfnJob } from '@aws-mdaa/glue-constructs';\nimport { MdaaRole } from '@aws-mdaa/iam-constructs';\nimport { MdaaBucket } from '@aws-mdaa/s3-constructs';\nimport { MdaaL3Construct, MdaaL3ConstructProps } from '@aws-mdaa/l3-construct';\nimport { CfnJob } from 'aws-cdk-lib/aws-glue';\nimport { BucketDeployment, ISource, Source } from 'aws-cdk-lib/aws-s3-deployment';\nimport { MdaaNagSuppressions } from '@aws-mdaa/construct'; //NOSONAR\nimport { Construct } from 'constructs';\nimport * as path from 'path';\nimport { SnsTopic } from 'aws-cdk-lib/aws-events-targets';\nimport { MdaaSnsTopic } from '@aws-mdaa/sns-constructs';\nimport { Rule } from 'aws-cdk-lib/aws-events';\nimport { Fn } from 'aws-cdk-lib';\nimport { ConfigurationElement } from '@aws-mdaa/config';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { MdaaLogGroup } from '@aws-mdaa/cloudwatch-constructs';\nimport { IKey, Key } from 'aws-cdk-lib/aws-kms';\nimport { updateProps } from '@aws-mdaa/cloudwatch-constructs/lib/loggroup-utils';\nimport { IBucket } from 'aws-cdk-lib/aws-s3';\nimport { IRole } from 'aws-cdk-lib/aws-iam';\n\nexport type JobCommandPythonVersion = '2' | '3' | undefined;\nexport type JobCommandName = 'glueetl' | 'pythonshell';\n\n/**\n * Configuration for Glue job logging with CloudWatch log retention management.\n *\n * Use cases: Log retention management; Compliance requirements; Cost optimization; Audit trail management\n *\n * AWS: CloudWatch log group retention for Glue job execution logs and monitoring\n *\n * Validation: logGroupRetentionDays is required with specific allowed values for retention period\n */\nexport interface LoggingConfig {\n  /** CloudWatch log group retention in days. Allowed: 1,3,5,7,14,30,60,90,120,150,180,365,400,545,731,1827,3653, or 0. */\n  readonly logGroupRetentionDays: number;\n}\n\n/**\n * Configuration for a Glue job command specifying script execution and runtime environment.\n *\n * Use cases: Job command configuration; Script execution; Runtime environment; ETL job setup\n *\n * AWS: Glue job command configuration for script execution and runtime environment\n *\n * Validation: name and scriptLocation are required; pythonVersion is optional with specific constraints\n */\nexport interface JobCommand {\n  /** Job type: 'glueetl' for Spark ETL or 'pythonshell' for Python scripts. */\n  readonly name: JobCommandName;\n  /** Python version for job runtime ('2' or '3'). */\n  readonly pythonVersion?: JobCommandPythonVersion;\n  /** Relative path to the Glue script for job execution, or `asset:<filename>` to use a pre-built script from the app's assets directory. Available assets: dq-main.py (DQ evaluation, with utils/ as additionalScripts). */\n  readonly scriptLocation: string;\n}\n\nexport type JobWorkerType = 'Standard' | 'G.1X' | 'G.2X';\n\n/**\n * Configuration for a Glue job including execution roles, commands, capacity, and monitoring.\n *\n * Use cases: ETL job configuration; Data transformation; Job resource management; DataOps processing\n *\n * AWS: Glue job configuration for ETL processing and data transformation workflows\n *\n * Validation: executionRoleArn, command, and description are required; other properties are optional with specific constraints\n */\nexport interface JobConfig {\n  /** IAM role ARN for Glue job execution permissions. */\n  readonly executionRoleArn: string;\n  /** Job command configuration defining script and runtime environment. */\n  readonly command: JobCommand;\n  /** Template name for configuration inheritance. */\n  readonly template?: string;\n  /** Number of capacity units allocated to the job. */\n  readonly allocatedCapacity?: number;\n  /** Connection names for database and external system access. */\n  readonly connections?: string[];\n  /** Default arguments passed to the job at runtime. */\n  readonly defaultArguments?: ConfigurationElement;\n  /** Job description for documentation and management. */\n  readonly description: string;\n  /** Execution properties including maximum concurrent runs. */\n  readonly executionProperty?: CfnJob.ExecutionPropertyProperty;\n  /** Glue runtime version for the job. */\n  readonly glueVersion?: string;\n  /** Maximum DPU capacity for the job. */\n  readonly maxCapacity?: number;\n  /** Maximum retry count before job failure. */\n  readonly maxRetries?: number;\n  /** Notification settings for job monitoring and alerting. */\n  readonly notificationProperty?: CfnJob.NotificationPropertyProperty;\n  /** Number of workers for parallel processing. */\n  readonly numberOfWorkers?: number;\n  /** Job timeout in minutes. */\n  readonly timeout?: number;\n  /** Worker type: 'Standard', 'G.1X', or 'G.2X'. */\n  readonly workerType?: JobWorkerType;\n  /** Relative paths to additional Python scripts for the job. */\n  readonly additionalScripts?: string[];\n  /** Relative paths to additional JAR files for the job. */\n  readonly additionalJars?: string[];\n  /** Relative paths to additional files for the job. */\n  readonly additionalFiles?: string[];\n  /** Continuous logging configuration for real-time monitoring. */\n  readonly continuousLogging?: LoggingConfig;\n}\n\nexport interface GlueJobL3ConstructProps extends MdaaL3ConstructProps {\n  /**\n   * Role which will be used to deploy the Job code. Should be obtained from the DataOps Project\n   */\n  readonly deploymentRoleArn?: string;\n  /**\n   * The name of the Data Ops project bucket where job resources will be deployed and which will be used as a temporary job location\n   */\n  readonly bucketName?: string;\n  /**\n   * Map of job names to job configurations\n   */\n  readonly jobConfigs: { [key: string]: JobConfig };\n  /**\n   * Name of the Glue Security configuration to be used for all jobs. Likely supplied by the DataOps Project.\n   */\n  readonly securityConfigurationName?: string;\n  /**\n   * Name of the dataops project to which the job will be associated.\n   */\n  readonly projectName?: string;\n\n  /**\n   * Notification topic Arn\n   */\n  readonly notificationTopicArn?: string;\n\n  /**\n   * Dataops project KMS key ARN.\n   */\n  readonly kmsArn?: string;\n\n  /**\n   * Base path for resolving `asset:` prefixed script locations.\n   * When a job's scriptLocation starts with `asset:`, it is resolved\n   * relative to this directory.\n   */\n  readonly assetBasePath?: string;\n}\n\nexport class GlueJobL3Construct extends MdaaL3Construct {\n  protected readonly props: GlueJobL3ConstructProps;\n  private readonly kmsKey: IKey;\n\n  constructor(scope: Construct, id: string, props: GlueJobL3ConstructProps) {\n    super(scope, id, props);\n    this.props = props;\n\n    if (!this.props.deploymentRoleArn) {\n      throw new Error('Deployment role ARN is required for job configuration');\n    }\n    const deploymentRole = MdaaRole.fromRoleArn(this.scope, `deployment-role`, this.props.deploymentRoleArn);\n    if (!this.props.bucketName) {\n      throw new Error('Project bucket name is required for job configuration');\n    }\n    const bucket = MdaaBucket.fromBucketName(this.scope, `project-bucket`, this.props.bucketName);\n    if (!this.props.kmsArn) {\n      throw new Error('Project KMS Key is required for job configuration');\n    }\n    this.kmsKey = Key.fromKeyArn(this, this.props.projectName ?? 'kms-key', this.props.kmsArn);\n\n    // Build our jobs!\n    const allJobs = this.props.jobConfigs;\n    for (const jobName of Object.keys(allJobs)) {\n      const jobConfig = allJobs[jobName];\n      this.createJob(jobName, jobConfig, deploymentRole, bucket);\n    }\n\n    // BucketDeployment adds an inline policy to the imported deployment role,\n    // which CDK synthesizes as a standalone AWS::IAM::Policy resource.\n    // Because the policy name is derived from the construct path (not the stack name),\n    // multiple stacks importing the same role produce colliding physical names.\n    // Remove the inline policy node — the deployment role already has the necessary\n    // S3 permissions granted in the dataops-project construct that owns it.\n    for (const child of deploymentRole.node.children) {\n      if (child.node.id === 'Policy') {\n        deploymentRole.node.tryRemoveChild(child.node.id);\n        break;\n      }\n    }\n\n    // Suppress nag warnings on the imported deployment role node itself\n    // (remaining after inline policy removal, e.g. wildcard resource warnings)\n    this.scope.node.children.forEach(child => {\n      if (child.node.id.startsWith('deployment-role')) {\n        MdaaNagSuppressions.addCodeResourceSuppressions(\n          child,\n          [{ id: 'AwsSolutions-IAM5', reason: 'Permissions granted on deployment role in owning stack.' }],\n          true,\n        );\n      }\n    });\n  }\n\n  private deployAdditionalFiles(\n    additionalFilesSources: ISource[],\n    bucket: IBucket,\n    deploymentRole: IRole,\n    deploymentId: string,\n    deploymentPath: string,\n    extract: boolean,\n  ): BucketDeployment {\n    // Deploy Source asset(s) to /deployment/libs/<job> location.\n    return new BucketDeployment(this.scope, deploymentId, {\n      sources: additionalFilesSources,\n      destinationBucket: bucket,\n      destinationKeyPrefix: deploymentPath,\n      role: deploymentRole,\n      extract: extract,\n    });\n  }\n\n  private addAdditionalScripts(\n    jobName: string,\n    jobConfig: JobConfig,\n    bucket: IBucket,\n    deploymentRole: IRole,\n    defaultArguments: ConfigurationElement,\n  ) {\n    if (jobConfig.additionalScripts) {\n      /**\n       * Group all files at parent directory level. This will allow creating zip lib assests at various directory levels\n       * ex. '/main/script1.py' , '/util/script2.py' , '/util/script3.py' will create 2 zip files representing 'main' and 'utils'\n       *  */\n      const directoryToFile: { [filePath: string]: string[] } = {};\n      jobConfig.additionalScripts.forEach(fileLocation => {\n        const filePath = path.dirname(fileLocation.trim());\n        if (filePath in directoryToFile) {\n          directoryToFile[filePath].push(`!${path.basename(fileLocation.trim())}`);\n        } else {\n          directoryToFile[filePath] = [`!${path.basename(fileLocation.trim())}`];\n        }\n      });\n\n      // Create Source asset for each directory\n      const additionalFilesSources = Object.entries(directoryToFile).map(([filePath, fileNames]) => {\n        return Source.asset(filePath, { exclude: ['**', ...fileNames] });\n      });\n      const deploymentPath = `deployment/libs/${jobName}`;\n      const additionalFileDeployment = this.deployAdditionalFiles(\n        additionalFilesSources,\n        bucket,\n        deploymentRole,\n        `job-deployment-${jobName}-additional-script`,\n        deploymentPath,\n        false, // Glue expects zip of additional scripts, hence disabling the extraction\n      );\n\n      // Extract zip name(s) for each source and create comma separated list of s3 locations\n      const libraryZipNames: string[] = [];\n\n      for (let i = 0; i < additionalFilesSources.length; i++) {\n        const libName = Fn.select(i, additionalFileDeployment.objectKeys); // Extract file name of zip containing additional scripts\n        libraryZipNames.push(`s3://${additionalFileDeployment.deployedBucket.bucketName}/${deploymentPath}/${libName}`);\n      }\n\n      // Add comma separated list of zip file names to default arguments.\n      if (defaultArguments['--extra-py-files']) {\n        defaultArguments['--extra-py-files'] += ',' + libraryZipNames.join(',');\n      } else {\n        defaultArguments['--extra-py-files'] = libraryZipNames.join(',');\n      }\n    }\n  }\n\n  private addAdditionalJars(\n    jobName: string,\n    jobConfig: JobConfig,\n    bucket: IBucket,\n    deploymentRole: IRole,\n    defaultArguments: ConfigurationElement,\n  ) {\n    if (jobConfig.additionalJars) {\n      // Create Source asset for each directory\n      const additionalFilesSources = jobConfig.additionalJars.map(fullFileName => {\n        const filePath = path.dirname(fullFileName.trim());\n        const fileName = path.basename(fullFileName.trim());\n        return Source.asset(filePath, { exclude: ['**', `!${fileName}`] });\n      });\n      const deploymentPath = `deployment/libs/${jobName}`;\n\n      const additionalFileDeployment = this.deployAdditionalFiles(\n        additionalFilesSources,\n        bucket,\n        deploymentRole,\n        `job-deployment-${jobName}-additional-jar`,\n        deploymentPath,\n        true,\n      );\n\n      const extraJarNames = jobConfig.additionalJars.map(fullFileName => {\n        const fileName = path.basename(fullFileName.trim());\n        return `s3://${additionalFileDeployment.deployedBucket.bucketName}/${deploymentPath}/${fileName}`;\n      });\n\n      // Add comma separated list of zip file names to default arguments.\n      if (defaultArguments['--extra-jars']) {\n        defaultArguments['--extra-jars'] += ',' + extraJarNames.join(',');\n      } else {\n        defaultArguments['--extra-jars'] = extraJarNames.join(',');\n      }\n    }\n  }\n\n  private addAdditionalFiles(\n    jobName: string,\n    jobConfig: JobConfig,\n    bucket: IBucket,\n    deploymentRole: IRole,\n    defaultArguments: ConfigurationElement,\n  ) {\n    if (jobConfig.additionalFiles) {\n      // Create Source asset for each directory\n      const additionalFilesSources = jobConfig.additionalFiles.map(fullFileName => {\n        const filePath = path.dirname(fullFileName.trim());\n        const fileName = path.basename(fullFileName.trim());\n        return Source.asset(filePath, { exclude: ['**', `!${fileName}`] });\n      });\n      const deploymentPath = `deployment/files/${jobName}`;\n      const additionalFileDeployment = this.deployAdditionalFiles(\n        additionalFilesSources,\n        bucket,\n        deploymentRole,\n        `job-deployment-${jobName}-additional-file`,\n        deploymentPath,\n        true,\n      );\n      const extraFileNames = jobConfig.additionalFiles.map(fullFileName => {\n        const fileName = path.basename(fullFileName.trim());\n        return `s3://${additionalFileDeployment.deployedBucket.bucketName}/${deploymentPath}/${fileName}`;\n      });\n\n      // Add comma separated list of zip file names to default arguments.\n      if (defaultArguments['--extra-files']) {\n        defaultArguments['--extra-files'] += ',' + extraFileNames.join(',');\n      } else {\n        defaultArguments['--extra-files'] = extraFileNames.join(',');\n      }\n    }\n  }\n\n  private resolveAssetPath(jobName: string, location: string): string {\n    if (location.startsWith('asset:')) {\n      const assetName = location.substring('asset:'.length);\n      if (!this.props.assetBasePath) {\n        throw new Error(\n          `Job '${jobName}' uses asset: prefix but no assetBasePath is configured. ` +\n            `Set assetBasePath in the construct props.`,\n        );\n      }\n      return path.join(this.props.assetBasePath, assetName);\n    }\n    return location;\n  }\n\n  private createJob(jobName: string, jobConfig: JobConfig, deploymentRole: IRole, bucket: IBucket) {\n    const defaultArguments = jobConfig.defaultArguments ? jobConfig.defaultArguments : {};\n    let scriptLocation = this.resolveAssetPath(jobName, jobConfig.command.scriptLocation.trim());\n\n    const scriptPath = path.dirname(scriptLocation);\n    const scriptName = path.basename(scriptLocation);\n    const scriptSource = Source.asset(scriptPath, { exclude: ['**', `!${scriptName}`] });\n\n    const scriptDeploymentPath = `deployment/jobs/${jobName}`;\n    const scriptDeployment = new BucketDeployment(this.scope, `job-deployment-${jobName}`, {\n      sources: [scriptSource],\n      destinationBucket: bucket,\n      destinationKeyPrefix: scriptDeploymentPath,\n      role: deploymentRole,\n      extract: true,\n    });\n\n    // Resolve asset: prefixes in additionalScripts before processing\n    if (jobConfig.additionalScripts) {\n      jobConfig = {\n        ...jobConfig,\n        additionalScripts: jobConfig.additionalScripts.map(s => this.resolveAssetPath(jobName, s.trim())),\n      };\n    }\n\n    this.addAdditionalScripts(jobName, jobConfig, bucket, deploymentRole, defaultArguments);\n    this.addAdditionalJars(jobName, jobConfig, bucket, deploymentRole, defaultArguments);\n    this.addAdditionalFiles(jobName, jobConfig, bucket, deploymentRole, defaultArguments);\n    MdaaNagSuppressions.addCodeResourceSuppressions(\n      this.scope,\n      [\n        { id: 'AwsSolutions-L1', reason: 'Function is used only as custom resource during CDK deployment.' },\n        {\n          id: 'NIST.800.53.R5-LambdaConcurrency',\n          reason: 'Function is used only as custom resource during CDK deployment.',\n        },\n        {\n          id: 'NIST.800.53.R5-LambdaInsideVPC',\n          reason: 'Function is used only as custom resource during CDK deployment and interacts only with S3.',\n        },\n        {\n          id: 'NIST.800.53.R5-LambdaDLQ',\n          reason:\n            'Function is used only as custom resource during CDK deployment. Errors will be handled by CloudFormation.',\n        },\n        {\n          id: 'HIPAA.Security-LambdaConcurrency',\n          reason: 'Function is used only as custom resource during CDK deployment.',\n        },\n        {\n          id: 'PCI.DSS.321-LambdaConcurrency',\n          reason: 'Function is used only as custom resource during CDK deployment.',\n        },\n        {\n          id: 'HIPAA.Security-LambdaInsideVPC',\n          reason: 'Function is used only as custom resource during CDK deployment and interacts only with S3.',\n        },\n        {\n          id: 'PCI.DSS.321-LambdaInsideVPC',\n          reason: 'Function is used only as custom resource during CDK deployment and interacts only with S3.',\n        },\n        {\n          id: 'HIPAA.Security-LambdaDLQ',\n          reason:\n            'Function is used only as custom resource during CDK deployment. Errors will be handled by CloudFormation.',\n        },\n        {\n          id: 'PCI.DSS.321-LambdaDLQ',\n          reason:\n            'Function is used only as custom resource during CDK deployment. Errors will be handled by CloudFormation.',\n        },\n      ],\n      true,\n    );\n    // Connections will require an array of references where they are defined\n    let connectionsConfigured: ConfigurationElement | undefined;\n    if (jobConfig.connections) {\n      connectionsConfigured = {\n        connections: jobConfig.connections,\n      };\n    }\n\n    defaultArguments['--TempDir'] = `s3://${this.props.bucketName}/temp/jobs/${jobName}`;\n\n    // add continuous logging unless explicitly disabled\n    if (jobConfig.continuousLogging) {\n      const logGroupName = jobName;\n      const logGroupNamePathPrefix = '/aws/glue';\n      defaultArguments['--continuous-log-logGroup'] = this.createLogGroup(\n        logGroupNamePathPrefix,\n        logGroupName,\n        jobConfig.continuousLogging,\n      );\n    } else {\n      console.log(`Continuous logging not enabled for job: ${jobName}`);\n    }\n\n    const inputParams = defaultArguments['--input_params'];\n    if (inputParams) {\n      defaultArguments['--input_params'] = JSON.stringify(inputParams);\n    }\n    if (!this.props.securityConfigurationName) {\n      throw new Error('Security configuration name is required for job monitoring event rule');\n    }\n    const job = new MdaaCfnJob(this.scope, `${jobName}-job`, {\n      command: {\n        name: jobConfig.command.name,\n        pythonVersion: jobConfig.command.pythonVersion,\n        scriptLocation: `s3://${scriptDeployment.deployedBucket.bucketName}/${scriptDeploymentPath}/${scriptName}`,\n      },\n      role: jobConfig.executionRoleArn,\n      allocatedCapacity: jobConfig.allocatedCapacity,\n      connections: connectionsConfigured,\n      defaultArguments: defaultArguments,\n      description: jobConfig.description,\n      executionProperty: jobConfig.executionProperty,\n      glueVersion: jobConfig.glueVersion,\n      maxCapacity: jobConfig.maxCapacity,\n      maxRetries: jobConfig.maxRetries,\n      name: jobName,\n      notificationProperty: jobConfig.notificationProperty,\n      numberOfWorkers: jobConfig.numberOfWorkers,\n      securityConfiguration: this.props.securityConfigurationName,\n      timeout: jobConfig.timeout,\n      workerType: jobConfig.workerType,\n      naming: this.props.naming,\n    });\n\n    if (job.name && this.props.projectName) {\n      DataOpsProjectUtils.createProjectSSMParam(\n        this.scope,\n        this.props.naming,\n        this.props.projectName,\n        `job/name/${jobName}`,\n        job.name,\n      );\n\n      const eventRule = this.createJobMonitoringEventRule(`${jobName}-monitor`, [job.name]);\n      if (!this.props.notificationTopicArn) {\n        throw new Error('Notification topic ARN is required for job monitoring event rule');\n      }\n      eventRule.addTarget(\n        new SnsTopic(MdaaSnsTopic.fromTopicArn(this.scope, `${jobName}-topic`, this.props.notificationTopicArn)),\n      );\n    }\n  }\n\n  private createJobMonitoringEventRule(ruleName: string, jobNames: string[]): Rule {\n    return EventBridgeHelper.createGlueMonitoringEventRule(\n      this.scope,\n      this.props.naming,\n      ruleName,\n      'Workflow Job failure events',\n      {\n        jobName: jobNames,\n        state: ['FAILED', 'TIMEOUT', 'STOPPED'],\n      },\n    );\n  }\n\n  private createLogGroup(logGroupNamePathPrefix: string, logGroupName: string, loggingConfig: LoggingConfig): string {\n    let logGroupRetentionDays: RetentionDays;\n\n    if (loggingConfig.logGroupRetentionDays != 0) {\n      logGroupRetentionDays = loggingConfig.logGroupRetentionDays;\n    } else {\n      logGroupRetentionDays = RetentionDays.INFINITE;\n    }\n\n    const logProps = {\n      naming: this.props.naming,\n      logGroupName: logGroupName,\n      logGroupNamePathPrefix: logGroupNamePathPrefix,\n      encryptionKey: this.kmsKey,\n      retention: logGroupRetentionDays,\n    };\n    new MdaaLogGroup(this, logGroupName, logProps);\n    // `updateProps` always returns a prop with a logGroupName\n    return updateProps(logProps).logGroupName!;\n  }\n}\n"]}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
# Don't include original .ts files when doing `npm pack`
|
|
2
|
+
*.ts
|
|
3
|
+
*.tsbuildinfo
|
|
4
|
+
.eslintrc.js
|
|
5
|
+
tsconfig.json
|
|
6
|
+
typedoc.json
|
|
7
|
+
|
|
8
|
+
# dev-related files
|
|
9
|
+
coverage
|
|
10
|
+
test/
|
|
11
|
+
|
|
12
|
+
# pack output
|
|
13
|
+
*.tgz
|
|
14
|
+
|
|
15
|
+
dist
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
#Exclude coverage reports for bundled deps
|
|
19
|
+
node_modules/**/coverage/*
|
|
20
|
+
# Exclude typescript source and config
|
|
21
|
+
#*.ts
|
|
22
|
+
#tsconfig.json
|
|
23
|
+
|
|
24
|
+
# Include javascript files and typescript declarations
|
|
25
|
+
!*.js
|
|
26
|
+
!*.d.ts
|
|
27
|
+
|
|
28
|
+
# Exclude jsii outdir
|
|
29
|
+
jsii-dist
|
|
30
|
+
|
|
31
|
+
# Include .jsii and .jsii.gz
|
|
32
|
+
!.jsii
|
|
33
|
+
!.jsii.gz
|
|
34
|
+
jest.config.js
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
/*!
|
|
2
|
+
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
3
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
*/
|
|
5
|
+
import { Construct } from 'constructs';
|
|
6
|
+
import { IMdaaResourceNaming } from '@aws-mdaa/naming';
|
|
7
|
+
import { IMdaaConfigValueTransformer, TransformResult } from './transformer';
|
|
8
|
+
export interface MdaaConfigBlueprintRefValueTransformerProps {
|
|
9
|
+
readonly naming: IMdaaResourceNaming;
|
|
10
|
+
readonly scope: Construct;
|
|
11
|
+
}
|
|
12
|
+
export declare class MdaaConfigBlueprintRefValueTransformer implements IMdaaConfigValueTransformer {
|
|
13
|
+
protected props: MdaaConfigBlueprintRefValueTransformerProps;
|
|
14
|
+
constructor(props: MdaaConfigBlueprintRefValueTransformerProps);
|
|
15
|
+
transformValue(value: string): TransformResult;
|
|
16
|
+
protected parseBlueprintRef(value: string, refMatch: string[]): string | number | object;
|
|
17
|
+
private substituteBlueprintReferences;
|
|
18
|
+
private convertToString;
|
|
19
|
+
private resolveBlueprintReference;
|
|
20
|
+
}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/*!
|
|
3
|
+
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
|
+
exports.MdaaConfigBlueprintRefValueTransformer = void 0;
|
|
8
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
9
|
+
const XRegExp = require("xregexp");
|
|
10
|
+
class MdaaConfigBlueprintRefValueTransformer {
|
|
11
|
+
constructor(props) {
|
|
12
|
+
this.props = props;
|
|
13
|
+
}
|
|
14
|
+
transformValue(value) {
|
|
15
|
+
const refMatch = XRegExp.matchRecursive(value, '{{', '}}', 'g', {
|
|
16
|
+
unbalanced: 'skip',
|
|
17
|
+
});
|
|
18
|
+
if (refMatch.length > 0) {
|
|
19
|
+
return this.parseBlueprintRef(value, refMatch);
|
|
20
|
+
}
|
|
21
|
+
else {
|
|
22
|
+
return value;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
parseBlueprintRef(value, refMatch) {
|
|
26
|
+
// Substitute all references in the string
|
|
27
|
+
return this.substituteBlueprintReferences(value, refMatch);
|
|
28
|
+
}
|
|
29
|
+
substituteBlueprintReferences(value, refMatch) {
|
|
30
|
+
return refMatch.reduce((result, ref) => {
|
|
31
|
+
const refInner = this.transformValue(ref).toString();
|
|
32
|
+
const resolvedValue = this.resolveBlueprintReference(refInner);
|
|
33
|
+
if (resolvedValue === undefined) {
|
|
34
|
+
return result;
|
|
35
|
+
}
|
|
36
|
+
const stringValue = this.convertToString(resolvedValue);
|
|
37
|
+
return result.replace(`{{${ref}}}`, stringValue);
|
|
38
|
+
}, value);
|
|
39
|
+
}
|
|
40
|
+
convertToString(value) {
|
|
41
|
+
if (typeof value === 'string') {
|
|
42
|
+
return value;
|
|
43
|
+
}
|
|
44
|
+
if (typeof value === 'number' || typeof value === 'boolean') {
|
|
45
|
+
return String(value);
|
|
46
|
+
}
|
|
47
|
+
// For objects, use JSON.stringify as a safe fallback
|
|
48
|
+
return JSON.stringify(value);
|
|
49
|
+
}
|
|
50
|
+
resolveBlueprintReference(refInner) {
|
|
51
|
+
if (refInner.startsWith('blueprint:')) {
|
|
52
|
+
if (!this.props.scope) {
|
|
53
|
+
throw new Error('Unable to resolve blueprint params outside of a Construct');
|
|
54
|
+
}
|
|
55
|
+
if (!this.props.naming) {
|
|
56
|
+
throw new Error('Unable to resolve blueprint params without a naming implementation');
|
|
57
|
+
}
|
|
58
|
+
const blueprintPath = refInner.replace(/^blueprint:/, '');
|
|
59
|
+
const ssmOrgPath = this.props.naming?.ssmOrgPath('', false);
|
|
60
|
+
const existingDzProjectParam = this.props.scope?.node.tryFindChild('datazoneEnvironmentProjectId');
|
|
61
|
+
const dzProjectParam = existingDzProjectParam ?? new aws_cdk_lib_1.CfnParameter(this.props.scope, 'datazoneEnvironmentProjectId');
|
|
62
|
+
const ssmPath = `{{resolve:ssm:${ssmOrgPath}${dzProjectParam.valueAsString}${blueprintPath}}}`;
|
|
63
|
+
console.log(`Resolving blueprint path: ${ssmPath}`);
|
|
64
|
+
return ssmPath;
|
|
65
|
+
}
|
|
66
|
+
return undefined;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
exports.MdaaConfigBlueprintRefValueTransformer = MdaaConfigBlueprintRefValueTransformer;
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmx1ZXByaW50LXZhbHVlLXRyYW5zZm9ybWVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiYmx1ZXByaW50LXZhbHVlLXRyYW5zZm9ybWVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7O0dBR0c7OztBQUVILDZDQUEyQztBQUkzQyxtQ0FBbUM7QUFRbkMsTUFBYSxzQ0FBc0M7SUFHakQsWUFBWSxLQUFrRDtRQUM1RCxJQUFJLENBQUMsS0FBSyxHQUFHLEtBQUssQ0FBQztJQUNyQixDQUFDO0lBRU0sY0FBYyxDQUFDLEtBQWE7UUFDakMsTUFBTSxRQUFRLEdBQUcsT0FBTyxDQUFDLGNBQWMsQ0FBQyxLQUFLLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxHQUFHLEVBQUU7WUFDOUQsVUFBVSxFQUFFLE1BQU07U0FDbkIsQ0FBQyxDQUFDO1FBQ0gsSUFBSSxRQUFRLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRSxDQUFDO1lBQ3hCLE9BQU8sSUFBSSxDQUFDLGlCQUFpQixDQUFDLEtBQUssRUFBRSxRQUFRLENBQW9CLENBQUM7UUFDcEUsQ0FBQzthQUFNLENBQUM7WUFDTixPQUFPLEtBQUssQ0FBQztRQUNmLENBQUM7SUFDSCxDQUFDO0lBRVMsaUJBQWlCLENBQUMsS0FBYSxFQUFFLFFBQWtCO1FBQzNELDBDQUEwQztRQUMxQyxPQUFPLElBQUksQ0FBQyw2QkFBNkIsQ0FBQyxLQUFLLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDN0QsQ0FBQztJQUVPLDZCQUE2QixDQUFDLEtBQWEsRUFBRSxRQUFrQjtRQUNyRSxPQUFPLFFBQVEsQ0FBQyxNQUFNLENBQUMsQ0FBQyxNQUFNLEVBQUUsR0FBRyxFQUFFLEVBQUU7WUFDckMsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLGNBQWMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxRQUFRLEVBQUUsQ0FBQztZQUNyRCxNQUFNLGFBQWEsR0FBRyxJQUFJLENBQUMseUJBQXlCLENBQUMsUUFBUSxDQUFDLENBQUM7WUFFL0QsSUFBSSxhQUFhLEtBQUssU0FBUyxFQUFFLENBQUM7Z0JBQ2hDLE9BQU8sTUFBTSxDQUFDO1lBQ2hCLENBQUM7WUFFRCxNQUFNLFdBQVcsR0FBRyxJQUFJLENBQUMsZUFBZSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1lBQ3hELE9BQU8sTUFBTSxDQUFDLE9BQU8sQ0FBQyxLQUFLLEdBQUcsSUFBSSxFQUFFLFdBQVcsQ0FBQyxDQUFDO1FBQ25ELENBQUMsRUFBRSxLQUFLLENBQUMsQ0FBQztJQUNaLENBQUM7SUFFTyxlQUFlLENBQUMsS0FBYztRQUNwQyxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzlCLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQztRQUNELElBQUksT0FBTyxLQUFLLEtBQUssUUFBUSxJQUFJLE9BQU8sS0FBSyxLQUFLLFNBQVMsRUFBRSxDQUFDO1lBQzVELE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3ZCLENBQUM7UUFDRCxxREFBcUQ7UUFDckQsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQy9CLENBQUM7SUFFTyx5QkFBeUIsQ0FBQyxRQUFnQjtRQUNoRCxJQUFJLFFBQVEsQ0FBQyxVQUFVLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQztZQUN0QyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLEVBQUUsQ0FBQztnQkFDdEIsTUFBTSxJQUFJLEtBQUssQ0FBQywyREFBMkQsQ0FBQyxDQUFDO1lBQy9FLENBQUM7WUFDRCxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLEVBQUUsQ0FBQztnQkFDdkIsTUFBTSxJQUFJLEtBQUssQ0FBQyxvRUFBb0UsQ0FBQyxDQUFDO1lBQ3hGLENBQUM7WUFDRCxNQUFNLGFBQWEsR0FBRyxRQUFRLENBQUMsT0FBTyxDQUFDLGFBQWEsRUFBRSxFQUFFLENBQUMsQ0FBQztZQUMxRCxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sRUFBRSxVQUFVLENBQUMsRUFBRSxFQUFFLEtBQUssQ0FBQyxDQUFDO1lBQzVELE1BQU0sc0JBQXNCLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLFlBQVksQ0FDaEUsOEJBQThCLENBQ2YsQ0FBQztZQUNsQixNQUFNLGNBQWMsR0FDbEIsc0JBQXNCLElBQUksSUFBSSwwQkFBWSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxFQUFFLDhCQUE4QixDQUFDLENBQUM7WUFDL0YsTUFBTSxPQUFPLEdBQUcsaUJBQWlCLFVBQVUsR0FBRyxjQUFjLENBQUMsYUFBYSxHQUFHLGFBQWEsSUFBSSxDQUFDO1lBQy9GLE9BQU8sQ0FBQyxHQUFHLENBQUMsNkJBQTZCLE9BQU8sRUFBRSxDQUFDLENBQUM7WUFDcEQsT0FBTyxPQUFPLENBQUM7UUFDakIsQ0FBQztRQUVELE9BQU8sU0FBUyxDQUFDO0lBQ25CLENBQUM7Q0FDRjtBQXRFRCx3RkFzRUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiFcbiAqIENvcHlyaWdodCBBbWF6b24uY29tLCBJbmMuIG9yIGl0cyBhZmZpbGlhdGVzLiBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICogU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcbiAqL1xuXG5pbXBvcnQgeyBDZm5QYXJhbWV0ZXIgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbi8vIG5vc2VtZ3JlcFxuaW1wb3J0IHsgSU1kYWFSZXNvdXJjZU5hbWluZyB9IGZyb20gJ0Bhd3MtbWRhYS9uYW1pbmcnO1xuaW1wb3J0ICogYXMgWFJlZ0V4cCBmcm9tICd4cmVnZXhwJztcbmltcG9ydCB7IElNZGFhQ29uZmlnVmFsdWVUcmFuc2Zvcm1lciwgVHJhbnNmb3JtUmVzdWx0IH0gZnJvbSAnLi90cmFuc2Zvcm1lcic7XG5cbmV4cG9ydCBpbnRlcmZhY2UgTWRhYUNvbmZpZ0JsdWVwcmludFJlZlZhbHVlVHJhbnNmb3JtZXJQcm9wcyB7XG4gIHJlYWRvbmx5IG5hbWluZzogSU1kYWFSZXNvdXJjZU5hbWluZztcbiAgcmVhZG9ubHkgc2NvcGU6IENvbnN0cnVjdDtcbn1cblxuZXhwb3J0IGNsYXNzIE1kYWFDb25maWdCbHVlcHJpbnRSZWZWYWx1ZVRyYW5zZm9ybWVyIGltcGxlbWVudHMgSU1kYWFDb25maWdWYWx1ZVRyYW5zZm9ybWVyIHtcbiAgcHJvdGVjdGVkIHByb3BzOiBNZGFhQ29uZmlnQmx1ZXByaW50UmVmVmFsdWVUcmFuc2Zvcm1lclByb3BzO1xuXG4gIGNvbnN0cnVjdG9yKHByb3BzOiBNZGFhQ29uZmlnQmx1ZXByaW50UmVmVmFsdWVUcmFuc2Zvcm1lclByb3BzKSB7XG4gICAgdGhpcy5wcm9wcyA9IHByb3BzO1xuICB9XG5cbiAgcHVibGljIHRyYW5zZm9ybVZhbHVlKHZhbHVlOiBzdHJpbmcpOiBUcmFuc2Zvcm1SZXN1bHQge1xuICAgIGNvbnN0IHJlZk1hdGNoID0gWFJlZ0V4cC5tYXRjaFJlY3Vyc2l2ZSh2YWx1ZSwgJ3t7JywgJ319JywgJ2cnLCB7XG4gICAgICB1bmJhbGFuY2VkOiAnc2tpcCcsXG4gICAgfSk7XG4gICAgaWYgKHJlZk1hdGNoLmxlbmd0aCA+IDApIHtcbiAgICAgIHJldHVybiB0aGlzLnBhcnNlQmx1ZXByaW50UmVmKHZhbHVlLCByZWZNYXRjaCkgYXMgVHJhbnNmb3JtUmVzdWx0O1xuICAgIH0gZWxzZSB7XG4gICAgICByZXR1cm4gdmFsdWU7XG4gICAgfVxuICB9XG5cbiAgcHJvdGVjdGVkIHBhcnNlQmx1ZXByaW50UmVmKHZhbHVlOiBzdHJpbmcsIHJlZk1hdGNoOiBzdHJpbmdbXSk6IHN0cmluZyB8IG51bWJlciB8IG9iamVjdCB7XG4gICAgLy8gU3Vic3RpdHV0ZSBhbGwgcmVmZXJlbmNlcyBpbiB0aGUgc3RyaW5nXG4gICAgcmV0dXJuIHRoaXMuc3Vic3RpdHV0ZUJsdWVwcmludFJlZmVyZW5jZXModmFsdWUsIHJlZk1hdGNoKTtcbiAgfVxuXG4gIHByaXZhdGUgc3Vic3RpdHV0ZUJsdWVwcmludFJlZmVyZW5jZXModmFsdWU6IHN0cmluZywgcmVmTWF0Y2g6IHN0cmluZ1tdKTogc3RyaW5nIHtcbiAgICByZXR1cm4gcmVmTWF0Y2gucmVkdWNlKChyZXN1bHQsIHJlZikgPT4ge1xuICAgICAgY29uc3QgcmVmSW5uZXIgPSB0aGlzLnRyYW5zZm9ybVZhbHVlKHJlZikudG9TdHJpbmcoKTtcbiAgICAgIGNvbnN0IHJlc29sdmVkVmFsdWUgPSB0aGlzLnJlc29sdmVCbHVlcHJpbnRSZWZlcmVuY2UocmVmSW5uZXIpO1xuXG4gICAgICBpZiAocmVzb2x2ZWRWYWx1ZSA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHN0cmluZ1ZhbHVlID0gdGhpcy5jb252ZXJ0VG9TdHJpbmcocmVzb2x2ZWRWYWx1ZSk7XG4gICAgICByZXR1cm4gcmVzdWx0LnJlcGxhY2UoYHt7JHtyZWZ9fX1gLCBzdHJpbmdWYWx1ZSk7XG4gICAgfSwgdmFsdWUpO1xuICB9XG5cbiAgcHJpdmF0ZSBjb252ZXJ0VG9TdHJpbmcodmFsdWU6IHVua25vd24pOiBzdHJpbmcge1xuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnKSB7XG4gICAgICByZXR1cm4gdmFsdWU7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09ICdudW1iZXInIHx8IHR5cGVvZiB2YWx1ZSA9PT0gJ2Jvb2xlYW4nKSB7XG4gICAgICByZXR1cm4gU3RyaW5nKHZhbHVlKTtcbiAgICB9XG4gICAgLy8gRm9yIG9iamVjdHMsIHVzZSBKU09OLnN0cmluZ2lmeSBhcyBhIHNhZmUgZmFsbGJhY2tcbiAgICByZXR1cm4gSlNPTi5zdHJpbmdpZnkodmFsdWUpO1xuICB9XG5cbiAgcHJpdmF0ZSByZXNvbHZlQmx1ZXByaW50UmVmZXJlbmNlKHJlZklubmVyOiBzdHJpbmcpIHtcbiAgICBpZiAocmVmSW5uZXIuc3RhcnRzV2l0aCgnYmx1ZXByaW50OicpKSB7XG4gICAgICBpZiAoIXRoaXMucHJvcHMuc2NvcGUpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdVbmFibGUgdG8gcmVzb2x2ZSBibHVlcHJpbnQgcGFyYW1zIG91dHNpZGUgb2YgYSBDb25zdHJ1Y3QnKTtcbiAgICAgIH1cbiAgICAgIGlmICghdGhpcy5wcm9wcy5uYW1pbmcpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdVbmFibGUgdG8gcmVzb2x2ZSBibHVlcHJpbnQgcGFyYW1zIHdpdGhvdXQgYSBuYW1pbmcgaW1wbGVtZW50YXRpb24nKTtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGJsdWVwcmludFBhdGggPSByZWZJbm5lci5yZXBsYWNlKC9eYmx1ZXByaW50Oi8sICcnKTtcbiAgICAgIGNvbnN0IHNzbU9yZ1BhdGggPSB0aGlzLnByb3BzLm5hbWluZz8uc3NtT3JnUGF0aCgnJywgZmFsc2UpO1xuICAgICAgY29uc3QgZXhpc3RpbmdEelByb2plY3RQYXJhbSA9IHRoaXMucHJvcHMuc2NvcGU/Lm5vZGUudHJ5RmluZENoaWxkKFxuICAgICAgICAnZGF0YXpvbmVFbnZpcm9ubWVudFByb2plY3RJZCcsXG4gICAgICApIGFzIENmblBhcmFtZXRlcjtcbiAgICAgIGNvbnN0IGR6UHJvamVjdFBhcmFtID1cbiAgICAgICAgZXhpc3RpbmdEelByb2plY3RQYXJhbSA/PyBuZXcgQ2ZuUGFyYW1ldGVyKHRoaXMucHJvcHMuc2NvcGUsICdkYXRhem9uZUVudmlyb25tZW50UHJvamVjdElkJyk7XG4gICAgICBjb25zdCBzc21QYXRoID0gYHt7cmVzb2x2ZTpzc206JHtzc21PcmdQYXRofSR7ZHpQcm9qZWN0UGFyYW0udmFsdWVBc1N0cmluZ30ke2JsdWVwcmludFBhdGh9fX1gO1xuICAgICAgY29uc29sZS5sb2coYFJlc29sdmluZyBibHVlcHJpbnQgcGF0aDogJHtzc21QYXRofWApO1xuICAgICAgcmV0dXJuIHNzbVBhdGg7XG4gICAgfVxuXG4gICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgfVxufVxuIl19
|