UNPKG

cdk8s-cli

Version:

This is the command line tool for Cloud Development Kit (CDK) for Kubernetes (cdk8s).

195 lines • 25 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.safeParseCrds = exports.ImportCustomResourceDefinition = exports.CustomResourceDefinition = void 0; const fs = __importStar(require("fs")); const path = __importStar(require("path")); const ajv_1 = __importDefault(require("ajv")); const codemaker_1 = require("codemaker"); const json2jsii_1 = require("json2jsii"); const base_1 = require("./base"); const codegen_1 = require("./codegen"); const reviver_1 = require("../reviver"); const util_1 = require("../util"); const CRD_KIND = 'CustomResourceDefinition'; // all these APIs are compatible from our perspective. const SUPPORTED_API_VERSIONS = [ 'apiextensions.k8s.io/v1beta1', 'apiextensions.k8s.io/v1', ]; class CustomResourceDefinition { constructor(manifest) { this.versions = []; const apiVersion = manifest?.apiVersion ?? 'undefined'; assert(SUPPORTED_API_VERSIONS.includes(apiVersion), `"apiVersion" is "${apiVersion}" but it should be one of: ${SUPPORTED_API_VERSIONS.map(x => `"${x}"`).join(', ')}`); assert(manifest.kind === CRD_KIND, `"kind" must be "${CRD_KIND}"`); const spec = manifest.spec; if (!spec) { throw new Error('manifest does not have a "spec" attribute'); } this.group = spec.group; this.kind = spec.names.kind; if (spec.version) { this.addVersions([{ name: spec.version, schema: spec.validation?.openAPIV3Schema }]); } else { this.addVersions((spec.versions ?? []).map(v => ({ name: v.name, schema: v.schema?.openAPIV3Schema ?? spec.validation?.openAPIV3Schema }))); } if (this.versions.length === 0) { throw new Error('unable to determine CRD versions'); } } merge(crd) { this.addVersions(crd.versions); } addVersions(versions) { for (const v of versions) { const existingVersions = this.versions.map(ver => ver.name); if (existingVersions.includes(v.name)) { throw new Error(`Found multiple occurrences of version ${v.name} for ${this.key}`); } this.versions.push({ name: v.name, schema: v.schema }); } } get key() { return `${this.group}/${this.kind.toLocaleLowerCase()}`; } async generateTypeScript(code, options) { for (let i = 0; i < this.versions.length; i++) { const version = this.versions[i]; // to preseve backwards compatiblity, only append a suffix for // the second version onwards. const suffix = i === 0 ? '' : (0, codemaker_1.toPascalCase)(version.name); const types = new json2jsii_1.TypeGenerator({}); (0, codegen_1.generateConstruct)(types, { group: this.group, version: version.name, kind: this.kind, fqn: `${this.kind}${suffix}`, schema: version.schema, custom: true, prefix: options.classNamePrefix, suffix, }); code.line(types.render()); } } } exports.CustomResourceDefinition = CustomResourceDefinition; class ImportCustomResourceDefinition extends base_1.ImportBase { static async fromSpec(importSpec) { const { source } = importSpec; const manifest = await (0, util_1.download)(source); return new ImportCustomResourceDefinition(manifest); } constructor(rawManifest) { super(); this.groups = {}; this.rawManifest = rawManifest; const manifest = safeParseCrds(rawManifest); const crds = {}; const groups = {}; for (const spec of manifest) { const crd = new CustomResourceDefinition(spec); const key = crd.key; if (key in crds) { // might contain different versions - lets try to merge them in crds[key].merge(crd); } else { crds[key] = crd; } } //sort to ensure consistent ordering for snapshot compare const sortedCrds = Object.values(crds).sort((a, b) => a.key.localeCompare(b.key)); for (const crd of sortedCrds) { const g = crd.group; if (!(g in groups)) { groups[g] = new Array(); } groups[g].push(crd); } this.groups = groups; } get moduleNames() { return Object.keys(this.groups); } async generateTypeScript(code, moduleName, options) { const crds = this.groups[moduleName]; (0, codegen_1.emitHeader)(code, true); for (const crd of crds) { console.log(` ${crd.key}`); await crd.generateTypeScript(code, options); } } } exports.ImportCustomResourceDefinition = ImportCustomResourceDefinition; function assert(condition, message) { if (!condition) { throw new Error(`invalid CustomResourceDefinition manifest: ${message}`); } } function safeParseCrds(manifest) { const schemaPath = path.join(__dirname, '..', 'schemas', 'crd.schema.json'); const schema = JSON.parse(fs.readFileSync(schemaPath, { encoding: 'utf8' })); const reviver = new reviver_1.SafeReviver({ sanitizers: [reviver_1.SafeReviver.DESCRIPTION_SANITIZER, reviver_1.SafeReviver.LEGAL_CHAR_SANITIZER], }); // first parse and strip const objects = (0, util_1.safeParseYaml)(manifest, reviver); // since the manifest can contain non crds as well, we first // collect all crds and only apply a schema validation on them. const crds = []; function collectCRDs(objs) { for (const obj of objs.filter(o => o)) { if (obj.kind === CRD_KIND) { crds.push(obj); } if (obj.kind === 'List') { collectCRDs(obj.items); } } } collectCRDs(objects); const ajv = new ajv_1.default(); const validate = ajv.compile(schema); const errors = []; for (const crd of crds) { validate(crd); if (validate.errors) { errors.push(...validate.errors); } ; } if (errors.length > 0) { throw new Error(`Schema validation errors detected\n ${errors.map(e => `* ${e.message}`).join('\n')}`); } return crds; } exports.safeParseCrds = safeParseCrds; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crd.js","sourceRoot":"","sources":["../../src/import/crd.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport Ajv from 'ajv';\nimport { CodeMaker, toPascalCase } from 'codemaker';\nimport { TypeGenerator } from 'json2jsii';\nimport { GenerateOptions, ImportBase } from './base';\nimport { emitHeader, generateConstruct } from './codegen';\nimport { ImportSpec } from '../config';\nimport { SafeReviver } from '../reviver';\nimport { download, safeParseYaml } from '../util';\n\nconst CRD_KIND = 'CustomResourceDefinition';\n\nexport interface ManifestObjectDefinition {\n  apiVersion?: string;\n  kind?: string;\n  items?: ManifestObjectDefinition[]; // if `kind` is \"List\"\n  metadata?: {\n    name?: string;\n  };\n  spec?: {\n    group: string;\n    names: {\n      kind: string;\n      [key: string]: any;\n    };\n    versions?: Array<{\n      name: string;\n      schema?: { openAPIV3Schema?: any };\n      [key: string]: any;\n    }>;\n    version?: string;\n    validation?: { openAPIV3Schema?: any };\n    [key: string]: any;\n  };\n}\n\n// all these APIs are compatible from our perspective.\nconst SUPPORTED_API_VERSIONS = [\n  'apiextensions.k8s.io/v1beta1',\n  'apiextensions.k8s.io/v1',\n];\n\ntype CustomResourceDefinitionVersion = { name: string; schema?: any };\n\nexport class CustomResourceDefinition {\n\n  private readonly kind: string;\n  private readonly versions: CustomResourceDefinitionVersion[] = [];\n\n  public readonly group: string;\n\n  constructor(manifest: ManifestObjectDefinition) {\n    const apiVersion = manifest?.apiVersion ?? 'undefined';\n    assert(SUPPORTED_API_VERSIONS.includes(apiVersion), `\"apiVersion\" is \"${apiVersion}\" but it should be one of: ${SUPPORTED_API_VERSIONS.map(x => `\"${x}\"`).join(', ')}`);\n    assert(manifest.kind === CRD_KIND, `\"kind\" must be \"${CRD_KIND}\"`);\n\n    const spec = manifest.spec;\n    if (!spec) {\n      throw new Error('manifest does not have a \"spec\" attribute');\n    }\n\n    this.group = spec.group;\n    this.kind = spec.names.kind;\n\n    if (spec.version) {\n      this.addVersions([{ name: spec.version, schema: spec.validation?.openAPIV3Schema }]);\n    } else {\n      this.addVersions((spec.versions ?? []).map(v => ({ name: v.name, schema: v.schema?.openAPIV3Schema ?? spec.validation?.openAPIV3Schema })));\n    }\n\n    if (this.versions.length === 0) {\n      throw new Error('unable to determine CRD versions');\n    }\n\n  }\n\n  public merge(crd: CustomResourceDefinition) {\n    this.addVersions(crd.versions);\n  }\n\n  private addVersions(versions: CustomResourceDefinitionVersion[]) {\n    for (const v of versions) {\n      const existingVersions = this.versions.map(ver => ver.name);\n      if (existingVersions.includes(v.name)) {\n        throw new Error(`Found multiple occurrences of version ${v.name} for ${this.key}`);\n      }\n      this.versions.push({ name: v.name, schema: v.schema });\n    }\n  }\n\n  public get key() {\n    return `${this.group}/${this.kind.toLocaleLowerCase()}`;\n  }\n\n  public async generateTypeScript(code: CodeMaker, options: GenerateOptions) {\n\n    for (let i = 0; i < this.versions.length; i++) {\n\n      const version = this.versions[i];\n\n      // to preseve backwards compatiblity, only append a suffix for\n      // the second version onwards.\n      const suffix = i === 0 ? '' : toPascalCase(version.name);\n\n      const types = new TypeGenerator({});\n\n      generateConstruct(types, {\n        group: this.group,\n        version: version.name,\n        kind: this.kind,\n        fqn: `${this.kind}${suffix}`,\n        schema: version.schema,\n        custom: true,\n        prefix: options.classNamePrefix,\n        suffix,\n      });\n\n      code.line(types.render());\n    }\n  }\n}\n\nexport class ImportCustomResourceDefinition extends ImportBase {\n  public static async fromSpec(importSpec: ImportSpec): Promise<ImportCustomResourceDefinition> {\n    const { source } = importSpec;\n    const manifest = await download(source);\n    return new ImportCustomResourceDefinition(manifest);\n  }\n\n  public readonly rawManifest: string;\n  private readonly groups: Record<string, CustomResourceDefinition[]> = { };\n\n  private constructor(rawManifest: string) {\n    super();\n\n    this.rawManifest = rawManifest;\n    const manifest = safeParseCrds(rawManifest);\n\n    const crds: Record<string, CustomResourceDefinition> = { };\n    const groups: Record<string, CustomResourceDefinition[]> = { };\n\n    for (const spec of manifest) {\n      const crd = new CustomResourceDefinition(spec);\n      const key = crd.key;\n\n      if (key in crds) {\n        // might contain different versions - lets try to merge them in\n        crds[key].merge(crd);\n      } else {\n        crds[key] = crd;\n      }\n    }\n\n    //sort to ensure consistent ordering for snapshot compare\n    const sortedCrds = Object.values(crds).sort((a: CustomResourceDefinition, b: CustomResourceDefinition) => a.key.localeCompare(b.key));\n\n    for (const crd of sortedCrds) {\n      const g = crd.group;\n      if ( !(g in groups) ) {\n        groups[g] = new Array<CustomResourceDefinition>();\n      }\n      groups[g].push(crd);\n    }\n\n    this.groups = groups;\n  }\n\n  public get moduleNames() {\n    return Object.keys(this.groups);\n  }\n\n  protected async generateTypeScript(code: CodeMaker, moduleName: string, options: GenerateOptions) {\n    const crds = this.groups[moduleName];\n\n\n    emitHeader(code, true);\n\n    for (const crd of crds) {\n      console.log(`  ${crd.key}`);\n      await crd.generateTypeScript(code, options);\n    }\n  }\n}\n\nfunction assert(condition: boolean, message: string) {\n  if (!condition) {\n    throw new Error(`invalid CustomResourceDefinition manifest: ${message}`);\n  }\n}\n\n\nexport function safeParseCrds(manifest: string): ManifestObjectDefinition[] {\n  const schemaPath = path.join(__dirname, '..', 'schemas', 'crd.schema.json');\n  const schema = JSON.parse(fs.readFileSync(schemaPath, { encoding: 'utf8' }));\n  const reviver = new SafeReviver({\n    sanitizers: [SafeReviver.DESCRIPTION_SANITIZER, SafeReviver.LEGAL_CHAR_SANITIZER],\n  });\n\n  // first parse and strip\n  const objects = safeParseYaml(manifest, reviver);\n\n  // since the manifest can contain non crds as well, we first\n  // collect all crds and only apply a schema validation on them.\n\n  const crds: any[] = [];\n\n  function collectCRDs(objs: any[]) {\n    for (const obj of objs.filter(o => o)) {\n      if (obj.kind === CRD_KIND) {\n        crds.push(obj);\n      }\n      if (obj.kind === 'List') {\n        collectCRDs(obj.items);\n      }\n    }\n  }\n\n  collectCRDs(objects);\n\n  const ajv = new Ajv();\n  const validate = ajv.compile(schema);\n  const errors = [];\n  for (const crd of crds) {\n    validate(crd);\n    if (validate.errors) {\n      errors.push(...validate.errors);\n    };\n  }\n  if (errors.length > 0) {\n    throw new Error(`Schema validation errors detected\\n ${errors.map(e => `* ${e.message}`).join('\\n')}`);\n  }\n  return crds;\n}\n"]}