@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
198 lines • 22.9 kB
JavaScript
;
var _a, _b;
Object.defineProperty(exports, "__esModule", { value: true });
exports.PDKNag = exports.PDKNagApp = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0 */
const aws_cdk_lib_1 = require("aws-cdk-lib");
const cdk_nag_1 = require("cdk-nag");
const memory_logger_1 = require("./loggers/memory-logger");
const metrics_aspect_1 = require("./metrics-aspect");
const CDK_NAG_MESSAGE_TYPES = {
ERROR: "aws:cdk:error",
WARNING: "aws:cdk:warning",
};
const CDK_NAG_MESSAGE_TYPES_SET = new Set(Object.values(CDK_NAG_MESSAGE_TYPES));
const DEFAULT_NAG_PACKS = [
new cdk_nag_1.AwsSolutionsChecks({
verbose: true,
reports: true,
}),
];
/**
* @inheritDoc
*/
class PDKNagApp extends aws_cdk_lib_1.App {
constructor(props) {
super(props);
this._nagResults = [];
this._extendedNagResults = [];
this.failOnError = props?.failOnError ?? false;
this.failOnWarning = props?.failOnWarning ?? false;
this.nagPacks = props?.nagPacks ?? DEFAULT_NAG_PACKS;
aws_cdk_lib_1.Aspects.of(this).add(new PDKNagAspect(this));
aws_cdk_lib_1.Aspects.of(this).add(new metrics_aspect_1.MetricsAspect());
}
synth(options) {
const assembly = super.synth(options);
const typesToFail = new Set([
this.failOnError && CDK_NAG_MESSAGE_TYPES.ERROR,
this.failOnWarning && CDK_NAG_MESSAGE_TYPES.WARNING,
].filter((t) => t));
if (this._nagResults.find((r) => r.messages.find((m) => typesToFail.has(m.messageType)))) {
throw new Error(JSON.stringify(this._nagResults, undefined, 2));
}
return assembly;
}
addNagResult(result) {
this._nagResults.push(result);
}
/**
* Returns a list of NagResult.
*
* Note: app.synth() must be called before this to retrieve results.
*/
nagResults() {
return this._nagResults;
}
addExtendedNagResults(...results) {
this._extendedNagResults.push(...results);
}
/**
* Returns a list of ExtendedNagResult.
*
* Note: app.synth() must be called before this to retrieve results.
*/
extendedNagResults() {
return this._extendedNagResults;
}
}
exports.PDKNagApp = PDKNagApp;
_a = JSII_RTTI_SYMBOL_1;
PDKNagApp[_a] = { fqn: "@aws/pdk.pdk_nag.PDKNagApp", version: "0.26.14" };
class PDKNagAspect {
constructor(app) {
this.app = app;
}
visit(node) {
const memoryLogger = new memory_logger_1.MemoryLogger();
this.app.nagPacks.forEach((nagPack) => {
// @ts-ignore loggers is private, but since we haven't called "visit" yet it's safe to add another
nagPack.loggers.push(memoryLogger);
nagPack.visit(node);
// @ts-ignore loggers is private, but remove the memory logger to clean up
nagPack.loggers.pop();
});
this.app.addExtendedNagResults(...memoryLogger.results);
const results = node.node.metadata.filter((m) => CDK_NAG_MESSAGE_TYPES_SET.has(m.type));
results.length > 0 &&
this.app.addNagResult({
resource: node.node.path,
messages: results.map((m) => ({
messageDescription: m.data,
messageType: m.type,
})),
});
}
}
/**
* Helper for create a Nag Enabled App.
*/
class PDKNag {
/**
* Returns an instance of an App with Nag enabled.
*
* @param props props to initialize the app with.
*/
static app(props) {
return new PDKNagApp(props);
}
/**
* Wrapper around NagSuppressions which does not throw.
*
* @param stack stack instance
* @param path resource path
* @param suppressions list of suppressions to apply.
* @param applyToChildren whether to apply to children.
*/
static addResourceSuppressionsByPathNoThrow(stack, path, suppressions, applyToChildren = false) {
try {
cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, path, suppressions, applyToChildren);
}
catch (e) {
// Do Nothing
}
}
/**
* Returns a prefix comprising of a delimited set of Stack Ids.
*
* For example: StackA/NestedStackB/
*
* @param stack stack instance.
*/
static getStackPrefix(stack) {
if (stack.nested) {
return `${PDKNag.getStackPrefix(stack.nestedStackParent)}${stack.node.id}/`;
}
else {
const stageName = aws_cdk_lib_1.Stage.of(stack)?.stageName;
const stagePrefix = stageName && `${stageName}-`;
let stackName = stack.stackName;
stackName =
stagePrefix && stackName.startsWith(stagePrefix)
? `${stageName}/${stackName.slice(stagePrefix.length)}`
: stackName;
return `${stackName}/`;
}
}
/**
* Returns a stack partition regex.
*
* @param stack stack instance.
*/
static getStackPartitionRegex(stack) {
if (stack.nested) {
return PDKNag.getStackPartitionRegex(stack.nestedStackParent);
}
else {
return stack.partition.startsWith("${Token")
? "<AWS::Partition>"
: `(<AWS::Partition>|${stack.partition})`;
}
}
/**
* Returns a stack region regex.
*
* @param stack stack instance.
*/
static getStackRegionRegex(stack) {
if (stack.nested) {
return PDKNag.getStackRegionRegex(stack.nestedStackParent);
}
else {
return stack.region.startsWith("${Token")
? "<AWS::Region>"
: `(<AWS::Region>|${stack.region})`;
}
}
/**
* Returns a stack account regex.
*
* @param stack stack instance.
*/
static getStackAccountRegex(stack) {
if (stack.nested) {
return PDKNag.getStackAccountRegex(stack.nestedStackParent);
}
else {
return stack.account.startsWith("${Token")
? "<AWS::AccountId>"
: `(<AWS::AccountId>|${stack.account})`;
}
}
}
exports.PDKNag = PDKNag;
_b = JSII_RTTI_SYMBOL_1;
PDKNag[_b] = { fqn: "@aws/pdk.pdk_nag.PDKNag", version: "0.26.14" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pdk-nag.js","sourceRoot":"","sources":["pdk-nag.ts"],"names":[],"mappings":"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","sourcesContent":["/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0 */\nimport {\n  App,\n  AppProps,\n  Aspects,\n  IAspect,\n  Stack,\n  Stage,\n  StageSynthesisOptions,\n} from \"aws-cdk-lib\";\nimport { CloudAssembly } from \"aws-cdk-lib/cx-api\";\nimport {\n  AwsSolutionsChecks,\n  NagPack,\n  NagPackSuppression,\n  NagSuppressions,\n} from \"cdk-nag\";\nimport { IConstruct } from \"constructs\";\nimport { MemoryLogger } from \"./loggers/memory-logger\";\nimport { ExtendedNagResult } from \"./loggers/types\";\nimport { MetricsAspect } from \"./metrics-aspect\";\n\nconst CDK_NAG_MESSAGE_TYPES = {\n  ERROR: \"aws:cdk:error\",\n  WARNING: \"aws:cdk:warning\",\n};\nconst CDK_NAG_MESSAGE_TYPES_SET = new Set(Object.values(CDK_NAG_MESSAGE_TYPES));\nconst DEFAULT_NAG_PACKS = [\n  new AwsSolutionsChecks({\n    verbose: true,\n    reports: true,\n  }),\n];\n\n/**\n * Message instance.\n */\nexport interface Message {\n  /**\n   * Message description.\n   */\n  readonly messageDescription: string;\n\n  /**\n   * Message type as returned from cdk-nag.\n   */\n  readonly messageType: string;\n}\n\n/**\n * Nag result.\n */\nexport interface NagResult {\n  /**\n   * Resource which triggered the message.\n   */\n  readonly resource: string;\n\n  /**\n   * List of messages.\n   */\n  readonly messages: Message[];\n}\n\n/**\n * @inheritDoc\n */\nexport interface PDKNagAppProps extends AppProps {\n  /**\n   * Determines whether any errors encountered should trigger a test failure.\n   *\n   * @default false\n   */\n  readonly failOnError?: boolean;\n\n  /**\n   * Determines whether any warnings encountered should trigger a test failure.\n   *\n   * @default false\n   */\n  readonly failOnWarning?: boolean;\n\n  /**\n   * Custom nag packs to execute.\n   *\n   * @default DEFAULT_NAG_PACKS\n   */\n  readonly nagPacks?: NagPack[];\n}\n\n/**\n * @inheritDoc\n */\nexport class PDKNagApp extends App {\n  private readonly _nagResults: NagResult[] = [];\n  private readonly _extendedNagResults: ExtendedNagResult[] = [];\n  private readonly failOnError: boolean;\n  private readonly failOnWarning: boolean;\n  public readonly nagPacks: NagPack[];\n\n  constructor(props?: PDKNagAppProps) {\n    super(props);\n    this.failOnError = props?.failOnError ?? false;\n    this.failOnWarning = props?.failOnWarning ?? false;\n    this.nagPacks = props?.nagPacks ?? DEFAULT_NAG_PACKS;\n\n    Aspects.of(this).add(new PDKNagAspect(this));\n    Aspects.of(this).add(new MetricsAspect());\n  }\n\n  synth(options?: StageSynthesisOptions): CloudAssembly {\n    const assembly = super.synth(options);\n\n    const typesToFail = new Set(\n      [\n        this.failOnError && CDK_NAG_MESSAGE_TYPES.ERROR,\n        this.failOnWarning && CDK_NAG_MESSAGE_TYPES.WARNING,\n      ].filter((t) => t)\n    );\n    if (\n      this._nagResults.find((r) =>\n        r.messages.find((m) => typesToFail.has(m.messageType))\n      )\n    ) {\n      throw new Error(JSON.stringify(this._nagResults, undefined, 2));\n    }\n\n    return assembly;\n  }\n\n  addNagResult(result: NagResult) {\n    this._nagResults.push(result);\n  }\n\n  /**\n   * Returns a list of NagResult.\n   *\n   * Note: app.synth() must be called before this to retrieve results.\n   */\n  public nagResults(): NagResult[] {\n    return this._nagResults;\n  }\n\n  addExtendedNagResults(...results: ExtendedNagResult[]) {\n    this._extendedNagResults.push(...results);\n  }\n\n  /**\n   * Returns a list of ExtendedNagResult.\n   *\n   * Note: app.synth() must be called before this to retrieve results.\n   */\n  public extendedNagResults(): ExtendedNagResult[] {\n    return this._extendedNagResults;\n  }\n}\n\nclass PDKNagAspect implements IAspect {\n  private readonly app: PDKNagApp;\n\n  constructor(app: PDKNagApp) {\n    this.app = app;\n  }\n\n  visit(node: IConstruct): void {\n    const memoryLogger = new MemoryLogger();\n\n    this.app.nagPacks.forEach((nagPack) => {\n      // @ts-ignore loggers is private, but since we haven't called \"visit\" yet it's safe to add another\n      nagPack.loggers.push(memoryLogger);\n      nagPack.visit(node);\n      // @ts-ignore loggers is private, but remove the memory logger to clean up\n      nagPack.loggers.pop();\n    });\n    this.app.addExtendedNagResults(...memoryLogger.results);\n\n    const results = node.node.metadata.filter((m) =>\n      CDK_NAG_MESSAGE_TYPES_SET.has(m.type)\n    );\n    results.length > 0 &&\n      this.app.addNagResult({\n        resource: node.node.path,\n        messages: results.map((m) => ({\n          messageDescription: m.data,\n          messageType: m.type,\n        })),\n      });\n  }\n}\n\n/**\n * Helper for create a Nag Enabled App.\n */\nexport class PDKNag {\n  /**\n   * Returns an instance of an App with Nag enabled.\n   *\n   * @param props props to initialize the app with.\n   */\n  public static app(props?: PDKNagAppProps): PDKNagApp {\n    return new PDKNagApp(props);\n  }\n\n  /**\n   * Wrapper around NagSuppressions which does not throw.\n   *\n   * @param stack stack instance\n   * @param path resource path\n   * @param suppressions list of suppressions to apply.\n   * @param applyToChildren whether to apply to children.\n   */\n  public static addResourceSuppressionsByPathNoThrow(\n    stack: Stack,\n    path: string,\n    suppressions: NagPackSuppression[],\n    applyToChildren: boolean = false\n  ): void {\n    try {\n      NagSuppressions.addResourceSuppressionsByPath(\n        stack,\n        path,\n        suppressions,\n        applyToChildren\n      );\n    } catch (e) {\n      // Do Nothing\n    }\n  }\n\n  /**\n   * Returns a prefix comprising of a delimited set of Stack Ids.\n   *\n   * For example: StackA/NestedStackB/\n   *\n   * @param stack stack instance.\n   */\n  public static getStackPrefix(stack: Stack): string {\n    if (stack.nested) {\n      return `${PDKNag.getStackPrefix(stack.nestedStackParent!)}${\n        stack.node.id\n      }/`;\n    } else {\n      const stageName = Stage.of(stack)?.stageName;\n      const stagePrefix = stageName && `${stageName}-`;\n      let stackName = stack.stackName;\n\n      stackName =\n        stagePrefix && stackName.startsWith(stagePrefix)\n          ? `${stageName}/${stackName.slice(stagePrefix.length)}`\n          : stackName;\n      return `${stackName}/`;\n    }\n  }\n\n  /**\n   * Returns a stack partition regex.\n   *\n   * @param stack stack instance.\n   */\n  public static getStackPartitionRegex(stack: Stack): string {\n    if (stack.nested) {\n      return PDKNag.getStackPartitionRegex(stack.nestedStackParent!);\n    } else {\n      return stack.partition.startsWith(\"${Token\")\n        ? \"<AWS::Partition>\"\n        : `(<AWS::Partition>|${stack.partition})`;\n    }\n  }\n\n  /**\n   * Returns a stack region regex.\n   *\n   * @param stack stack instance.\n   */\n  public static getStackRegionRegex(stack: Stack): string {\n    if (stack.nested) {\n      return PDKNag.getStackRegionRegex(stack.nestedStackParent!);\n    } else {\n      return stack.region.startsWith(\"${Token\")\n        ? \"<AWS::Region>\"\n        : `(<AWS::Region>|${stack.region})`;\n    }\n  }\n\n  /**\n   * Returns a stack account regex.\n   *\n   * @param stack stack instance.\n   */\n  public static getStackAccountRegex(stack: Stack): string {\n    if (stack.nested) {\n      return PDKNag.getStackAccountRegex(stack.nestedStackParent!);\n    } else {\n      return stack.account.startsWith(\"${Token\")\n        ? \"<AWS::AccountId>\"\n        : `(<AWS::AccountId>|${stack.account})`;\n    }\n  }\n}\n"]}