UNPKG

@aws-cdk/aws-eks-v2-alpha

Version:

The CDK Construct Library for AWS::EKS

143 lines 20.4 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.KubectlProvider = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const constructs_1 = require("constructs"); const cluster_1 = require("./cluster"); const iam = require("aws-cdk-lib/aws-iam"); const lambda = require("aws-cdk-lib/aws-lambda"); const core_1 = require("aws-cdk-lib/core"); const cr = require("aws-cdk-lib/custom-resources"); const lambda_layer_awscli_1 = require("aws-cdk-lib/lambda-layer-awscli"); const path = require("path"); /** * Implementation of Kubectl Lambda */ class KubectlProvider extends constructs_1.Construct { static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/aws-eks-v2-alpha.KubectlProvider", version: "2.222.0-alpha.0" }; /** * Take existing provider on cluster * * @param scope Construct * @param cluster k8s cluster */ static getKubectlProvider(scope, cluster) { try { jsiiDeprecationWarnings._aws_cdk_aws_eks_v2_alpha_ICluster(cluster); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.getKubectlProvider); } throw error; } // if this is an "owned" cluster, we need to wait for the kubectl barrier // before applying any resources. if (cluster instanceof cluster_1.Cluster) { cluster._dependOnKubectlBarrier(scope); } return cluster.kubectlProvider; } /** * Import an existing provider * * @param scope Construct * @param id an id of resource * @param attrs attributes for the provider */ static fromKubectlProviderAttributes(scope, id, attrs) { try { jsiiDeprecationWarnings._aws_cdk_aws_eks_v2_alpha_KubectlProviderAttributes(attrs); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.fromKubectlProviderAttributes); } throw error; } class Import extends constructs_1.Construct { serviceToken = attrs.serviceToken; role = attrs.role; } return new Import(scope, id); } /** * The custom resource provider's service token. */ serviceToken; /** * The IAM execution role of the handler. */ role; constructor(scope, id, props) { super(scope, id); try { jsiiDeprecationWarnings._aws_cdk_aws_eks_v2_alpha_KubectlProviderProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, KubectlProvider); } throw error; } const vpc = props.privateSubnets ? props.cluster.vpc : undefined; let securityGroups; if (props.privateSubnets && props.cluster.clusterSecurityGroup) { securityGroups = [props.cluster.clusterSecurityGroup]; } const privateSubnets = props.privateSubnets ? { subnets: props.privateSubnets } : undefined; const handler = new lambda.Function(this, 'Handler', { timeout: core_1.Duration.minutes(15), description: 'onEvent handler for EKS kubectl resource provider', memorySize: props.memory?.toMebibytes() ?? 1024, environment: { // required and recommended for boto3 AWS_STS_REGIONAL_ENDPOINTS: 'regional', ...props.environment, }, role: props.role, code: lambda.Code.fromAsset(path.join(__dirname, 'kubectl-handler')), handler: 'index.handler', runtime: lambda.Runtime.determineLatestPythonRuntime(this), // defined only when using private access vpc, securityGroups, vpcSubnets: privateSubnets, }); // allow user to customize the layers with the tools we need handler.addLayers(props.awscliLayer ?? new lambda_layer_awscli_1.AwsCliLayer(this, 'AwsCliLayer')); handler.addLayers(props.kubectlLayer); const handlerRole = handler.role; handlerRole.addToPrincipalPolicy(new iam.PolicyStatement({ actions: ['eks:DescribeCluster'], resources: [props.cluster.clusterArn], })); // taken from the lambda default role logic. // makes it easier for roles to be passed in. if (handler.isBoundToVpc) { handlerRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaVPCAccessExecutionRole')); } // For OCI helm chart authorization. handlerRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonEC2ContainerRegistryReadOnly')); /** * For OCI helm chart public ECR authorization. As ECR public is only available in `aws` partition, * we conditionally attach this policy when the AWS partition is `aws`. */ const hasEcrPublicCondition = new core_1.CfnCondition(handlerRole.node.scope, 'HasEcrPublic', { expression: core_1.Fn.conditionEquals(core_1.Aws.PARTITION, 'aws'), }); const conditionalPolicy = iam.ManagedPolicy.fromManagedPolicyArn(this, 'ConditionalPolicyArn', core_1.Fn.conditionIf(hasEcrPublicCondition.logicalId, iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonElasticContainerRegistryPublicReadOnly').managedPolicyArn, core_1.Aws.NO_VALUE).toString()); handlerRole.addManagedPolicy(iam.ManagedPolicy.fromManagedPolicyArn(this, 'conditionalPolicy', conditionalPolicy.managedPolicyArn)); const provider = new cr.Provider(this, 'Provider', { onEventHandler: handler, vpc: props.cluster.vpc, vpcSubnets: privateSubnets, securityGroups, }); this.serviceToken = provider.serviceToken; this.role = handlerRole; } } exports.KubectlProvider = KubectlProvider; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kubectl-provider.js","sourceRoot":"","sources":["kubectl-provider.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct, IConstruct } from 'constructs';\nimport { Cluster, ICluster } from './cluster';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport { Duration, CfnCondition, Fn, Aws, Size } from 'aws-cdk-lib/core';\nimport * as cr from 'aws-cdk-lib/custom-resources';\nimport { AwsCliLayer } from 'aws-cdk-lib/lambda-layer-awscli';\nimport * as path from 'path';\n\nexport interface KubectlProviderOptions {\n  /**\n   * An IAM role that can perform kubectl operations against this cluster.\n   *\n   * The role should be mapped to the `system:masters` Kubernetes RBAC role.\n   *\n   * This role is directly passed to the lambda handler that sends Kube Ctl commands to the cluster.\n   * @default - if not specified, the default role created by a lambda function will\n   * be used.\n   */\n  readonly role?: iam.IRole;\n\n  /**\n   * An AWS Lambda layer that contains the `aws` CLI.\n   *\n   * If not defined, a default layer will be used containing the AWS CLI 2.x.\n   */\n  readonly awscliLayer?: lambda.ILayerVersion;\n\n  /**\n   *\n   * Custom environment variables when running `kubectl` against this cluster.\n   */\n  readonly environment?: { [key: string]: string };\n\n  /**\n   * A security group to use for `kubectl` execution.\n   *\n   * @default - If not specified, the k8s endpoint is expected to be accessible\n   * publicly.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * The amount of memory allocated to the kubectl provider's lambda function.\n   */\n  readonly memory?: Size;\n\n  /**\n   * An AWS Lambda layer that includes `kubectl` and `helm`\n   */\n  readonly kubectlLayer: lambda.ILayerVersion;\n\n  /**\n   * Subnets to host the `kubectl` compute resources. If not specified, the k8s\n   * endpoint is expected to be accessible publicly.\n   */\n  readonly privateSubnets?: ec2.ISubnet[];\n}\n\n/**\n * Properties for a KubectlProvider\n */\nexport interface KubectlProviderProps extends KubectlProviderOptions {\n  /**\n   * The cluster to control.\n   */\n  readonly cluster: ICluster;\n}\n\n/**\n * Kubectl Provider Attributes\n */\nexport interface KubectlProviderAttributes {\n  /**\n   * The kubectl provider lambda arn\n   */\n  readonly serviceToken: string;\n\n  /**\n   * The role of the provider lambda function.\n   * Only required if you deploy helm charts using this imported provider.\n   *\n   * @default - no role.\n   */\n  readonly role?: iam.IRole;\n}\n\n/**\n * Imported KubectlProvider that can be used in place of the default one created by CDK\n */\nexport interface IKubectlProvider extends IConstruct {\n  /**\n   * The custom resource provider's service token.\n   */\n  readonly serviceToken: string;\n\n  /**\n   * The role of the provider lambda function. If undefined,\n   * you cannot use this provider to deploy helm charts.\n   */\n  readonly role?: iam.IRole;\n}\n\n/**\n * Implementation of Kubectl Lambda\n */\nexport class KubectlProvider extends Construct implements IKubectlProvider {\n  /**\n   * Take existing provider on cluster\n   *\n   * @param scope Construct\n   * @param cluster k8s cluster\n   */\n  public static getKubectlProvider(scope: Construct, cluster: ICluster) {\n    // if this is an \"owned\" cluster, we need to wait for the kubectl barrier\n    // before applying any resources.\n    if (cluster instanceof Cluster) {\n      cluster._dependOnKubectlBarrier(scope);\n    }\n\n    return cluster.kubectlProvider;\n  }\n\n  /**\n   * Import an existing provider\n   *\n   * @param scope Construct\n   * @param id an id of resource\n   * @param attrs attributes for the provider\n   */\n  public static fromKubectlProviderAttributes(scope: Construct, id: string, attrs: KubectlProviderAttributes): IKubectlProvider {\n    class Import extends Construct implements IKubectlProvider {\n      public readonly serviceToken: string = attrs.serviceToken;\n      public readonly role?: iam.IRole = attrs.role;\n    }\n    return new Import(scope, id);\n  }\n\n  /**\n   * The custom resource provider's service token.\n   */\n  public readonly serviceToken: string;\n\n  /**\n   * The IAM execution role of the handler.\n   */\n  public readonly role?: iam.IRole;\n\n  public constructor(scope: Construct, id: string, props: KubectlProviderProps) {\n    super(scope, id);\n\n    const vpc = props.privateSubnets ? props.cluster.vpc : undefined;\n    let securityGroups;\n    if (props.privateSubnets && props.cluster.clusterSecurityGroup) {\n      securityGroups = [props.cluster.clusterSecurityGroup];\n    }\n    const privateSubnets = props.privateSubnets ? { subnets: props.privateSubnets } : undefined;\n\n    const handler = new lambda.Function(this, 'Handler', {\n      timeout: Duration.minutes(15),\n      description: 'onEvent handler for EKS kubectl resource provider',\n      memorySize: props.memory?.toMebibytes() ?? 1024,\n      environment: {\n        // required and recommended for boto3\n        AWS_STS_REGIONAL_ENDPOINTS: 'regional',\n        ...props.environment,\n      },\n      role: props.role,\n      code: lambda.Code.fromAsset(path.join(__dirname, 'kubectl-handler')),\n      handler: 'index.handler',\n      runtime: lambda.Runtime.determineLatestPythonRuntime(this),\n      // defined only when using private access\n      vpc,\n      securityGroups,\n      vpcSubnets: privateSubnets,\n    });\n\n    // allow user to customize the layers with the tools we need\n    handler.addLayers(props.awscliLayer ?? new AwsCliLayer(this, 'AwsCliLayer'));\n    handler.addLayers(props.kubectlLayer);\n\n    const handlerRole = handler.role!;\n\n    handlerRole.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['eks:DescribeCluster'],\n      resources: [props.cluster.clusterArn],\n    }));\n\n    // taken from the lambda default role logic.\n    // makes it easier for roles to be passed in.\n    if (handler.isBoundToVpc) {\n      handlerRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaVPCAccessExecutionRole'));\n    }\n\n    // For OCI helm chart authorization.\n    handlerRole.addManagedPolicy(\n      iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonEC2ContainerRegistryReadOnly'),\n    );\n\n    /**\n     * For OCI helm chart public ECR authorization. As ECR public is only available in `aws` partition,\n     * we conditionally attach this policy when the AWS partition is `aws`.\n     */\n    const hasEcrPublicCondition = new CfnCondition(handlerRole.node.scope!, 'HasEcrPublic', {\n      expression: Fn.conditionEquals(Aws.PARTITION, 'aws'),\n    });\n\n    const conditionalPolicy = iam.ManagedPolicy.fromManagedPolicyArn(this, 'ConditionalPolicyArn',\n      Fn.conditionIf(hasEcrPublicCondition.logicalId,\n        iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonElasticContainerRegistryPublicReadOnly').managedPolicyArn,\n        Aws.NO_VALUE).toString(),\n    );\n\n    handlerRole.addManagedPolicy(iam.ManagedPolicy.fromManagedPolicyArn(this, 'conditionalPolicy', conditionalPolicy.managedPolicyArn));\n\n    const provider = new cr.Provider(this, 'Provider', {\n      onEventHandler: handler,\n      vpc: props.cluster.vpc,\n      vpcSubnets: privateSubnets,\n      securityGroups,\n    });\n\n    this.serviceToken = provider.serviceToken;\n    this.role = handlerRole;\n  }\n}\n"]}