cdk8s-cli
Version:
This is the command line tool for Cloud Development Kit (CDK) for Kubernetes (cdk8s).
143 lines • 19 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.findApiObjectDefinitions = exports.ImportKubernetesApi = exports.DEFAULT_API_VERSION = void 0;
const json2jsii_1 = require("json2jsii");
const base_1 = require("./base");
const codegen_1 = require("./codegen");
const k8s_util_1 = require("./k8s-util");
const util_1 = require("../util");
exports.DEFAULT_API_VERSION = '1.25.0';
const DEFAULT_CLASS_NAME_PREFIX = 'Kube';
class ImportKubernetesApi extends base_1.ImportBase {
static async match(importSpec, argv) {
var _a;
const { source } = importSpec;
if (source !== 'k8s' && !source.startsWith('k8s@')) {
return undefined;
}
let k8sVersion = (_a = source.split('@')[1]) !== null && _a !== void 0 ? _a : exports.DEFAULT_API_VERSION;
const k8sVersionRegex = /^\d+\.\d+\.\d+$/;
if (!k8sVersionRegex.test(k8sVersion)) {
throw new Error(`Expected k8s version "${k8sVersion}" to match format "<major>.<minor>.<patch>".`);
}
console.error(`Importing k8s v${k8sVersion}...`);
return {
apiVersion: k8sVersion,
exclude: argv.exclude,
};
}
constructor(options) {
super();
this.options = options;
}
get moduleNames() {
return ['k8s'];
}
async generateTypeScript(code, moduleName, options) {
var _a;
const schema = await downloadSchema(this.options.apiVersion);
if (moduleName !== 'k8s') {
throw new Error(`unexpected module name "${moduleName}" when importing k8s types (expected "k8s")`);
}
const prefix = (_a = options.classNamePrefix) !== null && _a !== void 0 ? _a : DEFAULT_CLASS_NAME_PREFIX;
const topLevelObjects = findApiObjectDefinitions(schema, prefix);
const typeGenerator = new json2jsii_1.TypeGenerator({
definitions: schema.definitions,
exclude: this.options.exclude,
renderTypeName: (def) => {
const parsed = (0, k8s_util_1.parseApiTypeName)(def);
if (!parsed.version) {
// not a versioned api type. return basename
return parsed.basename;
}
return (0, codegen_1.getTypeName)(false, parsed.basename, parsed.version.raw);
},
});
// rename "Props" type from their original name based on the API object kind
// (e.g. `Deployment`) to their actual props type (`KubeDeploymentProps`) in
// order to avoid confusion between constructs (`KubeDeployment`) and those
// types. This is done by simply replacing their definition in the schema
// with a $ref to the definition of the props type.
for (const o of topLevelObjects) {
typeGenerator.addDefinition(o.fqn, { $ref: `#/definitions/${(0, codegen_1.getPropsTypeName)(o)}` });
}
// emit construct types (recursive)
for (const o of topLevelObjects) {
(0, codegen_1.generateConstruct)(typeGenerator, o);
}
(0, codegen_1.emitHeader)(code, false);
code.line(typeGenerator.render());
}
}
exports.ImportKubernetesApi = ImportKubernetesApi;
/**
* Returns a map of all API objects in the spec (objects that have the
* 'x-kubernetes-group-version-kind' annotation).
*
* The key is the base name of the type (i.e. `Deployment`). Since API objects
* may have multiple versions, each value in the map is an array of type definitions
* along with version information.
*
* @see https://kubernetes.io/docs/concepts/overview/kubernetes-api/#api-versioning
*/
function findApiObjectDefinitions(schema, prefix) {
const result = new Array();
for (const [typename, apischema] of Object.entries(schema.definitions || {})) {
const objectName = tryGetObjectName(apischema);
if (!objectName) {
continue;
}
const type = (0, k8s_util_1.parseApiTypeName)(typename);
if (!type.version) {
throw new Error(`Unable to parse version for type: ${typename}`);
}
result.push({
custom: false,
fqn: type.fullname,
group: objectName.group,
kind: objectName.kind,
version: objectName.version,
schema: apischema,
prefix,
});
}
return result;
}
exports.findApiObjectDefinitions = findApiObjectDefinitions;
function tryGetObjectName(def) {
var _a;
const objectNames = def[X_GROUP_VERSION_KIND];
if (!objectNames) {
return undefined;
}
const objectName = objectNames[0];
if (!objectName) {
return undefined;
}
// skip definitions without "metadata". they are not API objects that can be defined
// in manifests (example: io.k8s.apimachinery.pkg.apis.meta.v1.DeleteOptions)
// they will be treated as data types
if (!((_a = def.properties) === null || _a === void 0 ? void 0 : _a.metadata)) {
return undefined;
}
return objectName;
}
const X_GROUP_VERSION_KIND = 'x-kubernetes-group-version-kind';
async function downloadSchema(apiVersion) {
const url = `https://raw.githubusercontent.com/cdk8s-team/cdk8s/master/kubernetes-schemas/v${apiVersion}/_definitions.json`;
let output;
try {
output = await (0, util_1.download)(url);
}
catch (e) {
console.error(`Could not find a schema for k8s version ${apiVersion}. The current list of available schemas is at https://github.com/cdk8s-team/cdk8s/tree/master/kubernetes-schemas.`);
throw e;
}
try {
return (0, k8s_util_1.safeParseJsonSchema)(output);
}
catch (e) {
throw new Error(`Unable to parse schema at ${url}: ${e}`);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"k8s.js","sourceRoot":"","sources":["../../src/import/k8s.ts"],"names":[],"mappings":"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","sourcesContent":["import { CodeMaker } from 'codemaker';\n\n// we just need the types from json-schema\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { JSONSchema4 } from 'json-schema';\n\nimport { TypeGenerator } from 'json2jsii';\nimport { GenerateOptions, ImportBase } from './base';\nimport { ApiObjectDefinition, emitHeader, generateConstruct, getPropsTypeName, getTypeName } from './codegen';\nimport { parseApiTypeName, safeParseJsonSchema } from './k8s-util';\nimport { ImportSpec } from '../config';\nimport { download } from '../util';\n\n\nexport const DEFAULT_API_VERSION = '1.25.0';\n\nconst DEFAULT_CLASS_NAME_PREFIX = 'Kube';\n\nexport interface ImportKubernetesApiOptions {\n  /**\n   * The API version to generate.\n   */\n  readonly apiVersion: string;\n\n  /**\n   * Do not import these types. Instead, represent them as \"any\".\n   *\n   * @default - include all types that derive from the root types.\n   */\n  readonly exclude?: string[];\n}\n\nexport class ImportKubernetesApi extends ImportBase {\n\n  public static async match(importSpec: ImportSpec, argv: any): Promise<ImportKubernetesApiOptions | undefined> {\n    const { source } = importSpec;\n    if (source !== 'k8s' && !source.startsWith('k8s@')) {\n      return undefined;\n    }\n\n    let k8sVersion = source.split('@')[1] ?? DEFAULT_API_VERSION;\n\n    const k8sVersionRegex = /^\\d+\\.\\d+\\.\\d+$/;\n    if (!k8sVersionRegex.test(k8sVersion)) {\n      throw new Error(`Expected k8s version \"${k8sVersion}\" to match format \"<major>.<minor>.<patch>\".`);\n    }\n\n    console.error(`Importing k8s v${k8sVersion}...`);\n\n    return {\n      apiVersion: k8sVersion,\n      exclude: argv.exclude,\n    };\n  }\n\n  constructor(private readonly options: ImportKubernetesApiOptions) {\n    super();\n  }\n\n  public get moduleNames() {\n    return ['k8s'];\n  }\n\n  protected async generateTypeScript(code: CodeMaker, moduleName: string, options: GenerateOptions) {\n    const schema = await downloadSchema(this.options.apiVersion);\n\n    if (moduleName !== 'k8s') {\n      throw new Error(`unexpected module name \"${moduleName}\" when importing k8s types (expected \"k8s\")`);\n    }\n\n    const prefix = options.classNamePrefix ?? DEFAULT_CLASS_NAME_PREFIX;\n    const topLevelObjects = findApiObjectDefinitions(schema, prefix);\n\n    const typeGenerator = new TypeGenerator({\n      definitions: schema.definitions,\n      exclude: this.options.exclude,\n      renderTypeName: (def: string) => {\n        const parsed = parseApiTypeName(def);\n        if (!parsed.version) {\n          // not a versioned api type. return basename\n          return parsed.basename;\n        }\n        return getTypeName(false, parsed.basename, parsed.version.raw);\n      },\n    });\n\n    // rename \"Props\" type from their original name based on the API object kind\n    // (e.g. `Deployment`) to their actual props type (`KubeDeploymentProps`) in\n    // order to avoid confusion between constructs (`KubeDeployment`) and those\n    // types. This is done by simply replacing their definition in the schema\n    // with a $ref to the definition of the props type.\n    for (const o of topLevelObjects) {\n      typeGenerator.addDefinition(o.fqn, { $ref: `#/definitions/${getPropsTypeName(o)}` });\n    }\n\n    // emit construct types (recursive)\n    for (const o of topLevelObjects) {\n      generateConstruct(typeGenerator, o);\n    }\n\n    emitHeader(code, false);\n\n    code.line(typeGenerator.render());\n  }\n}\n\n/**\n * Returns a map of all API objects in the spec (objects that have the\n * 'x-kubernetes-group-version-kind' annotation).\n *\n * The key is the base name of the type (i.e. `Deployment`). Since API objects\n * may have multiple versions, each value in the map is an array of type definitions\n * along with version information.\n *\n * @see https://kubernetes.io/docs/concepts/overview/kubernetes-api/#api-versioning\n */\nexport function findApiObjectDefinitions(schema: JSONSchema4, prefix: string): ApiObjectDefinition[] {\n  const result = new Array<ApiObjectDefinition>();\n\n  for (const [typename, apischema] of Object.entries(schema.definitions || { })) {\n    const objectName = tryGetObjectName(apischema);\n    if (!objectName) {\n      continue;\n    }\n\n    const type = parseApiTypeName(typename);\n    if (!type.version) {\n      throw new Error(`Unable to parse version for type: ${typename}`);\n    }\n    result.push({\n      custom: false, // not a CRD\n      fqn: type.fullname,\n      group: objectName.group,\n      kind: objectName.kind,\n      version: objectName.version,\n      schema: apischema,\n      prefix,\n    });\n  }\n\n  return result;\n}\n\nfunction tryGetObjectName(def: JSONSchema4): GroupVersionKind | undefined {\n  const objectNames = def[X_GROUP_VERSION_KIND] as GroupVersionKind[];\n  if (!objectNames) {\n    return undefined;\n  }\n\n  const objectName = objectNames[0];\n  if (!objectName) {\n    return undefined;\n  }\n\n  // skip definitions without \"metadata\". they are not API objects that can be defined\n  // in manifests (example: io.k8s.apimachinery.pkg.apis.meta.v1.DeleteOptions)\n  // they will be treated as data types\n  if (!def.properties?.metadata) {\n    return undefined;\n  }\n\n  return objectName;\n}\n\nexport interface GroupVersionKind {\n  readonly group: string;\n  readonly kind: string;\n  readonly version: string;\n}\n\nconst X_GROUP_VERSION_KIND = 'x-kubernetes-group-version-kind';\n\nasync function downloadSchema(apiVersion: string) {\n  const url = `https://raw.githubusercontent.com/cdk8s-team/cdk8s/master/kubernetes-schemas/v${apiVersion}/_definitions.json`;\n  let output;\n  try {\n    output = await download(url);\n  } catch (e) {\n    console.error(`Could not find a schema for k8s version ${apiVersion}. The current list of available schemas is at https://github.com/cdk8s-team/cdk8s/tree/master/kubernetes-schemas.`);\n    throw e;\n  }\n  try {\n    return safeParseJsonSchema(output) as JSONSchema4;\n  } catch (e) {\n    throw new Error(`Unable to parse schema at ${url}: ${e}`);\n  }\n}\n"]}
;