@aws-cdk/aws-ec2
Version:
The CDK Construct Library for AWS::EC2
271 lines • 27 kB
JavaScript
;
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"]}