aws-ddk-core
Version:
The AWS DataOps Development Kit is an open source development framework for customers that build data workflows and modern data architecture on AWS.
223 lines • 33.1 kB
JavaScript
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.MWAAEnvironment = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const cdk = require("aws-cdk-lib");
const ec2 = require("aws-cdk-lib/aws-ec2");
const iam = require("aws-cdk-lib/aws-iam");
const mwaa = require("aws-cdk-lib/aws-mwaa");
const s3deploy = require("aws-cdk-lib/aws-s3-deployment");
const constructs_1 = require("constructs");
const s3_factory_1 = require("./s3-factory");
class MWAAEnvironment extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
this.dagProcessingLogs = props.dagProcessingLogs ?? "INFO";
this.schedulerLogsLevel = props.schedulerLogsLevel ?? "INFO";
this.taskLogsLevel = props.taskLogsLevel ?? "INFO";
this.workerLogsLevel = props.workerLogsLevel ?? "INFO";
this.webserverLogsLevel = props.webserverLogsLevel ?? "INFO";
this.dagS3Path = props.dagS3Path ?? "dags";
if (props.vpcId) {
this.vpc = ec2.Vpc.fromLookup(scope, `${props.name} VPC`, { vpcId: props.vpcId });
}
else if (props.vpcCidr) {
this.vpc = this.createVpc(scope, props.name, props.vpcCidr);
}
else {
throw new Error("One of 'vpcId' or 'vpcCidr' must be provided");
}
const securityGroup = new ec2.SecurityGroup(scope, `${props.name} Security Group`, {
securityGroupName: `${props.name} Security Group`,
description: "Security group with a self-referencing inbound rule.",
vpc: this.vpc,
});
securityGroup.addIngressRule(securityGroup, ec2.Port.allTraffic(), "Self referencing rule");
if (props.s3Bucket) {
this.s3Bucket = props.s3Bucket;
}
else {
this.s3Bucket = s3_factory_1.S3Factory.bucket(this, `${props.name} Environment Bucket`, {
versioned: true,
});
}
if (props.dagFiles) {
var sources = [];
props.dagFiles.forEach((location) => {
sources.push(s3deploy.Source.asset(location));
});
new s3deploy.BucketDeployment(this, `${props.name} Deploy Dag Files`, {
sources: sources,
destinationBucket: this.s3Bucket,
destinationKeyPrefix: this.dagS3Path,
});
}
if (props.pluginFile) {
if (props.pluginsS3Path) {
this.pluginFile = new s3deploy.BucketDeployment(this, `${props.name} Deploy Plugin File`, {
sources: [s3deploy.Source.asset(props.pluginFile)],
destinationBucket: this.s3Bucket,
destinationKeyPrefix: props.pluginsS3Path,
});
}
else {
throw new Error("'pluginsS3Path' must be specified if a 'pluginFile' is specified.");
}
}
if (props.requirementsFile) {
if (props.requirementsS3Path) {
this.pluginFile = new s3deploy.BucketDeployment(this, `${props.name} Deploy Requirements File`, {
sources: [
s3deploy.Source.asset(props.requirementsFile.split("/").slice(0, -1).join("/"), {
exclude: ["**", `!${props.requirementsFile.split("/")[-1]}`],
}),
],
destinationBucket: this.s3Bucket,
destinationKeyPrefix: props.requirementsS3Path,
});
}
else {
throw new Error("'requirementsS3Path' must be specified if a 'requirementsFile' is specified.");
}
}
const mwaaExecutionRole = new iam.Role(scope, `${props.name} MWAA Execution Role`, {
assumedBy: new iam.CompositePrincipal(new iam.ServicePrincipal("airflow.amazonaws.com"), new iam.ServicePrincipal("airflow-env.amazonaws.com")),
path: "/service-role/",
});
mwaaExecutionRole.addManagedPolicy(new iam.ManagedPolicy(this, `${props.name} MWAA Execution Policy`, {
statements: [
new iam.PolicyStatement({
actions: ["airflow:PublishMetrics"],
resources: [
`arn:aws:airflow:${cdk.Stack.of(scope).region}:${cdk.Stack.of(scope).account}:environment/${props.name}`,
],
}),
new iam.PolicyStatement({
effect: iam.Effect.DENY,
actions: ["s3:ListAllMyBuckets"],
resources: [this.s3Bucket.bucketArn, `${this.s3Bucket.bucketArn}/*`],
}),
new iam.PolicyStatement({
actions: ["s3:GetObject*", "s3:GetBucket*", "s3:List*"],
resources: [this.s3Bucket.bucketArn, `${this.s3Bucket.bucketArn}/*`],
}),
new iam.PolicyStatement({
actions: ["logs:DescribeLogGroups"],
resources: ["*"],
}),
new iam.PolicyStatement({
actions: [
"logs:CreateLogStream",
"logs:CreateLogGroup",
"logs:PutLogEvents",
"logs:GetLogEvents",
"logs:GetLogRecord",
"logs:GetLogGroupFields",
"logs:GetQueryResults",
"logs:DescribeLogGroups",
],
resources: [
`arn:aws:logs:${cdk.Stack.of(scope).region}:${cdk.Stack.of(scope).account}:log-group:airflow-${cdk.Stack.of(scope).stackName}*`,
],
}),
new iam.PolicyStatement({
actions: ["cloudwatch:PutMetricData"],
resources: ["*"],
}),
new iam.PolicyStatement({
actions: [
"sqs:ChangeMessageVisibility",
"sqs:DeleteMessage",
"sqs:GetQueueAttributes",
"sqs:GetQueueUrl",
"sqs:ReceiveMessage",
"sqs:SendMessage",
],
resources: [`arn:aws:sqs:${cdk.Stack.of(scope).region}:*:airflow-celery-*`],
}),
new iam.PolicyStatement({
actions: ["kms:Decrypt", "kms:DescribeKey", "kms:GenerateDataKey*", "kms:Encrypt"],
notResources: [`arn:aws:kms:*:${cdk.Stack.of(scope).account}:key/*`],
conditions: {
StringLike: {
"kms:ViaService": [`sqs.${cdk.Stack.of(scope).region}.amazonaws.com`],
},
},
}),
],
}));
if (props.additionalPolicyStatements) {
props.additionalPolicyStatements.forEach((statement) => {
mwaaExecutionRole.addToPolicy(statement);
});
}
this.mwaaEnvironment = new mwaa.CfnEnvironment(this, `${props.name} MWAA Environment`, {
sourceBucketArn: this.s3Bucket.bucketArn,
executionRoleArn: mwaaExecutionRole.roleArn,
dagS3Path: this.dagS3Path,
networkConfiguration: {
securityGroupIds: [securityGroup.securityGroupId],
subnetIds: [
this.vpc.selectSubnets({ subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS }).subnetIds[0],
this.vpc.selectSubnets({ subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS }).subnetIds[1],
],
},
webserverAccessMode: "PUBLIC_ONLY",
loggingConfiguration: {
dagProcessingLogs: {
enabled: true,
logLevel: this.dagProcessingLogs,
},
schedulerLogs: {
enabled: true,
logLevel: this.schedulerLogsLevel,
},
taskLogs: {
enabled: true,
logLevel: this.taskLogsLevel,
},
webserverLogs: {
enabled: true,
logLevel: this.webserverLogsLevel,
},
workerLogs: {
enabled: false,
logLevel: this.workerLogsLevel,
},
},
...props,
});
}
createVpc(scope, environmentName, vpcCidr) {
const resourceName = `${environmentName}-MWAA`;
const vpcCIDRMask = +vpcCidr.split("/")[1];
if (vpcCIDRMask > 20 || vpcCIDRMask < 16) {
throw new Error("Vpc Cidr Range must of size >=16 and <=20");
}
const subnetCIDRMask = vpcCIDRMask + 4;
const vpc = new ec2.Vpc(scope, `${resourceName} Vpc`, {
ipAddresses: ec2.IpAddresses.cidr(vpcCidr),
enableDnsSupport: true,
enableDnsHostnames: true,
vpcName: resourceName,
subnetConfiguration: [
{
name: "Public",
subnetType: ec2.SubnetType.PUBLIC,
cidrMask: subnetCIDRMask,
},
{
name: "Private",
subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS,
cidrMask: subnetCIDRMask,
},
],
});
return vpc;
}
}
exports.MWAAEnvironment = MWAAEnvironment;
_a = JSII_RTTI_SYMBOL_1;
MWAAEnvironment[_a] = { fqn: "aws-ddk-core.MWAAEnvironment", version: "1.4.1" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mwaa-environment.js","sourceRoot":"","sources":["../../src/core/mwaa-environment.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\";\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as mwaa from \"aws-cdk-lib/aws-mwaa\";\nimport * as s3 from \"aws-cdk-lib/aws-s3\";\nimport * as s3deploy from \"aws-cdk-lib/aws-s3-deployment\";\nimport { Construct } from \"constructs\";\n\nimport { S3Factory } from \"./s3-factory\";\n\nexport interface MWAAEnvironmentProps extends mwaa.CfnEnvironmentProps {\n  /**\n   * Log level for DagProcessing\n   */\n  readonly dagProcessingLogs?: string;\n  /**\n   * Log level for SchedulerLogs\n   */\n  readonly schedulerLogsLevel?: string;\n  /**\n   * Log level for TaskLogs\n   */\n  readonly taskLogsLevel?: string;\n  /**\n   * Log level for WorkerLogs\n   */\n  readonly workerLogsLevel?: string;\n  /**\n   * Log level for WebserverLogs\n   */\n  readonly webserverLogsLevel?: string;\n  /**\n   * Existing vpc id\n   */\n  readonly vpcId?: string;\n  /**\n   * The IP range (CIDR notation) for this VPC\n   */\n  readonly vpcCidr?: string;\n  /**\n   * S3 Bucket\n   */\n  readonly s3Bucket?: s3.IBucket;\n  /**\n   * File(s) to be uploaded to dags location in s3 bucket.\n   */\n  readonly dagFiles?: string[];\n  /**\n   * Requirements file to be uploaded to plugin path in S3. 'requirementsS3Path' must be specified as well.\n   */\n  readonly requirementsFile?: string;\n  /**\n   * Plugin file to be uploaded to plugin path in S3. 'pluginsS3Path' must be specified as well.\n   */\n  readonly pluginFile?: string;\n  /**\n   * Additional policy statements to add to the airflow execution role\n   */\n  readonly additionalPolicyStatements?: iam.PolicyStatement[];\n}\n\nexport class MWAAEnvironment extends Construct {\n  readonly dagProcessingLogs: string;\n  readonly schedulerLogsLevel: string;\n  readonly taskLogsLevel: string;\n  readonly workerLogsLevel: string;\n  readonly webserverLogsLevel: string;\n  readonly vpc: ec2.IVpc;\n  readonly s3Bucket: s3.IBucket;\n  readonly mwaaEnvironment: mwaa.CfnEnvironment;\n  readonly dagS3Path: string;\n  readonly pluginFile?: s3deploy.BucketDeployment;\n\n  constructor(scope: Construct, id: string, props: MWAAEnvironmentProps) {\n    super(scope, id);\n\n    this.dagProcessingLogs = props.dagProcessingLogs ?? \"INFO\";\n    this.schedulerLogsLevel = props.schedulerLogsLevel ?? \"INFO\";\n    this.taskLogsLevel = props.taskLogsLevel ?? \"INFO\";\n    this.workerLogsLevel = props.workerLogsLevel ?? \"INFO\";\n    this.webserverLogsLevel = props.webserverLogsLevel ?? \"INFO\";\n    this.dagS3Path = props.dagS3Path ?? \"dags\";\n\n    if (props.vpcId) {\n      this.vpc = ec2.Vpc.fromLookup(scope, `${props.name} VPC`, { vpcId: props.vpcId });\n    } else if (props.vpcCidr) {\n      this.vpc = this.createVpc(scope, props.name, props.vpcCidr);\n    } else {\n      throw new Error(\"One of 'vpcId' or 'vpcCidr' must be provided\");\n    }\n\n    const securityGroup = new ec2.SecurityGroup(scope, `${props.name} Security Group`, {\n      securityGroupName: `${props.name} Security Group`,\n      description: \"Security group with a self-referencing inbound rule.\",\n      vpc: this.vpc,\n    });\n    securityGroup.addIngressRule(securityGroup, ec2.Port.allTraffic(), \"Self referencing rule\");\n\n    if (props.s3Bucket) {\n      this.s3Bucket = props.s3Bucket;\n    } else {\n      this.s3Bucket = S3Factory.bucket(this, `${props.name} Environment Bucket`, {\n        versioned: true,\n      });\n    }\n\n    if (props.dagFiles) {\n      var sources: s3deploy.ISource[] = [];\n      props.dagFiles.forEach((location) => {\n        sources.push(s3deploy.Source.asset(location));\n      });\n      new s3deploy.BucketDeployment(this, `${props.name} Deploy Dag Files`, {\n        sources: sources,\n        destinationBucket: this.s3Bucket,\n        destinationKeyPrefix: this.dagS3Path,\n      });\n    }\n\n    if (props.pluginFile) {\n      if (props.pluginsS3Path) {\n        this.pluginFile = new s3deploy.BucketDeployment(this, `${props.name} Deploy Plugin File`, {\n          sources: [s3deploy.Source.asset(props.pluginFile)],\n          destinationBucket: this.s3Bucket,\n          destinationKeyPrefix: props.pluginsS3Path,\n        });\n      } else {\n        throw new Error(\"'pluginsS3Path' must be specified if a 'pluginFile' is specified.\");\n      }\n    }\n\n    if (props.requirementsFile) {\n      if (props.requirementsS3Path) {\n        this.pluginFile = new s3deploy.BucketDeployment(this, `${props.name} Deploy Requirements File`, {\n          sources: [\n            s3deploy.Source.asset(props.requirementsFile.split(\"/\").slice(0, -1).join(\"/\"), {\n              exclude: [\"**\", `!${props.requirementsFile.split(\"/\")[-1]}`],\n            }),\n          ],\n          destinationBucket: this.s3Bucket,\n          destinationKeyPrefix: props.requirementsS3Path,\n        });\n      } else {\n        throw new Error(\"'requirementsS3Path' must be specified if a 'requirementsFile' is specified.\");\n      }\n    }\n\n    const mwaaExecutionRole = new iam.Role(scope, `${props.name} MWAA Execution Role`, {\n      assumedBy: new iam.CompositePrincipal(\n        new iam.ServicePrincipal(\"airflow.amazonaws.com\"),\n        new iam.ServicePrincipal(\"airflow-env.amazonaws.com\"),\n      ),\n      path: \"/service-role/\",\n    });\n    mwaaExecutionRole.addManagedPolicy(\n      new iam.ManagedPolicy(this, `${props.name} MWAA Execution Policy`, {\n        statements: [\n          new iam.PolicyStatement({\n            actions: [\"airflow:PublishMetrics\"],\n            resources: [\n              `arn:aws:airflow:${cdk.Stack.of(scope).region}:${cdk.Stack.of(scope).account}:environment/${props.name}`,\n            ],\n          }),\n          new iam.PolicyStatement({\n            effect: iam.Effect.DENY,\n            actions: [\"s3:ListAllMyBuckets\"],\n            resources: [this.s3Bucket.bucketArn, `${this.s3Bucket.bucketArn}/*`],\n          }),\n          new iam.PolicyStatement({\n            actions: [\"s3:GetObject*\", \"s3:GetBucket*\", \"s3:List*\"],\n            resources: [this.s3Bucket.bucketArn, `${this.s3Bucket.bucketArn}/*`],\n          }),\n          new iam.PolicyStatement({\n            actions: [\"logs:DescribeLogGroups\"],\n            resources: [\"*\"],\n          }),\n          new iam.PolicyStatement({\n            actions: [\n              \"logs:CreateLogStream\",\n              \"logs:CreateLogGroup\",\n              \"logs:PutLogEvents\",\n              \"logs:GetLogEvents\",\n              \"logs:GetLogRecord\",\n              \"logs:GetLogGroupFields\",\n              \"logs:GetQueryResults\",\n              \"logs:DescribeLogGroups\",\n            ],\n            resources: [\n              `arn:aws:logs:${cdk.Stack.of(scope).region}:${cdk.Stack.of(scope).account}:log-group:airflow-${\n                cdk.Stack.of(scope).stackName\n              }*`,\n            ],\n          }),\n          new iam.PolicyStatement({\n            actions: [\"cloudwatch:PutMetricData\"],\n            resources: [\"*\"],\n          }),\n          new iam.PolicyStatement({\n            actions: [\n              \"sqs:ChangeMessageVisibility\",\n              \"sqs:DeleteMessage\",\n              \"sqs:GetQueueAttributes\",\n              \"sqs:GetQueueUrl\",\n              \"sqs:ReceiveMessage\",\n              \"sqs:SendMessage\",\n            ],\n            resources: [`arn:aws:sqs:${cdk.Stack.of(scope).region}:*:airflow-celery-*`],\n          }),\n          new iam.PolicyStatement({\n            actions: [\"kms:Decrypt\", \"kms:DescribeKey\", \"kms:GenerateDataKey*\", \"kms:Encrypt\"],\n            notResources: [`arn:aws:kms:*:${cdk.Stack.of(scope).account}:key/*`],\n            conditions: {\n              StringLike: {\n                \"kms:ViaService\": [`sqs.${cdk.Stack.of(scope).region}.amazonaws.com`],\n              },\n            },\n          }),\n        ],\n      }),\n    );\n\n    if (props.additionalPolicyStatements) {\n      props.additionalPolicyStatements.forEach((statement) => {\n        mwaaExecutionRole.addToPolicy(statement);\n      });\n    }\n\n    this.mwaaEnvironment = new mwaa.CfnEnvironment(this, `${props.name} MWAA Environment`, {\n      sourceBucketArn: this.s3Bucket.bucketArn,\n      executionRoleArn: mwaaExecutionRole.roleArn,\n      dagS3Path: this.dagS3Path,\n      networkConfiguration: {\n        securityGroupIds: [securityGroup.securityGroupId],\n        subnetIds: [\n          this.vpc.selectSubnets({ subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS }).subnetIds[0],\n          this.vpc.selectSubnets({ subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS }).subnetIds[1],\n        ],\n      },\n      webserverAccessMode: \"PUBLIC_ONLY\",\n      loggingConfiguration: {\n        dagProcessingLogs: {\n          enabled: true,\n          logLevel: this.dagProcessingLogs,\n        },\n        schedulerLogs: {\n          enabled: true,\n          logLevel: this.schedulerLogsLevel,\n        },\n        taskLogs: {\n          enabled: true,\n          logLevel: this.taskLogsLevel,\n        },\n        webserverLogs: {\n          enabled: true,\n          logLevel: this.webserverLogsLevel,\n        },\n        workerLogs: {\n          enabled: false,\n          logLevel: this.workerLogsLevel,\n        },\n      },\n      ...props,\n    });\n  }\n  createVpc(scope: Construct, environmentName: string, vpcCidr: string): ec2.IVpc {\n    const resourceName = `${environmentName}-MWAA`;\n    const vpcCIDRMask = +vpcCidr.split(\"/\")[1];\n    if (vpcCIDRMask > 20 || vpcCIDRMask < 16) {\n      throw new Error(\"Vpc Cidr Range must of size >=16 and <=20\");\n    }\n    const subnetCIDRMask = vpcCIDRMask + 4;\n    const vpc = new ec2.Vpc(scope, `${resourceName} Vpc`, {\n      ipAddresses: ec2.IpAddresses.cidr(vpcCidr),\n      enableDnsSupport: true,\n      enableDnsHostnames: true,\n      vpcName: resourceName,\n      subnetConfiguration: [\n        {\n          name: \"Public\",\n          subnetType: ec2.SubnetType.PUBLIC,\n          cidrMask: subnetCIDRMask,\n        },\n        {\n          name: \"Private\",\n          subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS,\n          cidrMask: subnetCIDRMask,\n        },\n      ],\n    });\n\n    return vpc;\n  }\n}\n"]}
;