UNPKG

cdk8s

Version:

This is the core library of Cloud Development Kit (CDK) for Kubernetes (cdk8s). cdk8s apps synthesize into standard Kubernetes manifests which can be applied to any Kubernetes cluster.

148 lines (147 loc) • 18.7 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.ApiObject = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const constructs_1 = require("constructs"); const _util_1 = require("./_util"); const chart_1 = require("./chart"); const json_patch_1 = require("./json-patch"); const metadata_1 = require("./metadata"); const resolve_1 = require("./resolve"); const API_OBJECT_SYMBOL = Symbol.for('cdk8s.ApiObject'); class ApiObject extends constructs_1.Construct { /** * Return whether the given object is an `ApiObject`. * * We do attribute detection since we can't reliably use 'instanceof'. * @param o The object to check */ static isApiObject(o) { return o !== null && typeof o === 'object' && API_OBJECT_SYMBOL in o; } /** * Implements `instanceof ApiObject` using the more reliable `ApiObject.isApiObject` static method * * @param o The object to check * @internal */ static [(_a = JSII_RTTI_SYMBOL_1, Symbol.hasInstance)](o) { return ApiObject.isApiObject(o); } /** * Returns the `ApiObject` named `Resource` which is a child of the given * construct. If `c` is an `ApiObject`, it is returned directly. Throws an * exception if the construct does not have a child named `Default` _or_ if * this child is not an `ApiObject`. * * @param c The higher-level construct */ static of(c) { if (c instanceof ApiObject) { return c; } const child = c.node.defaultChild; if (!child) { throw new Error(`cannot find a (direct or indirect) child of type ApiObject for construct ${c.node.path}`); } return ApiObject.of(child); } /** * Defines an API object. * * @param scope the construct scope * @param id namespace * @param props options */ constructor(scope, id, props) { super(scope, id); this.props = props; this.patches = new Array(); this.chart = chart_1.Chart.of(this); this.kind = props.kind; this.apiVersion = props.apiVersion; this.apiGroup = parseApiGroup(this.apiVersion); this.name = props.metadata?.name ?? this.chart.generateObjectName(this); this.metadata = new metadata_1.ApiObjectMetadataDefinition({ name: this.name, // user defined values ...props.metadata, namespace: props.metadata?.namespace ?? this.chart.namespace, labels: { ...this.chart.labels, ...props.metadata?.labels, }, apiObject: this, }); Object.defineProperty(this, API_OBJECT_SYMBOL, { value: true }); } /** * Create a dependency between this ApiObject and other constructs. * These can be other ApiObjects, Charts, or custom. * * @param dependencies the dependencies to add. */ addDependency(...dependencies) { this.node.addDependency(...dependencies); } /** * Applies a set of RFC-6902 JSON-Patch operations to the manifest * synthesized for this API object. * * @param ops The JSON-Patch operations to apply. * * @example * * kubePod.addJsonPatch(JsonPatch.replace('/spec/enableServiceLinks', true)); * */ addJsonPatch(...ops) { this.patches.push(...ops); } /** * Renders the object to Kubernetes JSON. * * To disable sorting of dictionary keys in output object set the * `CDK8S_DISABLE_SORT` environment variable to any non-empty value. */ toJson() { try { const data = { ...this.props, metadata: this.metadata.toJson(), }; const sortKeys = process.env.CDK8S_DISABLE_SORT ? false : true; const json = (0, _util_1.sanitizeValue)((0, resolve_1.resolve)([], data, this), { sortKeys }); const patched = json_patch_1.JsonPatch.apply(json, ...this.patches); // reorder top-level keys so that we first have "apiVersion", "kind" and // "metadata" and then all the rest const result = {}; const orderedKeys = ['apiVersion', 'kind', 'metadata', ...Object.keys(patched)]; for (const k of orderedKeys) { if (k in patched) { result[k] = patched[k]; } } return result; } catch (e) { throw new Error(`Failed serializing construct at path '${this.node.path}' with name '${this.name}': ${e}`); } } } exports.ApiObject = ApiObject; ApiObject[_a] = { fqn: "cdk8s.ApiObject", version: "2.70.4" }; function parseApiGroup(apiVersion) { const v = apiVersion.split('/'); // no group means "core" // https://kubernetes.io/docs/reference/using-api/api-overview/#api-groups if (v.length === 1) { return 'core'; } if (v.length === 2) { return v[0]; } throw new Error(`invalid apiVersion ${apiVersion}, expecting GROUP/VERSION. See https://kubernetes.io/docs/reference/using-api/api-overview/#api-groups`); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-object.js","sourceRoot":"","sources":["../src/api-object.ts"],"names":[],"mappings":";;;;;AAAA,2CAAmD;AACnD,mCAAwC;AACxC,mCAAgC;AAChC,6CAAyC;AACzC,yCAA4E;AAC5E,uCAAoC;AA4CpC,MAAM,iBAAiB,GAAG,MAAM,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;AAExD,MAAa,SAAU,SAAQ,sBAAS;IAEtC;;;;;;OAMG;IACH,MAAM,CAAC,WAAW,CAAC,CAAM;QACvB,OAAO,CAAC,KAAK,IAAI,IAAI,OAAO,CAAC,KAAK,QAAQ,IAAI,iBAAiB,IAAI,CAAC,CAAC;IACvE,CAAC;IAED;;;;;OAKG;IACH,MAAM,CAAC,2BAAC,MAAM,CAAC,WAAW,EAAC,CAAC,CAAU;QACpC,OAAO,SAAS,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;IAClC,CAAC;IACD;;;;;;;OAOG;IACI,MAAM,CAAC,EAAE,CAAC,CAAa;QAC5B,IAAI,CAAC,YAAY,SAAS,EAAE,CAAC;YAC3B,OAAO,CAAC,CAAC;QACX,CAAC;QAED,MAAM,KAAK,GAAG,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC;QAClC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,4EAA4E,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QAC7G,CAAC;QAED,OAAO,SAAS,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;IAC7B,CAAC;IA0CD;;;;;;OAMG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAmB,KAAqB;QAC9E,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QADwC,UAAK,GAAL,KAAK,CAAgB;QAE9E,IAAI,CAAC,OAAO,GAAG,IAAI,KAAK,EAAa,CAAC;QACtC,IAAI,CAAC,KAAK,GAAG,aAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QAC5B,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC;QACvB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,UAAU,CAAC;QACnC,IAAI,CAAC,QAAQ,GAAG,aAAa,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAE/C,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC,QAAQ,EAAE,IAAI,IAAI,IAAI,CAAC,KAAK,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC;QAExE,IAAI,CAAC,QAAQ,GAAG,IAAI,sCAA2B,CAAC;YAC9C,IAAI,EAAE,IAAI,CAAC,IAAI;YAEf,sBAAsB;YACtB,GAAG,KAAK,CAAC,QAAQ;YAEjB,SAAS,EAAE,KAAK,CAAC,QAAQ,EAAE,SAAS,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS;YAC5D,MAAM,EAAE;gBACN,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM;gBACpB,GAAG,KAAK,CAAC,QAAQ,EAAE,MAAM;aAC1B;YACD,SAAS,EAAE,IAAI;SAChB,CAAC,CAAC;QAEH,MAAM,CAAC,cAAc,CAAC,IAAI,EAAE,iBAAiB,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IAClE,CAAC;IAED;;;;;OAKG;IACI,aAAa,CAAC,GAAG,YAA0B;QAChD,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG,YAAY,CAAC,CAAC;IAC3C,CAAC;IAED;;;;;;;;;;OAUG;IACI,YAAY,CAAC,GAAG,GAAgB;QACrC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC;IAC5B,CAAC;IAED;;;;;OAKG;IACI,MAAM;QAEX,IAAI,CAAC;YACH,MAAM,IAAI,GAAQ;gBAChB,GAAG,IAAI,CAAC,KAAK;gBACb,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE;aACjC,CAAC;YAEF,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;YAC/D,MAAM,IAAI,GAAG,IAAA,qBAAa,EAAC,IAAA,iBAAO,EAAC,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;YAClE,MAAM,OAAO,GAAG,sBAAS,CAAC,KAAK,CAAC,IAAI,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC;YAEvD,wEAAwE;YACxE,mCAAmC;YACnC,MAAM,MAAM,GAAQ,EAAE,CAAC;YACvB,MAAM,WAAW,GAAG,CAAC,YAAY,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;YAChF,KAAK,MAAM,CAAC,IAAI,WAAW,EAAE,CAAC;gBAC5B,IAAI,CAAC,IAAI,OAAO,EAAE,CAAC;oBACjB,MAAM,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;gBACzB,CAAC;YACH,CAAC;YAED,OAAO,MAAM,CAAC;QAChB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,yCAAyC,IAAI,CAAC,IAAI,CAAC,IAAI,gBAAgB,IAAI,CAAC,IAAI,MAAM,CAAC,EAAE,CAAC,CAAC;QAC7G,CAAC;IACH,CAAC;;AA9KH,8BA+KC;;AAED,SAAS,aAAa,CAAC,UAAkB;IACvC,MAAM,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAEhC,wBAAwB;IACxB,0EAA0E;IAC1E,IAAI,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACnB,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,IAAI,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACnB,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;IACd,CAAC;IAED,MAAM,IAAI,KAAK,CAAC,sBAAsB,UAAU,wGAAwG,CAAC,CAAC;AAC5J,CAAC","sourcesContent":["import { Construct, IConstruct } from 'constructs';\nimport { sanitizeValue } from './_util';\nimport { Chart } from './chart';\nimport { JsonPatch } from './json-patch';\nimport { ApiObjectMetadata, ApiObjectMetadataDefinition } from './metadata';\nimport { resolve } from './resolve';\n\n/**\n * Options for defining API objects.\n */\nexport interface ApiObjectProps {\n  /**\n   * Object metadata.\n   *\n   * If `name` is not specified, an app-unique name will be allocated by the\n   * framework based on the path of the construct within thes construct tree.\n   */\n  readonly metadata?: ApiObjectMetadata;\n\n  /**\n   * API version.\n   */\n  readonly apiVersion: string;\n\n  /**\n   * Resource kind.\n   */\n  readonly kind: string;\n\n  /**\n   * Additional attributes for this API object.\n   * @jsii ignore\n   * @see https://github.com/cdk8s-team/cdk8s-core/issues/1297\n   */\n  readonly [key: string]: any;\n}\n\nexport interface GroupVersionKind {\n  /**\n   * The object's API version (e.g. `authorization.k8s.io/v1`)\n   */\n  readonly apiVersion: string;\n\n  /**\n   * The object kind.\n   */\n  readonly kind: string;\n}\n\nconst API_OBJECT_SYMBOL = Symbol.for('cdk8s.ApiObject');\n\nexport class ApiObject extends Construct {\n\n  /**\n   * Return whether the given object is an `ApiObject`.\n   *\n   * We do attribute detection since we can't reliably use 'instanceof'.\n\n   * @param o The object to check\n   */\n  static isApiObject(o: any): o is ApiObject {\n    return o !== null && typeof o === 'object' && API_OBJECT_SYMBOL in o;\n  }\n\n  /**\n   * Implements `instanceof ApiObject` using the more reliable `ApiObject.isApiObject` static method\n   *\n   * @param o The object to check\n   * @internal\n   */\n  static [Symbol.hasInstance](o: unknown) {\n    return ApiObject.isApiObject(o);\n  }\n  /**\n   * Returns the `ApiObject` named `Resource` which is a child of the given\n   * construct. If `c` is an `ApiObject`, it is returned directly. Throws an\n   * exception if the construct does not have a child named `Default` _or_ if\n   * this child is not an `ApiObject`.\n   *\n   * @param c The higher-level construct\n   */\n  public static of(c: IConstruct): ApiObject {\n    if (c instanceof ApiObject) {\n      return c;\n    }\n\n    const child = c.node.defaultChild;\n    if (!child) {\n      throw new Error(`cannot find a (direct or indirect) child of type ApiObject for construct ${c.node.path}`);\n    }\n\n    return ApiObject.of(child);\n  }\n\n  /**\n   * The name of the API object.\n   *\n   * If a name is specified in `metadata.name` this will be the name returned.\n   * Otherwise, a name will be generated by calling\n   * `Chart.of(this).generatedObjectName(this)`, which by default uses the\n   * construct path to generate a DNS-compatible name for the resource.\n   */\n  public readonly name: string;\n\n  /**\n   * The object's API version (e.g. `authorization.k8s.io/v1`)\n   */\n  public readonly apiVersion: string;\n\n  /**\n   * The group portion of the API version (e.g. `authorization.k8s.io`)\n   */\n  public readonly apiGroup: string;\n\n  /**\n   * The object kind.\n   */\n  public readonly kind: string;\n\n  /**\n   * The chart in which this object is defined.\n   */\n  public readonly chart: Chart;\n\n  /**\n   * Metadata associated with this API object.\n   */\n  public readonly metadata: ApiObjectMetadataDefinition;\n\n  /**\n   * A set of JSON patch operations to apply to the document after synthesis.\n   */\n  private readonly patches: Array<JsonPatch>;\n\n  /**\n   * Defines an API object.\n   *\n   * @param scope the construct scope\n   * @param id namespace\n   * @param props options\n   */\n  constructor(scope: Construct, id: string, private readonly props: ApiObjectProps) {\n    super(scope, id);\n    this.patches = new Array<JsonPatch>();\n    this.chart = Chart.of(this);\n    this.kind = props.kind;\n    this.apiVersion = props.apiVersion;\n    this.apiGroup = parseApiGroup(this.apiVersion);\n\n    this.name = props.metadata?.name ?? this.chart.generateObjectName(this);\n\n    this.metadata = new ApiObjectMetadataDefinition({\n      name: this.name,\n\n      // user defined values\n      ...props.metadata,\n\n      namespace: props.metadata?.namespace ?? this.chart.namespace,\n      labels: {\n        ...this.chart.labels,\n        ...props.metadata?.labels,\n      },\n      apiObject: this,\n    });\n\n    Object.defineProperty(this, API_OBJECT_SYMBOL, { value: true });\n  }\n\n  /**\n   * Create a dependency between this ApiObject and other constructs.\n   * These can be other ApiObjects, Charts, or custom.\n   *\n   * @param dependencies the dependencies to add.\n   */\n  public addDependency(...dependencies: IConstruct[]) {\n    this.node.addDependency(...dependencies);\n  }\n\n  /**\n   * Applies a set of RFC-6902 JSON-Patch operations to the manifest\n   * synthesized for this API object.\n   *\n   * @param ops The JSON-Patch operations to apply.\n   *\n   * @example\n   *\n   *   kubePod.addJsonPatch(JsonPatch.replace('/spec/enableServiceLinks', true));\n   *\n   */\n  public addJsonPatch(...ops: JsonPatch[]) {\n    this.patches.push(...ops);\n  }\n\n  /**\n   * Renders the object to Kubernetes JSON.\n   *\n   * To disable sorting of dictionary keys in output object set the\n   * `CDK8S_DISABLE_SORT` environment variable to any non-empty value.\n   */\n  public toJson(): any {\n\n    try {\n      const data: any = {\n        ...this.props,\n        metadata: this.metadata.toJson(),\n      };\n\n      const sortKeys = process.env.CDK8S_DISABLE_SORT ? false : true;\n      const json = sanitizeValue(resolve([], data, this), { sortKeys });\n      const patched = JsonPatch.apply(json, ...this.patches);\n\n      // reorder top-level keys so that we first have \"apiVersion\", \"kind\" and\n      // \"metadata\" and then all the rest\n      const result: any = {};\n      const orderedKeys = ['apiVersion', 'kind', 'metadata', ...Object.keys(patched)];\n      for (const k of orderedKeys) {\n        if (k in patched) {\n          result[k] = patched[k];\n        }\n      }\n\n      return result;\n    } catch (e) {\n      throw new Error(`Failed serializing construct at path '${this.node.path}' with name '${this.name}': ${e}`);\n    }\n  }\n}\n\nfunction parseApiGroup(apiVersion: string) {\n  const v = apiVersion.split('/');\n\n  // no group means \"core\"\n  // https://kubernetes.io/docs/reference/using-api/api-overview/#api-groups\n  if (v.length === 1) {\n    return 'core';\n  }\n\n  if (v.length === 2) {\n    return v[0];\n  }\n\n  throw new Error(`invalid apiVersion ${apiVersion}, expecting GROUP/VERSION. See https://kubernetes.io/docs/reference/using-api/api-overview/#api-groups`);\n}\n"]}