@aws-cdk/aws-eks-v2-alpha
Version:
The CDK Construct Library for AWS::EKS
143 lines • 20.4 kB
JavaScript
;
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"]}