UNPKG

@aws-cdk/aws-ec2

Version:

The CDK Construct Library for AWS::EC2

271 lines 27 kB
"use strict"; var _a, _b, _c; Object.defineProperty(exports, "__esModule", { value: true }); exports.FlowLog = exports.FlowLogDestination = exports.FlowLogResourceType = exports.FlowLogDestinationType = exports.FlowLogTrafficType = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const iam = require("@aws-cdk/aws-iam"); const logs = require("@aws-cdk/aws-logs"); const s3 = require("@aws-cdk/aws-s3"); const core_1 = require("@aws-cdk/core"); const ec2_generated_1 = require("./ec2.generated"); /** * The type of VPC traffic to log * * */ var FlowLogTrafficType; (function (FlowLogTrafficType) { /** * Only log accepts */ FlowLogTrafficType["ACCEPT"] = "ACCEPT"; /** * Log all requests */ FlowLogTrafficType["ALL"] = "ALL"; /** * Only log rejects */ FlowLogTrafficType["REJECT"] = "REJECT"; })(FlowLogTrafficType = exports.FlowLogTrafficType || (exports.FlowLogTrafficType = {})); /** * The available destination types for Flow Logs * */ var FlowLogDestinationType; (function (FlowLogDestinationType) { /** * Send flow logs to CloudWatch Logs Group */ FlowLogDestinationType["CLOUD_WATCH_LOGS"] = "cloud-watch-logs"; /** * Send flow logs to S3 Bucket */ FlowLogDestinationType["S3"] = "s3"; })(FlowLogDestinationType = exports.FlowLogDestinationType || (exports.FlowLogDestinationType = {})); /** * The type of resource to create the flow log for * * */ class FlowLogResourceType { /** * The subnet to attach the Flow Log to */ static fromSubnet(subnet) { try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_ISubnet(subnet); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.fromSubnet); } throw error; } return { resourceType: 'Subnet', resourceId: subnet.subnetId, }; } /** * The VPC to attach the Flow Log to */ static fromVpc(vpc) { try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_IVpc(vpc); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.fromVpc); } throw error; } return { resourceType: 'VPC', resourceId: vpc.vpcId, }; } /** * The Network Interface to attach the Flow Log to */ static fromNetworkInterfaceId(id) { return { resourceType: 'NetworkInterface', resourceId: id, }; } } exports.FlowLogResourceType = FlowLogResourceType; _a = JSII_RTTI_SYMBOL_1; FlowLogResourceType[_a] = { fqn: "@aws-cdk/aws-ec2.FlowLogResourceType", version: "1.204.0" }; /** * The destination type for the flow log * * */ class FlowLogDestination { /** * Use CloudWatch logs as the destination */ static toCloudWatchLogs(logGroup, iamRole) { return new CloudWatchLogsDestination({ logDestinationType: FlowLogDestinationType.CLOUD_WATCH_LOGS, logGroup, iamRole, }); } /** * Use S3 as the destination */ static toS3(bucket, keyPrefix) { return new S3Destination({ logDestinationType: FlowLogDestinationType.S3, s3Bucket: bucket, keyPrefix, }); } } exports.FlowLogDestination = FlowLogDestination; _b = JSII_RTTI_SYMBOL_1; FlowLogDestination[_b] = { fqn: "@aws-cdk/aws-ec2.FlowLogDestination", version: "1.204.0" }; /** * */ class S3Destination extends FlowLogDestination { constructor(props) { super(); this.props = props; } bind(scope, _flowLog) { let s3Bucket; if (this.props.s3Bucket === undefined) { s3Bucket = new s3.Bucket(scope, 'Bucket', { encryption: s3.BucketEncryption.UNENCRYPTED, removalPolicy: core_1.RemovalPolicy.RETAIN, }); } else { s3Bucket = this.props.s3Bucket; } return { logDestinationType: FlowLogDestinationType.S3, s3Bucket, keyPrefix: this.props.keyPrefix, }; } } /** * */ class CloudWatchLogsDestination extends FlowLogDestination { constructor(props) { super(); this.props = props; } bind(scope, _flowLog) { let iamRole; let logGroup; if (this.props.iamRole === undefined) { iamRole = new iam.Role(scope, 'IAMRole', { roleName: core_1.PhysicalName.GENERATE_IF_NEEDED, assumedBy: new iam.ServicePrincipal('vpc-flow-logs.amazonaws.com'), }); } else { iamRole = this.props.iamRole; } if (this.props.logGroup === undefined) { logGroup = new logs.LogGroup(scope, 'LogGroup'); } else { logGroup = this.props.logGroup; } iamRole.addToPrincipalPolicy(new iam.PolicyStatement({ actions: [ 'logs:CreateLogStream', 'logs:PutLogEvents', 'logs:DescribeLogStreams', ], effect: iam.Effect.ALLOW, resources: [logGroup.logGroupArn], })); iamRole.addToPrincipalPolicy(new iam.PolicyStatement({ actions: ['iam:PassRole'], effect: iam.Effect.ALLOW, resources: [iamRole.roleArn], })); return { logDestinationType: FlowLogDestinationType.CLOUD_WATCH_LOGS, logGroup, iamRole, }; } } /** * The base class for a Flow Log * * */ class FlowLogBase extends core_1.Resource { } /** * A VPC flow log. * @resource AWS::EC2::FlowLog * * */ class FlowLog extends FlowLogBase { constructor(scope, id, props) { super(scope, id, { physicalName: props.flowLogName, }); try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_FlowLogProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, FlowLog); } throw error; } const destination = props.destination || FlowLogDestination.toCloudWatchLogs(); const destinationConfig = destination.bind(this, this); this.logGroup = destinationConfig.logGroup; this.bucket = destinationConfig.s3Bucket; this.iamRole = destinationConfig.iamRole; this.keyPrefix = destinationConfig.keyPrefix; let logDestination = undefined; if (this.bucket) { logDestination = this.keyPrefix ? this.bucket.arnForObjects(this.keyPrefix) : this.bucket.bucketArn; } const flowLog = new ec2_generated_1.CfnFlowLog(this, 'FlowLog', { deliverLogsPermissionArn: this.iamRole ? this.iamRole.roleArn : undefined, logDestinationType: destinationConfig.logDestinationType, logGroupName: this.logGroup ? this.logGroup.logGroupName : undefined, resourceId: props.resourceType.resourceId, resourceType: props.resourceType.resourceType, trafficType: props.trafficType ? props.trafficType : FlowLogTrafficType.ALL, logDestination, }); this.flowLogId = flowLog.ref; } /** * Import a Flow Log by it's Id */ static fromFlowLogId(scope, id, flowLogId) { class Import extends FlowLogBase { constructor() { super(...arguments); this.flowLogId = flowLogId; } } return new Import(scope, id); } } exports.FlowLog = FlowLog; _c = JSII_RTTI_SYMBOL_1; FlowLog[_c] = { fqn: "@aws-cdk/aws-ec2.FlowLog", version: "1.204.0" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vpc-flow-logs.js","sourceRoot":"","sources":["vpc-flow-logs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as iam from '@aws-cdk/aws-iam';\nimport * as logs from '@aws-cdk/aws-logs';\nimport * as s3 from '@aws-cdk/aws-s3';\nimport { IResource, PhysicalName, RemovalPolicy, Resource } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { CfnFlowLog } from './ec2.generated';\nimport { ISubnet, IVpc } from './vpc';\n\n// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.\n// eslint-disable-next-line\nimport { Construct as CoreConstruct } from '@aws-cdk/core';\n\n/**\n * A FlowLog\n *\n *\n */\nexport interface IFlowLog extends IResource {\n  /**\n   * The Id of the VPC Flow Log\n   *\n   * @attribute\n   */\n  readonly flowLogId: string;\n}\n\n/**\n * The type of VPC traffic to log\n *\n *\n */\nexport enum FlowLogTrafficType {\n  /**\n   * Only log accepts\n   */\n  ACCEPT = 'ACCEPT',\n\n  /**\n   * Log all requests\n   */\n  ALL = 'ALL',\n\n  /**\n   * Only log rejects\n   */\n  REJECT = 'REJECT'\n}\n\n/**\n * The available destination types for Flow Logs\n *\n */\nexport enum FlowLogDestinationType {\n  /**\n   * Send flow logs to CloudWatch Logs Group\n   */\n  CLOUD_WATCH_LOGS = 'cloud-watch-logs',\n\n  /**\n   * Send flow logs to S3 Bucket\n   */\n  S3 = 's3'\n}\n\n/**\n * The type of resource to create the flow log for\n *\n *\n */\nexport abstract class FlowLogResourceType {\n  /**\n   * The subnet to attach the Flow Log to\n   */\n  public static fromSubnet(subnet: ISubnet): FlowLogResourceType {\n    return {\n      resourceType: 'Subnet',\n      resourceId: subnet.subnetId,\n    };\n  }\n\n  /**\n   * The VPC to attach the Flow Log to\n   */\n  public static fromVpc(vpc: IVpc): FlowLogResourceType {\n    return {\n      resourceType: 'VPC',\n      resourceId: vpc.vpcId,\n    };\n  }\n\n  /**\n   * The Network Interface to attach the Flow Log to\n   */\n  public static fromNetworkInterfaceId(id: string): FlowLogResourceType {\n    return {\n      resourceType: 'NetworkInterface',\n      resourceId: id,\n    };\n  }\n\n  /**\n   * The type of resource to attach a flow log to.\n   */\n  public abstract resourceType: string;\n\n  /**\n   * The Id of the resource that the flow log should be attached to.\n   */\n  public abstract resourceId: string;\n}\n\n/**\n * The destination type for the flow log\n *\n *\n */\nexport abstract class FlowLogDestination {\n  /**\n   * Use CloudWatch logs as the destination\n   */\n  public static toCloudWatchLogs(logGroup?: logs.ILogGroup, iamRole?: iam.IRole): FlowLogDestination {\n    return new CloudWatchLogsDestination({\n      logDestinationType: FlowLogDestinationType.CLOUD_WATCH_LOGS,\n      logGroup,\n      iamRole,\n    });\n  }\n\n  /**\n   * Use S3 as the destination\n   */\n  public static toS3(bucket?: s3.IBucket, keyPrefix?: string): FlowLogDestination {\n    return new S3Destination({\n      logDestinationType: FlowLogDestinationType.S3,\n      s3Bucket: bucket,\n      keyPrefix,\n    });\n  }\n\n  /**\n   * Generates a flow log destination configuration\n   */\n  public abstract bind(scope: CoreConstruct, flowLog: FlowLog): FlowLogDestinationConfig;\n}\n\n/**\n * Flow Log Destination configuration\n *\n *\n */\nexport interface FlowLogDestinationConfig {\n  /**\n   * The type of destination to publish the flow logs to.\n   *\n   * @default - CLOUD_WATCH_LOGS\n   */\n  readonly logDestinationType: FlowLogDestinationType;\n\n  /**\n   * The IAM Role that has access to publish to CloudWatch logs\n   *\n   * @default - default IAM role is created for you\n   */\n  readonly iamRole?: iam.IRole;\n\n  /**\n   * The CloudWatch Logs Log Group to publish the flow logs to\n   *\n   * @default - default log group is created for you\n   */\n  readonly logGroup?: logs.ILogGroup;\n\n  /**\n   * S3 bucket to publish the flow logs to\n   *\n   * @default - undefined\n   */\n  readonly s3Bucket?: s3.IBucket;\n\n  /**\n   * S3 bucket key prefix to publish the flow logs to\n   *\n   * @default - undefined\n   */\n  readonly keyPrefix?: string;\n}\n\n/**\n *\n */\nclass S3Destination extends FlowLogDestination {\n  constructor(private readonly props: FlowLogDestinationConfig) {\n    super();\n  }\n\n  public bind(scope: CoreConstruct, _flowLog: FlowLog): FlowLogDestinationConfig {\n    let s3Bucket: s3.IBucket;\n    if (this.props.s3Bucket === undefined) {\n      s3Bucket = new s3.Bucket(scope, 'Bucket', {\n        encryption: s3.BucketEncryption.UNENCRYPTED,\n        removalPolicy: RemovalPolicy.RETAIN,\n      });\n    } else {\n      s3Bucket = this.props.s3Bucket;\n    }\n    return {\n      logDestinationType: FlowLogDestinationType.S3,\n      s3Bucket,\n      keyPrefix: this.props.keyPrefix,\n    };\n  }\n}\n\n/**\n *\n */\nclass CloudWatchLogsDestination extends FlowLogDestination {\n  constructor(private readonly props: FlowLogDestinationConfig) {\n    super();\n  }\n\n  public bind(scope: CoreConstruct, _flowLog: FlowLog): FlowLogDestinationConfig {\n    let iamRole: iam.IRole;\n    let logGroup: logs.ILogGroup;\n    if (this.props.iamRole === undefined) {\n      iamRole = new iam.Role(scope, 'IAMRole', {\n        roleName: PhysicalName.GENERATE_IF_NEEDED,\n        assumedBy: new iam.ServicePrincipal('vpc-flow-logs.amazonaws.com'),\n      });\n    } else {\n      iamRole = this.props.iamRole;\n    }\n\n    if (this.props.logGroup === undefined) {\n      logGroup = new logs.LogGroup(scope, 'LogGroup');\n    } else {\n      logGroup = this.props.logGroup;\n    }\n\n    iamRole.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          'logs:CreateLogStream',\n          'logs:PutLogEvents',\n          'logs:DescribeLogStreams',\n        ],\n        effect: iam.Effect.ALLOW,\n        resources: [logGroup.logGroupArn],\n      }),\n    );\n\n    iamRole.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        actions: ['iam:PassRole'],\n        effect: iam.Effect.ALLOW,\n        resources: [iamRole.roleArn],\n      }),\n    );\n\n    return {\n      logDestinationType: FlowLogDestinationType.CLOUD_WATCH_LOGS,\n      logGroup,\n      iamRole,\n    };\n  }\n}\n\n/**\n * Options to add a flow log to a VPC\n *\n *\n */\nexport interface FlowLogOptions {\n  /**\n   * The type of traffic to log. You can log traffic that the resource accepts or rejects, or all traffic.\n   *\n   * @default ALL\n   */\n  readonly trafficType?: FlowLogTrafficType;\n\n  /**\n   * Specifies the type of destination to which the flow log data is to be published.\n   * Flow log data can be published to CloudWatch Logs or Amazon S3\n   *\n   * @default FlowLogDestinationType.toCloudWatchLogs()\n   */\n  readonly destination?: FlowLogDestination;\n}\n\n/**\n * Properties of a VPC Flow Log\n *\n *\n */\nexport interface FlowLogProps extends FlowLogOptions {\n  /**\n   * The name of the FlowLog\n   *\n   * It is not recommended to use an explicit name.\n   *\n   * @default If you don't specify a flowLogName, AWS CloudFormation generates a\n   * unique physical ID and uses that ID for the group name.\n   */\n  readonly flowLogName?: string;\n\n  /**\n   * The type of resource for which to create the flow log\n   */\n  readonly resourceType: FlowLogResourceType;\n}\n\n/**\n * The base class for a Flow Log\n *\n *\n */\nabstract class FlowLogBase extends Resource implements IFlowLog {\n  /**\n   * The Id of the VPC Flow Log\n   *\n   * @attribute\n   */\n  public abstract readonly flowLogId: string;\n}\n\n/**\n * A VPC flow log.\n * @resource AWS::EC2::FlowLog\n *\n *\n */\nexport class FlowLog extends FlowLogBase {\n  /**\n   * Import a Flow Log by it's Id\n   */\n  public static fromFlowLogId(scope: Construct, id: string, flowLogId: string): IFlowLog {\n    class Import extends FlowLogBase {\n      public flowLogId = flowLogId;\n    }\n\n    return new Import(scope, id);\n  }\n\n  /**\n   * The Id of the VPC Flow Log\n   *\n   * @attribute\n   */\n  public readonly flowLogId: string;\n\n  /**\n   * The S3 bucket to publish flow logs to\n   */\n  public readonly bucket?: s3.IBucket;\n\n  /**\n   * S3 bucket key prefix to publish the flow logs under\n   */\n  readonly keyPrefix?: string;\n\n  /**\n   * The iam role used to publish logs to CloudWatch\n   */\n  public readonly iamRole?: iam.IRole;\n\n  /**\n   * The CloudWatch Logs LogGroup to publish flow logs to\n   */\n  public readonly logGroup?: logs.ILogGroup;\n\n  constructor(scope: Construct, id: string, props: FlowLogProps) {\n    super(scope, id, {\n      physicalName: props.flowLogName,\n    });\n\n    const destination = props.destination || FlowLogDestination.toCloudWatchLogs();\n\n    const destinationConfig = destination.bind(this, this);\n    this.logGroup = destinationConfig.logGroup;\n    this.bucket = destinationConfig.s3Bucket;\n    this.iamRole = destinationConfig.iamRole;\n    this.keyPrefix = destinationConfig.keyPrefix;\n\n    let logDestination: string | undefined = undefined;\n    if (this.bucket) {\n      logDestination = this.keyPrefix ? this.bucket.arnForObjects(this.keyPrefix) : this.bucket.bucketArn;\n    }\n\n    const flowLog = new CfnFlowLog(this, 'FlowLog', {\n      deliverLogsPermissionArn: this.iamRole ? this.iamRole.roleArn : undefined,\n      logDestinationType: destinationConfig.logDestinationType,\n      logGroupName: this.logGroup ? this.logGroup.logGroupName : undefined,\n      resourceId: props.resourceType.resourceId,\n      resourceType: props.resourceType.resourceType,\n      trafficType: props.trafficType\n        ? props.trafficType\n        : FlowLogTrafficType.ALL,\n      logDestination,\n    });\n\n    this.flowLogId = flowLog.ref;\n  }\n}\n"]}