UNPKG

@aws-cdk/aws-logs

Version:

The CDK Construct Library for AWS::Logs

133 lines 19.8 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.LogRetention = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const path = require("path"); const iam = require("@aws-cdk/aws-iam"); const s3_assets = require("@aws-cdk/aws-s3-assets"); const cdk = require("@aws-cdk/core"); const log_group_1 = require("./log-group"); // keep this import separate from other imports to reduce chance for merge conflicts with v2-main // eslint-disable-next-line no-duplicate-imports, import/order const core_1 = require("@aws-cdk/core"); // keep this import separate from other imports to reduce chance for merge conflicts with v2-main // eslint-disable-next-line no-duplicate-imports, import/order const core_2 = require("@aws-cdk/core"); /** * Creates a custom resource to control the retention policy of a CloudWatch Logs * log group. The log group is created if it doesn't already exist. The policy * is removed when `retentionDays` is `undefined` or equal to `Infinity`. * Log group can be created in the region that is different from stack region by * specifying `logGroupRegion` */ class LogRetention extends core_2.Construct { constructor(scope, id, props) { super(scope, id); try { jsiiDeprecationWarnings._aws_cdk_aws_logs_LogRetentionProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, LogRetention); } throw error; } // Custom resource provider const provider = this.ensureSingletonLogRetentionFunction(props); // Need to use a CfnResource here to prevent lerna dependency cycles // @aws-cdk/aws-cloudformation -> @aws-cdk/aws-lambda -> @aws-cdk/aws-cloudformation const retryOptions = props.logRetentionRetryOptions; const resource = new cdk.CfnResource(this, 'Resource', { type: 'Custom::LogRetention', properties: { ServiceToken: provider.functionArn, LogGroupName: props.logGroupName, LogGroupRegion: props.logGroupRegion, SdkRetry: retryOptions ? { maxRetries: retryOptions.maxRetries, base: retryOptions.base?.toMilliseconds(), } : undefined, RetentionInDays: props.retention === log_group_1.RetentionDays.INFINITE ? undefined : props.retention, }, }); const logGroupName = resource.getAtt('LogGroupName').toString(); // Append ':*' at the end of the ARN to match with how CloudFormation does this for LogGroup ARNs // See https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-logs-loggroup.html#aws-resource-logs-loggroup-return-values this.logGroupArn = cdk.Stack.of(this).formatArn({ region: props.logGroupRegion, service: 'logs', resource: 'log-group', resourceName: `${logGroupName}:*`, arnFormat: core_1.ArnFormat.COLON_RESOURCE_NAME, }); } /** * Helper method to ensure that only one instance of LogRetentionFunction resources are in the stack mimicking the * behaviour of @aws-cdk/aws-lambda's SingletonFunction to prevent circular dependencies */ ensureSingletonLogRetentionFunction(props) { const functionLogicalId = 'LogRetentionaae0aa3c5b4d4f87b02d85b201efdd8a'; const existing = cdk.Stack.of(this).node.tryFindChild(functionLogicalId); if (existing) { return existing; } return new LogRetentionFunction(cdk.Stack.of(this), functionLogicalId, props); } } exports.LogRetention = LogRetention; _a = JSII_RTTI_SYMBOL_1; LogRetention[_a] = { fqn: "@aws-cdk/aws-logs.LogRetention", version: "1.204.0" }; /** * Private provider Lambda function to support the log retention custom resource. */ class LogRetentionFunction extends core_2.Construct { constructor(scope, id, props) { super(scope, id); this.tags = new cdk.TagManager(cdk.TagType.KEY_VALUE, 'AWS::Lambda::Function'); // Code const asset = new s3_assets.Asset(this, 'Code', { path: path.join(__dirname, 'log-retention-provider'), }); // Role const role = props.role || new iam.Role(this, 'ServiceRole', { assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'), managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')], }); // Duplicate statements will be deduplicated by `PolicyDocument` role.addToPrincipalPolicy(new iam.PolicyStatement({ actions: ['logs:PutRetentionPolicy', 'logs:DeleteRetentionPolicy'], // We need '*' here because we will also put a retention policy on // the log group of the provider function. Referencing its name // creates a CF circular dependency. resources: ['*'], })); // Lambda function const resource = new cdk.CfnResource(this, 'Resource', { type: 'AWS::Lambda::Function', properties: { Handler: 'index.handler', Runtime: 'nodejs14.x', Code: { S3Bucket: asset.s3BucketName, S3Key: asset.s3ObjectKey, }, Role: role.roleArn, Tags: this.tags.renderedTags, }, }); this.functionArn = resource.getAtt('Arn'); asset.addResourceMetadata(resource, 'Code'); // Function dependencies role.node.children.forEach((child) => { if (cdk.CfnResource.isCfnResource(child)) { resource.addDependsOn(child); } if (cdk.Construct.isConstruct(child) && child.node.defaultChild && cdk.CfnResource.isCfnResource(child.node.defaultChild)) { resource.addDependsOn(child.node.defaultChild); } }); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"log-retention.js","sourceRoot":"","sources":["log-retention.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as s3_assets from '@aws-cdk/aws-s3-assets';\nimport * as cdk from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { RetentionDays } from './log-group';\n\n// keep this import separate from other imports to reduce chance for merge conflicts with v2-main\n// eslint-disable-next-line no-duplicate-imports, import/order\nimport { ArnFormat } from '@aws-cdk/core';\n\n// keep this import separate from other imports to reduce chance for merge conflicts with v2-main\n// eslint-disable-next-line no-duplicate-imports, import/order\nimport { Construct as CoreConstruct } from '@aws-cdk/core';\n\n/**\n * Construction properties for a LogRetention.\n */\nexport interface LogRetentionProps {\n  /**\n   * The log group name.\n   */\n  readonly logGroupName: string;\n\n  /**\n   * The region where the log group should be created\n   * @default - same region as the stack\n   */\n  readonly logGroupRegion?: string;\n\n  /**\n   * The number of days log events are kept in CloudWatch Logs.\n   */\n  readonly retention: RetentionDays;\n\n  /**\n   * The IAM role for the Lambda function associated with the custom resource.\n   *\n   * @default - A new role is created\n   */\n  readonly role?: iam.IRole;\n\n  /**\n   * Retry options for all AWS API calls.\n   *\n   * @default - AWS SDK default retry options\n   */\n  readonly logRetentionRetryOptions?: LogRetentionRetryOptions;\n}\n\n/**\n * Retry options for all AWS API calls.\n */\nexport interface LogRetentionRetryOptions {\n  /**\n   * The maximum amount of retries.\n   *\n   * @default 3 (AWS SDK default)\n   */\n  readonly maxRetries?: number;\n  /**\n   * The base duration to use in the exponential backoff for operation retries.\n   *\n   * @default Duration.millis(100) (AWS SDK default)\n   */\n  readonly base?: cdk.Duration;\n}\n\n/**\n * Creates a custom resource to control the retention policy of a CloudWatch Logs\n * log group. The log group is created if it doesn't already exist. The policy\n * is removed when `retentionDays` is `undefined` or equal to `Infinity`.\n * Log group can be created in the region that is different from stack region by\n * specifying `logGroupRegion`\n */\nexport class LogRetention extends CoreConstruct {\n\n  /**\n   * The ARN of the LogGroup.\n   */\n  public readonly logGroupArn: string;\n\n  constructor(scope: Construct, id: string, props: LogRetentionProps) {\n    super(scope, id);\n\n    // Custom resource provider\n    const provider = this.ensureSingletonLogRetentionFunction(props);\n\n    // Need to use a CfnResource here to prevent lerna dependency cycles\n    // @aws-cdk/aws-cloudformation -> @aws-cdk/aws-lambda -> @aws-cdk/aws-cloudformation\n    const retryOptions = props.logRetentionRetryOptions;\n    const resource = new cdk.CfnResource(this, 'Resource', {\n      type: 'Custom::LogRetention',\n      properties: {\n        ServiceToken: provider.functionArn,\n        LogGroupName: props.logGroupName,\n        LogGroupRegion: props.logGroupRegion,\n        SdkRetry: retryOptions ? {\n          maxRetries: retryOptions.maxRetries,\n          base: retryOptions.base?.toMilliseconds(),\n        } : undefined,\n        RetentionInDays: props.retention === RetentionDays.INFINITE ? undefined : props.retention,\n      },\n    });\n\n    const logGroupName = resource.getAtt('LogGroupName').toString();\n    // Append ':*' at the end of the ARN to match with how CloudFormation does this for LogGroup ARNs\n    // See https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-logs-loggroup.html#aws-resource-logs-loggroup-return-values\n    this.logGroupArn = cdk.Stack.of(this).formatArn({\n      region: props.logGroupRegion,\n      service: 'logs',\n      resource: 'log-group',\n      resourceName: `${logGroupName}:*`,\n      arnFormat: ArnFormat.COLON_RESOURCE_NAME,\n    });\n  }\n\n  /**\n   * Helper method to ensure that only one instance of LogRetentionFunction resources are in the stack mimicking the\n   * behaviour of @aws-cdk/aws-lambda's SingletonFunction to prevent circular dependencies\n   */\n  private ensureSingletonLogRetentionFunction(props: LogRetentionProps) {\n    const functionLogicalId = 'LogRetentionaae0aa3c5b4d4f87b02d85b201efdd8a';\n    const existing = cdk.Stack.of(this).node.tryFindChild(functionLogicalId);\n    if (existing) {\n      return existing as LogRetentionFunction;\n    }\n    return new LogRetentionFunction(cdk.Stack.of(this), functionLogicalId, props);\n  }\n}\n\n/**\n * Private provider Lambda function to support the log retention custom resource.\n */\nclass LogRetentionFunction extends CoreConstruct implements cdk.ITaggable {\n  public readonly functionArn: cdk.Reference;\n\n  public readonly tags: cdk.TagManager = new cdk.TagManager(cdk.TagType.KEY_VALUE, 'AWS::Lambda::Function');\n\n  constructor(scope: Construct, id: string, props: LogRetentionProps) {\n    super(scope, id);\n\n    // Code\n    const asset = new s3_assets.Asset(this, 'Code', {\n      path: path.join(__dirname, 'log-retention-provider'),\n    });\n\n    // Role\n    const role = props.role || new iam.Role(this, 'ServiceRole', {\n      assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n      managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')],\n    });\n    // Duplicate statements will be deduplicated by `PolicyDocument`\n    role.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['logs:PutRetentionPolicy', 'logs:DeleteRetentionPolicy'],\n      // We need '*' here because we will also put a retention policy on\n      // the log group of the provider function. Referencing its name\n      // creates a CF circular dependency.\n      resources: ['*'],\n    }));\n\n    // Lambda function\n    const resource = new cdk.CfnResource(this, 'Resource', {\n      type: 'AWS::Lambda::Function',\n      properties: {\n        Handler: 'index.handler',\n        Runtime: 'nodejs14.x', // Equivalent to Runtime.NODEJS_14_X\n        Code: {\n          S3Bucket: asset.s3BucketName,\n          S3Key: asset.s3ObjectKey,\n        },\n        Role: role.roleArn,\n        Tags: this.tags.renderedTags,\n      },\n    });\n    this.functionArn = resource.getAtt('Arn');\n\n    asset.addResourceMetadata(resource, 'Code');\n\n    // Function dependencies\n    role.node.children.forEach((child) => {\n      if (cdk.CfnResource.isCfnResource(child)) {\n        resource.addDependsOn(child);\n      }\n      if (cdk.Construct.isConstruct(child) && child.node.defaultChild && cdk.CfnResource.isCfnResource(child.node.defaultChild)) {\n        resource.addDependsOn(child.node.defaultChild);\n      }\n    });\n  }\n}\n"]}