cdk8s-cli
Version:
This is the command line tool for Cloud Development Kit (CDK) for Kubernetes (cdk8s).
196 lines • 25.4 kB
JavaScript
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) {
var _a, _b, _c;
this.versions = [];
const apiVersion = (_a = manifest === null || manifest === void 0 ? void 0 : manifest.apiVersion) !== null && _a !== void 0 ? _a : '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: (_b = spec.validation) === null || _b === void 0 ? void 0 : _b.openAPIV3Schema }]);
}
else {
this.addVersions(((_c = spec.versions) !== null && _c !== void 0 ? _c : []).map(v => { var _a, _b, _c; return ({ name: v.name, schema: (_b = (_a = v.schema) === null || _a === void 0 ? void 0 : _a.openAPIV3Schema) !== null && _b !== void 0 ? _b : (_c = spec.validation) === null || _c === void 0 ? void 0 : _c.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"]}
;