cdk8s-cli
Version:
This is the command line tool for Cloud Development Kit (CDK) for Kubernetes (cdk8s).
242 lines • 32.3 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.ValidationPlugin = exports.ValidationReport = exports.ValidationLogger = exports.ValidationContext = void 0;
const fs = __importStar(require("fs"));
const os = __importStar(require("os"));
const table_1 = require("table");
const yaml = __importStar(require("yaml"));
/**
* Context available to plugins during validation.
*/
class ValidationContext {
constructor(
/**
* The list of manifests to validate.
*/
manifests,
/**
* The NodeJS package name of the plugin running the validation.
*/
pkg,
/**
* The version of the NodeJS package that runs the validation.
*/
version,
/**
* Construct metadata of resources in the application.
*/
metadata = {},
/**
* Whether or not the synth command was executed with --stdout.
*/
stdout) {
var _a;
this.manifests = manifests;
this.pkg = pkg;
this.version = version;
this.metadata = metadata;
this.stdout = stdout;
this.report = new ValidationReport(this.pkg, this.version, (_a = this.metadata) !== null && _a !== void 0 ? _a : {}, stdout !== null && stdout !== void 0 ? stdout : false);
this.logger = new ValidationLogger();
}
parseManifest(manifestPath) {
const parsed = yaml.parseAllDocuments(fs.readFileSync(manifestPath, { encoding: 'utf-8' }));
const resources = Array.isArray(parsed) ? parsed : [parsed];
return resources.map(r => r.toJS());
}
}
exports.ValidationContext = ValidationContext;
/**
* Logger available to plugins during validation. Use this instead of `console.log`.
*/
class ValidationLogger {
/**
* Log a message.
*/
log(message) {
console.log(message);
}
}
exports.ValidationLogger = ValidationLogger;
/**
* The report emitted by the plugin after evaluation.
*/
class ValidationReport {
constructor(pkg, version, metadata, stdout) {
this.pkg = pkg;
this.version = version;
this.metadata = metadata;
this.stdout = stdout;
this.violations = [];
}
/**
* Add a violation to the report.
*/
addViolation(violation) {
var _a;
if (this._summary) {
throw new Error('Violations cannot be added to report after its submitted');
}
const violatingConstructs = [];
for (const resource of violation.violatingResources) {
const constructPath = (_a = this.metadata[resource.resourceName]) === null || _a === void 0 ? void 0 : _a.path;
violatingConstructs.push({
...resource,
// augment with construct metadata
constructPath: constructPath,
// if synth is executed with --stdout, the manifest path
// here is temporary and will be deleted once the command finishes.
manifestPath: this.stdout ? 'STDOUT' : resource.manifestPath,
});
}
this.violations.push({
ruleName: violation.ruleName,
recommendation: violation.recommendation,
violatingConstructs: violatingConstructs,
fix: violation.fix,
});
}
/**
* Submit the report with a status and additional metadata.
*/
submit(status, metadata) {
this._summary = { status, plugin: this.pkg, version: this.version, metadata };
}
/**
* Whether or not the report was successfull.
*/
get success() {
if (!this._summary) {
throw new Error('Unable to determine report status: Report is incomplete. Call \'report.submit\'');
}
return this._summary.status === 'success';
}
/**
* Transform the report to a well formatted table string.
*/
toString() {
var _a, _b;
const json = this.toJson();
const output = [json.title];
output.push('-'.repeat(json.title.length));
output.push('');
output.push('(Summary)');
output.push('');
output.push((0, table_1.table)([
['Status', json.summary.status],
['Plugin', json.summary.plugin],
['Version', json.summary.version],
...Object.entries((_a = json.summary.metadata) !== null && _a !== void 0 ? _a : {}),
]));
if (json.violations) {
output.push('');
output.push('(Violations)');
}
for (const violation of json.violations) {
const occurrences = violation.violatingConstructs.flatMap(c => c.locations).length;
const title = reset(red(bright(`${violation.ruleName} (${occurrences} occurrences)`)));
output.push('');
output.push(title);
output.push('');
output.push(' Occurrences:');
for (const construct of violation.violatingConstructs) {
output.push('');
output.push(` - Construct Path: ${(_b = construct.constructPath) !== null && _b !== void 0 ? _b : 'N/A'}`);
output.push(` - Manifest Path: ${construct.manifestPath}`);
output.push(` - Resource Name: ${construct.resourceName}`);
if (construct.locations) {
output.push(' - Locations:');
for (const location of construct.locations) {
output.push(` > ${location}`);
}
}
}
output.push('');
output.push(` Recommendation: ${violation.recommendation}`);
output.push(` How to fix: ${violation.fix}`);
}
return output.join(os.EOL);
}
/**
* Transform the report into a JSON object.
*/
toJson() {
if (!this._summary) {
throw new Error('Unable to determine report result: Report is incomplete. Call \'report.submit\'');
}
return {
title: `Validation Report (${this.pkg}@${this.version})`,
violations: this.violations,
summary: this._summary,
};
}
}
exports.ValidationReport = ValidationReport;
/**
* Utiliy class for loading validation plugins.
*/
class ValidationPlugin {
/**
* Load the validation plugin and create the necessary context for its execution.
*/
static load(validation, app, stdout, pluginManager) {
const plugin = pluginManager.load({
pkg: validation.package,
version: validation.version,
class: validation.class,
properties: validation.properties,
installEnv: validation.installEnv,
});
if (typeof (plugin.instance.validate) !== 'function') {
throw new Error(`Instance of class '${validation.class}' from package '${validation.package}@${validation.version}' is not a validation plugin. Are you sure you specified the correct class?`);
}
const metadata = app.constructMetadata ? this.loadConstructMetadata(app.constructMetadata) : {};
const context = new ValidationContext(app.manifests, plugin.package.pkg, plugin.package.version, metadata, stdout);
return { plugin: plugin.instance, context };
}
static loadConstructMetadata(constructMetadataPath) {
const contents = JSON.parse(fs.readFileSync(constructMetadataPath, { encoding: 'utf-8' }));
const resources = {};
if (contents.version !== '1.0.0') {
throw new Error(`Unexpected version of construct metadata at ${constructMetadataPath}: ${contents.version}. Supported versions are: [1.0.0]`);
}
for (const [name, metadata] of Object.entries(contents.resources)) {
resources[name] = { path: metadata.path };
}
return resources;
}
}
exports.ValidationPlugin = ValidationPlugin;
function reset(s) {
return `${s}\x1b[0m`;
}
function red(s) {
return `\x1b[31m${s}`;
}
function bright(s) {
return `\x1b[1m${s}`;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validation.js","sourceRoot":"","sources":["../../src/plugins/validation.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as os from 'os';\nimport { table } from 'table';\nimport * as yaml from 'yaml';\nimport { PluginManager } from './_manager';\nimport { ValidationConfig } from '../config';\nimport { SynthesizedApp } from '../util';\n\n/**\n * Context available to plugins during validation.\n */\nexport class ValidationContext {\n\n  /**\n   * Report emitted by the validation.\n   *\n   * Plugins should interact with this object to generate the report.\n   */\n  public readonly report: ValidationReport;\n\n  /**\n   * Logger for the validation.\n   *\n   * Plugins should interact with this object to log messages during validation.\n   */\n  public readonly logger: ValidationLogger;\n\n  constructor(\n\n    /**\n     * The list of manifests to validate.\n     */\n    public readonly manifests: readonly string[],\n\n    /**\n     * The NodeJS package name of the plugin running the validation.\n     */\n    public readonly pkg: string,\n\n    /**\n     * The version of the NodeJS package that runs the validation.\n     */\n    public readonly version: string,\n\n    /**\n     * Construct metadata of resources in the application.\n     */\n    public readonly metadata: {readonly [key: string]: ResourceConstructMetadata} = {},\n\n    /**\n     * Whether or not the synth command was executed with --stdout.\n     */\n    public readonly stdout?: boolean) {\n\n    this.report = new ValidationReport(this.pkg, this.version, this.metadata ?? {}, stdout ?? false);\n    this.logger = new ValidationLogger();\n  }\n\n  public parseManifest(manifestPath: string): any[] {\n    const parsed = yaml.parseAllDocuments(fs.readFileSync(manifestPath, { encoding: 'utf-8' }));\n    const resources = Array.isArray(parsed) ? parsed : [parsed];\n    return resources.map(r => r.toJS());\n  }\n}\n\n/**\n * Logger available to plugins during validation. Use this instead of `console.log`.\n */\nexport class ValidationLogger {\n\n  /**\n   * Log a message.\n   */\n  public log(message: string) {\n    console.log(message);\n  }\n}\n\n/**\n * Contract between cdk8s and third-parties looking to implement validation plugins.\n */\nexport interface Validation {\n\n  /**\n   * Run the validation logic.\n   *\n   * - Use `context.manifests` to retrieve the list of manifests to validate.\n   * - Use `context.report` to access and build the resulting report.\n   *\n   * Make sure to call `context.report.pass()` or `context.report.fail()` before returning, otherwise the validation is considered incomplete.\n   */\n  validate(context: ValidationContext): Promise<void>;\n\n}\n\n/**\n * Resource violating a specific rule.\n */\nexport interface ValidationViolatingResource {\n\n  /**\n   * The resource name.\n   */\n  readonly resourceName: string;\n\n  /**\n   * The locations in its config that pose the violations.\n   */\n  readonly locations: readonly string[];\n\n  /**\n   * The manifest this resource is defined in.\n   */\n  readonly manifestPath: string;\n\n}\n\n/**\n * Construct violating a specific rule.\n */\nexport interface ValidationViolatingConstruct extends ValidationViolatingResource {\n\n  /**\n   * The construct path as defined in the application.\n   */\n  readonly constructPath?: string;\n\n}\n\n/**\n * Violation produced by the validation plugin.\n */\nexport interface ValidationViolation {\n\n  /**\n   * The name of the rule.\n   */\n  readonly ruleName: string;\n\n  /**\n   * The recommendation to resolve the violation.\n   */\n  readonly recommendation: string;\n\n  /**\n   * How to fix the recommendation.\n   */\n  readonly fix: string;\n\n  /**\n   * The resources violating this rule.\n   */\n  readonly violatingResources: readonly ValidationViolatingResource[];\n\n}\n\n/**\n * Validation produced by the validation plugin, in construct terms.\n */\nexport interface ValidationViolationConstructAware extends Omit<ValidationViolation, 'violatingResources'> {\n\n  /**\n   * The constructs violating this rule.\n   */\n  readonly violatingConstructs: readonly ValidationViolatingConstruct[];\n}\n\n// we intentionally don't use an enum so that\n// plugins don't have to import the cli at runtime.\nexport type ValidationReportStatus = 'success' | 'failure';\n\n/**\n * Summary of the report.\n */\nexport interface ValidationReportSummary {\n\n  readonly status: ValidationReportStatus;\n\n  readonly plugin: string;\n\n  readonly version: string;\n\n  readonly metadata?: { readonly [key: string]: string };\n\n}\n\n/**\n * JSON representation of the report.\n */\nexport interface ValidationReportJson {\n\n  /**\n   * Report title.\n   */\n  readonly title: string;\n\n  /**\n   * List of violations in the rerpot.\n   */\n  readonly violations: readonly ValidationViolationConstructAware[];\n\n  /**\n   * Report summary.\n   */\n  readonly summary: ValidationReportSummary;\n\n}\n\n/**\n * The report emitted by the plugin after evaluation.\n */\nexport class ValidationReport {\n\n  private readonly violations: ValidationViolationConstructAware[] = [];\n\n  private _summary?: ValidationReportSummary;\n\n  constructor(\n    private readonly pkg: string,\n    private readonly version: string,\n    private readonly metadata: {readonly [key: string]: ResourceConstructMetadata},\n    private readonly stdout: boolean) {\n  }\n\n  /**\n   * Add a violation to the report.\n   */\n  public addViolation(violation: ValidationViolation) {\n    if (this._summary) {\n      throw new Error('Violations cannot be added to report after its submitted');\n    }\n\n    const violatingConstructs: ValidationViolatingConstruct[] = [];\n\n    for (const resource of violation.violatingResources) {\n      const constructPath = this.metadata[resource.resourceName]?.path;\n      violatingConstructs.push({\n        ...resource,\n\n        // augment with construct metadata\n        constructPath: constructPath,\n\n        // if synth is executed with --stdout, the manifest path\n        // here is temporary and will be deleted once the command finishes.\n        manifestPath: this.stdout ? 'STDOUT' : resource.manifestPath,\n      });\n    }\n\n    this.violations.push({\n      ruleName: violation.ruleName,\n      recommendation: violation.recommendation,\n      violatingConstructs: violatingConstructs,\n      fix: violation.fix,\n    });\n  }\n\n  /**\n   * Submit the report with a status and additional metadata.\n   */\n  public submit(status: ValidationReportStatus, metadata?: { readonly [key: string]: string }) {\n    this._summary = { status, plugin: this.pkg, version: this.version, metadata };\n  }\n\n  /**\n   * Whether or not the report was successfull.\n   */\n  public get success(): boolean {\n    if (!this._summary) {\n      throw new Error('Unable to determine report status: Report is incomplete. Call \\'report.submit\\'');\n    }\n    return this._summary.status === 'success';\n  }\n\n  /**\n   * Transform the report to a well formatted table string.\n   */\n  public toString(): string {\n\n    const json = this.toJson();\n    const output = [json.title];\n\n    output.push('-'.repeat(json.title.length));\n    output.push('');\n    output.push('(Summary)');\n    output.push('');\n    output.push(table([\n      ['Status', json.summary.status],\n      ['Plugin', json.summary.plugin],\n      ['Version', json.summary.version],\n      ...Object.entries(json.summary.metadata ?? {}),\n    ]));\n\n    if (json.violations) {\n      output.push('');\n      output.push('(Violations)');\n    }\n    for (const violation of json.violations) {\n      const occurrences = violation.violatingConstructs.flatMap(c => c.locations).length;\n      const title = reset(red(bright(`${violation.ruleName} (${occurrences} occurrences)`)));\n      output.push('');\n      output.push(title);\n      output.push('');\n      output.push('  Occurrences:');\n      for (const construct of violation.violatingConstructs) {\n        output.push('');\n        output.push(`    - Construct Path: ${construct.constructPath ?? 'N/A'}`);\n        output.push(`    - Manifest Path: ${construct.manifestPath}`);\n        output.push(`    - Resource Name: ${construct.resourceName}`);\n        if (construct.locations) {\n          output.push('    - Locations:');\n          for (const location of construct.locations) {\n            output.push(`      > ${location}`);\n          }\n        }\n      }\n      output.push('');\n      output.push(`  Recommendation: ${violation.recommendation}`);\n      output.push(`  How to fix: ${violation.fix}`);\n    }\n\n    return output.join(os.EOL);\n\n  }\n\n  /**\n   * Transform the report into a JSON object.\n   */\n  public toJson(): ValidationReportJson {\n    if (!this._summary) {\n      throw new Error('Unable to determine report result: Report is incomplete. Call \\'report.submit\\'');\n    }\n    return {\n      title: `Validation Report (${this.pkg}@${this.version})`,\n      violations: this.violations,\n      summary: this._summary,\n    };\n  }\n\n}\n\n/**\n * Utiliy class for loading validation plugins.\n */\nexport class ValidationPlugin {\n\n  /**\n   * Load the validation plugin and create the necessary context for its execution.\n   */\n  public static load(\n    validation: ValidationConfig,\n    app: SynthesizedApp,\n    stdout: boolean,\n    pluginManager: PluginManager): { plugin: Validation; context: ValidationContext } {\n\n    const plugin = pluginManager.load({\n      pkg: validation.package,\n      version: validation.version,\n      class: validation.class,\n      properties: validation.properties,\n      installEnv: validation.installEnv,\n    });\n\n    if (typeof((plugin.instance as any).validate) !== 'function') {\n      throw new Error(`Instance of class '${validation.class}' from package '${validation.package}@${validation.version}' is not a validation plugin. Are you sure you specified the correct class?`);\n    }\n\n    const metadata = app.constructMetadata ? this.loadConstructMetadata(app.constructMetadata) : {};\n    const context = new ValidationContext(app.manifests, plugin.package.pkg, plugin.package.version, metadata, stdout);\n    return { plugin: plugin.instance as Validation, context };\n\n  }\n\n  private static loadConstructMetadata(constructMetadataPath: string): { readonly [key: string]: ResourceConstructMetadata } {\n    const contents = JSON.parse(fs.readFileSync(constructMetadataPath, { encoding: 'utf-8' }));\n    const resources: { [key: string]: ResourceConstructMetadata } = {};\n    if (contents.version !== '1.0.0') {\n      throw new Error(`Unexpected version of construct metadata at ${constructMetadataPath}: ${contents.version}. Supported versions are: [1.0.0]`);\n    }\n    for (const [name, metadata] of Object.entries(contents.resources)) {\n      resources[name] = { path: (metadata as any).path };\n    }\n    return resources;\n\n  }\n\n}\n\n/**\n * Construct related metadata on resources.\n */\ninterface ResourceConstructMetadata {\n\n  /**\n   * The path of the construct in the application.\n   */\n  readonly path: string;\n\n}\n\nfunction reset(s: string) {\n  return `${s}\\x1b[0m`;\n}\n\nfunction red(s: string) {\n  return `\\x1b[31m${s}`;\n}\n\nfunction bright(s: string) {\n  return `\\x1b[1m${s}`;\n}"]}
;