cdk8s-plus
Advanced tools
Comparing version 0.29.0 to 0.30.0-pre.14ac668f5f88445f14864ac3d271f8a9afbd40c7
@@ -128,2 +128,18 @@ import { IConfigMap } from './config-map'; | ||
/** | ||
* Arguments to the entrypoint. The docker image's CMD is used if `command` is | ||
* not provided. | ||
* | ||
* Variable references $(VAR_NAME) are expanded using the container's | ||
* environment. If a variable cannot be resolved, the reference in the input | ||
* string will be unchanged. The $(VAR_NAME) syntax can be escaped with a | ||
* double $$, ie: $$(VAR_NAME). Escaped references will never be expanded, | ||
* regardless of whether the variable exists or not. | ||
* | ||
* Cannot be updated. | ||
* | ||
* @see https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell | ||
* @default [] | ||
*/ | ||
readonly args?: string[]; | ||
/** | ||
* Container's working directory. If not specified, the container runtime's default will be used, which might be configured in the container image. Cannot be updated. | ||
@@ -181,2 +197,3 @@ * | ||
private readonly _command?; | ||
private readonly _args?; | ||
private readonly _env; | ||
@@ -190,2 +207,8 @@ constructor(props: ContainerProps); | ||
/** | ||
* Arguments to the entrypoint. | ||
* | ||
* @returns a copy of the arguments array, cannot be modified. | ||
*/ | ||
get args(): string[] | undefined; | ||
/** | ||
* Add an environment value to the container. The variable value can come | ||
@@ -192,0 +215,0 @@ * from various dynamic sources such a secrets of config maps. |
@@ -103,2 +103,3 @@ "use strict"; | ||
this._command = props.command; | ||
this._args = props.args; | ||
this._env = (_b = props.env) !== null && _b !== void 0 ? _b : {}; | ||
@@ -117,2 +118,10 @@ this.workingDir = props.workingDir; | ||
/** | ||
* Arguments to the entrypoint. | ||
* | ||
* @returns a copy of the arguments array, cannot be modified. | ||
*/ | ||
get args() { | ||
return this._args ? [...this._args] : undefined; | ||
} | ||
/** | ||
* Add an environment value to the container. The variable value can come | ||
@@ -175,2 +184,3 @@ * from various dynamic sources such a secrets of config maps. | ||
command: this.command, | ||
args: this.args, | ||
workingDir: this.workingDir, | ||
@@ -241,2 +251,2 @@ env: renderEnv(this._env), | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"container.js","sourceRoot":"","sources":["../src/container.ts"],"names":[],"mappings":"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","sourcesContent":["import { IConfigMap } from './config-map';\nimport { ISecret } from './secret';\nimport * as k8s from './imports/k8s';\nimport { Volume } from './volume';\n\n/**\n * Options to specify an envionment variable value from a ConfigMap key.\n */\nexport interface EnvValueFromConfigMapOptions {\n\n  /**\n   * Specify whether the ConfigMap or its key must be defined.\n   *\n   * @default false\n   */\n  readonly optional?: boolean;\n}\n\n/**\n * Options to specify an environment variable value from a Secret.\n */\nexport interface EnvValueFromSecretOptions {\n\n  /**\n   * Specify whether the Secret or its key must be defined.\n   *\n   * @default false\n   */\n  readonly optional?: boolean;\n\n}\n\n/**\n * Options to specify an environment variable value from the process environment.\n */\nexport interface EnvValueFromProcessOptions {\n\n  /**\n   * Specify whether the key must exist in the environment.\n   * If this is set to true, and the key does not exist, an error will thrown.\n   *\n   * @default false\n   */\n  readonly required?: boolean;\n}\n\n\n/**\n * Utility class for creating reading env values from various sources.\n */\nexport class EnvValue {\n\n  /**\n   * Create a value by reading a specific key inside a config map.\n   *\n   * @param configMap - The config map.\n   * @param key - The key to extract the value from.\n   * @param options - Additional options.\n   */\n  public static fromConfigMap(configMap: IConfigMap, key: string, options: EnvValueFromConfigMapOptions = { }): EnvValue {\n\n    const source: k8s.EnvVarSource = {\n      configMapKeyRef: {\n        name: configMap.name,\n        key,\n        optional: options.optional,\n      },\n    };\n    return new EnvValue(undefined, source);\n  }\n\n  /**\n   * Create a by reading a specific key inside a secret.\n   *\n   * @param secret - The secret.\n   * @param key - The key.\n   * @param options - Additional options.\n   */\n  public static fromSecret(secret: ISecret, key: string, options: EnvValueFromSecretOptions = {}): EnvValue {\n\n    const source: k8s.EnvVarSource = {\n      secretKeyRef: {\n        name: secret.name,\n        key,\n        optional: options.optional,\n      },\n    };\n\n    return new EnvValue(undefined, source);\n  }\n\n  /**\n   * Create a value from the given argument.\n   *\n   * @param value - The value.\n   */\n  public static fromValue(value: string): EnvValue {\n    return new EnvValue(value);\n  }\n\n  /**\n   * Create a value from a key in the current process environment.\n   *\n   * @param key - The key to read.\n   * @param options - Additional options.\n   */\n  public static fromProcess(key: string, options: EnvValueFromProcessOptions = {}): EnvValue {\n\n    const value = process.env[key];\n\n    if (options.required && !value) {\n      throw new Error(`Missing ${key} env variable`);\n    }\n\n    return EnvValue.fromValue(value!);\n  }\n\n  private constructor(public readonly value?: any, public readonly valueFrom?: any) {}\n}\n\nexport enum ImagePullPolicy {\n  /**\n   * Every time the kubelet launches a container, the kubelet queries the container image registry\n   * to resolve the name to an image digest. If the kubelet has a container image with that exact\n   * digest cached locally, the kubelet uses its cached image; otherwise, the kubelet downloads\n   * (pulls) the image with the resolved digest, and uses that image to launch the container.\n   * \n   * Default is Always if ImagePullPolicy is omitted and either the image tag is :latest or\n   * the image tag is omitted.\n   */\n  ALWAYS = 'Always',\n\n  /**\n   * The image is pulled only if it is not already present locally.\n   * \n   * Default is IfNotPresent if ImagePullPolicy is omitted and the image tag is present but\n   * not :latest\n   */\n  IF_NOT_PRESENT = 'IfNotPresent',\n\n  /**\n   * The image is assumed to exist locally. No attempt is made to pull the image.\n   */\n  NEVER = 'Never',\n}\n\n/**\n * Properties for creating a container.\n */\nexport interface ContainerProps {\n\n  /**\n   * Docker image name.\n   */\n  readonly image: string;\n\n  /**\n   * Name of the container specified as a DNS_LABEL. Each container in a pod must have a unique name (DNS_LABEL). Cannot be updated.\n   *\n   * @default 'main'\n   */\n  readonly name?: string;\n\n  /**\n   * Number of port to expose on the pod's IP address. This must be a valid port number, 0 < x < 65536.\n   *\n   * @default - No port is exposed.\n   */\n  readonly port?: number\n\n  /**\n   * Entrypoint array. Not executed within a shell. The docker image's ENTRYPOINT is used if this is not provided. Variable references $(VAR_NAME) are expanded using the container's environment.\n   * If a variable cannot be resolved, the reference in the input string will be unchanged. The $(VAR_NAME) syntax can be escaped with a double $$, ie: $$(VAR_NAME).\n   * Escaped references will never be expanded, regardless of whether the variable exists or not. Cannot be updated.\n   * More info: https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell\n   *\n   * @default - The docker image's ENTRYPOINT.\n   */\n  readonly command?: string[];\n\n  /**\n   * Container's working directory. If not specified, the container runtime's default will be used, which might be configured in the container image. Cannot be updated.\n   *\n   * @default - The container runtime's default.\n   */\n  readonly workingDir?: string;\n\n  /**\n   * List of environment variables to set in the container. Cannot be updated.\n   *\n   * @default - No environment variables.\n   */\n  readonly env?: { [name: string]: EnvValue };\n\n  /**\n   * Pod volumes to mount into the container's filesystem. Cannot be updated.\n   */\n  readonly volumeMounts?: VolumeMount[];\n\n  /**\n   * Image pull policy for this container\n   * @default ImagePullPolicy.ALWAYS\n   */\n  readonly imagePullPolicy?: ImagePullPolicy\n}\n\n/**\n * A single application container that you want to run within a pod.\n */\nexport class Container {\n\n  /**\n   * The port this container exposes.\n   */\n  public readonly port?: number;\n\n  /**\n   * Volume mounts configured for this container.\n   */\n  public readonly mounts: VolumeMount[];\n\n  /**\n   * Image pull policy for this container\n   */\n  public readonly imagePullPolicy: ImagePullPolicy;\n\n  /**\n   * The container image.\n   */\n  public readonly image: string;\n\n  /**\n   * The name of the container.\n   */\n  public readonly name: string;\n\n  /**\n   * The working directory inside the container.\n   */\n  public readonly workingDir?: string;\n\n  private readonly _command?: readonly string[];\n  private readonly _env: { [name: string]: EnvValue };\n\n  constructor(props: ContainerProps) {\n    this.name = props.name ?? 'main';\n    this.image = props.image;\n    this.port = props.port;\n    this._command = props.command;\n    this._env = props.env ?? { };\n    this.workingDir = props.workingDir;\n    this.mounts = props.volumeMounts ?? [];\n    this.imagePullPolicy = props.imagePullPolicy ?? ImagePullPolicy.ALWAYS;\n  }\n\n  /**\n   * Entrypoint array (the command to execute when the container starts).\n   * @returns a copy of the entrypoint array, cannot be modified\n   */\n  public get command(): string[] | undefined {\n    return this._command ? [ ...this._command ] : undefined;\n  }\n\n  /**\n   * Add an environment value to the container. The variable value can come\n   * from various dynamic sources such a secrets of config maps.\n   *\n   * @see EnvValue.fromXXX\n   *\n   * @param name - The variable name.\n   * @param value - The variable value.\n   */\n  public addEnv(name: string, value: EnvValue) {\n    this._env[name] = value;\n  }\n\n  /**\n   * The environment variables for this container.\n   *\n   * Returns a copy. To add environment variables use `addEnv()`.\n   */\n  public get env(): Record<string, EnvValue> {\n    return { ...this._env };\n  }\n\n  /**\n   * Mount a volume to a specific path so that it is accessible by the container.\n   * Every pod that is configured to use this container will autmoatically have access to the volume.\n   *\n   * @param path - The desired path in the container.\n   * @param volume - The volume to mount.\n   */\n  public mount(path: string, volume: Volume, options: MountOptions = { }) {\n    this.mounts.push({ path, volume, ...options });\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.Container {\n    const volumeMounts: k8s.VolumeMount[] = [];\n\n    for (const mount of this.mounts) {\n      volumeMounts.push({\n        name: mount.volume.name,\n        mountPath: mount.path,\n        readOnly: mount.readOnly,\n        mountPropagation: mount.propagation,\n        subPath: mount.subPath,\n        subPathExpr: mount.subPathExpr,\n      });\n    }\n\n    const ports = new Array<k8s.ContainerPort>();\n\n    if (this.port) {\n      ports.push({\n        containerPort: this.port,\n      });\n    }\n\n    return {\n      name: this.name,\n      image: this.image,\n      imagePullPolicy: this.imagePullPolicy,\n      ports,\n      volumeMounts,\n      command: this.command,\n      workingDir: this.workingDir,\n      env: renderEnv(this._env),\n    };\n  }\n}\n\n/**\n * Options for mounts.\n */\nexport interface MountOptions {\n  /**\n   * Determines how mounts are propagated from the host to container and the\n   * other way around. When not set, MountPropagationNone is used.\n   *\n   * Mount propagation allows for sharing volumes mounted by a Container to\n   * other Containers in the same Pod, or even to other Pods on the same node.\n   *\n   * This field is beta in 1.10.\n   *\n   * @default MountPropagation.NONE\n   */\n  readonly propagation?: MountPropagation;\n\n  /**\n   * Mounted read-only if true, read-write otherwise (false or unspecified).\n   * Defaults to false.\n   *\n   * @default false\n   */\n  readonly readOnly?: boolean;\n\n  /**\n   * Path within the volume from which the container's volume should be mounted.).\n   *\n   * @default \"\" the volume's root\n   */\n  readonly subPath?: string;\n\n  /**\n   * Expanded path within the volume from which the container's volume should be\n   * mounted. Behaves similarly to SubPath but environment variable references\n   * $(VAR_NAME) are expanded using the container's environment. Defaults to \"\"\n   * (volume's root). SubPathExpr and SubPath are mutually exclusive. This field\n   * is beta in 1.15.\n   *\n   * `subPathExpr` and `subPath` are mutually exclusive. This field is beta in\n   * 1.15.\n   *\n   * @default \"\" volume's root.\n   */\n  readonly subPathExpr?: string;\n}\n\n/**\n * Mount a volume from the pod to the container.\n */\nexport interface VolumeMount extends MountOptions {\n  /**\n   * The volume to mount.\n   */\n  readonly volume: Volume;\n\n  /**\n   * Path within the container at which the volume should be mounted. Must not\n   * contain ':'.\n   */\n  readonly path: string;\n}\n\nexport enum MountPropagation {\n  /**\n   * This volume mount will not receive any subsequent mounts that are mounted\n   * to this volume or any of its subdirectories by the host. In similar\n   * fashion, no mounts created by the Container will be visible on the host.\n   *\n   * This is the default mode.\n   *\n   * This mode is equal to `private` mount propagation as described in the Linux\n   * kernel documentation\n   */\n  NONE = 'None',\n\n  /**\n   * This volume mount will receive all subsequent mounts that are mounted to\n   * this volume or any of its subdirectories.\n   *\n   * In other words, if the host mounts anything inside the volume mount, the\n   * Container will see it mounted there.\n   *\n   * Similarly, if any Pod with Bidirectional mount propagation to the same\n   * volume mounts anything there, the Container with HostToContainer mount\n   * propagation will see it.\n   *\n   * This mode is equal to `rslave` mount propagation as described in the Linux\n   * kernel documentation\n   */\n  HOST_TO_CONTAINER = 'HostToContainer',\n\n  /**\n   * This volume mount behaves the same the HostToContainer mount. In addition,\n   * all volume mounts created by the Container will be propagated back to the\n   * host and to all Containers of all Pods that use the same volume\n   *\n   * A typical use case for this mode is a Pod with a FlexVolume or CSI driver\n   * or a Pod that needs to mount something on the host using a hostPath volume.\n   *\n   * This mode is equal to `rshared` mount propagation as described in the Linux\n   * kernel documentation\n   *\n   * Caution: Bidirectional mount propagation can be dangerous. It can damage\n   * the host operating system and therefore it is allowed only in privileged\n   * Containers. Familiarity with Linux kernel behavior is strongly recommended.\n   * In addition, any volume mounts created by Containers in Pods must be\n   * destroyed (unmounted) by the Containers on termination.\n   *\n   */\n  BIDIRECTIONAL = 'Bidirectional',\n}\n\nfunction renderEnv(env: { [name: string]: EnvValue }): k8s.EnvVar[] {\n  const result = new Array<k8s.EnvVar>();\n  for (const [name, v] of Object.entries(env)) {\n    result.push({\n      name,\n      value: v.value,\n      valueFrom: v.valueFrom,\n    });\n  }\n  return result;\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"container.js","sourceRoot":"","sources":["../src/container.ts"],"names":[],"mappings":"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","sourcesContent":["import { IConfigMap } from './config-map';\nimport { ISecret } from './secret';\nimport * as k8s from './imports/k8s';\nimport { Volume } from './volume';\n\n/**\n * Options to specify an envionment variable value from a ConfigMap key.\n */\nexport interface EnvValueFromConfigMapOptions {\n\n  /**\n   * Specify whether the ConfigMap or its key must be defined.\n   *\n   * @default false\n   */\n  readonly optional?: boolean;\n}\n\n/**\n * Options to specify an environment variable value from a Secret.\n */\nexport interface EnvValueFromSecretOptions {\n\n  /**\n   * Specify whether the Secret or its key must be defined.\n   *\n   * @default false\n   */\n  readonly optional?: boolean;\n\n}\n\n/**\n * Options to specify an environment variable value from the process environment.\n */\nexport interface EnvValueFromProcessOptions {\n\n  /**\n   * Specify whether the key must exist in the environment.\n   * If this is set to true, and the key does not exist, an error will thrown.\n   *\n   * @default false\n   */\n  readonly required?: boolean;\n}\n\n\n/**\n * Utility class for creating reading env values from various sources.\n */\nexport class EnvValue {\n\n  /**\n   * Create a value by reading a specific key inside a config map.\n   *\n   * @param configMap - The config map.\n   * @param key - The key to extract the value from.\n   * @param options - Additional options.\n   */\n  public static fromConfigMap(configMap: IConfigMap, key: string, options: EnvValueFromConfigMapOptions = { }): EnvValue {\n\n    const source: k8s.EnvVarSource = {\n      configMapKeyRef: {\n        name: configMap.name,\n        key,\n        optional: options.optional,\n      },\n    };\n    return new EnvValue(undefined, source);\n  }\n\n  /**\n   * Create a by reading a specific key inside a secret.\n   *\n   * @param secret - The secret.\n   * @param key - The key.\n   * @param options - Additional options.\n   */\n  public static fromSecret(secret: ISecret, key: string, options: EnvValueFromSecretOptions = {}): EnvValue {\n\n    const source: k8s.EnvVarSource = {\n      secretKeyRef: {\n        name: secret.name,\n        key,\n        optional: options.optional,\n      },\n    };\n\n    return new EnvValue(undefined, source);\n  }\n\n  /**\n   * Create a value from the given argument.\n   *\n   * @param value - The value.\n   */\n  public static fromValue(value: string): EnvValue {\n    return new EnvValue(value);\n  }\n\n  /**\n   * Create a value from a key in the current process environment.\n   *\n   * @param key - The key to read.\n   * @param options - Additional options.\n   */\n  public static fromProcess(key: string, options: EnvValueFromProcessOptions = {}): EnvValue {\n\n    const value = process.env[key];\n\n    if (options.required && !value) {\n      throw new Error(`Missing ${key} env variable`);\n    }\n\n    return EnvValue.fromValue(value!);\n  }\n\n  private constructor(public readonly value?: any, public readonly valueFrom?: any) {}\n}\n\nexport enum ImagePullPolicy {\n  /**\n   * Every time the kubelet launches a container, the kubelet queries the container image registry\n   * to resolve the name to an image digest. If the kubelet has a container image with that exact\n   * digest cached locally, the kubelet uses its cached image; otherwise, the kubelet downloads\n   * (pulls) the image with the resolved digest, and uses that image to launch the container.\n   * \n   * Default is Always if ImagePullPolicy is omitted and either the image tag is :latest or\n   * the image tag is omitted.\n   */\n  ALWAYS = 'Always',\n\n  /**\n   * The image is pulled only if it is not already present locally.\n   * \n   * Default is IfNotPresent if ImagePullPolicy is omitted and the image tag is present but\n   * not :latest\n   */\n  IF_NOT_PRESENT = 'IfNotPresent',\n\n  /**\n   * The image is assumed to exist locally. No attempt is made to pull the image.\n   */\n  NEVER = 'Never',\n}\n\n/**\n * Properties for creating a container.\n */\nexport interface ContainerProps {\n\n  /**\n   * Docker image name.\n   */\n  readonly image: string;\n\n  /**\n   * Name of the container specified as a DNS_LABEL. Each container in a pod must have a unique name (DNS_LABEL). Cannot be updated.\n   *\n   * @default 'main'\n   */\n  readonly name?: string;\n\n  /**\n   * Number of port to expose on the pod's IP address. This must be a valid port number, 0 < x < 65536.\n   *\n   * @default - No port is exposed.\n   */\n  readonly port?: number\n\n  /**\n   * Entrypoint array. Not executed within a shell. The docker image's ENTRYPOINT is used if this is not provided. Variable references $(VAR_NAME) are expanded using the container's environment.\n   * If a variable cannot be resolved, the reference in the input string will be unchanged. The $(VAR_NAME) syntax can be escaped with a double $$, ie: $$(VAR_NAME).\n   * Escaped references will never be expanded, regardless of whether the variable exists or not. Cannot be updated.\n   * More info: https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell\n   *\n   * @default - The docker image's ENTRYPOINT.\n   */\n  readonly command?: string[];\n\n  /**\n   * Arguments to the entrypoint. The docker image's CMD is used if `command` is\n   * not provided. \n   *\n   * Variable references $(VAR_NAME) are expanded using the container's\n   * environment. If a variable cannot be resolved, the reference in the input\n   * string will be unchanged. The $(VAR_NAME) syntax can be escaped with a\n   * double $$, ie: $$(VAR_NAME). Escaped references will never be expanded,\n   * regardless of whether the variable exists or not. \n   *\n   * Cannot be updated. \n   * \n   * @see https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell\n   * @default []\n   */\n  readonly args?: string[];\n\n  /**\n   * Container's working directory. If not specified, the container runtime's default will be used, which might be configured in the container image. Cannot be updated.\n   *\n   * @default - The container runtime's default.\n   */\n  readonly workingDir?: string;\n\n  /**\n   * List of environment variables to set in the container. Cannot be updated.\n   *\n   * @default - No environment variables.\n   */\n  readonly env?: { [name: string]: EnvValue };\n\n  /**\n   * Pod volumes to mount into the container's filesystem. Cannot be updated.\n   */\n  readonly volumeMounts?: VolumeMount[];\n\n  /**\n   * Image pull policy for this container\n   * @default ImagePullPolicy.ALWAYS\n   */\n  readonly imagePullPolicy?: ImagePullPolicy\n}\n\n/**\n * A single application container that you want to run within a pod.\n */\nexport class Container {\n\n  /**\n   * The port this container exposes.\n   */\n  public readonly port?: number;\n\n  /**\n   * Volume mounts configured for this container.\n   */\n  public readonly mounts: VolumeMount[];\n\n  /**\n   * Image pull policy for this container\n   */\n  public readonly imagePullPolicy: ImagePullPolicy;\n\n  /**\n   * The container image.\n   */\n  public readonly image: string;\n\n  /**\n   * The name of the container.\n   */\n  public readonly name: string;\n\n  /**\n   * The working directory inside the container.\n   */\n  public readonly workingDir?: string;\n\n  private readonly _command?: readonly string[];\n  private readonly _args?: readonly string[];\n  private readonly _env: { [name: string]: EnvValue };\n\n  constructor(props: ContainerProps) {\n    this.name = props.name ?? 'main';\n    this.image = props.image;\n    this.port = props.port;\n    this._command = props.command;\n    this._args = props.args;\n    this._env = props.env ?? { };\n    this.workingDir = props.workingDir;\n    this.mounts = props.volumeMounts ?? [];\n    this.imagePullPolicy = props.imagePullPolicy ?? ImagePullPolicy.ALWAYS;\n  }\n\n  /**\n   * Entrypoint array (the command to execute when the container starts).\n   * @returns a copy of the entrypoint array, cannot be modified\n   */\n  public get command(): string[] | undefined {\n    return this._command ? [ ...this._command ] : undefined;\n  }\n\n  /**\n   * Arguments to the entrypoint.\n   * \n   * @returns a copy of the arguments array, cannot be modified.\n   */\n  public get args(): string[] | undefined {\n    return this._args ? [ ...this._args ] : undefined;\n  }\n\n  /**\n   * Add an environment value to the container. The variable value can come\n   * from various dynamic sources such a secrets of config maps.\n   *\n   * @see EnvValue.fromXXX\n   *\n   * @param name - The variable name.\n   * @param value - The variable value.\n   */\n  public addEnv(name: string, value: EnvValue) {\n    this._env[name] = value;\n  }\n\n  /**\n   * The environment variables for this container.\n   *\n   * Returns a copy. To add environment variables use `addEnv()`.\n   */\n  public get env(): Record<string, EnvValue> {\n    return { ...this._env };\n  }\n\n  /**\n   * Mount a volume to a specific path so that it is accessible by the container.\n   * Every pod that is configured to use this container will autmoatically have access to the volume.\n   *\n   * @param path - The desired path in the container.\n   * @param volume - The volume to mount.\n   */\n  public mount(path: string, volume: Volume, options: MountOptions = { }) {\n    this.mounts.push({ path, volume, ...options });\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.Container {\n    const volumeMounts: k8s.VolumeMount[] = [];\n\n    for (const mount of this.mounts) {\n      volumeMounts.push({\n        name: mount.volume.name,\n        mountPath: mount.path,\n        readOnly: mount.readOnly,\n        mountPropagation: mount.propagation,\n        subPath: mount.subPath,\n        subPathExpr: mount.subPathExpr,\n      });\n    }\n\n    const ports = new Array<k8s.ContainerPort>();\n\n    if (this.port) {\n      ports.push({\n        containerPort: this.port,\n      });\n    }\n\n    return {\n      name: this.name,\n      image: this.image,\n      imagePullPolicy: this.imagePullPolicy,\n      ports,\n      volumeMounts,\n      command: this.command,\n      args: this.args,\n      workingDir: this.workingDir,\n      env: renderEnv(this._env),\n    };\n  }\n}\n\n/**\n * Options for mounts.\n */\nexport interface MountOptions {\n  /**\n   * Determines how mounts are propagated from the host to container and the\n   * other way around. When not set, MountPropagationNone is used.\n   *\n   * Mount propagation allows for sharing volumes mounted by a Container to\n   * other Containers in the same Pod, or even to other Pods on the same node.\n   *\n   * This field is beta in 1.10.\n   *\n   * @default MountPropagation.NONE\n   */\n  readonly propagation?: MountPropagation;\n\n  /**\n   * Mounted read-only if true, read-write otherwise (false or unspecified).\n   * Defaults to false.\n   *\n   * @default false\n   */\n  readonly readOnly?: boolean;\n\n  /**\n   * Path within the volume from which the container's volume should be mounted.).\n   *\n   * @default \"\" the volume's root\n   */\n  readonly subPath?: string;\n\n  /**\n   * Expanded path within the volume from which the container's volume should be\n   * mounted. Behaves similarly to SubPath but environment variable references\n   * $(VAR_NAME) are expanded using the container's environment. Defaults to \"\"\n   * (volume's root). SubPathExpr and SubPath are mutually exclusive. This field\n   * is beta in 1.15.\n   *\n   * `subPathExpr` and `subPath` are mutually exclusive. This field is beta in\n   * 1.15.\n   *\n   * @default \"\" volume's root.\n   */\n  readonly subPathExpr?: string;\n}\n\n/**\n * Mount a volume from the pod to the container.\n */\nexport interface VolumeMount extends MountOptions {\n  /**\n   * The volume to mount.\n   */\n  readonly volume: Volume;\n\n  /**\n   * Path within the container at which the volume should be mounted. Must not\n   * contain ':'.\n   */\n  readonly path: string;\n}\n\nexport enum MountPropagation {\n  /**\n   * This volume mount will not receive any subsequent mounts that are mounted\n   * to this volume or any of its subdirectories by the host. In similar\n   * fashion, no mounts created by the Container will be visible on the host.\n   *\n   * This is the default mode.\n   *\n   * This mode is equal to `private` mount propagation as described in the Linux\n   * kernel documentation\n   */\n  NONE = 'None',\n\n  /**\n   * This volume mount will receive all subsequent mounts that are mounted to\n   * this volume or any of its subdirectories.\n   *\n   * In other words, if the host mounts anything inside the volume mount, the\n   * Container will see it mounted there.\n   *\n   * Similarly, if any Pod with Bidirectional mount propagation to the same\n   * volume mounts anything there, the Container with HostToContainer mount\n   * propagation will see it.\n   *\n   * This mode is equal to `rslave` mount propagation as described in the Linux\n   * kernel documentation\n   */\n  HOST_TO_CONTAINER = 'HostToContainer',\n\n  /**\n   * This volume mount behaves the same the HostToContainer mount. In addition,\n   * all volume mounts created by the Container will be propagated back to the\n   * host and to all Containers of all Pods that use the same volume\n   *\n   * A typical use case for this mode is a Pod with a FlexVolume or CSI driver\n   * or a Pod that needs to mount something on the host using a hostPath volume.\n   *\n   * This mode is equal to `rshared` mount propagation as described in the Linux\n   * kernel documentation\n   *\n   * Caution: Bidirectional mount propagation can be dangerous. It can damage\n   * the host operating system and therefore it is allowed only in privileged\n   * Containers. Familiarity with Linux kernel behavior is strongly recommended.\n   * In addition, any volume mounts created by Containers in Pods must be\n   * destroyed (unmounted) by the Containers on termination.\n   *\n   */\n  BIDIRECTIONAL = 'Bidirectional',\n}\n\nfunction renderEnv(env: { [name: string]: EnvValue }): k8s.EnvVar[] {\n  const result = new Array<k8s.EnvVar>();\n  for (const [name, v] of Object.entries(env)) {\n    result.push({\n      name,\n      value: v.value,\n      valueFrom: v.valueFrom,\n    });\n  }\n  return result;\n}\n"]} |
@@ -63,3 +63,3 @@ "use strict"; | ||
const selector = 'cdk8s.deployment'; | ||
const matcher = constructs_1.Node.of(this).uniqueId; | ||
const matcher = cdk8s_1.Names.toLabelValue(constructs_1.Node.of(this).path); | ||
const service = new service_1.Service(this, 'Service', { | ||
@@ -115,3 +115,3 @@ spec: { | ||
const selector = 'cdk8s.deployment'; | ||
const matcher = constructs_1.Node.of(deployment).uniqueId; | ||
const matcher = cdk8s_1.Names.toLabelValue(constructs_1.Node.of(deployment).path); | ||
this.podMetadataTemplate.addLabel(selector, matcher); | ||
@@ -132,2 +132,2 @@ this.selectByLabel(selector, matcher); | ||
exports.DeploymentSpecDefinition = DeploymentSpecDefinition; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deployment.js","sourceRoot":"","sources":["../src/deployment.ts"],"names":[],"mappings":"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","sourcesContent":["import * as k8s from './imports/k8s';\nimport { Construct, Node } from 'constructs';\nimport { Service, ServiceType } from './service';\nimport { Resource, ResourceProps } from './base';\nimport * as cdk8s from 'cdk8s';\nimport { PodSpecDefinition, PodSpec } from './pod';\nimport { ApiObjectMetadata, ApiObjectMetadataDefinition } from 'cdk8s';\n\n/**\n * Properties for initialization of `Deployment`.\n */\nexport interface DeploymentProps extends ResourceProps {\n  /**\n   * The spec of the deployment. Use `deployment.spec` to apply post instatiation mutations.\n   *\n   * @default - An empty spec will be created.\n   */\n  readonly spec?: DeploymentSpec;\n}\n\n/**\n * Options for exposing a deployment via a service.\n */\nexport interface ExposeOptions {\n  /**\n   * The port number the service will bind to.\n   */\n  readonly port: number;\n\n  /**\n   * The type of the exposed service.\n   *\n   * @default - ClusterIP.\n   */\n  readonly serviceType?: ServiceType;\n}\n\n/**\n*\n* A Deployment provides declarative updates for Pods and ReplicaSets.\n*\n* You describe a desired state in a Deployment, and the Deployment Controller changes the actual\n* state to the desired state at a controlled rate. You can define Deployments to create new ReplicaSets, or to remove\n* existing Deployments and adopt all their resources with new Deployments.\n*\n* > Note: Do not manage ReplicaSets owned by a Deployment. Consider opening an issue in the main Kubernetes repository if your use case is not covered below.\n*\n* Use Case\n* ---------\n*\n* The following are typical use cases for Deployments:\n*\n* - Create a Deployment to rollout a ReplicaSet. The ReplicaSet creates Pods in the background.\n*   Check the status of the rollout to see if it succeeds or not.\n* - Declare the new state of the Pods by updating the PodTemplateSpec of the Deployment.\n*   A new ReplicaSet is created and the Deployment manages moving the Pods from the old ReplicaSet to the new one at a controlled rate.\n*   Each new ReplicaSet updates the revision of the Deployment.\n* - Rollback to an earlier Deployment revision if the current state of the Deployment is not stable.\n*   Each rollback updates the revision of the Deployment.\n* - Scale up the Deployment to facilitate more load.\n* - Pause the Deployment to apply multiple fixes to its PodTemplateSpec and then resume it to start a new rollout.\n* - Use the status of the Deployment as an indicator that a rollout has stuck.\n* - Clean up older ReplicaSets that you don't need anymore.\n*\n**/\nexport class Deployment extends Resource {\n  /**\n   * @see base.Resource.apiObject\n   */\n  protected readonly apiObject: cdk8s.ApiObject;\n\n  /**\n   * Provides access to the underlying spec.\n   *\n   * You can use this field to apply post instantiation mutations\n   * to the spec.\n   */\n  public readonly spec: DeploymentSpecDefinition;\n\n  constructor(scope: Construct, id: string, props: DeploymentProps = {}) {\n    super(scope, id, props);\n\n    this.spec = new DeploymentSpecDefinition(props.spec);\n\n    this.apiObject = new k8s.Deployment(this, 'Pod', {\n      metadata: props.metadata,\n      spec: cdk8s.Lazy.any({ produce: () => this.spec._toKube(this) }),\n    });\n  }\n\n  /**\n   * Expose a deployment via a service.\n   *\n   * This is equivalent to running `kubectl expose deployment <deployment-name>`.\n   *\n   * @param options - Options.\n   */\n  public expose(options: ExposeOptions): Service {\n    const containers = this.spec.podSpecTemplate.containers;\n    if (containers.length === 0) {\n      throw new Error('Cannot expose a deployment without containers');\n    }\n\n    // create a label and attach it to the deployment pods\n    const selector = 'cdk8s.deployment';\n    const matcher = Node.of(this).uniqueId;\n\n    const service = new Service(this, 'Service', {\n      spec: {\n        type: options.serviceType ?? ServiceType.CLUSTER_IP,\n      },\n    });\n\n    service.spec.addSelector(selector, matcher);\n    service.spec.serve(options.port, {\n      // just a PoC, we assume the first container is the main one.\n      // TODO: figure out what the correct thing to do here.\n      targetPort: containers[0].port,\n    });\n\n    return service;\n  }\n}\n\n/**\n * Properties for initialization of `DeploymentSpec`.\n */\nexport interface DeploymentSpec {\n\n  /**\n   * Number of desired pods.\n   * @default 1\n   */\n  readonly replicas?: number;\n\n  /**\n   * Template for pod specs.\n   */\n  readonly podSpecTemplate?: PodSpec;\n\n  /**\n   * Template for pod metadata.\n   */\n  readonly podMetadataTemplate?: ApiObjectMetadata;\n}\n\n/**\n * DeploymentSpec is the specification of the desired behavior of the Deployment.\n */\nexport class DeploymentSpecDefinition {\n  /**\n   * Number of desired pods.\n   */\n  public readonly replicas?: number;\n\n  /**\n   * Provides access to the underlying pod template spec.\n   *\n   * You can use this field to apply post instatiation mutations\n   * to the spec.\n   */\n  public readonly podSpecTemplate: PodSpecDefinition;\n\n  /**\n   * Template for pod metadata.\n   */\n  public readonly podMetadataTemplate: ApiObjectMetadataDefinition;\n\n  private readonly _labelSelector: Record<string, string>;\n\n  constructor(props: DeploymentSpec = {}) {\n    this.replicas = props.replicas ?? 1;\n    this.podSpecTemplate = new PodSpecDefinition(props.podSpecTemplate);\n    this.podMetadataTemplate = new ApiObjectMetadataDefinition(props.podMetadataTemplate);\n\n    this._labelSelector = {};\n  }\n\n  /**\n   * Configure a label selector to this deployment.\n   * Pods that have the label will be selected by deployments configured with this spec.\n   *\n   * @param key - The label key.\n   * @param value - The label value.\n   */\n  public selectByLabel(key: string, value: string) {\n    this._labelSelector[key] = value;\n  }\n\n  /**\n   * The labels this deployment will match against in order to select pods.\n   *\n   * Returns a a copy. Use `selectByLabel()` to add labels.\n   */\n  public get labelSelector(): Record<string, string> {\n    return { ...this._labelSelector };\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(deployment: Deployment): k8s.DeploymentSpec {\n\n    // automatically select pods in this deployment\n\n    const selector = 'cdk8s.deployment';\n    const matcher = Node.of(deployment).uniqueId;\n\n    this.podMetadataTemplate.addLabel(selector, matcher);\n\n    this.selectByLabel(selector, matcher);\n\n    return {\n      replicas: this.replicas,\n      template: {\n        metadata: this.podMetadataTemplate.toJson(),\n        spec: this.podSpecTemplate._toKube(),\n      },\n      selector: {\n        matchLabels: this._labelSelector,\n      },\n    };\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deployment.js","sourceRoot":"","sources":["../src/deployment.ts"],"names":[],"mappings":"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","sourcesContent":["import * as k8s from './imports/k8s';\nimport { Construct, Node } from 'constructs';\nimport { Service, ServiceType } from './service';\nimport { Resource, ResourceProps } from './base';\nimport * as cdk8s from 'cdk8s';\nimport { PodSpecDefinition, PodSpec } from './pod';\nimport { ApiObjectMetadata, ApiObjectMetadataDefinition, Names } from 'cdk8s';\n\n/**\n * Properties for initialization of `Deployment`.\n */\nexport interface DeploymentProps extends ResourceProps {\n  /**\n   * The spec of the deployment. Use `deployment.spec` to apply post instatiation mutations.\n   *\n   * @default - An empty spec will be created.\n   */\n  readonly spec?: DeploymentSpec;\n}\n\n/**\n * Options for exposing a deployment via a service.\n */\nexport interface ExposeOptions {\n  /**\n   * The port number the service will bind to.\n   */\n  readonly port: number;\n\n  /**\n   * The type of the exposed service.\n   *\n   * @default - ClusterIP.\n   */\n  readonly serviceType?: ServiceType;\n}\n\n/**\n*\n* A Deployment provides declarative updates for Pods and ReplicaSets.\n*\n* You describe a desired state in a Deployment, and the Deployment Controller changes the actual\n* state to the desired state at a controlled rate. You can define Deployments to create new ReplicaSets, or to remove\n* existing Deployments and adopt all their resources with new Deployments.\n*\n* > Note: Do not manage ReplicaSets owned by a Deployment. Consider opening an issue in the main Kubernetes repository if your use case is not covered below.\n*\n* Use Case\n* ---------\n*\n* The following are typical use cases for Deployments:\n*\n* - Create a Deployment to rollout a ReplicaSet. The ReplicaSet creates Pods in the background.\n*   Check the status of the rollout to see if it succeeds or not.\n* - Declare the new state of the Pods by updating the PodTemplateSpec of the Deployment.\n*   A new ReplicaSet is created and the Deployment manages moving the Pods from the old ReplicaSet to the new one at a controlled rate.\n*   Each new ReplicaSet updates the revision of the Deployment.\n* - Rollback to an earlier Deployment revision if the current state of the Deployment is not stable.\n*   Each rollback updates the revision of the Deployment.\n* - Scale up the Deployment to facilitate more load.\n* - Pause the Deployment to apply multiple fixes to its PodTemplateSpec and then resume it to start a new rollout.\n* - Use the status of the Deployment as an indicator that a rollout has stuck.\n* - Clean up older ReplicaSets that you don't need anymore.\n*\n**/\nexport class Deployment extends Resource {\n  /**\n   * @see base.Resource.apiObject\n   */\n  protected readonly apiObject: cdk8s.ApiObject;\n\n  /**\n   * Provides access to the underlying spec.\n   *\n   * You can use this field to apply post instantiation mutations\n   * to the spec.\n   */\n  public readonly spec: DeploymentSpecDefinition;\n\n  constructor(scope: Construct, id: string, props: DeploymentProps = {}) {\n    super(scope, id, props);\n\n    this.spec = new DeploymentSpecDefinition(props.spec);\n\n    this.apiObject = new k8s.Deployment(this, 'Pod', {\n      metadata: props.metadata,\n      spec: cdk8s.Lazy.any({ produce: () => this.spec._toKube(this) }),\n    });\n  }\n\n  /**\n   * Expose a deployment via a service.\n   *\n   * This is equivalent to running `kubectl expose deployment <deployment-name>`.\n   *\n   * @param options - Options.\n   */\n  public expose(options: ExposeOptions): Service {\n    const containers = this.spec.podSpecTemplate.containers;\n    if (containers.length === 0) {\n      throw new Error('Cannot expose a deployment without containers');\n    }\n\n    // create a label and attach it to the deployment pods\n    const selector = 'cdk8s.deployment';\n    const matcher = Names.toLabelValue(Node.of(this).path);\n\n    const service = new Service(this, 'Service', {\n      spec: {\n        type: options.serviceType ?? ServiceType.CLUSTER_IP,\n      },\n    });\n\n    service.spec.addSelector(selector, matcher);\n    service.spec.serve(options.port, {\n      // just a PoC, we assume the first container is the main one.\n      // TODO: figure out what the correct thing to do here.\n      targetPort: containers[0].port,\n    });\n\n    return service;\n  }\n}\n\n/**\n * Properties for initialization of `DeploymentSpec`.\n */\nexport interface DeploymentSpec {\n\n  /**\n   * Number of desired pods.\n   * @default 1\n   */\n  readonly replicas?: number;\n\n  /**\n   * Template for pod specs.\n   */\n  readonly podSpecTemplate?: PodSpec;\n\n  /**\n   * Template for pod metadata.\n   */\n  readonly podMetadataTemplate?: ApiObjectMetadata;\n}\n\n/**\n * DeploymentSpec is the specification of the desired behavior of the Deployment.\n */\nexport class DeploymentSpecDefinition {\n  /**\n   * Number of desired pods.\n   */\n  public readonly replicas?: number;\n\n  /**\n   * Provides access to the underlying pod template spec.\n   *\n   * You can use this field to apply post instatiation mutations\n   * to the spec.\n   */\n  public readonly podSpecTemplate: PodSpecDefinition;\n\n  /**\n   * Template for pod metadata.\n   */\n  public readonly podMetadataTemplate: ApiObjectMetadataDefinition;\n\n  private readonly _labelSelector: Record<string, string>;\n\n  constructor(props: DeploymentSpec = {}) {\n    this.replicas = props.replicas ?? 1;\n    this.podSpecTemplate = new PodSpecDefinition(props.podSpecTemplate);\n    this.podMetadataTemplate = new ApiObjectMetadataDefinition(props.podMetadataTemplate);\n\n    this._labelSelector = {};\n  }\n\n  /**\n   * Configure a label selector to this deployment.\n   * Pods that have the label will be selected by deployments configured with this spec.\n   *\n   * @param key - The label key.\n   * @param value - The label value.\n   */\n  public selectByLabel(key: string, value: string) {\n    this._labelSelector[key] = value;\n  }\n\n  /**\n   * The labels this deployment will match against in order to select pods.\n   *\n   * Returns a a copy. Use `selectByLabel()` to add labels.\n   */\n  public get labelSelector(): Record<string, string> {\n    return { ...this._labelSelector };\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(deployment: Deployment): k8s.DeploymentSpec {\n\n    // automatically select pods in this deployment\n\n    const selector = 'cdk8s.deployment';\n    const matcher = Names.toLabelValue(Node.of(deployment).path);\n\n    this.podMetadataTemplate.addLabel(selector, matcher);\n\n    this.selectByLabel(selector, matcher);\n\n    return {\n      replicas: this.replicas,\n      template: {\n        metadata: this.podMetadataTemplate.toJson(),\n        spec: this.podSpecTemplate._toKube(),\n      },\n      selector: {\n        matchLabels: this._labelSelector,\n      },\n    };\n  }\n}\n"]} |
@@ -13,1 +13,2 @@ export * from './base'; | ||
export * from './size'; | ||
export * from './ingress'; |
@@ -25,2 +25,3 @@ "use strict"; | ||
__exportStar(require("./size"), exports); | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBQUEseUNBQXVCO0FBQ3ZCLCtDQUE2QjtBQUM3Qiw4Q0FBNEI7QUFDNUIsK0NBQTZCO0FBQzdCLDZDQUEyQjtBQUMzQix3Q0FBc0I7QUFDdEIsd0NBQXNCO0FBQ3RCLDJDQUF5QjtBQUN6QixvREFBa0M7QUFDbEMsNENBQTBCO0FBQzFCLDJDQUF5QjtBQUN6Qix5Q0FBdUIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2Jhc2UnO1xuZXhwb3J0ICogZnJvbSAnLi9jb25maWctbWFwJztcbmV4cG9ydCAqIGZyb20gJy4vY29udGFpbmVyJztcbmV4cG9ydCAqIGZyb20gJy4vZGVwbG95bWVudCc7XG5leHBvcnQgKiBmcm9tICcuL2R1cmF0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vam9iJztcbmV4cG9ydCAqIGZyb20gJy4vcG9kJztcbmV4cG9ydCAqIGZyb20gJy4vc2VjcmV0JztcbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZS1hY2NvdW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3ZvbHVtZSc7XG5leHBvcnQgKiBmcm9tICcuL3NpemUnOyJdfQ== | ||
__exportStar(require("./ingress"), exports); | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBQUEseUNBQXVCO0FBQ3ZCLCtDQUE2QjtBQUM3Qiw4Q0FBNEI7QUFDNUIsK0NBQTZCO0FBQzdCLDZDQUEyQjtBQUMzQix3Q0FBc0I7QUFDdEIsd0NBQXNCO0FBQ3RCLDJDQUF5QjtBQUN6QixvREFBa0M7QUFDbEMsNENBQTBCO0FBQzFCLDJDQUF5QjtBQUN6Qix5Q0FBdUI7QUFDdkIsNENBQTBCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9iYXNlJztcbmV4cG9ydCAqIGZyb20gJy4vY29uZmlnLW1hcCc7XG5leHBvcnQgKiBmcm9tICcuL2NvbnRhaW5lcic7XG5leHBvcnQgKiBmcm9tICcuL2RlcGxveW1lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9kdXJhdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL2pvYic7XG5leHBvcnQgKiBmcm9tICcuL3BvZCc7XG5leHBvcnQgKiBmcm9tICcuL3NlY3JldCc7XG5leHBvcnQgKiBmcm9tICcuL3NlcnZpY2UtYWNjb3VudCc7XG5leHBvcnQgKiBmcm9tICcuL3NlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi92b2x1bWUnO1xuZXhwb3J0ICogZnJvbSAnLi9zaXplJztcbmV4cG9ydCAqIGZyb20gJy4vaW5ncmVzcyc7Il19 |
@@ -204,2 +204,8 @@ import { Construct } from 'constructs'; | ||
serve(port: number, options?: ServicePortOptions): void; | ||
/** | ||
* Ports for this service. | ||
* | ||
* Use `serve()` to expose additional service ports. | ||
*/ | ||
get ports(): ServicePort[]; | ||
} |
@@ -113,2 +113,10 @@ "use strict"; | ||
/** | ||
* Ports for this service. | ||
* | ||
* Use `serve()` to expose additional service ports. | ||
*/ | ||
get ports() { | ||
return [...this._ports]; | ||
} | ||
/** | ||
* @internal | ||
@@ -138,2 +146,2 @@ */ | ||
exports.ServiceSpecDefinition = ServiceSpecDefinition; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service.js","sourceRoot":"","sources":["../src/service.ts"],"names":[],"mappings":"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","sourcesContent":["import * as k8s from './imports/k8s';\nimport { Construct } from 'constructs';\nimport { ResourceProps, Resource } from './base';\nimport * as cdk8s from 'cdk8s';\n\n\n/**\n * Properties for initialization of `Service`.\n */\nexport interface ServiceProps extends ResourceProps {\n\n  /**\n   * The spec of the service. Use `service.spec` to apply post instantiation mutations.\n   *\n   * @default - An empty spec will be created.\n   */\n  readonly spec?: ServiceSpec;\n\n}\n\n/**\n * For some parts of your application (for example, frontends) you may want to expose a Service onto an\n * external IP address, that's outside of your cluster.\n * Kubernetes ServiceTypes allow you to specify what kind of Service you want.\n * The default is ClusterIP.\n */\nexport enum ServiceType {\n\n  /**\n   * Exposes the Service on a cluster-internal IP.\n   * Choosing this value makes the Service only reachable from within the cluster.\n   * This is the default ServiceType\n   */\n  CLUSTER_IP = 'ClusterIP',\n\n  /**\n   * Exposes the Service on each Node's IP at a static port (the NodePort).\n   * A ClusterIP Service, to which the NodePort Service routes, is automatically created.\n   * You'll be able to contact the NodePort Service, from outside the cluster,\n   * by requesting <NodeIP>:<NodePort>.\n   */\n  NODE_PORT = 'NodePort',\n\n  /**\n   * Exposes the Service externally using a cloud provider's load balancer.\n   * NodePort and ClusterIP Services, to which the external load balancer routes,\n   * are automatically created.\n   */\n  LOAD_BALANCER = 'LoadBalancer',\n\n  /**\n   * Maps the Service to the contents of the externalName field (e.g. foo.bar.example.com), by returning a CNAME record with its value.\n   * No proxying of any kind is set up.\n   *\n   * > Note: You need either kube-dns version 1.7 or CoreDNS version 0.0.8 or higher to use the ExternalName type.\n   */\n  EXTERNAL_NAME = 'ExternalName'\n}\n\n/**\n * An abstract way to expose an application running on a set of Pods as a network service.\n * With Kubernetes you don't need to modify your application to use an unfamiliar service discovery mechanism.\n * Kubernetes gives Pods their own IP addresses and a single DNS name for a set of Pods, and can load-balance across them.\n *\n * For example, consider a stateless image-processing backend which is running with 3 replicas. Those replicas are fungible—frontends do not care which backend they use.\n * While the actual Pods that compose the backend set may change, the frontend clients should not need to be aware of that,\n * nor should they need to keep track of the set of backends themselves.\n * The Service abstraction enables this decoupling.\n *\n * If you're able to use Kubernetes APIs for service discovery in your application, you can query the API server for Endpoints,\n * that get updated whenever the set of Pods in a Service changes. For non-native applications, Kubernetes offers ways to place a network port\n * or load balancer in between your application and the backend Pods.\n */\nexport class Service extends Resource {\n\n  protected readonly apiObject: cdk8s.ApiObject;\n\n  /**\n   * Provides access to the underlying spec.\n   *\n   * You can use this field to apply post instantiation mutations\n   * to the spec.\n   */\n  public readonly spec: ServiceSpecDefinition;\n\n  constructor(scope: Construct, id: string, props: ServiceProps = {}) {\n    super(scope, id, props);\n\n    this.spec = new ServiceSpecDefinition(props.spec);\n\n    this.apiObject = new k8s.Service(this, 'Pod', {\n      metadata: props.metadata,\n      spec: cdk8s.Lazy.any({ produce: () => this.spec._toKube() }),\n    });\n  }\n\n}\n\nexport enum Protocol {\n  TCP = 'TCP',\n  UDP = 'UDP',\n  SCTP = 'SCTP'\n}\n\nexport interface ServicePortOptions {\n  /**\n   * The name of this port within the service. This must be a DNS_LABEL. All\n   * ports within a ServiceSpec must have unique names. This maps to the 'Name'\n   * field in EndpointPort objects. Optional if only one ServicePort is defined\n   * on this service.\n   */\n  readonly name?: string;\n\n  /**\n   * The port on each node on which this service is exposed when type=NodePort\n   * or LoadBalancer. Usually assigned by the system. If specified, it will be\n   * allocated to the service if unused or else creation of the service will\n   * fail. Default is to auto-allocate a port if the ServiceType of this Service\n   * requires one.\n   *\n   * @see https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport\n   *\n   * @default to auto-allocate a port if the ServiceType of this Service\n   * requires one.\n   */\n  readonly nodePort?: number;\n\n  /**\n   * The IP protocol for this port. Supports \"TCP\", \"UDP\", and \"SCTP\". Default is TCP.\n   *\n   * @default Protocol.TCP\n   */\n  readonly protocol?: Protocol;\n\n  /**\n   * The port number the service will redirect to.\n   *\n   * @default - The value of `port` will be used.\n   */\n  readonly targetPort?: number;\n}\n\n/**\n * Definition of a service port.\n */\nexport interface ServicePort extends ServicePortOptions {\n\n  /**\n   * The port number the service will bind to.\n   */\n  readonly port: number;\n}\n\n/**\n * Properties for initialization of `ServiceSpec`.\n */\nexport interface ServiceSpec {\n\n  /**\n   * The IP address of the service and is usually assigned randomly by the\n   * master. If an address is specified manually and is not in use by others, it\n   * will be allocated to the service; otherwise, creation of the service will\n   * fail. This field can not be changed through updates. Valid values are\n   * \"None\", empty string (\"\"), or a valid IP address. \"None\" can be specified\n   * for headless services when proxying is not required. Only applies to types\n   * ClusterIP, NodePort, and LoadBalancer. Ignored if type is ExternalName.\n   *\n   * @see https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies\n   * @default - Automatically assigned.\n   *\n   */\n  readonly clusterIP?: string;\n\n  /**\n   * A list of IP addresses for which nodes in the cluster will also accept\n   * traffic for this service. These IPs are not managed by Kubernetes. The user\n   * is responsible for ensuring that traffic arrives at a node with this IP. A\n   * common example is external load-balancers that are not part of the\n   * Kubernetes system.\n   *\n   * @default - No external IPs.\n   */\n  readonly externalIPs?: string[];\n\n  /**\n   * Determines how the Service is exposed.\n   *\n   * More info: https://kubernetes.io/docs/concepts/services-networking/service/#publishing-services-service-types\n   *\n   * @default ServiceType.ClusterIP\n   */\n  readonly type?: ServiceType;\n\n  /**\n   * The port exposed by this service.\n   *\n   * More info: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies\n   */\n  readonly ports?: ServicePort[];\n}\n\n/**\n * A description of a service.\n */\nexport class ServiceSpecDefinition {\n  /**\n   * The IP address of the service and is usually assigned randomly by the\n   * master.\n   */\n  public readonly clusterIP?: string;\n\n  /**\n   * A list of IP addresses for which nodes in the cluster will also accept\n   * traffic for this service.\n   */\n  private readonly externalIPs: string[];\n\n  /**\n   * Determines how the Service is exposed.\n   */\n  public readonly type: ServiceType;\n\n  private readonly _selector: Record<string, string>;\n\n  private readonly _ports: ServicePort[];\n\n  constructor(props: ServiceSpec = {}) {\n    this.clusterIP = props.clusterIP;\n    this.externalIPs = props.externalIPs ?? [];\n    this.type = props.type ?? ServiceType.CLUSTER_IP;\n    this._ports = [];\n    this._selector = { };\n\n    for (const portAndOptions of props.ports ?? []) {\n      this.serve(portAndOptions.port, portAndOptions);\n    }\n  }\n\n  /**\n   * Returns the labels which are used to select pods for this service.\n   */\n  public get selector() {\n    return this._selector;\n  }\n\n  /**\n   * Services defined using this spec will select pods according the provided label.\n   *\n   * @param label The label key.\n   * @param value The label value.\n   */\n  public addSelector(label: string, value: string) {\n    this._selector[label] = value;\n  }\n\n  /**\n   * Configure a port the service will bind to.\n   * This method can be called multiple times.\n   *\n   * @param port The port definition.\n   */\n  public serve(port: number, options: ServicePortOptions = { }) {\n    this._ports.push({ port, ...options });\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.ServiceSpec {\n    if (this._ports.length === 0) {\n      throw new Error('A service must be configured with a port');\n    }\n\n    const ports: k8s.ServicePort[] = [];\n\n    for (const port of this._ports) {\n      ports.push({\n        port: port.port,\n        targetPort: port.targetPort,\n        nodePort: port.nodePort,\n      });\n    }\n\n    return {\n      clusterIP: this.clusterIP,\n      externalIPs: this.externalIPs,\n      type: this.type,\n      selector: this._selector,\n      ports: ports,\n    };\n  }\n}"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service.js","sourceRoot":"","sources":["../src/service.ts"],"names":[],"mappings":"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","sourcesContent":["import * as k8s from './imports/k8s';\nimport { Construct } from 'constructs';\nimport { ResourceProps, Resource } from './base';\nimport * as cdk8s from 'cdk8s';\n\n\n/**\n * Properties for initialization of `Service`.\n */\nexport interface ServiceProps extends ResourceProps {\n\n  /**\n   * The spec of the service. Use `service.spec` to apply post instantiation mutations.\n   *\n   * @default - An empty spec will be created.\n   */\n  readonly spec?: ServiceSpec;\n\n}\n\n/**\n * For some parts of your application (for example, frontends) you may want to expose a Service onto an\n * external IP address, that's outside of your cluster.\n * Kubernetes ServiceTypes allow you to specify what kind of Service you want.\n * The default is ClusterIP.\n */\nexport enum ServiceType {\n\n  /**\n   * Exposes the Service on a cluster-internal IP.\n   * Choosing this value makes the Service only reachable from within the cluster.\n   * This is the default ServiceType\n   */\n  CLUSTER_IP = 'ClusterIP',\n\n  /**\n   * Exposes the Service on each Node's IP at a static port (the NodePort).\n   * A ClusterIP Service, to which the NodePort Service routes, is automatically created.\n   * You'll be able to contact the NodePort Service, from outside the cluster,\n   * by requesting <NodeIP>:<NodePort>.\n   */\n  NODE_PORT = 'NodePort',\n\n  /**\n   * Exposes the Service externally using a cloud provider's load balancer.\n   * NodePort and ClusterIP Services, to which the external load balancer routes,\n   * are automatically created.\n   */\n  LOAD_BALANCER = 'LoadBalancer',\n\n  /**\n   * Maps the Service to the contents of the externalName field (e.g. foo.bar.example.com), by returning a CNAME record with its value.\n   * No proxying of any kind is set up.\n   *\n   * > Note: You need either kube-dns version 1.7 or CoreDNS version 0.0.8 or higher to use the ExternalName type.\n   */\n  EXTERNAL_NAME = 'ExternalName'\n}\n\n/**\n * An abstract way to expose an application running on a set of Pods as a network service.\n * With Kubernetes you don't need to modify your application to use an unfamiliar service discovery mechanism.\n * Kubernetes gives Pods their own IP addresses and a single DNS name for a set of Pods, and can load-balance across them.\n *\n * For example, consider a stateless image-processing backend which is running with 3 replicas. Those replicas are fungible—frontends do not care which backend they use.\n * While the actual Pods that compose the backend set may change, the frontend clients should not need to be aware of that,\n * nor should they need to keep track of the set of backends themselves.\n * The Service abstraction enables this decoupling.\n *\n * If you're able to use Kubernetes APIs for service discovery in your application, you can query the API server for Endpoints,\n * that get updated whenever the set of Pods in a Service changes. For non-native applications, Kubernetes offers ways to place a network port\n * or load balancer in between your application and the backend Pods.\n */\nexport class Service extends Resource {\n\n  protected readonly apiObject: cdk8s.ApiObject;\n\n  /**\n   * Provides access to the underlying spec.\n   *\n   * You can use this field to apply post instantiation mutations\n   * to the spec.\n   */\n  public readonly spec: ServiceSpecDefinition;\n\n  constructor(scope: Construct, id: string, props: ServiceProps = {}) {\n    super(scope, id, props);\n\n    this.spec = new ServiceSpecDefinition(props.spec);\n\n    this.apiObject = new k8s.Service(this, 'Pod', {\n      metadata: props.metadata,\n      spec: cdk8s.Lazy.any({ produce: () => this.spec._toKube() }),\n    });\n  }\n\n}\n\nexport enum Protocol {\n  TCP = 'TCP',\n  UDP = 'UDP',\n  SCTP = 'SCTP'\n}\n\nexport interface ServicePortOptions {\n  /**\n   * The name of this port within the service. This must be a DNS_LABEL. All\n   * ports within a ServiceSpec must have unique names. This maps to the 'Name'\n   * field in EndpointPort objects. Optional if only one ServicePort is defined\n   * on this service.\n   */\n  readonly name?: string;\n\n  /**\n   * The port on each node on which this service is exposed when type=NodePort\n   * or LoadBalancer. Usually assigned by the system. If specified, it will be\n   * allocated to the service if unused or else creation of the service will\n   * fail. Default is to auto-allocate a port if the ServiceType of this Service\n   * requires one.\n   *\n   * @see https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport\n   *\n   * @default to auto-allocate a port if the ServiceType of this Service\n   * requires one.\n   */\n  readonly nodePort?: number;\n\n  /**\n   * The IP protocol for this port. Supports \"TCP\", \"UDP\", and \"SCTP\". Default is TCP.\n   *\n   * @default Protocol.TCP\n   */\n  readonly protocol?: Protocol;\n\n  /**\n   * The port number the service will redirect to.\n   *\n   * @default - The value of `port` will be used.\n   */\n  readonly targetPort?: number;\n}\n\n/**\n * Definition of a service port.\n */\nexport interface ServicePort extends ServicePortOptions {\n\n  /**\n   * The port number the service will bind to.\n   */\n  readonly port: number;\n}\n\n/**\n * Properties for initialization of `ServiceSpec`.\n */\nexport interface ServiceSpec {\n\n  /**\n   * The IP address of the service and is usually assigned randomly by the\n   * master. If an address is specified manually and is not in use by others, it\n   * will be allocated to the service; otherwise, creation of the service will\n   * fail. This field can not be changed through updates. Valid values are\n   * \"None\", empty string (\"\"), or a valid IP address. \"None\" can be specified\n   * for headless services when proxying is not required. Only applies to types\n   * ClusterIP, NodePort, and LoadBalancer. Ignored if type is ExternalName.\n   *\n   * @see https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies\n   * @default - Automatically assigned.\n   *\n   */\n  readonly clusterIP?: string;\n\n  /**\n   * A list of IP addresses for which nodes in the cluster will also accept\n   * traffic for this service. These IPs are not managed by Kubernetes. The user\n   * is responsible for ensuring that traffic arrives at a node with this IP. A\n   * common example is external load-balancers that are not part of the\n   * Kubernetes system.\n   *\n   * @default - No external IPs.\n   */\n  readonly externalIPs?: string[];\n\n  /**\n   * Determines how the Service is exposed.\n   *\n   * More info: https://kubernetes.io/docs/concepts/services-networking/service/#publishing-services-service-types\n   *\n   * @default ServiceType.ClusterIP\n   */\n  readonly type?: ServiceType;\n\n  /**\n   * The port exposed by this service.\n   *\n   * More info: https://kubernetes.io/docs/concepts/services-networking/service/#virtual-ips-and-service-proxies\n   */\n  readonly ports?: ServicePort[];\n}\n\n/**\n * A description of a service.\n */\nexport class ServiceSpecDefinition {\n  /**\n   * The IP address of the service and is usually assigned randomly by the\n   * master.\n   */\n  public readonly clusterIP?: string;\n\n  /**\n   * A list of IP addresses for which nodes in the cluster will also accept\n   * traffic for this service.\n   */\n  private readonly externalIPs: string[];\n\n  /**\n   * Determines how the Service is exposed.\n   */\n  public readonly type: ServiceType;\n\n  private readonly _selector: Record<string, string>;\n\n  private readonly _ports: ServicePort[];\n\n  constructor(props: ServiceSpec = {}) {\n    this.clusterIP = props.clusterIP;\n    this.externalIPs = props.externalIPs ?? [];\n    this.type = props.type ?? ServiceType.CLUSTER_IP;\n    this._ports = [];\n    this._selector = { };\n\n    for (const portAndOptions of props.ports ?? []) {\n      this.serve(portAndOptions.port, portAndOptions);\n    }\n  }\n\n  /**\n   * Returns the labels which are used to select pods for this service.\n   */\n  public get selector() {\n    return this._selector;\n  }\n\n  /**\n   * Services defined using this spec will select pods according the provided label.\n   *\n   * @param label The label key.\n   * @param value The label value.\n   */\n  public addSelector(label: string, value: string) {\n    this._selector[label] = value;\n  }\n\n  /**\n   * Configure a port the service will bind to.\n   * This method can be called multiple times.\n   *\n   * @param port The port definition.\n   */\n  public serve(port: number, options: ServicePortOptions = { }) {\n    this._ports.push({ port, ...options });\n  }\n\n  /**\n   * Ports for this service.\n   * \n   * Use `serve()` to expose additional service ports.\n   */\n  public get ports() {\n    return [...this._ports];\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.ServiceSpec {\n    if (this._ports.length === 0) {\n      throw new Error('A service must be configured with a port');\n    }\n\n    const ports: k8s.ServicePort[] = [];\n\n    for (const port of this._ports) {\n      ports.push({\n        port: port.port,\n        targetPort: port.targetPort,\n        nodePort: port.nodePort,\n      });\n    }\n\n    return {\n      clusterIP: this.clusterIP,\n      externalIPs: this.externalIPs,\n      type: this.type,\n      selector: this._selector,\n      ports: ports,\n    };\n  }\n}"]} |
@@ -34,3 +34,3 @@ { | ||
"devDependencies": { | ||
"cdk8s": "^0.29.0", | ||
"cdk8s": "0.30.0-pre.14ac668f5f88445f14864ac3d271f8a9afbd40c7", | ||
"constructs": "^3.0.4", | ||
@@ -58,3 +58,3 @@ "@types/minimatch": "^3.0.3", | ||
"peerDependencies": { | ||
"cdk8s": "^0.29.0", | ||
"cdk8s": "0.30.0-pre.14ac668f5f88445f14864ac3d271f8a9afbd40c7", | ||
"constructs": "^3.0.4" | ||
@@ -78,3 +78,3 @@ }, | ||
"license": "Apache-2.0", | ||
"version": "0.29.0", | ||
"version": "0.30.0-pre.14ac668f5f88445f14864ac3d271f8a9afbd40c7", | ||
"types": "lib/index.d.ts", | ||
@@ -81,0 +81,0 @@ "jest": { |
# cdk8s+ (cdk8s-plus) | ||
> ![Experimental](https://img.shields.io/badge/experimental-important.svg?style=for-the-badge)<br><br> | ||
> This library is in very early stages of development, as such, and in correspondence with a `0.x` semantic major version line, its `API` is | ||
likely to rapidly change in breaking ways. It is therefore not recommended to use library for production workloads. | ||
**cdk8s+** is a software development framework that provides high level abstractions for authoring Kubernetes applications. | ||
@@ -7,7 +11,9 @@ Built on top of the auto generated building blocks provided by [cdk8s](../cdk8s), this library includes a hand crafted *construct* | ||
> **You should not use this library in production environments.**<br><br> | ||
> ![Experimental](https://img.shields.io/badge/experimental-important.svg?style=for-the-badge)<br><br> | ||
> This library is in very early stages of development, as such, and in correspondence with a `0.x` semantic major version line, its `API` is | ||
likely to rapidly change in breaking ways. We therefore highly discourage from using this library in production workloads. | ||
## Kubernetes Spec | ||
**cdk8s+** is currently built on top of version [1.17.0](https://github.com/instrumenta/kubernetes-json-schema/tree/master/v1.17.0) of the kubernetes API specifications. | ||
If you are deploying manifests produced by `cdk8s+` onto clusters of a lower version, you might encounter some unsupported spec properties or invalid manifests. | ||
> See [Supporting various k8s API specs](https://github.com/awslabs/cdk8s/issues/299) for more details and progress on this issue. | ||
## Letter Of Intent | ||
@@ -189,3 +195,2 @@ | ||
## In Depth | ||
@@ -654,2 +659,53 @@ | ||
awsService.addSecret(awsCreds); | ||
``` | ||
``` | ||
### `Ingress` | ||
[Ingress] manages external access to services in a cluster, typically through | ||
HTTP. Ingress may provide load balancing, SSL termination and name-based virtual | ||
hosting. | ||
You must have an [Ingress controller] to satisfy an Ingress. Only creating an | ||
Ingress resource has no effect. | ||
> API Reference: [Ingress](./API.md#cdk8s-plus-ingress) | ||
[Ingress]: https://kubernetes.io/docs/concepts/services-networking/ingress/ | ||
[Ingress controller]: https://kubernetes.io/docs/concepts/services-networking/ingress-controllers | ||
The following example will route HTTP requests sent to the `/hello` url prefix | ||
to a service associated with a deployment of the | ||
[hashicorp/http-echo](https://github.com/hashicorp/http-echo) image. | ||
```ts | ||
const helloDeployment = new kplus.Deployment(this, text, { | ||
spec: { | ||
podSpecTemplate: { | ||
containers: [ | ||
new kplus.Container({ | ||
image: 'hashicorp/http-echo', | ||
args: [ '-text', 'hello ingress' ] | ||
}) | ||
] | ||
} | ||
} | ||
}); | ||
const helloService = helloDeployment.expose({ port: 5678 }); | ||
const ingress = new Ingress(this, 'ingress'); | ||
ingress.addRule('/hello', kplus.IngressBackend.fromService(helloService)); | ||
``` | ||
Yo can use `addHostRule(host, path, backend)` to define a route that will only | ||
apply to requests with this `Host` header. This can be used to implement virtual | ||
hosts. | ||
The `addDefaultBackend(backend)` and `addHostDefaultBackend(host, backend)` | ||
methods can be used to define backends that will accept all requests that do not | ||
match any other rules. | ||
The TCP port used to route requests to services will be determined based on | ||
which ports are exposed by the service (e.g. through `serve()`). If the service | ||
exposes multiple ports, then a port must be specified via | ||
`IngressBackend.fromService(service, { port }`. |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
2015672
59
18593
709