UNPKG

cdk-nag

Version:

Check CDK v2 applications for best practices using a combination on available rule packs.

200 lines • 34.4 kB
"use strict"; var _a, _b; Object.defineProperty(exports, "__esModule", { value: true }); exports.NagReportLogger = exports.NagReportFormat = exports.AnnotationLogger = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); /* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. SPDX-License-Identifier: Apache-2.0 */ const fs_1 = require("fs"); const path_1 = require("path"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const nag_rules_1 = require("./nag-rules"); /** * A NagLogger that outputs to the CDK Annotations system. */ class AnnotationLogger { constructor(props) { this.suppressionId = 'CdkNagSuppression'; this.verbose = props?.verbose ?? false; this.logIgnores = props?.logIgnores ?? false; } onCompliance(_data) { return; } onNonCompliance(data) { const message = this.createMessage(data.ruleId, data.findingId, data.ruleInfo, data.ruleExplanation, this.verbose); if (data.ruleLevel == nag_rules_1.NagMessageLevel.ERROR) { aws_cdk_lib_1.Annotations.of(data.resource).addError(message); } else if (data.ruleLevel == nag_rules_1.NagMessageLevel.WARN) { aws_cdk_lib_1.Annotations.of(data.resource).addWarning(message); } } onSuppressed(data) { if (this.logIgnores) { const message = this.createMessage(this.suppressionId, data.findingId, `${data.ruleId} was triggered but suppressed.`, `Provided reason: "${data.suppressionReason}"`, this.verbose); aws_cdk_lib_1.Annotations.of(data.resource).addInfo(message); } } onError(data) { const information = `'${data.ruleId}' threw an error during validation. This is generally caused by a parameter referencing an intrinsic function. You can suppress the "${nag_rules_1.VALIDATION_FAILURE_ID}" to get rid of this error. For more details enable verbose logging.'`; const message = this.createMessage(nag_rules_1.VALIDATION_FAILURE_ID, '', information, data.errorMessage, this.verbose); aws_cdk_lib_1.Annotations.of(data.resource).addWarning(message); } onSuppressedError(data) { if (this.logIgnores === true) { const message = this.createMessage(this.suppressionId, '', `${nag_rules_1.VALIDATION_FAILURE_ID} was triggered but suppressed.`, data.errorSuppressionReason, this.verbose); aws_cdk_lib_1.Annotations.of(data.resource).addInfo(message); } } onNotApplicable(_data) { return; } createMessage(ruleId, findingId, ruleInfo, ruleExplanation, verbose) { let message = findingId ? `${ruleId}[${findingId}]: ${ruleInfo}` : `${ruleId}: ${ruleInfo}`; return verbose ? `${message} ${ruleExplanation}\n` : `${message}\n`; } } exports.AnnotationLogger = AnnotationLogger; _a = JSII_RTTI_SYMBOL_1; AnnotationLogger[_a] = { fqn: "cdk-nag.AnnotationLogger", version: "2.28.194" }; /** * Possible output formats of the NagReport */ var NagReportFormat; (function (NagReportFormat) { NagReportFormat["CSV"] = "csv"; NagReportFormat["JSON"] = "json"; })(NagReportFormat = exports.NagReportFormat || (exports.NagReportFormat = {})); /** * A NagLogger that creates compliance reports */ class NagReportLogger { constructor(props) { this.reportStacks = new Map(); if (props.formats.length === 0) { throw new Error('Must provide at least 1 NagReportFormat.'); } this.formats = props.formats; } onCompliance(data) { this.initializeStackReport(data); this.writeToStackComplianceReport(data, nag_rules_1.NagRuleCompliance.COMPLIANT); } onNonCompliance(data) { this.initializeStackReport(data); this.writeToStackComplianceReport(data, nag_rules_1.NagRuleCompliance.NON_COMPLIANT); } onSuppressed(data) { this.initializeStackReport(data); this.writeToStackComplianceReport(data, nag_rules_1.NagRulePostValidationStates.SUPPRESSED); } onError(data) { this.initializeStackReport(data); this.writeToStackComplianceReport(data, nag_rules_1.NagRulePostValidationStates.UNKNOWN); } onSuppressedError(data) { this.initializeStackReport(data); this.writeToStackComplianceReport(data, nag_rules_1.NagRulePostValidationStates.SUPPRESSED); } onNotApplicable(data) { this.initializeStackReport(data); } getFormatStacks(format) { return this.reportStacks.get(format) ?? []; } /** * Initialize the report for the rule pack's compliance report for the resource's Stack if it doesn't exist * @param data */ initializeStackReport(data) { for (const format of this.formats) { const stackName = data.resource.stack.nested ? aws_cdk_lib_1.Names.uniqueId(data.resource.stack) : data.resource.stack.stackName; const fileName = `${data.nagPackName}-${stackName}-NagReport.${format}`; const stacks = this.getFormatStacks(format); if (!stacks.includes(fileName)) { const filePath = path_1.join(aws_cdk_lib_1.App.of(data.resource)?.outdir ?? '', fileName); this.reportStacks.set(format, [...stacks, fileName]); let body = ''; if (format === NagReportFormat.CSV) { body = 'Rule ID,Resource ID,Compliance,Exception Reason,Rule Level,Rule Info\n'; } else if (format === NagReportFormat.JSON) { body = JSON.stringify({ lines: [] }); } else { throw new Error(`Unrecognized output format ${format} for the NagReportLogger`); } fs_1.writeFileSync(filePath, body); } } } writeToStackComplianceReport(data, compliance) { for (const format of this.formats) { const stackName = data.resource.stack.nested ? aws_cdk_lib_1.Names.uniqueId(data.resource.stack) : data.resource.stack.stackName; const fileName = `${data.nagPackName}-${stackName}-NagReport.${format}`; const filePath = path_1.join(aws_cdk_lib_1.App.of(data.resource)?.outdir ?? '', fileName); if (format === NagReportFormat.CSV) { //| Rule ID | Resource ID | Compliance | Exception Reason | Rule Level | Rule Info const line = Array(); line.push(data.ruleId); line.push(data.resource.node.path); if (compliance === nag_rules_1.NagRulePostValidationStates.SUPPRESSED) { line.push(nag_rules_1.NagRulePostValidationStates.SUPPRESSED); if (data.suppressionReason !== undefined) { line.push(data.suppressionReason); } else { line.push(data.errorSuppressionReason); } } else { line.push(compliance); line.push('N/A'); } line.push(data.ruleLevel); line.push(data.ruleInfo); fs_1.appendFileSync(filePath, line.map((i) => '"' + i.replace(/"/g, '""') + '"').join(',') + '\n'); } else if (format === NagReportFormat.JSON) { const report = JSON.parse(fs_1.readFileSync(filePath, 'utf8')); let exceptionReason = 'N/A'; if (compliance === nag_rules_1.NagRulePostValidationStates.SUPPRESSED) { if (data.suppressionReason !== undefined) { exceptionReason = data .suppressionReason; } else { exceptionReason = data .errorSuppressionReason; } } report.lines.push({ ruleId: data.ruleId, resourceId: data.resource.node.path, compliance, exceptionReason, ruleLevel: data.ruleLevel, ruleInfo: data.ruleInfo, }); fs_1.writeFileSync(filePath, JSON.stringify(report)); } else { throw new Error(`Unrecognized output format ${format} for the NagReportLogger`); } } } } exports.NagReportLogger = NagReportLogger; _b = JSII_RTTI_SYMBOL_1; NagReportLogger[_b] = { fqn: "cdk-nag.NagReportLogger", version: "2.28.194" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nag-logger.js","sourceRoot":"","sources":["../src/nag-logger.ts"],"names":[],"mappings":"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","sourcesContent":["/*\nCopyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0\n*/\nimport { appendFileSync, readFileSync, writeFileSync } from 'fs';\nimport { join } from 'path';\nimport { Annotations, App, CfnResource, Names } from 'aws-cdk-lib';\nimport {\n  NagMessageLevel,\n  NagRuleCompliance,\n  NagRulePostValidationStates,\n  NagRuleStates,\n  VALIDATION_FAILURE_ID,\n} from './nag-rules';\n\n/**\n * Shared data for all INagLogger methods\n * @param nagPackName The name of the NagPack that the rule belongs to.\n * @param resource The resource the suppression is applied to.\n * @param ruleId The id of the rule to ignore.\n * @param ruleOriginalName Original name of the rule.\n * @param ruleInfo Why the rule was triggered.\n * @param ruleExplanation Why the rule exists.\n * @param ruleLevel The severity level of the rule.\n */\nexport interface NagLoggerBaseData {\n  readonly nagPackName: string;\n  readonly resource: CfnResource;\n  readonly ruleId: string;\n  readonly ruleOriginalName: string;\n  readonly ruleInfo: string;\n  readonly ruleExplanation: string;\n  readonly ruleLevel: NagMessageLevel;\n}\n\n/**\n * Data for onCompliance method of an INagLogger.\n */\nexport interface NagLoggerComplianceData extends NagLoggerBaseData {}\n/**\n * Data for onNonCompliance method of an INagLogger.\n * @param findingId The id of the finding that is being checked.\n */\nexport interface NagLoggerNonComplianceData extends NagLoggerBaseData {\n  readonly findingId: string;\n}\n/**\n * Data for onSuppressed method of an INagLogger.\n * @param suppressionReason The reason given for the suppression.\n */\nexport interface NagLoggerSuppressedData extends NagLoggerNonComplianceData {\n  readonly suppressionReason: string;\n}\n/**\n * Data for onError method of an INagLogger.\n * @param errorMessage: The error that was thrown.\n * @param shouldLogIgnored Whether or not the NagPack user has indicated that they want to log suppression details.\n */\nexport interface NagLoggerErrorData extends NagLoggerBaseData {\n  readonly errorMessage: string;\n}\n/**\n * Data for onSuppressedError method of an INagLogger.\n * @param errorSuppressionReason The reason given for the validation error suppression.\n */\nexport interface NagLoggerSuppressedErrorData extends NagLoggerErrorData {\n  readonly errorSuppressionReason: string;\n}\n\n/**\n * Data for onNotApplicable method of an INagLogger.\n */\nexport interface NagLoggerNotApplicableData extends NagLoggerBaseData {}\n\n/**\n * Interface for creating NagSuppression Ignores\n */\nexport interface INagLogger {\n  /**\n   * Called when a CfnResource passes the compliance check for a given rule.\n   */\n  onCompliance(data: NagLoggerComplianceData): void;\n  /**\n   * Called when a CfnResource does not pass the compliance check for a given rule and the the rule violation is not suppressed by the user.\n   */\n  onNonCompliance(data: NagLoggerNonComplianceData): void;\n  /**\n   * Called when a CfnResource does not pass the compliance check for a given rule and the rule violation is suppressed by the user.\n   */\n  onSuppressed(data: NagLoggerSuppressedData): void;\n  /**\n   * Called when a rule throws an error during while validating a CfnResource for compliance.\n   */\n  onError(data: NagLoggerErrorData): void;\n  /**\n   * Called when a rule throws an error during while validating a CfnResource for compliance and the error is suppressed.\n   */\n  onSuppressedError(data: NagLoggerSuppressedErrorData): void;\n  /**\n   * Called when a rule does not apply to the given CfnResource.\n   */\n  onNotApplicable(data: NagLoggerNotApplicableData): void;\n}\n\n/**\n * Props for the AnnotationLogger\n */\nexport interface AnnotationLoggerProps {\n  /**\n   * Whether or not to enable extended explanatory descriptions on warning, error, and logged ignore messages.\n   */\n  readonly verbose?: boolean;\n\n  /**\n   * Whether or not to log suppressed rule violations as informational messages (default: false).\n   */\n  readonly logIgnores?: boolean;\n}\n/**\n * A NagLogger that outputs to the CDK Annotations system.\n */\nexport class AnnotationLogger implements INagLogger {\n  suppressionId = 'CdkNagSuppression';\n  readonly verbose: boolean;\n  readonly logIgnores: boolean;\n  constructor(props?: AnnotationLoggerProps) {\n    this.verbose = props?.verbose ?? false;\n    this.logIgnores = props?.logIgnores ?? false;\n  }\n  onCompliance(_data: NagLoggerComplianceData): void {\n    return;\n  }\n  onNonCompliance(data: NagLoggerNonComplianceData): void {\n    const message = this.createMessage(\n      data.ruleId,\n      data.findingId,\n      data.ruleInfo,\n      data.ruleExplanation,\n      this.verbose\n    );\n    if (data.ruleLevel == NagMessageLevel.ERROR) {\n      Annotations.of(data.resource).addError(message);\n    } else if (data.ruleLevel == NagMessageLevel.WARN) {\n      Annotations.of(data.resource).addWarning(message);\n    }\n  }\n  onSuppressed(data: NagLoggerSuppressedData): void {\n    if (this.logIgnores) {\n      const message = this.createMessage(\n        this.suppressionId,\n        data.findingId,\n        `${data.ruleId} was triggered but suppressed.`,\n        `Provided reason: \"${data.suppressionReason}\"`,\n        this.verbose\n      );\n      Annotations.of(data.resource).addInfo(message);\n    }\n  }\n  onError(data: NagLoggerErrorData): void {\n    const information = `'${data.ruleId}' threw an error during validation. This is generally caused by a parameter referencing an intrinsic function. You can suppress the \"${VALIDATION_FAILURE_ID}\" to get rid of this error. For more details enable verbose logging.'`;\n    const message = this.createMessage(\n      VALIDATION_FAILURE_ID,\n      '',\n      information,\n      data.errorMessage,\n      this.verbose\n    );\n    Annotations.of(data.resource).addWarning(message);\n  }\n  onSuppressedError(data: NagLoggerSuppressedErrorData): void {\n    if (this.logIgnores === true) {\n      const message = this.createMessage(\n        this.suppressionId,\n        '',\n        `${VALIDATION_FAILURE_ID} was triggered but suppressed.`,\n        data.errorSuppressionReason,\n        this.verbose\n      );\n      Annotations.of(data.resource).addInfo(message);\n    }\n  }\n  onNotApplicable(_data: NagLoggerNotApplicableData): void {\n    return;\n  }\n\n  protected createMessage(\n    ruleId: string,\n    findingId: string,\n    ruleInfo: string,\n    ruleExplanation: string,\n    verbose: boolean\n  ): string {\n    let message = findingId\n      ? `${ruleId}[${findingId}]: ${ruleInfo}`\n      : `${ruleId}: ${ruleInfo}`;\n    return verbose ? `${message} ${ruleExplanation}\\n` : `${message}\\n`;\n  }\n}\n\nexport interface NagReportSchema {\n  readonly lines: NagReportLine[];\n}\n\nexport interface NagReportLine {\n  readonly ruleId: string;\n  readonly resourceId: string;\n  readonly compliance: string;\n  readonly exceptionReason: string;\n  readonly ruleLevel: string;\n  readonly ruleInfo: string;\n}\n\n/**\n * Possible output formats of the NagReport\n */\nexport enum NagReportFormat {\n  CSV = 'csv',\n  JSON = 'json',\n}\n\n/**\n * Props for the NagReportLogger\n */\nexport interface NagReportLoggerProps {\n  readonly formats: NagReportFormat[];\n}\n\n/**\n * A NagLogger that creates compliance reports\n */\nexport class NagReportLogger implements INagLogger {\n  private reportStacks = new Map<NagReportFormat, Array<string>>();\n  readonly formats: NagReportFormat[];\n  constructor(props: NagReportLoggerProps) {\n    if (props.formats.length === 0) {\n      throw new Error('Must provide at least 1 NagReportFormat.');\n    }\n    this.formats = props.formats;\n  }\n\n  onCompliance(data: NagLoggerComplianceData): void {\n    this.initializeStackReport(data);\n    this.writeToStackComplianceReport(data, NagRuleCompliance.COMPLIANT);\n  }\n  onNonCompliance(data: NagLoggerNonComplianceData): void {\n    this.initializeStackReport(data);\n    this.writeToStackComplianceReport(data, NagRuleCompliance.NON_COMPLIANT);\n  }\n  onSuppressed(data: NagLoggerSuppressedData): void {\n    this.initializeStackReport(data);\n    this.writeToStackComplianceReport(\n      data,\n      NagRulePostValidationStates.SUPPRESSED\n    );\n  }\n  onError(data: NagLoggerErrorData): void {\n    this.initializeStackReport(data);\n    this.writeToStackComplianceReport(\n      data,\n      NagRulePostValidationStates.UNKNOWN\n    );\n  }\n  onSuppressedError(data: NagLoggerSuppressedErrorData): void {\n    this.initializeStackReport(data);\n    this.writeToStackComplianceReport(\n      data,\n      NagRulePostValidationStates.SUPPRESSED\n    );\n  }\n  onNotApplicable(data: NagLoggerNotApplicableData): void {\n    this.initializeStackReport(data);\n  }\n\n  public getFormatStacks(format: NagReportFormat): string[] {\n    return this.reportStacks.get(format) ?? [];\n  }\n\n  /**\n   * Initialize the report for the rule pack's compliance report for the resource's Stack if it doesn't exist\n   * @param data\n   */\n  protected initializeStackReport(data: NagLoggerBaseData): void {\n    for (const format of this.formats) {\n      const stackName = data.resource.stack.nested\n        ? Names.uniqueId(data.resource.stack)\n        : data.resource.stack.stackName;\n      const fileName = `${data.nagPackName}-${stackName}-NagReport.${format}`;\n      const stacks = this.getFormatStacks(format);\n      if (!stacks.includes(fileName)) {\n        const filePath = join(App.of(data.resource)?.outdir ?? '', fileName);\n        this.reportStacks.set(format, [...stacks, fileName]);\n        let body = '';\n        if (format === NagReportFormat.CSV) {\n          body =\n            'Rule ID,Resource ID,Compliance,Exception Reason,Rule Level,Rule Info\\n';\n        } else if (format === NagReportFormat.JSON) {\n          body = JSON.stringify({ lines: [] } as NagReportSchema);\n        } else {\n          throw new Error(\n            `Unrecognized output format ${format} for the NagReportLogger`\n          );\n        }\n        writeFileSync(filePath, body);\n      }\n    }\n  }\n\n  protected writeToStackComplianceReport(\n    data: NagLoggerBaseData,\n    compliance: NagRuleStates\n  ): void {\n    for (const format of this.formats) {\n      const stackName = data.resource.stack.nested\n        ? Names.uniqueId(data.resource.stack)\n        : data.resource.stack.stackName;\n      const fileName = `${data.nagPackName}-${stackName}-NagReport.${format}`;\n      const filePath = join(App.of(data.resource)?.outdir ?? '', fileName);\n      if (format === NagReportFormat.CSV) {\n        //| Rule ID | Resource ID | Compliance | Exception Reason | Rule Level | Rule Info\n        const line = Array<string>();\n        line.push(data.ruleId);\n        line.push(data.resource.node.path);\n        if (compliance === NagRulePostValidationStates.SUPPRESSED) {\n          line.push(NagRulePostValidationStates.SUPPRESSED);\n          if (\n            (data as NagLoggerSuppressedData).suppressionReason !== undefined\n          ) {\n            line.push((data as NagLoggerSuppressedData).suppressionReason);\n          } else {\n            line.push(\n              (data as NagLoggerSuppressedErrorData).errorSuppressionReason\n            );\n          }\n        } else {\n          line.push(compliance);\n          line.push('N/A');\n        }\n        line.push(data.ruleLevel);\n        line.push(data.ruleInfo);\n        appendFileSync(\n          filePath,\n          line.map((i) => '\"' + i.replace(/\"/g, '\"\"') + '\"').join(',') + '\\n'\n        );\n      } else if (format === NagReportFormat.JSON) {\n        const report = JSON.parse(\n          readFileSync(filePath, 'utf8')\n        ) as NagReportSchema;\n        let exceptionReason = 'N/A';\n        if (compliance === NagRulePostValidationStates.SUPPRESSED) {\n          if (\n            (data as NagLoggerSuppressedData).suppressionReason !== undefined\n          ) {\n            exceptionReason = (data as NagLoggerSuppressedData)\n              .suppressionReason;\n          } else {\n            exceptionReason = (data as NagLoggerSuppressedErrorData)\n              .errorSuppressionReason;\n          }\n        }\n        report.lines.push({\n          ruleId: data.ruleId,\n          resourceId: data.resource.node.path,\n          compliance,\n          exceptionReason,\n          ruleLevel: data.ruleLevel,\n          ruleInfo: data.ruleInfo,\n        });\n        writeFileSync(filePath, JSON.stringify(report));\n      } else {\n        throw new Error(\n          `Unrecognized output format ${format} for the NagReportLogger`\n        );\n      }\n    }\n  }\n}\n"]}