cdk8s-cli
Version:
This is the command line tool for Cloud Development Kit (CDK) for Kubernetes (cdk8s).
301 lines • 36.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;
};
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}"]}
;