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.
129 lines • 17.6 kB
JavaScript
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.EMRServerlessCluster = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const ec2 = require("aws-cdk-lib/aws-ec2");
const emr = require("aws-cdk-lib/aws-emrserverless");
const iam = require("aws-cdk-lib/aws-iam");
const constructs_1 = require("constructs");
const s3_factory_1 = require("./s3-factory");
const utils_1 = require("./utils");
class EMRServerlessCluster extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
const name = props.name ?? "DDKEmrServerlessCluster";
if (props.vpcId) {
this.vpc = ec2.Vpc.fromLookup(scope, "VPC", { vpcId: props.vpcId });
}
else if (props.vpcCidr) {
this.vpc = this.createVpc(scope, name, props.vpcCidr);
}
else {
throw new Error("One of 'vpcId' or 'vpcCidr' must be provided");
}
if (this.vpc) {
this.securityGroup =
props.securityGroup ??
new ec2.SecurityGroup(scope, "EMR Serverless Security Group", {
securityGroupName: `EMR Serverless Security Group`,
description: "Security group with a self-referencing inbound rule.",
vpc: this.vpc,
});
this.securityGroup.addIngressRule(this.securityGroup, ec2.Port.allTraffic(), "Self referencing rule");
}
if (props.s3Bucket) {
this.s3Bucket = props.s3Bucket;
}
else {
this.s3Bucket = s3_factory_1.S3Factory.bucket(this, "EMR Serverless Cluster Bucket", {
versioned: true,
});
}
this.role = new iam.Role(scope, "EMR Serverless Cluster Role", {
assumedBy: new iam.ServicePrincipal("emr-serverless.amazonaws.com"),
path: "/service-role/",
});
this.role.addManagedPolicy(new iam.ManagedPolicy(this, "MWAA Execution Policy", {
statements: [
new iam.PolicyStatement({
actions: ["s3:GetObject*", "s3:ListBucket"],
resources: ["arn:aws:s3:::*.elasticmapreduce", "arn:aws:s3:::*.elasticmapreduce/*"],
}),
new iam.PolicyStatement({
actions: ["s3:DeleteObject", "s3:PutObject", "s3:GetObject*", "s3:GetBucket*", "s3:List*"],
resources: [this.s3Bucket.bucketArn, `${this.s3Bucket.bucketArn}/*`],
}),
new iam.PolicyStatement({
actions: [
"glue:GetDatabase",
"glue:CreateDatabase",
"glue:GetDataBases",
"glue:CreateTable",
"glue:GetTable",
"glue:UpdateTable",
"glue:DeleteTable",
"glue:GetTables",
"glue:GetPartition",
"glue:GetPartitions",
"glue:CreatePartition",
"glue:BatchCreatePartition",
"glue:GetUserDefinedFunctions",
],
resources: ["*"],
}),
],
}));
if (props.additionalPolicyStatements) {
props.additionalPolicyStatements.forEach((statement) => {
this.role.addToPolicy(statement);
});
}
if (!props.networkConfiguration && this.vpc && this.securityGroup) {
this.networkConfiguration = {
securityGroupIds: [this.securityGroup.securityGroupId],
subnetIds: this.vpc.selectSubnets({ subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS }).subnetIds,
};
}
else {
this.networkConfiguration = {};
}
const defaultProps = {
networkConfiguration: this.networkConfiguration,
};
const mergedProps = utils_1.overrideProps(defaultProps, props);
this.emrServerlessApplication = new emr.CfnApplication(this, "DDK EMR Serverless Application", {
...mergedProps,
});
}
createVpc(scope, resourceName, vpcCidr) {
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, "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.EMRServerlessCluster = EMRServerlessCluster;
_a = JSII_RTTI_SYMBOL_1;
EMRServerlessCluster[_a] = { fqn: "aws-ddk-core.EMRServerlessCluster", version: "1.4.1" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"emr-serverless-cluster.js","sourceRoot":"","sources":["../../src/core/emr-serverless-cluster.ts"],"names":[],"mappings":"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","sourcesContent":["import * as ec2 from \"aws-cdk-lib/aws-ec2\";\nimport * as emr from \"aws-cdk-lib/aws-emrserverless\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as s3 from \"aws-cdk-lib/aws-s3\";\nimport { Construct } from \"constructs\";\n\nimport { S3Factory } from \"./s3-factory\";\nimport { overrideProps } from \"./utils\";\n\nexport interface EMRServerlessClusterProps extends emr.CfnApplicationProps {\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   * Security Group\n   */\n  readonly securityGroup?: ec2.SecurityGroup;\n  /**\n   * Additional policy statements to add to the emr role\n   */\n  readonly additionalPolicyStatements?: iam.PolicyStatement[];\n}\n\nexport class EMRServerlessCluster extends Construct {\n  readonly vpc?: ec2.IVpc;\n  readonly s3Bucket: s3.IBucket;\n  readonly role: iam.Role;\n  readonly emrServerlessApplication: emr.CfnApplication;\n  readonly securityGroup?: ec2.SecurityGroup;\n  readonly networkConfiguration: emr.CfnApplication.NetworkConfigurationProperty;\n\n  constructor(scope: Construct, id: string, props: EMRServerlessClusterProps) {\n    super(scope, id);\n\n    const name = props.name ?? \"DDKEmrServerlessCluster\";\n\n    if (props.vpcId) {\n      this.vpc = ec2.Vpc.fromLookup(scope, \"VPC\", { vpcId: props.vpcId });\n    } else if (props.vpcCidr) {\n      this.vpc = this.createVpc(scope, name, props.vpcCidr);\n    } else {\n      throw new Error(\"One of 'vpcId' or 'vpcCidr' must be provided\");\n    }\n\n    if (this.vpc) {\n      this.securityGroup =\n        props.securityGroup ??\n        new ec2.SecurityGroup(scope, \"EMR Serverless Security Group\", {\n          securityGroupName: `EMR Serverless Security Group`,\n          description: \"Security group with a self-referencing inbound rule.\",\n          vpc: this.vpc,\n        });\n      this.securityGroup.addIngressRule(this.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, \"EMR Serverless Cluster Bucket\", {\n        versioned: true,\n      });\n    }\n\n    this.role = new iam.Role(scope, \"EMR Serverless Cluster Role\", {\n      assumedBy: new iam.ServicePrincipal(\"emr-serverless.amazonaws.com\"),\n      path: \"/service-role/\",\n    });\n    this.role.addManagedPolicy(\n      new iam.ManagedPolicy(this, \"MWAA Execution Policy\", {\n        statements: [\n          new iam.PolicyStatement({\n            actions: [\"s3:GetObject*\", \"s3:ListBucket\"],\n            resources: [\"arn:aws:s3:::*.elasticmapreduce\", \"arn:aws:s3:::*.elasticmapreduce/*\"],\n          }),\n          new iam.PolicyStatement({\n            actions: [\"s3:DeleteObject\", \"s3:PutObject\", \"s3:GetObject*\", \"s3:GetBucket*\", \"s3:List*\"],\n            resources: [this.s3Bucket.bucketArn, `${this.s3Bucket.bucketArn}/*`],\n          }),\n          new iam.PolicyStatement({\n            actions: [\n              \"glue:GetDatabase\",\n              \"glue:CreateDatabase\",\n              \"glue:GetDataBases\",\n              \"glue:CreateTable\",\n              \"glue:GetTable\",\n              \"glue:UpdateTable\",\n              \"glue:DeleteTable\",\n              \"glue:GetTables\",\n              \"glue:GetPartition\",\n              \"glue:GetPartitions\",\n              \"glue:CreatePartition\",\n              \"glue:BatchCreatePartition\",\n              \"glue:GetUserDefinedFunctions\",\n            ],\n            resources: [\"*\"],\n          }),\n        ],\n      }),\n    );\n\n    if (props.additionalPolicyStatements) {\n      props.additionalPolicyStatements.forEach((statement) => {\n        this.role.addToPolicy(statement);\n      });\n    }\n\n    if (!props.networkConfiguration && this.vpc && this.securityGroup) {\n      this.networkConfiguration = {\n        securityGroupIds: [this.securityGroup.securityGroupId],\n        subnetIds: this.vpc.selectSubnets({ subnetType: ec2.SubnetType.PRIVATE_WITH_EGRESS }).subnetIds,\n      };\n    } else {\n      this.networkConfiguration = {};\n    }\n\n    const defaultProps: Partial<emr.CfnApplicationProps> = {\n      networkConfiguration: this.networkConfiguration,\n    };\n\n    const mergedProps = overrideProps(defaultProps, props);\n\n    this.emrServerlessApplication = new emr.CfnApplication(this, \"DDK EMR Serverless Application\", {\n      ...mergedProps,\n    });\n  }\n  createVpc(scope: Construct, resourceName: string, vpcCidr: string): ec2.IVpc {\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, \"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"]}
;