UNPKG

@renovosolutions/cdk-aspects-library-encryption-enforcement

Version:
140 lines 17.5 kB
"use strict"; var _a, _b, _c; Object.defineProperty(exports, "__esModule", { value: true }); exports.EncryptionEnforcement = exports.RDSEncryptionEnforcementAspect = exports.EFSEncryptionEnforcementAspect = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_cdk_lib_1 = require("aws-cdk-lib"); ; /** * Common base class for all aspects in this module. * * This exists to define the common properties and methods for all encryption enforcement aspects. * Right now, that's just exposing the `excludeResources` property. * * @abstract * @class EncryptionEnforcementAspect * @implements {IAspect} */ class EncryptionEnforcementAspect { /** * Constructs a new EncryptionEnforcementAspect. * * @param props - Optional properties to configure the aspect. */ constructor(props) { this.excludeResources = props?.excludeResources ?? []; } /** * Dummy visit method that must be overridden by subclasses. * @param _node - The construct to visit. */ visit(_node) { } } /** * An aspect that enforces encryption on all EFS FileSystems in the stack. */ class EFSEncryptionEnforcementAspect extends EncryptionEnforcementAspect { /** * Visits each construct in the stack and enforces encryption on EFS FileSystems. * * @param node - The construct to visit. */ visit(node) { // Check if this is an EFS FileSystem if (node instanceof aws_cdk_lib_1.CfnResource && node.cfnResourceType == 'AWS::EFS::FileSystem') { // Get the logical ID of the parent construct, so we can properly check for exclusions const parent = node.node.scope.node.id; /** * If it's an L1 resource, the first condition is relevant. * If it's an L2 resource, the second condition is the one. * We do this so both L1 and L2 constructs can be excluded by ID. */ if (this.excludeResources?.includes(node.node.id) || this.excludeResources?.includes(parent)) { return; // Skip this resource } // Check if the FileSystem is encrypted const encrypted = node.encrypted; if (!encrypted || encrypted.toString() !== 'true') { // If not encrypted, annotate an error aws_cdk_lib_1.Annotations.of(node).addError('EFS FileSystem must be encrypted. Please set the \'encrypted\' property to true.'); } } } } exports.EFSEncryptionEnforcementAspect = EFSEncryptionEnforcementAspect; _a = JSII_RTTI_SYMBOL_1; EFSEncryptionEnforcementAspect[_a] = { fqn: "@renovosolutions/cdk-aspects-library-encryption-enforcement.EFSEncryptionEnforcementAspect", version: "0.0.1" }; /** * An aspect that enforces encryption on all RDS databases in the stack. * Covers both single instances and clusters. */ class RDSEncryptionEnforcementAspect extends EncryptionEnforcementAspect { /** * Visits each construct in the stack and enforces encryption on RDS databases. * * @param node - The construct to visit. */ visit(node) { // Check if this is an RDS DBInstance or DBCluster if (node instanceof aws_cdk_lib_1.CfnResource && ['AWS::RDS::DBInstance', 'AWS::RDS::DBCluster'].includes(node.cfnResourceType)) { // Get the logical ID of the parent construct, so we can properly check for exclusions const parent = node.node.scope.node.id; /** * If it's an L1 resource, the first condition is relevant. * If it's an L2 resource, the second condition is the one. * We do this so both L1 and L2 constructs can be excluded by ID. */ if (this.excludeResources?.includes(node.node.id) || this.excludeResources?.includes(parent)) { return; // Skip this resource } if (node.dbClusterIdentifier !== undefined || node.sourceDbClusterIdentifier !== undefined) { /** * If this is a DBInstance that is part of a DBCluster, we skip it. * The L2 ClusterInstance class does not have `storageEncrypted` property, * it just inherits from the parent DBCluster. We have to skip it or we'd * tell the user to change a property that doesn't exist. * The L1 CfnDBInstance class does have `storageEncrypted`, but we still skip it * because Cfn will not allow it to be set to a conflicting value with the parent DBCluster. */ return; } /** * Check if the database is encrypted * This works even if it's actually a DBCluster. */ const encrypted = node.storageEncrypted; if (!encrypted || encrypted.toString() !== 'true') { // If not encrypted, annotate an error aws_cdk_lib_1.Annotations.of(node).addError('RDS database must have storage encryption enabled. Please set the \'storageEncrypted\' property to true.'); } } } } exports.RDSEncryptionEnforcementAspect = RDSEncryptionEnforcementAspect; _b = JSII_RTTI_SYMBOL_1; RDSEncryptionEnforcementAspect[_b] = { fqn: "@renovosolutions/cdk-aspects-library-encryption-enforcement.RDSEncryptionEnforcementAspect", version: "0.0.1" }; /** * An convenience class with a static function that adds all of the aspects in this module. * It's only a class because jsii skips standalone functions. */ class EncryptionEnforcement { /** * Adds all encryption enforcement aspects to the given scope. * * This is a convenience method to add all aspects in this module at once. * It can be used in the `main` function of your CDK app or in a stack constructor. * * @param scope - The scope to add the aspects to. * @param props - Optional properties to configure the aspects. * @returns void */ static addAllAspects(scope, props) { aws_cdk_lib_1.Aspects.of(scope).add(new EFSEncryptionEnforcementAspect(props)); aws_cdk_lib_1.Aspects.of(scope).add(new RDSEncryptionEnforcementAspect(props)); } } exports.EncryptionEnforcement = EncryptionEnforcement; _c = JSII_RTTI_SYMBOL_1; EncryptionEnforcement[_c] = { fqn: "@renovosolutions/cdk-aspects-library-encryption-enforcement.EncryptionEnforcement", version: "0.0.1" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Aspects,\n  IAspect,\n  CfnResource,\n  Annotations,\n  aws_efs as efs,\n  aws_rds as rds,\n} from 'aws-cdk-lib';\nimport { IConstruct } from 'constructs';\n\n/**\n * Common properties for all aspects in this module.\n */\nexport interface EncryptionEnforcementAspectProps {\n  /**\n   * The resources to exclude from enforcement.\n   *\n   * Use a resource's ID to exclude a specific resource.\n   * Supports both CfnResource and L2 construct IDs.\n   *\n   * @default []\n   */\n  readonly excludeResources?: string[];\n};\n\n/**\n * Common base class for all aspects in this module.\n *\n * This exists to define the common properties and methods for all encryption enforcement aspects.\n * Right now, that's just exposing the `excludeResources` property.\n *\n * @abstract\n * @class EncryptionEnforcementAspect\n * @implements {IAspect}\n */\nabstract class EncryptionEnforcementAspect implements IAspect {\n  /**\n   * The resources to exclude from enforcement.\n   *\n   * Use a resource's ID to exclude a specific resource.\n   * Supports both CfnResource and L2 construct IDs.\n   *\n   * @default []\n   */\n  public readonly excludeResources: string[];\n\n  /**\n   * Constructs a new EncryptionEnforcementAspect.\n   *\n   * @param props - Optional properties to configure the aspect.\n   */\n  constructor(props?: EncryptionEnforcementAspectProps) {\n    this.excludeResources = props?.excludeResources ?? [];\n  }\n\n  /**\n   * Dummy visit method that must be overridden by subclasses.\n   * @param _node - The construct to visit.\n   */\n  visit(_node: IConstruct): void {}\n}\n\n/**\n * An aspect that enforces encryption on all EFS FileSystems in the stack.\n */\nexport class EFSEncryptionEnforcementAspect extends EncryptionEnforcementAspect {\n  /**\n   * Visits each construct in the stack and enforces encryption on EFS FileSystems.\n   *\n   * @param node - The construct to visit.\n   */\n  visit(node: IConstruct) {\n    // Check if this is an EFS FileSystem\n    if (node instanceof CfnResource && node.cfnResourceType == 'AWS::EFS::FileSystem') {\n      // Get the logical ID of the parent construct, so we can properly check for exclusions\n      const parent = node.node.scope!.node.id;\n\n      /**\n       * If it's an L1 resource, the first condition is relevant.\n       * If it's an L2 resource, the second condition is the one.\n       * We do this so both L1 and L2 constructs can be excluded by ID.\n       */\n      if (this.excludeResources?.includes(node.node.id) || this.excludeResources?.includes(parent)) {\n        return; // Skip this resource\n      }\n\n      // Check if the FileSystem is encrypted\n      const encrypted = (node as efs.CfnFileSystem).encrypted;\n      if (!encrypted || encrypted.toString() !== 'true') {\n        // If not encrypted, annotate an error\n        Annotations.of(node).addError(\n          'EFS FileSystem must be encrypted. Please set the \\'encrypted\\' property to true.');\n      }\n    }\n  }\n}\n\n/**\n * An aspect that enforces encryption on all RDS databases in the stack.\n * Covers both single instances and clusters.\n */\nexport class RDSEncryptionEnforcementAspect extends EncryptionEnforcementAspect {\n  /**\n   * Visits each construct in the stack and enforces encryption on RDS databases.\n   *\n   * @param node - The construct to visit.\n   */\n  visit(node: IConstruct) {\n    // Check if this is an RDS DBInstance or DBCluster\n    if (node instanceof CfnResource && ['AWS::RDS::DBInstance', 'AWS::RDS::DBCluster'].includes(node.cfnResourceType)) {\n      // Get the logical ID of the parent construct, so we can properly check for exclusions\n      const parent = node.node.scope!.node.id;\n\n      /**\n       * If it's an L1 resource, the first condition is relevant.\n       * If it's an L2 resource, the second condition is the one.\n       * We do this so both L1 and L2 constructs can be excluded by ID.\n       */\n      if (this.excludeResources?.includes(node.node.id) || this.excludeResources?.includes(parent)) {\n        return; // Skip this resource\n      }\n\n      if ((node as rds.CfnDBInstance).dbClusterIdentifier !== undefined ||\n        (node as rds.CfnDBInstance).sourceDbClusterIdentifier !== undefined) {\n        /**\n         * If this is a DBInstance that is part of a DBCluster, we skip it.\n         * The L2 ClusterInstance class does not have `storageEncrypted` property,\n         * it just inherits from the parent DBCluster. We have to skip it or we'd\n         * tell the user to change a property that doesn't exist.\n         * The L1 CfnDBInstance class does have `storageEncrypted`, but we still skip it\n         * because Cfn will not allow it to be set to a conflicting value with the parent DBCluster.\n         */\n        return;\n      }\n\n      /**\n       * Check if the database is encrypted\n       * This works even if it's actually a DBCluster.\n       */\n      const encrypted = (node as rds.CfnDBInstance).storageEncrypted;\n      if (!encrypted || encrypted.toString() !== 'true') {\n        // If not encrypted, annotate an error\n        Annotations.of(node).addError(\n          'RDS database must have storage encryption enabled. Please set the \\'storageEncrypted\\' property to true.');\n      }\n    }\n  }\n}\n\n/**\n * An convenience class with a static function that adds all of the aspects in this module.\n * It's only a class because jsii skips standalone functions.\n */\nexport class EncryptionEnforcement {\n/**\n * Adds all encryption enforcement aspects to the given scope.\n *\n * This is a convenience method to add all aspects in this module at once.\n * It can be used in the `main` function of your CDK app or in a stack constructor.\n *\n * @param scope - The scope to add the aspects to.\n * @param props - Optional properties to configure the aspects.\n * @returns void\n */\n  public static addAllAspects(scope: IConstruct, props?: EncryptionEnforcementAspectProps) {\n    Aspects.of(scope).add(new EFSEncryptionEnforcementAspect(props));\n    Aspects.of(scope).add(new RDSEncryptionEnforcementAspect(props));\n  }\n}\n"]}