UNPKG

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

Version:

The CDK Construct Library for AWS::EKS

191 lines 27.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.ServiceAccount = exports.IdentityType = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const constructs_1 = require("constructs"); // import { FargateCluster } from './index'; const aws_eks_1 = require("aws-cdk-lib/aws-eks"); const k8s_manifest_1 = require("./k8s-manifest"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const core_1 = require("aws-cdk-lib/core"); /** * Enum representing the different identity types that can be used for a Kubernetes service account. */ var IdentityType; (function (IdentityType) { /** * Use the IAM Roles for Service Accounts (IRSA) identity type. * IRSA allows you to associate an IAM role with a Kubernetes service account. * This provides a way to grant permissions to Kubernetes pods by associating an IAM role with a Kubernetes service account. * The IAM role can then be used to provide AWS credentials to the pods, allowing them to access other AWS resources. * * When enabled, the openIdConnectProvider of the cluster would be created when you create the ServiceAccount. * * @see https://docs.aws.amazon.com/eks/latest/userguide/iam-roles-for-service-accounts.html */ IdentityType["IRSA"] = "IRSA"; /** * Use the EKS Pod Identities identity type. * EKS Pod Identities provide the ability to manage credentials for your applications, similar to the way that Amazon EC2 instance profiles * provide credentials to Amazon EC2 instances. Instead of creating and distributing your AWS credentials to the containers or using the * Amazon EC2 instance's role, you associate an IAM role with a Kubernetes service account and configure your Pods to use the service account. * * When enabled, the Pod Identity Agent AddOn of the cluster would be created when you create the ServiceAccount. * * @see https://docs.aws.amazon.com/eks/latest/userguide/pod-identities.html */ IdentityType["POD_IDENTITY"] = "POD_IDENTITY"; })(IdentityType || (exports.IdentityType = IdentityType = {})); /** * Service Account */ class ServiceAccount extends constructs_1.Construct { static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/aws-eks-v2-alpha.ServiceAccount", version: "2.222.0-alpha.0" }; /** * The role which is linked to the service account. */ role; assumeRoleAction; grantPrincipal; policyFragment; /** * The name of the service account. */ serviceAccountName; /** * The namespace where the service account is located in. */ serviceAccountNamespace; constructor(scope, id, props) { super(scope, id); try { jsiiDeprecationWarnings._aws_cdk_aws_eks_v2_alpha_ServiceAccountProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, ServiceAccount); } throw error; } const { cluster } = props; this.serviceAccountName = props.name ?? core_1.Names.uniqueId(this).toLowerCase(); this.serviceAccountNamespace = props.namespace ?? 'default'; // From K8s docs: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/ if (!this.isValidDnsSubdomainName(this.serviceAccountName)) { throw RangeError('The name of a ServiceAccount object must be a valid DNS subdomain name.'); } // From K8s docs: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/#namespaces-and-dns if (!this.isValidDnsLabelName(this.serviceAccountNamespace)) { throw RangeError('All namespace names must be valid RFC 1123 DNS labels.'); } let principal; if (props.identityType !== IdentityType.POD_IDENTITY) { /* Add conditions to the role to improve security. This prevents other pods in the same namespace to assume the role. * See documentation: https://docs.aws.amazon.com/eks/latest/userguide/create-service-account-iam-policy-and-role.html */ const conditions = new core_1.CfnJson(this, 'ConditionJson', { value: { [`${cluster.openIdConnectProvider.openIdConnectProviderIssuer}:aud`]: 'sts.amazonaws.com', [`${cluster.openIdConnectProvider.openIdConnectProviderIssuer}:sub`]: `system:serviceaccount:${this.serviceAccountNamespace}:${this.serviceAccountName}`, }, }); principal = new aws_iam_1.OpenIdConnectPrincipal(cluster.openIdConnectProvider).withConditions({ StringEquals: conditions, }); } else { /** * Identity type is POD_IDENTITY. * Create a service principal with "Service": "pods.eks.amazonaws.com" * See https://docs.aws.amazon.com/eks/latest/userguide/pod-identities.html */ // EKS Pod Identity does not support Fargate // TODO: raise an error when using Fargate principal = new aws_iam_1.ServicePrincipal('pods.eks.amazonaws.com'); } const role = new aws_iam_1.Role(this, 'Role', { assumedBy: principal }); // pod identities requires 'sts:TagSession' in its principal actions if (props.identityType === IdentityType.POD_IDENTITY) { /** * EKS Pod Identities requires both assumed role actions otherwise it would fail. */ role.assumeRolePolicy.addStatements(new aws_iam_1.PolicyStatement({ actions: ['sts:AssumeRole', 'sts:TagSession'], principals: [new aws_iam_1.ServicePrincipal('pods.eks.amazonaws.com')], })); // ensure the pod identity agent cluster.eksPodIdentityAgent; // associate this service account with the pod role we just created for the cluster new aws_eks_1.CfnPodIdentityAssociation(this, 'Association', { clusterName: cluster.clusterName, namespace: props.namespace ?? 'default', roleArn: role.roleArn, serviceAccount: this.serviceAccountName, }); } this.role = role; this.assumeRoleAction = this.role.assumeRoleAction; this.grantPrincipal = this.role.grantPrincipal; this.policyFragment = this.role.policyFragment; // Note that we cannot use `cluster.addManifest` here because that would create the manifest // constrct in the scope of the cluster stack, which might be a different stack than this one. // This means that the cluster stack would depend on this stack because of the role, // and since this stack inherintely depends on the cluster stack, we will have a circular dependency. new k8s_manifest_1.KubernetesManifest(this, `manifest-${id}ServiceAccountResource`, { cluster, manifest: [{ apiVersion: 'v1', kind: 'ServiceAccount', metadata: { name: this.serviceAccountName, namespace: this.serviceAccountNamespace, labels: { 'app.kubernetes.io/name': this.serviceAccountName, ...props.labels, }, annotations: { 'eks.amazonaws.com/role-arn': this.role.roleArn, ...props.annotations, }, }, }], }); } /** * @deprecated use `addToPrincipalPolicy()` */ addToPolicy(statement) { try { jsiiDeprecationWarnings.print("@aws-cdk/aws-eks-v2-alpha.ServiceAccount#addToPolicy", "use `addToPrincipalPolicy()`"); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.addToPolicy); } throw error; } return this.addToPrincipalPolicy(statement).statementAdded; } addToPrincipalPolicy(statement) { return this.role.addToPrincipalPolicy(statement); } /** * If the value is a DNS subdomain name as defined in RFC 1123, from K8s docs. * * https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#dns-subdomain-names */ isValidDnsSubdomainName(value) { return value.length <= 253 && /^[a-z0-9]+[a-z0-9-.]*[a-z0-9]+$/.test(value); } /** * If the value follows DNS label standard as defined in RFC 1123, from K8s docs. * * https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#dns-label-names */ isValidDnsLabelName(value) { return value.length <= 63 && /^[a-z0-9]+[a-z0-9-]*[a-z0-9]+$/.test(value); } } exports.ServiceAccount = ServiceAccount; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service-account.js","sourceRoot":"","sources":["service-account.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from 'constructs';\nimport { ICluster } from './cluster';\n// import { FargateCluster } from './index';\nimport { CfnPodIdentityAssociation } from 'aws-cdk-lib/aws-eks';\nimport { KubernetesManifest } from './k8s-manifest';\nimport {\n  AddToPrincipalPolicyResult, IPrincipal, IRole, OpenIdConnectPrincipal, PolicyStatement, PrincipalPolicyFragment, Role,\n  ServicePrincipal,\n} from 'aws-cdk-lib/aws-iam';\nimport { CfnJson, Names } from 'aws-cdk-lib/core';\n\n/**\n * Enum representing the different identity types that can be used for a Kubernetes service account.\n */\nexport enum IdentityType {\n  /**\n   * Use the IAM Roles for Service Accounts (IRSA) identity type.\n   * IRSA allows you to associate an IAM role with a Kubernetes service account.\n   * This provides a way to grant permissions to Kubernetes pods by associating an IAM role with a Kubernetes service account.\n   * The IAM role can then be used to provide AWS credentials to the pods, allowing them to access other AWS resources.\n   *\n   * When enabled, the openIdConnectProvider of the cluster would be created when you create the ServiceAccount.\n   *\n   * @see https://docs.aws.amazon.com/eks/latest/userguide/iam-roles-for-service-accounts.html\n   */\n  IRSA = 'IRSA',\n\n  /**\n   * Use the EKS Pod Identities identity type.\n   * EKS Pod Identities provide the ability to manage credentials for your applications, similar to the way that Amazon EC2 instance profiles\n   * provide credentials to Amazon EC2 instances. Instead of creating and distributing your AWS credentials to the containers or using the\n   * Amazon EC2 instance's role, you associate an IAM role with a Kubernetes service account and configure your Pods to use the service account.\n   *\n   * When enabled, the Pod Identity Agent AddOn of the cluster would be created when you create the ServiceAccount.\n   *\n   * @see https://docs.aws.amazon.com/eks/latest/userguide/pod-identities.html\n   */\n  POD_IDENTITY = 'POD_IDENTITY',\n}\n\n/**\n * Options for `ServiceAccount`\n */\nexport interface ServiceAccountOptions {\n  /**\n   * The name of the service account.\n   *\n   * The name of a ServiceAccount object must be a valid DNS subdomain name.\n   * https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/\n   * @default - If no name is given, it will use the id of the resource.\n   */\n  readonly name?: string;\n\n  /**\n   * The namespace of the service account.\n   *\n   * All namespace names must be valid RFC 1123 DNS labels.\n   * https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/#namespaces-and-dns\n   * @default \"default\"\n   */\n  readonly namespace?: string;\n\n  /**\n   * Additional annotations of the service account.\n   *\n   * @default - no additional annotations\n   */\n  readonly annotations?: { [key: string]: string };\n\n  /**\n   * Additional labels of the service account.\n   *\n   * @default - no additional labels\n   */\n  readonly labels?: { [key: string]: string };\n\n  /**\n   * The identity type to use for the service account.\n   * @default IdentityType.IRSA\n   */\n  readonly identityType?: IdentityType;\n}\nexport interface ServiceAccountOptions {\n  /**\n   * The name of the service account.\n   *\n   * The name of a ServiceAccount object must be a valid DNS subdomain name.\n   * https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/\n   * @default - If no name is given, it will use the id of the resource.\n   */\n  readonly name?: string;\n\n  /**\n   * The namespace of the service account.\n   *\n   * All namespace names must be valid RFC 1123 DNS labels.\n   * https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/#namespaces-and-dns\n   * @default \"default\"\n   */\n  readonly namespace?: string;\n\n  /**\n   * Additional annotations of the service account.\n   *\n   * @default - no additional annotations\n   */\n  readonly annotations?: {[key:string]: string};\n\n  /**\n   * Additional labels of the service account.\n   *\n   * @default - no additional labels\n   */\n  readonly labels?: {[key:string]: string};\n}\n\n/**\n * Properties for defining service accounts\n */\nexport interface ServiceAccountProps extends ServiceAccountOptions {\n  /**\n   * The cluster to apply the patch to.\n   */\n  readonly cluster: ICluster;\n}\n\n/**\n * Service Account\n */\nexport class ServiceAccount extends Construct implements IPrincipal {\n  /**\n   * The role which is linked to the service account.\n   */\n  public readonly role: IRole;\n\n  public readonly assumeRoleAction: string;\n  public readonly grantPrincipal: IPrincipal;\n  public readonly policyFragment: PrincipalPolicyFragment;\n\n  /**\n   * The name of the service account.\n   */\n  public readonly serviceAccountName: string;\n\n  /**\n   * The namespace where the service account is located in.\n   */\n  public readonly serviceAccountNamespace: string;\n\n  constructor(scope: Construct, id: string, props: ServiceAccountProps) {\n    super(scope, id);\n\n    const { cluster } = props;\n    this.serviceAccountName = props.name ?? Names.uniqueId(this).toLowerCase();\n    this.serviceAccountNamespace = props.namespace ?? 'default';\n\n    // From K8s docs: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/\n    if (!this.isValidDnsSubdomainName(this.serviceAccountName)) {\n      throw RangeError('The name of a ServiceAccount object must be a valid DNS subdomain name.');\n    }\n\n    // From K8s docs: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/#namespaces-and-dns\n    if (!this.isValidDnsLabelName(this.serviceAccountNamespace)) {\n      throw RangeError('All namespace names must be valid RFC 1123 DNS labels.');\n    }\n\n    let principal: IPrincipal;\n    if (props.identityType !== IdentityType.POD_IDENTITY) {\n      /* Add conditions to the role to improve security. This prevents other pods in the same namespace to assume the role.\n      * See documentation: https://docs.aws.amazon.com/eks/latest/userguide/create-service-account-iam-policy-and-role.html\n      */\n      const conditions = new CfnJson(this, 'ConditionJson', {\n        value: {\n          [`${cluster.openIdConnectProvider.openIdConnectProviderIssuer}:aud`]: 'sts.amazonaws.com',\n          [`${cluster.openIdConnectProvider.openIdConnectProviderIssuer}:sub`]: `system:serviceaccount:${this.serviceAccountNamespace}:${this.serviceAccountName}`,\n        },\n      });\n      principal = new OpenIdConnectPrincipal(cluster.openIdConnectProvider).withConditions({\n        StringEquals: conditions,\n      });\n    } else {\n      /**\n       * Identity type is POD_IDENTITY.\n       * Create a service principal with \"Service\": \"pods.eks.amazonaws.com\"\n       * See https://docs.aws.amazon.com/eks/latest/userguide/pod-identities.html\n       */\n\n      // EKS Pod Identity does not support Fargate\n      // TODO: raise an error when using Fargate\n      principal = new ServicePrincipal('pods.eks.amazonaws.com');\n    }\n\n    const role = new Role(this, 'Role', { assumedBy: principal });\n\n    // pod identities requires 'sts:TagSession' in its principal actions\n    if (props.identityType === IdentityType.POD_IDENTITY) {\n      /**\n       * EKS Pod Identities requires both assumed role actions otherwise it would fail.\n       */\n      role.assumeRolePolicy!.addStatements(new PolicyStatement({\n        actions: ['sts:AssumeRole', 'sts:TagSession'],\n        principals: [new ServicePrincipal('pods.eks.amazonaws.com')],\n      }));\n\n      // ensure the pod identity agent\n      cluster.eksPodIdentityAgent;\n\n      // associate this service account with the pod role we just created for the cluster\n      new CfnPodIdentityAssociation(this, 'Association', {\n        clusterName: cluster.clusterName,\n        namespace: props.namespace ?? 'default',\n        roleArn: role.roleArn,\n        serviceAccount: this.serviceAccountName,\n      });\n    }\n\n    this.role = role;\n\n    this.assumeRoleAction = this.role.assumeRoleAction;\n    this.grantPrincipal = this.role.grantPrincipal;\n    this.policyFragment = this.role.policyFragment;\n\n    // Note that we cannot use `cluster.addManifest` here because that would create the manifest\n    // constrct in the scope of the cluster stack, which might be a different stack than this one.\n    // This means that the cluster stack would depend on this stack because of the role,\n    // and since this stack inherintely depends on the cluster stack, we will have a circular dependency.\n    new KubernetesManifest(this, `manifest-${id}ServiceAccountResource`, {\n      cluster,\n      manifest: [{\n        apiVersion: 'v1',\n        kind: 'ServiceAccount',\n        metadata: {\n          name: this.serviceAccountName,\n          namespace: this.serviceAccountNamespace,\n          labels: {\n            'app.kubernetes.io/name': this.serviceAccountName,\n            ...props.labels,\n          },\n          annotations: {\n            'eks.amazonaws.com/role-arn': this.role.roleArn,\n            ...props.annotations,\n          },\n        },\n      }],\n    });\n  }\n\n  /**\n   * @deprecated use `addToPrincipalPolicy()`\n   */\n  public addToPolicy(statement: PolicyStatement): boolean {\n    return this.addToPrincipalPolicy(statement).statementAdded;\n  }\n\n  public addToPrincipalPolicy(statement: PolicyStatement): AddToPrincipalPolicyResult {\n    return this.role.addToPrincipalPolicy(statement);\n  }\n\n  /**\n   * If the value is a DNS subdomain name as defined in RFC 1123, from K8s docs.\n   *\n   * https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#dns-subdomain-names\n   */\n  private isValidDnsSubdomainName(value: string): boolean {\n    return value.length <= 253 && /^[a-z0-9]+[a-z0-9-.]*[a-z0-9]+$/.test(value);\n  }\n\n  /**\n   * If the value follows DNS label standard as defined in RFC 1123, from K8s docs.\n   *\n   * https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#dns-label-names\n   */\n  private isValidDnsLabelName(value: string): boolean {\n    return value.length <= 63 && /^[a-z0-9]+[a-z0-9-]*[a-z0-9]+$/.test(value);\n  }\n}\n"]}