UNPKG

cdk8s-cli

Version:

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

173 lines • 21.1 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; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.ImportHelm = void 0; const child_process_1 = require("child_process"); const fs = __importStar(require("fs")); const os = __importStar(require("os")); const path = __importStar(require("path")); const cdk8s_1 = require("cdk8s"); const json2jsii_1 = require("json2jsii"); const semver = __importStar(require("semver")); const base_1 = require("./base"); const codegen_1 = require("./codegen"); const MAX_HELM_BUFFER = 10 * 1024 * 1024; const CHART_SCHEMA = 'values.schema.json'; const CHART_YAML = 'Chart.yaml'; class ImportHelm extends base_1.ImportBase { static async fromSpec(importSpec) { const { source } = importSpec; return new ImportHelm(source); } constructor(source) { super(); this.chartDependencies = []; const [chartUrl, chartName, chartVersion] = extractHelmChartDetails(source); this.chartName = chartName; this.chartUrl = chartUrl; this.chartVersion = chartVersion; const tmpDir = pullHelmRepo(chartUrl, chartName, chartVersion); const chartYamlFilePath = path.join(tmpDir, this.chartName, CHART_YAML); const contents = cdk8s_1.Yaml.load(chartYamlFilePath); if (contents.length === 1 && contents[0].dependencies) { for (const dependency of contents[0].dependencies) { this.chartDependencies.push(dependency.name); } } const potentialSchemaPath = path.join(tmpDir, this.chartName, CHART_SCHEMA); this.chartSchemaPath = fs.existsSync(potentialSchemaPath) ? potentialSchemaPath : undefined; this.schema = this.chartSchemaPath ? JSON.parse(fs.readFileSync(this.chartSchemaPath, 'utf-8')) : undefined; cleanup(tmpDir); } get moduleNames() { return [this.chartName]; } async generateTypeScript(code) { (0, codegen_1.emitHelmHeader)(code); const types = new json2jsii_1.TypeGenerator({ definitions: this.schema?.definitions ?? this.schema?.$defs, toJson: true, sanitizeEnums: true, }); (0, codegen_1.generateHelmConstruct)(types, { schema: this.schema, chartName: this.chartName, chartUrl: this.chartUrl, chartVersion: this.chartVersion, chartDependencies: this.chartDependencies, fqn: this.chartName, }); code.line(types.render()); } } exports.ImportHelm = ImportHelm; /** * Gets information about the helm chart from the helm url * @param url * @returns chartUrl, chartName and chartVersion */ function extractHelmChartDetails(url) { let chartUrl; let chartName; let chartVersion; if (url.startsWith('helm:oci://')) { // URL: helm:oci://registry-1.docker.io/bitnamicharts/wordpress@17.1.17 const helmRegex = /^helm:(oci:\/\/[A-Za-z0-9_.-:\-]+)\@(v?[0-9]+)\.([0-9]+)\.([A-Za-z0-9-+]+)$/; const helmDetails = helmRegex.exec(url); if (!helmDetails) { throw Error(`Invalid helm URL: ${url}. Must match the format: 'helm:<oci-registry-url>@<chart-version>'.`); } chartUrl = helmDetails[1]; const lastIndexOfSlash = chartUrl.lastIndexOf('/'); chartName = chartUrl.substring(lastIndexOfSlash + 1); const major = helmDetails[2]; const minor = helmDetails[3]; const patch = helmDetails[4]; chartVersion = `${major}.${minor}.${patch}`; } else { // URL: helm:https://lacework.github.io/helm-charts/lacework-agent@6.9.0 const helmRegex = /^helm:([A-Za-z0-9_.-:\-]+)\/([A-Za-z0-9_.-:\-]+)\@(v?[0-9]+)\.([0-9]+)\.([A-Za-z0-9-+]+)$/; const helmDetails = helmRegex.exec(url); if (!helmDetails) { throw Error(`Invalid helm URL: ${url}. Must match the format: 'helm:<repo-url>/<chart-name>@<chart-version>'.`); } chartUrl = helmDetails[1]; chartName = helmDetails[2]; const major = helmDetails[3]; const minor = helmDetails[4]; const patch = helmDetails[5]; chartVersion = `${major}.${minor}.${patch}`; } if (!semver.valid(chartVersion)) { throw new Error(`Invalid chart version (${chartVersion}) in URL: ${url}. Must follow SemVer-2 (see https://semver.org/).`); } return [chartUrl, chartName, chartVersion]; } /** * Pulls the helm chart in a temporary directory * @param chartUrl Chart url * @param chartName Chart name * @param chartVersion Chart version * @returns Temporary directory path */ function pullHelmRepo(chartUrl, chartName, chartVersion) { const workdir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk8s-helm-')); const args = new Array(); args.push('pull'); if (!chartUrl.startsWith('oci://')) { args.push(chartName); args.push('--repo', chartUrl); } else { args.push(chartUrl); } args.push('--version', chartVersion); args.push('--untar'); args.push('--untardir', workdir); const command = 'helm'; const helm = (0, child_process_1.spawnSync)(command, args, { maxBuffer: MAX_HELM_BUFFER, }); if (helm.error) { const err = helm.error.message; if (err.includes('ENOENT')) { throw new Error(`Unable to execute '${command}' to pull the Helm chart. Is helm installed on your system?`); } throw new Error(`Failed pulling helm chart from URL (${chartUrl}): ${err}`); } if (helm.status !== 0) { throw new Error(helm.stderr.toString()); } return workdir; } /** * Cleanup temp directory created * @param tmpDir Temporary directory path */ function cleanup(tmpDir) { fs.rmSync(tmpDir, { recursive: true }); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"helm.js","sourceRoot":"","sources":["../../src/import/helm.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawnSync } from 'child_process';\nimport * as fs from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { Yaml } from 'cdk8s';\nimport { CodeMaker } from 'codemaker';\nimport type { JSONSchema4 } from 'json-schema';\nimport { TypeGenerator } from 'json2jsii';\nimport * as semver from 'semver';\nimport { ImportBase } from './base';\nimport { emitHelmHeader, generateHelmConstruct } from './codegen';\nimport { ImportSpec } from '../config';\n\nconst MAX_HELM_BUFFER = 10 * 1024 * 1024;\nconst CHART_SCHEMA = 'values.schema.json';\nconst CHART_YAML = 'Chart.yaml';\n\nexport class ImportHelm extends ImportBase {\n  public static async fromSpec(importSpec: ImportSpec): Promise<ImportHelm> {\n    const { source } = importSpec;\n    return new ImportHelm(source);\n  }\n\n  private readonly chartName: string;\n  private readonly chartUrl: string;\n  private readonly chartVersion: string;\n  private readonly chartSchemaPath: string | undefined;\n  private readonly chartDependencies: string[] = [];\n  private readonly schema: JSONSchema4 | undefined;\n\n  private constructor(source: string) {\n    super();\n\n    const [chartUrl, chartName, chartVersion] = extractHelmChartDetails(source);\n\n    this.chartName = chartName;\n    this.chartUrl = chartUrl;\n    this.chartVersion = chartVersion;\n    const tmpDir = pullHelmRepo(chartUrl, chartName, chartVersion);\n\n    const chartYamlFilePath = path.join(tmpDir, this.chartName, CHART_YAML);\n    const contents = Yaml.load(chartYamlFilePath);\n\n    if (contents.length === 1 && contents[0].dependencies) {\n      for (const dependency of contents[0].dependencies) {\n        this.chartDependencies.push(dependency.name);\n      }\n    }\n\n    const potentialSchemaPath = path.join(tmpDir, this.chartName, CHART_SCHEMA);\n    this.chartSchemaPath = fs.existsSync(potentialSchemaPath) ? potentialSchemaPath : undefined;\n    this.schema = this.chartSchemaPath ? JSON.parse(fs.readFileSync(this.chartSchemaPath, 'utf-8')) : undefined;\n\n    cleanup(tmpDir);\n  }\n\n  public get moduleNames() {\n    return [this.chartName];\n  }\n\n  protected async generateTypeScript(code: CodeMaker) {\n    emitHelmHeader(code);\n\n    const types = new TypeGenerator({\n      definitions: this.schema?.definitions ?? this.schema?.$defs,\n      toJson: true,\n      sanitizeEnums: true,\n    });\n\n    generateHelmConstruct(types, {\n      schema: this.schema,\n      chartName: this.chartName,\n      chartUrl: this.chartUrl,\n      chartVersion: this.chartVersion,\n      chartDependencies: this.chartDependencies,\n      fqn: this.chartName,\n    });\n\n    code.line(types.render());\n  }\n}\n\n/**\n * Gets information about the helm chart from the helm url\n * @param url\n * @returns chartUrl, chartName and chartVersion\n */\nfunction extractHelmChartDetails(url: string) {\n\n  let chartUrl;\n  let chartName;\n  let chartVersion;\n\n  if (url.startsWith('helm:oci://')) {\n    // URL: helm:oci://registry-1.docker.io/bitnamicharts/wordpress@17.1.17\n    const helmRegex = /^helm:(oci:\\/\\/[A-Za-z0-9_.-:\\-]+)\\@(v?[0-9]+)\\.([0-9]+)\\.([A-Za-z0-9-+]+)$/;\n    const helmDetails = helmRegex.exec(url);\n\n    if (!helmDetails) {\n      throw Error(`Invalid helm URL: ${url}. Must match the format: 'helm:<oci-registry-url>@<chart-version>'.`);\n    }\n\n    chartUrl = helmDetails[1];\n    const lastIndexOfSlash = chartUrl.lastIndexOf('/');\n    chartName = chartUrl.substring(lastIndexOfSlash + 1);\n\n    const major = helmDetails[2];\n    const minor = helmDetails[3];\n    const patch = helmDetails[4];\n    chartVersion = `${major}.${minor}.${patch}`;\n\n  } else {\n    // URL: helm:https://lacework.github.io/helm-charts/lacework-agent@6.9.0\n    const helmRegex = /^helm:([A-Za-z0-9_.-:\\-]+)\\/([A-Za-z0-9_.-:\\-]+)\\@(v?[0-9]+)\\.([0-9]+)\\.([A-Za-z0-9-+]+)$/;\n    const helmDetails = helmRegex.exec(url);\n\n    if (!helmDetails) {\n      throw Error(`Invalid helm URL: ${url}. Must match the format: 'helm:<repo-url>/<chart-name>@<chart-version>'.`);\n    }\n\n    chartUrl = helmDetails[1];\n    chartName = helmDetails[2];\n\n    const major = helmDetails[3];\n    const minor = helmDetails[4];\n    const patch = helmDetails[5];\n    chartVersion = `${major}.${minor}.${patch}`;\n  }\n\n  if (!semver.valid(chartVersion)) {\n    throw new Error(`Invalid chart version (${chartVersion}) in URL: ${url}. Must follow SemVer-2 (see https://semver.org/).`);\n  }\n\n  return [chartUrl, chartName, chartVersion];\n}\n\n/**\n * Pulls the helm chart in a temporary directory\n * @param chartUrl Chart url\n * @param chartName Chart name\n * @param chartVersion Chart version\n * @returns Temporary directory path\n */\nfunction pullHelmRepo(chartUrl: string, chartName: string, chartVersion: string): string {\n  const workdir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk8s-helm-'));\n\n  const args = new Array<string>();\n  args.push('pull');\n\n  if (!chartUrl.startsWith('oci://')) {\n    args.push(chartName);\n    args.push('--repo', chartUrl);\n  } else {\n    args.push(chartUrl);\n  }\n\n  args.push('--version', chartVersion);\n  args.push('--untar');\n  args.push('--untardir', workdir);\n\n  const command = 'helm';\n\n  const helm = spawnSync(command, args, {\n    maxBuffer: MAX_HELM_BUFFER,\n  });\n\n  if (helm.error) {\n    const err = helm.error.message;\n    if (err.includes('ENOENT')) {\n      throw new Error(`Unable to execute '${command}' to pull the Helm chart. Is helm installed on your system?`);\n    }\n\n    throw new Error(`Failed pulling helm chart from URL (${chartUrl}): ${err}`);\n  }\n\n  if (helm.status !== 0) {\n    throw new Error(helm.stderr.toString());\n  }\n\n  return workdir;\n}\n\n/**\n * Cleanup temp directory created\n * @param tmpDir Temporary directory path\n */\nfunction cleanup(tmpDir: string) {\n  fs.rmSync(tmpDir, { recursive: true });\n}"]}