UNPKG

cdk8s-cli

Version:

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

301 lines • 36.4 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.crdsArePresent = exports.isHelmImport = exports.isK8sImport = exports.deriveFileName = exports.hashAndEncode = exports.parseImports = exports.findConstructMetadata = exports.findManifests = exports.download = exports.safeParseYaml = exports.safeParseJson = exports.validateApp = exports.synthApp = exports.mkdtemp = exports.shell = exports.PREFIX_DELIM = void 0; const child_process_1 = require("child_process"); const crypto_1 = require("crypto"); const fs_1 = require("fs"); const http = __importStar(require("http")); const https = __importStar(require("https")); const os = __importStar(require("os")); const path = __importStar(require("path")); const url_1 = require("url"); const fs = __importStar(require("fs-extra")); const yaml = __importStar(require("yaml")); const crds_dev_1 = require("./import/crds-dev"); const validation_1 = require("./plugins/validation"); exports.PREFIX_DELIM = ':='; async function shell(program, args = [], options = {}) { const command = `"${program} ${args.join(' ')}" at ${path.resolve(options.cwd?.toString() ?? '.')}`; return new Promise((ok, ko) => { const child = (0, child_process_1.spawn)(program, args, { stdio: ['inherit', 'pipe', 'inherit'], ...options }); const data = new Array(); child.stdout?.on('data', chunk => data.push(chunk)); child.once('error', err => ko(new Error(`command ${command} failed: ${err}`))); child.once('exit', code => { if (code === 0) { return ok(Buffer.concat(data).toString('utf-8')); } else { return ko(new Error(`command ${command} returned a non-zero exit code ${code}`)); } }); }); } exports.shell = shell; async function mkdtemp(closure) { const workdir = await fs.mkdtemp(path.join(os.tmpdir(), 'cdk8s-')); try { await closure(workdir); } finally { await fs.remove(workdir); } } exports.mkdtemp = mkdtemp; async function synthApp(command, outdir, stdout, metadata) { if (!stdout) { console.log('Synthesizing application'); } await shell(command, [], { shell: true, env: { ...process.env, CDK8S_OUTDIR: outdir, // record metadata so that the validation report // has contruct aware context. CDK8S_RECORD_CONSTRUCT_METADATA: process.env.CDK8S_RECORD_CONSTRUCT_METADATA ?? (metadata ? 'true' : 'false'), }, }); if (!await fs.pathExists(outdir)) { console.error(`ERROR: synthesis failed, app expected to create "${outdir}"`); process.exit(1); } let found = false; const yamlFiles = await findManifests(outdir); if (yamlFiles?.length) { if (!stdout) { for (const yamlFile of yamlFiles) { console.log(` - ${yamlFile}`); } } found = true; } if (!found) { console.error('No manifests synthesized'); } const constructMetadata = findConstructMetadata(outdir); return { manifests: yamlFiles, constructMetadata }; } exports.synthApp = synthApp; async function validateApp(app, stdout, validations, pluginManager, reportsFile) { const validators = []; for (const validation of validations) { const { plugin, context } = validation_1.ValidationPlugin.load(validation, app, stdout, pluginManager); validators.push({ plugin, context }); } const reports = []; let success = true; console.log('Performing validations'); for (const validator of validators) { await validator.plugin.validate(validator.context); const report = validator.context.report; success = success && report.success; reports.push(report); } console.log('Validations finished'); // now we can print them. we don't incrementally print // so to not clutter the terminal in case of errors. for (const report of reports) { console.log(''); console.log(report.toString()); console.log(''); } if (reportsFile) { if (fs.existsSync(reportsFile)) { throw new Error(`Unable to write validation reports file. Already exists: ${reportsFile}`); } // write the reports in JSON to a file fs.writeFileSync(reportsFile, JSON.stringify({ reports: reports.map(r => r.toJson()), }, null, 2)); } // exit with failure if any report resulted in a failure if (!success) { console.error('Validation failed. See above reports for details'); process.exit(2); } console.log('Validations ended succesfully'); } exports.validateApp = validateApp; function safeParseJson(text, reviver) { const json = JSON.parse(text); reviver.sanitize(json); return json; } exports.safeParseJson = safeParseJson; function safeParseYaml(text, reviver) { // parseAllDocuments doesnt accept a reviver // so we first parse normally and than transform // to JS using the reviver. const parsed = yaml.parseAllDocuments(text); const docs = []; for (const doc of parsed) { const json = doc.toJS(); reviver.sanitize(json); docs.push(json); } return docs; } exports.safeParseYaml = safeParseYaml; async function download(url) { let client; const proto = (0, url_1.parse)(url).protocol; if (!proto || proto === 'file:') { return fs.readFile(url, 'utf-8'); } switch (proto) { case 'https:': client = https; break; case 'http:': client = http; break; default: throw new Error(`unsupported protocol ${proto}`); } return new Promise((ok, ko) => { const req = client.get(url, res => { switch (res.statusCode) { case 200: { const data = new Array(); res.on('data', chunk => data.push(chunk)); res.once('end', () => ok(Buffer.concat(data).toString('utf-8'))); res.once('error', ko); break; } case 301: case 302: { if (res.headers.location) { ok(download(res.headers.location)); } break; } default: { ko(new Error(`${res.statusMessage}: ${url}`)); } } }); req.once('error', ko); req.end(); }); } exports.download = download; async function findManifests(directory) { // Ensure path is valid try { await fs_1.promises.access(directory); } catch { return []; } // Read Path contents const entries = await fs_1.promises.readdir(directory, { withFileTypes: true }); // Get files within the current directory const files = entries .filter(file => (!file.isDirectory() && file.name.endsWith('.yaml'))) .map(file => (directory + '/' + file.name)); // Get sub-folders within the current folder const folders = entries.filter(folder => folder.isDirectory()); for (const folder of folders) { files.push(...await findManifests(`${directory}/${folder.name}`)); } return files; } exports.findManifests = findManifests; function findConstructMetadata(directory) { // this file is optionally created during synthesis const p = path.join(directory, 'construct-metadata.json'); return fs.existsSync(p) ? p : undefined; } exports.findConstructMetadata = findConstructMetadata; function parseImports(spec) { const splitImport = spec.split(exports.PREFIX_DELIM); // k8s@x.y.z // crd.yaml // url.com/crd.yaml if (splitImport.length === 1) { return { source: spec, }; } // crd:=crd.yaml // crd:=url.com/crd.yaml if (splitImport.length === 2) { return { moduleNamePrefix: splitImport[0], source: splitImport[1], }; } throw new Error('Unable to parse import specification. Syntax is [NAME:=]SPEC'); } exports.parseImports = parseImports; function hashAndEncode(input, algorithm = 'sha256', encoding = 'hex') { const hash = (0, crypto_1.createHash)(algorithm); hash.update(input); return hash.digest(encoding); } exports.hashAndEncode = hashAndEncode; function deriveFileName(url) { const devUrl = (0, crds_dev_1.matchCrdsDevUrl)(url); let filename = undefined; if (devUrl) { const lastIndexOfSlash = devUrl.lastIndexOf('/'); const lastIndexOfAt = devUrl.lastIndexOf('@'); filename = (lastIndexOfSlash > 0 && lastIndexOfAt > 0) ? devUrl.slice(lastIndexOfSlash + 1, lastIndexOfAt) : undefined; } else { const lastIndexOfSlash = url.lastIndexOf('/'); const lastIndexOfYaml = url.lastIndexOf('.yaml') > 0 ? url.lastIndexOf('.yaml') : url.lastIndexOf('.yml'); filename = (lastIndexOfSlash > 0 && lastIndexOfYaml > 0) ? url.slice(lastIndexOfSlash + 1, lastIndexOfYaml) : undefined; } if (!filename) { // If the url if for a local file, then just encode the filename and not the entire path // Since path can depend on platform let file = (0, fs_1.existsSync)(url) ? path.basename(url) : url; filename = hashAndEncode(file); } return filename; } exports.deriveFileName = deriveFileName; function isK8sImport(value) { if (value !== 'k8s' && !value.startsWith('k8s@')) { return false; } return true; } exports.isK8sImport = isK8sImport; function isHelmImport(value) { if (!value.startsWith('helm:') && !value.includes(':=helm:')) { return false; } return true; } exports.isHelmImport = isHelmImport; function crdsArePresent(imprts) { return (imprts ?? []).some(imprt => (!isK8sImport(imprt) && !isHelmImport(imprt))); } exports.crdsArePresent = crdsArePresent; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"util.js","sourceRoot":"","sources":["../src/util.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn, SpawnOptions } from 'child_process';\nimport { BinaryToTextEncoding, createHash } from 'crypto';\nimport { existsSync, promises } from 'fs';\nimport * as http from 'http';\nimport * as https from 'https';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { parse } from 'url';\nimport * as fs from 'fs-extra';\nimport * as yaml from 'yaml';\nimport { ImportSpec, ValidationConfig } from './config';\nimport { matchCrdsDevUrl } from './import/crds-dev';\nimport { PluginManager } from './plugins/_manager';\nimport { ValidationPlugin, ValidationContext, ValidationReport, Validation } from './plugins/validation';\nimport { SafeReviver } from './reviver';\n\nexport const PREFIX_DELIM = ':=';\n\nexport async function shell(program: string, args: string[] = [], options: SpawnOptions = { }): Promise<string> {\n  const command = `\"${program} ${args.join(' ')}\" at ${path.resolve(options.cwd?.toString() ?? '.')}`;\n  return new Promise((ok, ko) => {\n    const child = spawn(program, args, { stdio: ['inherit', 'pipe', 'inherit'], ...options });\n    const data = new Array<Buffer>();\n    child.stdout?.on('data', chunk => data.push(chunk));\n\n    child.once('error', err => ko(new Error(`command ${command} failed: ${err}`)));\n    child.once('exit', code => {\n      if (code === 0) {\n        return ok(Buffer.concat(data).toString('utf-8'));\n      } else {\n        return ko(new Error(`command ${command} returned a non-zero exit code ${code}`));\n      }\n    });\n  });\n}\n\nexport async function mkdtemp(closure: (dir: string) => Promise<void>) {\n  const workdir = await fs.mkdtemp(path.join(os.tmpdir(), 'cdk8s-'));\n  try {\n    await closure(workdir);\n  } finally {\n    await fs.remove(workdir);\n  }\n}\n\nexport async function synthApp(command: string, outdir: string, stdout: boolean, metadata: boolean): Promise<SynthesizedApp> {\n  if (!stdout) {\n    console.log('Synthesizing application');\n  }\n  await shell(command, [], {\n    shell: true,\n    env: {\n      ...process.env,\n      CDK8S_OUTDIR: outdir,\n      // record metadata so that the validation report\n      // has contruct aware context.\n      CDK8S_RECORD_CONSTRUCT_METADATA: process.env.CDK8S_RECORD_CONSTRUCT_METADATA ?? (metadata ? 'true' : 'false'),\n    },\n  });\n\n  if (!await fs.pathExists(outdir)) {\n    console.error(`ERROR: synthesis failed, app expected to create \"${outdir}\"`);\n    process.exit(1);\n  }\n\n  let found = false;\n  const yamlFiles = await findManifests(outdir);\n  if (yamlFiles?.length) {\n    if (!stdout) {\n      for (const yamlFile of yamlFiles) {\n        console.log(`  - ${yamlFile}`);\n      }\n    }\n    found = true;\n  }\n\n  if (!found) {\n    console.error('No manifests synthesized');\n  }\n\n  const constructMetadata = findConstructMetadata(outdir);\n\n  return { manifests: yamlFiles, constructMetadata };\n}\n\nexport async function validateApp(\n  app: SynthesizedApp,\n  stdout: boolean,\n  validations: ValidationConfig[],\n  pluginManager: PluginManager,\n  reportsFile?: string) {\n\n  const validators: { plugin: Validation; context: ValidationContext}[] = [];\n\n  for (const validation of validations) {\n    const { plugin, context } = ValidationPlugin.load(validation, app, stdout, pluginManager);\n    validators.push({ plugin, context });\n  }\n\n  const reports: ValidationReport[] = [];\n  let success = true;\n\n  console.log('Performing validations');\n\n  for (const validator of validators) {\n    await validator.plugin.validate(validator.context);\n    const report = validator.context.report;\n    success = success && report.success;\n    reports.push(report);\n  }\n\n  console.log('Validations finished');\n\n  // now we can print them. we don't incrementally print\n  // so to not clutter the terminal in case of errors.\n  for (const report of reports) {\n    console.log('');\n    console.log(report.toString());\n    console.log('');\n  }\n\n  if (reportsFile) {\n\n    if (fs.existsSync(reportsFile)) {\n      throw new Error(`Unable to write validation reports file. Already exists: ${reportsFile}`);\n    }\n    // write the reports in JSON to a file\n    fs.writeFileSync(reportsFile, JSON.stringify({\n      reports: reports.map(r => r.toJson()),\n    }, null, 2));\n  }\n\n  // exit with failure if any report resulted in a failure\n  if (!success) {\n    console.error('Validation failed. See above reports for details');\n    process.exit(2);\n  }\n\n  console.log('Validations ended succesfully');\n\n}\n\nexport function safeParseJson(text: string, reviver: SafeReviver): any {\n  const json = JSON.parse(text);\n  reviver.sanitize(json);\n  return json;\n}\n\nexport function safeParseYaml(text: string, reviver: SafeReviver): any[] {\n\n  // parseAllDocuments doesnt accept a reviver\n  // so we first parse normally and than transform\n  // to JS using the reviver.\n  const parsed = yaml.parseAllDocuments(text);\n  const docs = [];\n  for (const doc of parsed) {\n    const json = doc.toJS();\n    reviver.sanitize(json);\n    docs.push(json);\n  }\n  return docs;\n}\n\nexport async function download(url: string): Promise<string> {\n\n  let client: typeof http | typeof https;\n  const proto = parse(url).protocol;\n\n  if (!proto || proto === 'file:') {\n    return fs.readFile(url, 'utf-8');\n  }\n\n  switch (proto) {\n    case 'https:':\n      client = https;\n      break;\n\n    case 'http:':\n      client = http;\n      break;\n\n    default:\n      throw new Error(`unsupported protocol ${proto}`);\n  }\n\n  return new Promise((ok, ko) => {\n    const req = client.get(url, res => {\n      switch (res.statusCode) {\n        case 200: {\n          const data = new Array<Buffer>();\n          res.on('data', chunk => data.push(chunk));\n          res.once('end', () => ok(Buffer.concat(data).toString('utf-8')));\n          res.once('error', ko);\n          break;\n        }\n\n        case 301:\n        case 302: {\n          if (res.headers.location) {\n            ok(download(res.headers.location));\n          }\n          break;\n        }\n\n        default: {\n          ko(new Error(`${res.statusMessage}: ${url}`));\n        }\n      }\n    });\n\n    req.once('error', ko);\n    req.end();\n  });\n}\n\nexport async function findManifests(directory: string): Promise<string[]> {\n  // Ensure path is valid\n  try {\n    await promises.access(directory);\n  } catch {\n    return [];\n  }\n\n  // Read Path contents\n  const entries = await promises.readdir(directory, { withFileTypes: true });\n\n  // Get files within the current directory\n  const files = entries\n    .filter(file => (!file.isDirectory() && file.name.endsWith('.yaml')))\n    .map(file => (directory + '/' + file.name));\n\n  // Get sub-folders within the current folder\n  const folders = entries.filter(folder => folder.isDirectory());\n\n  for (const folder of folders) {\n    files.push(...await findManifests(`${directory}/${folder.name}`));\n  }\n\n  return files;\n}\n\nexport function findConstructMetadata(directory: string): string | undefined {\n  // this file is optionally created during synthesis\n  const p = path.join(directory, 'construct-metadata.json');\n  return fs.existsSync(p) ? p : undefined;\n}\n\n/**\n * Result of synthesizing an application.\n */\nexport interface SynthesizedApp {\n\n  /**\n   * The list of manifests produced by the app.\n   */\n  readonly manifests: readonly string[];\n\n  /**\n   * The construct metadata file (if exists).\n   */\n  readonly constructMetadata?: string;\n}\n\nexport function parseImports(spec: string): ImportSpec {\n  const splitImport = spec.split(PREFIX_DELIM);\n\n  // k8s@x.y.z\n  // crd.yaml\n  // url.com/crd.yaml\n  if (splitImport.length === 1) {\n    return {\n      source: spec,\n    };\n  }\n\n  // crd:=crd.yaml\n  // crd:=url.com/crd.yaml\n  if (splitImport.length === 2) {\n    return {\n      moduleNamePrefix: splitImport[0],\n      source: splitImport[1],\n    };\n  }\n\n  throw new Error('Unable to parse import specification. Syntax is [NAME:=]SPEC');\n}\n\nexport function hashAndEncode(input: string, algorithm: string = 'sha256', encoding: BinaryToTextEncoding = 'hex'): string {\n  const hash = createHash(algorithm);\n  hash.update(input);\n  return hash.digest(encoding);\n}\n\nexport function deriveFileName(url: string): string {\n  const devUrl = matchCrdsDevUrl(url);\n  let filename = undefined;\n\n  if (devUrl) {\n    const lastIndexOfSlash = devUrl.lastIndexOf('/');\n    const lastIndexOfAt = devUrl.lastIndexOf('@');\n    filename = (lastIndexOfSlash > 0 && lastIndexOfAt > 0) ? devUrl.slice(lastIndexOfSlash+1, lastIndexOfAt): undefined;\n  } else {\n    const lastIndexOfSlash = url.lastIndexOf('/');\n    const lastIndexOfYaml = url.lastIndexOf('.yaml') > 0 ? url.lastIndexOf('.yaml') : url.lastIndexOf('.yml');\n\n    filename = (lastIndexOfSlash > 0 && lastIndexOfYaml > 0) ? url.slice(lastIndexOfSlash+1, lastIndexOfYaml): undefined;\n  }\n\n  if (!filename) {\n    // If the url if for a local file, then just encode the filename and not the entire path\n    // Since path can depend on platform\n    let file = existsSync(url) ? path.basename(url) : url;\n\n    filename = hashAndEncode(file);\n  }\n\n  return filename;\n}\n\nexport function isK8sImport(value: string): boolean {\n  if (value !== 'k8s' && !value.startsWith('k8s@')) {\n    return false;\n  }\n\n  return true;\n}\n\nexport function isHelmImport(value: string): boolean {\n  if (!value.startsWith('helm:') && !value.includes(':=helm:')) {\n    return false;\n  }\n\n  return true;\n}\n\nexport function crdsArePresent(imprts: string[] | undefined): boolean {\n  return (imprts ?? []).some(imprt => (!isK8sImport(imprt) && !isHelmImport(imprt)));\n}"]}