UNPKG

@aws-cdk/aws-ecs

Version:

The CDK Construct Library for AWS::ECS

178 lines 22.7 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.FirelensLogRouter = exports.obtainDefaultFluentBitECRImage = exports.FirelensConfigFileType = exports.FirelensLogRouterType = 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 ssm = require("@aws-cdk/aws-ssm"); const cdk = require("@aws-cdk/core"); const container_definition_1 = require("./container-definition"); const container_image_1 = require("./container-image"); /** * Firelens log router type, fluentbit or fluentd. * https://docs.aws.amazon.com/AmazonECS/latest/developerguide/using_firelens.html */ var FirelensLogRouterType; (function (FirelensLogRouterType) { /** * fluentbit */ FirelensLogRouterType["FLUENTBIT"] = "fluentbit"; /** * fluentd */ FirelensLogRouterType["FLUENTD"] = "fluentd"; })(FirelensLogRouterType = exports.FirelensLogRouterType || (exports.FirelensLogRouterType = {})); /** * Firelens configuration file type, s3 or file path. * https://docs.aws.amazon.com/AmazonECS/latest/developerguide/using_firelens.html#firelens-taskdef-customconfig */ var FirelensConfigFileType; (function (FirelensConfigFileType) { /** * s3 */ FirelensConfigFileType["S3"] = "s3"; /** * fluentd */ FirelensConfigFileType["FILE"] = "file"; })(FirelensConfigFileType = exports.FirelensConfigFileType || (exports.FirelensConfigFileType = {})); /** * Render to CfnTaskDefinition.FirelensConfigurationProperty from FirelensConfig */ function renderFirelensConfig(firelensConfig) { if (!firelensConfig.options) { return { type: firelensConfig.type }; } else { // firelensConfig.options.configFileType has been filled with s3 or file type in constructor. return { type: firelensConfig.type, options: { 'enable-ecs-log-metadata': firelensConfig.options.enableECSLogMetadata ? 'true' : 'false', 'config-file-type': firelensConfig.options.configFileType, 'config-file-value': firelensConfig.options.configFileValue, }, }; } } /** * SSM parameters for latest fluent bit docker image in ECR * https://github.com/aws/aws-for-fluent-bit#using-ssm-to-find-available-versions */ const fluentBitImageSSMPath = '/aws/service/aws-for-fluent-bit'; /** * Obtain Fluent Bit image in Amazon ECR and setup corresponding IAM permissions. * ECR image pull permissions will be granted in task execution role. * Cloudwatch logs, Kinesis data stream or firehose permissions will be grant by check options in logDriverConfig. * https://docs.aws.amazon.com/AmazonECS/latest/developerguide/using_firelens.html#firelens-using-fluentbit */ function obtainDefaultFluentBitECRImage(task, logDriverConfig, imageTag) { // grant ECR image pull permissions to executor role task.addToExecutionRolePolicy(new iam.PolicyStatement({ actions: [ 'ecr:GetAuthorizationToken', 'ecr:BatchCheckLayerAvailability', 'ecr:GetDownloadUrlForLayer', 'ecr:BatchGetImage', ], resources: ['*'], })); // grant cloudwatch or firehose permissions to task role const logName = logDriverConfig && logDriverConfig.logDriver === 'awsfirelens' && logDriverConfig.options && logDriverConfig.options.Name; if (logName === 'cloudwatch') { task.addToTaskRolePolicy(new iam.PolicyStatement({ actions: [ 'logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:DescribeLogStreams', 'logs:PutLogEvents', ], resources: ['*'], })); } else if (logName === 'firehose') { task.addToTaskRolePolicy(new iam.PolicyStatement({ actions: [ 'firehose:PutRecordBatch', ], resources: ['*'], })); } else if (logName === 'kinesis') { task.addToTaskRolePolicy(new iam.PolicyStatement({ actions: [ 'kinesis:PutRecords', ], resources: ['*'], })); } const fluentBitImageTag = imageTag || 'latest'; const fluentBitImage = `${fluentBitImageSSMPath}/${fluentBitImageTag}`; // Not use ContainerImage.fromEcrRepository since it's not support parsing ECR repo URI, // use repo ARN might result in complex Fn:: functions in cloudformation template. return container_image_1.ContainerImage.fromRegistry(ssm.StringParameter.valueForStringParameter(task, fluentBitImage)); } exports.obtainDefaultFluentBitECRImage = obtainDefaultFluentBitECRImage; /** * Firelens log router */ class FirelensLogRouter extends container_definition_1.ContainerDefinition { /** * Constructs a new instance of the FirelensLogRouter class. */ constructor(scope, id, props) { super(scope, id, props); jsiiDeprecationWarnings._aws_cdk_aws_ecs_FirelensLogRouterProps(props); const options = props.firelensConfig.options; if (options) { const enableECSLogMetadata = options.enableECSLogMetadata || options.enableECSLogMetadata === undefined; const configFileType = (options.configFileType === undefined || options.configFileType === FirelensConfigFileType.S3) && (cdk.Token.isUnresolved(options.configFileValue) || /arn:aws[a-zA-Z-]*:s3:::.+/.test(options.configFileValue)) ? FirelensConfigFileType.S3 : FirelensConfigFileType.FILE; this.firelensConfig = { type: props.firelensConfig.type, options: { enableECSLogMetadata, configFileType, configFileValue: options.configFileValue, }, }; // grant s3 access permissions if (configFileType === FirelensConfigFileType.S3) { props.taskDefinition.addToExecutionRolePolicy(new iam.PolicyStatement({ actions: [ 's3:GetObject', ], resources: [options.configFileValue], })); props.taskDefinition.addToExecutionRolePolicy(new iam.PolicyStatement({ actions: [ 's3:GetBucketLocation', ], resources: [options.configFileValue.split('/')[0]], })); } } else { this.firelensConfig = props.firelensConfig; } } /** * Render this container definition to a CloudFormation object */ renderContainerDefinition(_taskDefinition) { jsiiDeprecationWarnings._aws_cdk_aws_ecs_TaskDefinition(_taskDefinition); return { ...(super.renderContainerDefinition()), firelensConfiguration: this.firelensConfig && renderFirelensConfig(this.firelensConfig), }; } } exports.FirelensLogRouter = FirelensLogRouter; _a = JSII_RTTI_SYMBOL_1; FirelensLogRouter[_a] = { fqn: "@aws-cdk/aws-ecs.FirelensLogRouter", version: "1.149.0" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firelens-log-router.js","sourceRoot":"","sources":["firelens-log-router.ts"],"names":[],"mappings":"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","sourcesContent":["import * as iam from '@aws-cdk/aws-iam';\nimport * as ssm from '@aws-cdk/aws-ssm';\nimport * as cdk from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { TaskDefinition } from './base/task-definition';\nimport { ContainerDefinition, ContainerDefinitionOptions, ContainerDefinitionProps } from './container-definition';\nimport { ContainerImage } from './container-image';\nimport { CfnTaskDefinition } from './ecs.generated';\nimport { LogDriverConfig } from './log-drivers/log-driver';\n\n/**\n * Firelens log router type, fluentbit or fluentd.\n * https://docs.aws.amazon.com/AmazonECS/latest/developerguide/using_firelens.html\n */\nexport enum FirelensLogRouterType {\n  /**\n   * fluentbit\n   */\n  FLUENTBIT = 'fluentbit',\n\n  /**\n   * fluentd\n   */\n  FLUENTD = 'fluentd',\n}\n\n/**\n * Firelens configuration file type, s3 or file path.\n * https://docs.aws.amazon.com/AmazonECS/latest/developerguide/using_firelens.html#firelens-taskdef-customconfig\n */\nexport enum FirelensConfigFileType {\n  /**\n   * s3\n   */\n  S3 = 's3',\n\n  /**\n   * fluentd\n   */\n  FILE = 'file',\n}\n\n/**\n * The options for firelens log router\n * https://docs.aws.amazon.com/AmazonECS/latest/developerguide/using_firelens.html#firelens-taskdef-customconfig\n */\nexport interface FirelensOptions {\n  /**\n   * By default, Amazon ECS adds additional fields in your log entries that help identify the source of the logs.\n   * You can disable this action by setting enable-ecs-log-metadata to false.\n   * @default - true\n   */\n  readonly enableECSLogMetadata?: boolean;\n\n  /**\n   * Custom configuration file, s3 or file\n   * @default - determined by checking configFileValue with S3 ARN.\n   */\n  readonly configFileType?: FirelensConfigFileType;\n\n  /**\n   * Custom configuration file, S3 ARN or a file path\n   */\n  readonly configFileValue: string;\n}\n\n/**\n * Firelens Configuration\n * https://docs.aws.amazon.com/AmazonECS/latest/developerguide/using_firelens.html#firelens-taskdef\n */\nexport interface FirelensConfig {\n\n  /**\n   * The log router to use\n   * @default - fluentbit\n   */\n  readonly type: FirelensLogRouterType;\n\n  /**\n   * Firelens options\n   * @default - no additional options\n   */\n  readonly options?: FirelensOptions;\n}\n\n/**\n * The properties in a firelens log router.\n */\nexport interface FirelensLogRouterProps extends ContainerDefinitionProps {\n  /**\n   * Firelens configuration\n   */\n  readonly firelensConfig: FirelensConfig;\n}\n\n/**\n * The options for creating a firelens log router.\n */\nexport interface FirelensLogRouterDefinitionOptions extends ContainerDefinitionOptions {\n  /**\n   * Firelens configuration\n   */\n  readonly firelensConfig: FirelensConfig;\n}\n\n/**\n * Render to CfnTaskDefinition.FirelensConfigurationProperty from FirelensConfig\n */\nfunction renderFirelensConfig(firelensConfig: FirelensConfig): CfnTaskDefinition.FirelensConfigurationProperty {\n  if (!firelensConfig.options) {\n    return { type: firelensConfig.type };\n  } else {\n    // firelensConfig.options.configFileType has been filled with s3 or file type in constructor.\n    return {\n      type: firelensConfig.type,\n      options: {\n        'enable-ecs-log-metadata': firelensConfig.options.enableECSLogMetadata ? 'true' : 'false',\n        'config-file-type': firelensConfig.options.configFileType!,\n        'config-file-value': firelensConfig.options.configFileValue,\n      },\n    };\n  }\n\n}\n\n/**\n * SSM parameters for latest fluent bit docker image in ECR\n * https://github.com/aws/aws-for-fluent-bit#using-ssm-to-find-available-versions\n */\nconst fluentBitImageSSMPath = '/aws/service/aws-for-fluent-bit';\n\n/**\n * Obtain Fluent Bit image in Amazon ECR and setup corresponding IAM permissions.\n * ECR image pull permissions will be granted in task execution role.\n * Cloudwatch logs, Kinesis data stream or firehose permissions will be grant by check options in logDriverConfig.\n * https://docs.aws.amazon.com/AmazonECS/latest/developerguide/using_firelens.html#firelens-using-fluentbit\n */\nexport function obtainDefaultFluentBitECRImage(task: TaskDefinition, logDriverConfig?: LogDriverConfig, imageTag?: string): ContainerImage {\n  // grant ECR image pull permissions to executor role\n  task.addToExecutionRolePolicy(new iam.PolicyStatement({\n    actions: [\n      'ecr:GetAuthorizationToken',\n      'ecr:BatchCheckLayerAvailability',\n      'ecr:GetDownloadUrlForLayer',\n      'ecr:BatchGetImage',\n    ],\n    resources: ['*'],\n  }));\n\n  // grant cloudwatch or firehose permissions to task role\n  const logName = logDriverConfig && logDriverConfig.logDriver === 'awsfirelens'\n    && logDriverConfig.options && logDriverConfig.options.Name;\n  if (logName === 'cloudwatch') {\n    task.addToTaskRolePolicy(new iam.PolicyStatement({\n      actions: [\n        'logs:CreateLogGroup',\n        'logs:CreateLogStream',\n        'logs:DescribeLogStreams',\n        'logs:PutLogEvents',\n      ],\n      resources: ['*'],\n    }));\n  } else if (logName === 'firehose') {\n    task.addToTaskRolePolicy(new iam.PolicyStatement({\n      actions: [\n        'firehose:PutRecordBatch',\n      ],\n      resources: ['*'],\n    }));\n  } else if (logName === 'kinesis') {\n    task.addToTaskRolePolicy(new iam.PolicyStatement({\n      actions: [\n        'kinesis:PutRecords',\n      ],\n      resources: ['*'],\n    }));\n  }\n\n  const fluentBitImageTag = imageTag || 'latest';\n  const fluentBitImage = `${fluentBitImageSSMPath}/${fluentBitImageTag}`;\n\n  // Not use ContainerImage.fromEcrRepository since it's not support parsing ECR repo URI,\n  // use repo ARN might result in complex Fn:: functions in cloudformation template.\n  return ContainerImage.fromRegistry(ssm.StringParameter.valueForStringParameter(task, fluentBitImage));\n}\n\n/**\n * Firelens log router\n */\nexport class FirelensLogRouter extends ContainerDefinition {\n\n  /**\n   * Firelens configuration\n   */\n  public readonly firelensConfig: FirelensConfig;\n\n  /**\n   * Constructs a new instance of the FirelensLogRouter class.\n   */\n  constructor(scope: Construct, id: string, props: FirelensLogRouterProps) {\n    super(scope, id, props);\n    const options = props.firelensConfig.options;\n    if (options) {\n      const enableECSLogMetadata = options.enableECSLogMetadata || options.enableECSLogMetadata === undefined;\n      const configFileType = (options.configFileType === undefined || options.configFileType === FirelensConfigFileType.S3) &&\n        (cdk.Token.isUnresolved(options.configFileValue) || /arn:aws[a-zA-Z-]*:s3:::.+/.test(options.configFileValue))\n        ? FirelensConfigFileType.S3 : FirelensConfigFileType.FILE;\n      this.firelensConfig = {\n        type: props.firelensConfig.type,\n        options: {\n          enableECSLogMetadata,\n          configFileType,\n          configFileValue: options.configFileValue,\n        },\n      };\n\n      // grant s3 access permissions\n      if (configFileType === FirelensConfigFileType.S3) {\n        props.taskDefinition.addToExecutionRolePolicy(new iam.PolicyStatement({\n          actions: [\n            's3:GetObject',\n          ],\n          resources: [options.configFileValue],\n        }));\n        props.taskDefinition.addToExecutionRolePolicy(new iam.PolicyStatement({\n          actions: [\n            's3:GetBucketLocation',\n          ],\n          resources: [options.configFileValue.split('/')[0]],\n        }));\n      }\n    } else {\n      this.firelensConfig = props.firelensConfig;\n    }\n  }\n\n  /**\n   * Render this container definition to a CloudFormation object\n   */\n  public renderContainerDefinition(_taskDefinition?: TaskDefinition): CfnTaskDefinition.ContainerDefinitionProperty {\n    return {\n      ...(super.renderContainerDefinition()),\n      firelensConfiguration: this.firelensConfig && renderFirelensConfig(this.firelensConfig),\n    };\n  }\n}\n"]}