@aws-cdk/aws-eks-v2-alpha
Version:
The CDK Construct Library for AWS::EKS
114 lines • 17.6 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.KubernetesManifest = void 0;
const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const constructs_1 = require("constructs");
const alb_controller_1 = require("./alb-controller");
const kubectl_provider_1 = require("./kubectl-provider");
const core_1 = require("aws-cdk-lib/core");
const PRUNE_LABEL_PREFIX = 'aws.cdk.eks/prune-';
/**
* Represents a manifest within the Kubernetes system.
*
* Alternatively, you can use `cluster.addManifest(resource[, resource, ...])`
* to define resources on this cluster.
*
* Applies/deletes the manifest using `kubectl`.
*/
class KubernetesManifest extends constructs_1.Construct {
static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/aws-eks-v2-alpha.KubernetesManifest", version: "2.222.0-alpha.0" };
/**
* The CloudFormation resource type.
*/
static RESOURCE_TYPE = 'Custom::AWSCDK-EKS-KubernetesResource';
constructor(scope, id, props) {
super(scope, id);
try {
jsiiDeprecationWarnings._aws_cdk_aws_eks_v2_alpha_KubernetesManifestProps(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, KubernetesManifest);
}
throw error;
}
const stack = core_1.Stack.of(this);
const provider = kubectl_provider_1.KubectlProvider.getKubectlProvider(this, props.cluster);
if (!provider) {
throw new Error('Kubectl Provider is not defined in this cluster. Define it when creating the cluster');
}
const prune = props.prune ?? props.cluster.prune;
const pruneLabel = prune
? this.injectPruneLabel(props.manifest)
: undefined;
if (props.ingressAlb ?? false) {
this.injectIngressAlbAnnotations(props.manifest, props.ingressAlbScheme ?? alb_controller_1.AlbScheme.INTERNAL);
}
const customResource = new core_1.CustomResource(this, 'Resource', {
serviceToken: provider.serviceToken,
resourceType: KubernetesManifest.RESOURCE_TYPE,
properties: {
// `toJsonString` enables embedding CDK tokens in the manifest and will
// render a CloudFormation-compatible JSON string (similar to
// StepFunctions, CloudWatch Dashboards etc).
Manifest: stack.toJsonString(props.manifest),
ClusterName: props.cluster.clusterName,
PruneLabel: pruneLabel,
Overwrite: props.overwrite,
SkipValidation: props.skipValidation,
},
});
this.node.defaultChild = customResource.node.defaultChild;
}
/**
* Injects a generated prune label to all resources in this manifest. The
* label name will be `awscdk.eks/manifest-ADDR` where `ADDR` is the address
* of the construct in the construct tree.
*
* @returns the label name
*/
injectPruneLabel(manifest) {
// max label name is 64 chars and addrs is always 42.
const pruneLabel = PRUNE_LABEL_PREFIX + constructs_1.Node.of(this).addr;
for (const resource of manifest) {
// skip resource if it's not an object or if it does not have a "kind"
if (typeof (resource) !== 'object' || !resource.kind) {
continue;
}
if (!resource.metadata) {
resource.metadata = {};
}
if (!resource.metadata.labels) {
resource.metadata.labels = {};
}
resource.metadata.labels = {
[pruneLabel]: '',
...resource.metadata.labels,
};
}
return pruneLabel;
}
/**
* Inject the necessary ingress annontations if possible (and requested).
*
* @see https://kubernetes-sigs.github.io/aws-load-balancer-controller/v2.2/guide/ingress/annotations/
*/
injectIngressAlbAnnotations(manifest, scheme) {
for (const resource of manifest) {
// skip resource if it's not an object or if it does not have a "kind"
if (typeof (resource) !== 'object' || !resource.kind) {
continue;
}
if (resource.kind === 'Ingress') {
resource.metadata.annotations = {
'kubernetes.io/ingress.class': 'alb',
'alb.ingress.kubernetes.io/scheme': scheme,
...resource.metadata.annotations,
};
}
}
}
}
exports.KubernetesManifest = KubernetesManifest;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"k8s-manifest.js","sourceRoot":"","sources":["k8s-manifest.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct, Node } from 'constructs';\nimport { AlbScheme } from './alb-controller';\nimport { ICluster } from './cluster';\nimport { KubectlProvider } from './kubectl-provider';\nimport { CustomResource, Stack } from 'aws-cdk-lib/core';\n\nconst PRUNE_LABEL_PREFIX = 'aws.cdk.eks/prune-';\n\n/**\n * Options for `KubernetesManifest`.\n */\nexport interface KubernetesManifestOptions {\n  /**\n   * When a resource is removed from a Kubernetes manifest, it no longer appears\n   * in the manifest, and there is no way to know that this resource needs to be\n   * deleted. To address this, `kubectl apply` has a `--prune` option which will\n   * query the cluster for all resources with a specific label and will remove\n   * all the labeld resources that are not part of the applied manifest. If this\n   * option is disabled and a resource is removed, it will become \"orphaned\" and\n   * will not be deleted from the cluster.\n   *\n   * When this option is enabled (default), the construct will inject a label to\n   * all Kubernetes resources included in this manifest which will be used to\n   * prune resources when the manifest changes via `kubectl apply --prune`.\n   *\n   * The label name will be `aws.cdk.eks/prune-<ADDR>` where `<ADDR>` is the\n   * 42-char unique address of this construct in the construct tree. Value is\n   * empty.\n   *\n   * @see\n   * https://kubernetes.io/docs/tasks/manage-kubernetes-objects/declarative-config/#alternative-kubectl-apply-f-directory-prune-l-your-label\n   *\n   * @default - based on the prune option of the cluster, which is `true` unless\n   * otherwise specified.\n   */\n  readonly prune?: boolean;\n\n  /**\n   * A flag to signify if the manifest validation should be skipped\n   *\n   * @default false\n   */\n  readonly skipValidation?: boolean;\n\n  /**\n   * Automatically detect `Ingress` resources in the manifest and annotate them so they\n   * are picked up by an ALB Ingress Controller.\n   *\n   * @default false\n   */\n  readonly ingressAlb?: boolean;\n\n  /**\n   * Specify the ALB scheme that should be applied to `Ingress` resources.\n   * Only applicable if `ingressAlb` is set to `true`.\n   *\n   * @default AlbScheme.INTERNAL\n   */\n  readonly ingressAlbScheme?: AlbScheme;\n\n}\n\n/**\n * Properties for KubernetesManifest\n */\nexport interface KubernetesManifestProps extends KubernetesManifestOptions {\n  /**\n   * The EKS cluster to apply this manifest to.\n   *\n   * [disable-awslint:ref-via-interface]\n   */\n  readonly cluster: ICluster;\n\n  /**\n   * The manifest to apply.\n   *\n   * Consists of any number of child resources.\n   *\n   * When the resources are created/updated, this manifest will be applied to the\n   * cluster through `kubectl apply` and when the resources or the stack is\n   * deleted, the resources in the manifest will be deleted through `kubectl delete`.\n   *\n   * @example\n   *\n   * [{\n   *   apiVersion: 'v1',\n   *   kind: 'Pod',\n   *   metadata: { name: 'mypod' },\n   *   spec: {\n   *     containers: [ { name: 'hello', image: 'paulbouwer/hello-kubernetes:1.5', ports: [ { containerPort: 8080 } ] } ]\n   *   }\n   * }]\n   *\n   */\n  readonly manifest: Record<string, any>[];\n\n  /**\n   * Overwrite any existing resources.\n   *\n   * If this is set, we will use `kubectl apply` instead of `kubectl create`\n   * when the resource is created. Otherwise, if there is already a resource\n   * in the cluster with the same name, the operation will fail.\n   *\n   * @default false\n   */\n  readonly overwrite?: boolean;\n}\n\n/**\n * Represents a manifest within the Kubernetes system.\n *\n * Alternatively, you can use `cluster.addManifest(resource[, resource, ...])`\n * to define resources on this cluster.\n *\n * Applies/deletes the manifest using `kubectl`.\n */\nexport class KubernetesManifest extends Construct {\n  /**\n   * The CloudFormation resource type.\n   */\n  public static readonly RESOURCE_TYPE = 'Custom::AWSCDK-EKS-KubernetesResource';\n\n  constructor(scope: Construct, id: string, props: KubernetesManifestProps) {\n    super(scope, id);\n\n    const stack = Stack.of(this);\n    const provider = KubectlProvider.getKubectlProvider(this, props.cluster);\n    if (!provider) {\n      throw new Error('Kubectl Provider is not defined in this cluster. Define it when creating the cluster');\n    }\n\n    const prune = props.prune ?? props.cluster.prune;\n    const pruneLabel = prune\n      ? this.injectPruneLabel(props.manifest)\n      : undefined;\n\n    if (props.ingressAlb ?? false) {\n      this.injectIngressAlbAnnotations(props.manifest, props.ingressAlbScheme ?? AlbScheme.INTERNAL);\n    }\n\n    const customResource = new CustomResource(this, 'Resource', {\n      serviceToken: provider.serviceToken,\n      resourceType: KubernetesManifest.RESOURCE_TYPE,\n      properties: {\n        // `toJsonString` enables embedding CDK tokens in the manifest and will\n        // render a CloudFormation-compatible JSON string (similar to\n        // StepFunctions, CloudWatch Dashboards etc).\n        Manifest: stack.toJsonString(props.manifest),\n        ClusterName: props.cluster.clusterName,\n        PruneLabel: pruneLabel,\n        Overwrite: props.overwrite,\n        SkipValidation: props.skipValidation,\n      },\n    });\n\n    this.node.defaultChild = customResource.node.defaultChild;\n  }\n\n  /**\n   * Injects a generated prune label to all resources in this manifest. The\n   * label name will be `awscdk.eks/manifest-ADDR` where `ADDR` is the address\n   * of the construct in the construct tree.\n   *\n   * @returns the label name\n   */\n  private injectPruneLabel(manifest: Record<string, any>[]): string {\n    // max label name is 64 chars and addrs is always 42.\n    const pruneLabel = PRUNE_LABEL_PREFIX + Node.of(this).addr;\n\n    for (const resource of manifest) {\n      // skip resource if it's not an object or if it does not have a \"kind\"\n      if (typeof(resource) !== 'object' || !resource.kind) {\n        continue;\n      }\n\n      if (!resource.metadata) {\n        resource.metadata = {};\n      }\n\n      if (!resource.metadata.labels) {\n        resource.metadata.labels = {};\n      }\n\n      resource.metadata.labels = {\n        [pruneLabel]: '',\n        ...resource.metadata.labels,\n      };\n    }\n\n    return pruneLabel;\n  }\n\n  /**\n   * Inject the necessary ingress annontations if possible (and requested).\n   *\n   * @see https://kubernetes-sigs.github.io/aws-load-balancer-controller/v2.2/guide/ingress/annotations/\n   */\n  private injectIngressAlbAnnotations(manifest: Record<string, any>[], scheme: AlbScheme) {\n    for (const resource of manifest) {\n      // skip resource if it's not an object or if it does not have a \"kind\"\n      if (typeof(resource) !== 'object' || !resource.kind) {\n        continue;\n      }\n\n      if (resource.kind === 'Ingress') {\n        resource.metadata.annotations = {\n          'kubernetes.io/ingress.class': 'alb',\n          'alb.ingress.kubernetes.io/scheme': scheme,\n          ...resource.metadata.annotations,\n        };\n      }\n    }\n  }\n}\n"]}