UNPKG

@aws-solutions-constructs/core

Version:
138 lines 20.1 kB
"use strict"; /** * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance * with the License. A copy of the License is located at * * http://www.apache.org/licenses/LICENSE-2.0 * * or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES * OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions * and limitations under the License. */ Object.defineProperty(exports, "__esModule", { value: true }); exports.buildTopic = buildTopic; exports.CheckSnsProps = CheckSnsProps; /* * The functions found here in the core library are for internal use and can be changed * or removed outside of a major release. We recommend against calling them directly from client code. */ // Imports const sns = require("aws-cdk-lib/aws-sns"); const kms = require("aws-cdk-lib/aws-kms"); const sns_defaults_1 = require("./sns-defaults"); const kms_helper_1 = require("./kms-helper"); const utils_1 = require("./utils"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const aws_cdk_lib_1 = require("aws-cdk-lib"); function applySecureTopicPolicy(topic) { // Apply topic policy to enforce only the topic owner can publish and subscribe to this topic topic.addToResourcePolicy(new aws_iam_1.PolicyStatement({ sid: 'TopicOwnerOnlyAccess', resources: [ `${topic.topicArn}` ], actions: [ "SNS:Publish", "SNS:RemovePermission", "SNS:SetTopicAttributes", "SNS:DeleteTopic", "SNS:ListSubscriptionsByTopic", "SNS:GetTopicAttributes", "SNS:Receive", "SNS:AddPermission", "SNS:Subscribe" ], principals: [new aws_iam_1.AccountPrincipal(aws_cdk_lib_1.Stack.of(topic).account)], effect: aws_iam_1.Effect.ALLOW, conditions: { StringEquals: { "AWS:SourceOwner": aws_cdk_lib_1.Stack.of(topic).account } } })); // Apply Topic policy to enforce encryption of data in transit topic.addToResourcePolicy(new aws_iam_1.PolicyStatement({ sid: 'HttpsOnly', resources: [ `${topic.topicArn}` ], actions: [ "SNS:Publish", "SNS:RemovePermission", "SNS:SetTopicAttributes", "SNS:DeleteTopic", "SNS:ListSubscriptionsByTopic", "SNS:GetTopicAttributes", "SNS:Receive", "SNS:AddPermission", "SNS:Subscribe" ], principals: [new aws_iam_1.AnyPrincipal()], effect: aws_iam_1.Effect.DENY, conditions: { Bool: { 'aws:SecureTransport': 'false' } } })); } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function buildTopic(scope, id, props) { if (!props.existingTopicObj) { // Setup the topic properties const snsTopicProps = (0, utils_1.consolidateProps)(sns_defaults_1.defaultSnsTopicProps, props.topicProps); // Set encryption properties if (props.topicProps?.masterKey) { snsTopicProps.masterKey = props.topicProps?.masterKey; } else if (props.encryptionKey) { snsTopicProps.masterKey = props.encryptionKey; } else if (props.encryptionKeyProps || props.enableEncryptionWithCustomerManagedKey === true) { snsTopicProps.masterKey = (0, kms_helper_1.buildEncryptionKey)(scope, id, props.encryptionKeyProps); } else { snsTopicProps.masterKey = kms.Alias.fromAliasName(scope, 'aws-managed-key', 'alias/aws/sns'); } // Create the SNS Topic // NOSONAR (typescript:S6327) - The masterKey is set in the if statement above, SONAR is // not catching it. Behavior is confirmed in the // 'Test deployment with no properties using AWS Managed KMS Key' unit test const topic = new sns.Topic(scope, 'SnsTopic', snsTopicProps); // NOSONAR applySecureTopicPolicy(topic); return { topic, key: snsTopicProps.masterKey }; } else { return { topic: props.existingTopicObj, key: props.existingTopicEncryptionKey }; } } function CheckSnsProps(propsObject) { let errorMessages = ''; let errorFound = false; // FargateToSns used TopicObject instead of TopicObj - to fix would be a breaking change, so we // must look for both here. if (propsObject.topicProps && (propsObject.existingTopicObj || propsObject.existingTopicObject)) { errorMessages += 'Error - Either provide topicProps or existingTopicObj, but not both.\n'; errorFound = true; } if (propsObject.topicProps?.masterKey && propsObject.encryptionKey) { errorMessages += 'Error - Either provide topicProps.masterKey or encryptionKey, but not both.\n'; errorFound = true; } if (propsObject.topicProps?.masterKey && propsObject.encryptionKeyProps) { errorMessages += 'Error - Either provide topicProps.masterKey or encryptionKeyProps, but not both.\n'; errorFound = true; } if (propsObject.encryptionKey && propsObject.encryptionKeyProps) { errorMessages += 'Error - Either provide encryptionKey or encryptionKeyProps, but not both.\n'; errorFound = true; } if (errorFound) { throw new Error(errorMessages); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sns-helper.js","sourceRoot":"","sources":["sns-helper.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n *  Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n *\n *  Licensed under the Apache License, Version 2.0 (the \"License\"). You may not use this file except in compliance\n *  with the License. A copy of the License is located at\n *\n *      http://www.apache.org/licenses/LICENSE-2.0\n *\n *  or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES\n *  OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions\n *  and limitations under the License.\n */\n\n/*\n *  The functions found here in the core library are for internal use and can be changed\n *  or removed outside of a major release. We recommend against calling them directly from client code.\n */\n\n// Imports\nimport * as sns from 'aws-cdk-lib/aws-sns';\nimport * as kms from 'aws-cdk-lib/aws-kms';\nimport { defaultSnsTopicProps } from './sns-defaults';\nimport { buildEncryptionKey } from './kms-helper';\nimport { consolidateProps } from './utils';\nimport { PolicyStatement, AnyPrincipal, Effect, AccountPrincipal } from 'aws-cdk-lib/aws-iam';\nimport { Stack } from 'aws-cdk-lib';\n// Note: To ensure CDKv2 compatibility, keep the import statement for Construct separate\nimport { Construct } from 'constructs';\n\nexport interface BuildTopicProps {\n    /**\n     * Existing SNS topic to be used instead of the default topic. Providing both this and `topicProps` will cause an error.\n     * If the SNS Topic is encrypted with a Customer-Managed managed KMS key, the key must be specified in the\n     * `existingTopicEncryptionKey` property.\n     *\n     * @default - Default props are used\n     */\n    readonly existingTopicObj?: sns.Topic;\n     /**\n      * If an existing topic is provided in the `existingTopicObj` property, and that topic is encrypted with a customer managed KMS key,\n      * this property also needs to be set with same CMK.\n      *\n      * @default - None\n      */\n    readonly existingTopicEncryptionKey?: kms.Key;\n    /**\n     * Optional user provided props to override the default props for the SNS topic.\n     *\n     * @default - Default props are used.\n     */\n    readonly topicProps?: sns.TopicProps;\n    /**\n     * If no key is provided, this flag determines whether the topic is encrypted with a new CMK or an AWS managed key.\n     * This flag is ignored if any of the following are defined: topicProps.masterKey, encryptionKey or encryptionKeyProps.\n     *\n     * @default - False if topicProps.masterKey, encryptionKey, and encryptionKeyProps are all undefined.\n     */\n    readonly enableEncryptionWithCustomerManagedKey?: boolean;\n    /**\n     * An optional, imported encryption key to encrypt the SNS topic with.\n     *\n     * @default - None\n     */\n    readonly encryptionKey?: kms.Key;\n    /**\n     * Optional user provided properties to override the default properties for the KMS encryption key used to encrypt the SNS topic with.\n     *\n     * @default - None\n     */\n    readonly encryptionKeyProps?: kms.KeyProps;\n}\n\nfunction applySecureTopicPolicy(topic: sns.Topic): void {\n\n  // Apply topic policy to enforce only the topic owner can publish and subscribe to this topic\n  topic.addToResourcePolicy(\n    new PolicyStatement({\n      sid: 'TopicOwnerOnlyAccess',\n      resources: [\n        `${topic.topicArn}`\n      ],\n      actions: [\n        \"SNS:Publish\",\n        \"SNS:RemovePermission\",\n        \"SNS:SetTopicAttributes\",\n        \"SNS:DeleteTopic\",\n        \"SNS:ListSubscriptionsByTopic\",\n        \"SNS:GetTopicAttributes\",\n        \"SNS:Receive\",\n        \"SNS:AddPermission\",\n        \"SNS:Subscribe\"\n      ],\n      principals: [new AccountPrincipal(Stack.of(topic).account)],\n      effect: Effect.ALLOW,\n      conditions:\n            {\n              StringEquals: {\n                \"AWS:SourceOwner\": Stack.of(topic).account\n              }\n            }\n    })\n  );\n\n  // Apply Topic policy to enforce encryption of data in transit\n  topic.addToResourcePolicy(\n    new PolicyStatement({\n      sid: 'HttpsOnly',\n      resources: [\n        `${topic.topicArn}`\n      ],\n      actions: [\n        \"SNS:Publish\",\n        \"SNS:RemovePermission\",\n        \"SNS:SetTopicAttributes\",\n        \"SNS:DeleteTopic\",\n        \"SNS:ListSubscriptionsByTopic\",\n        \"SNS:GetTopicAttributes\",\n        \"SNS:Receive\",\n        \"SNS:AddPermission\",\n        \"SNS:Subscribe\"\n      ],\n      principals: [new AnyPrincipal()],\n      effect: Effect.DENY,\n      conditions:\n            {\n              Bool: {\n                'aws:SecureTransport': 'false'\n              }\n            }\n    })\n  );\n}\n\nexport interface BuildTopicResponse {\n  readonly topic: sns.Topic,\n  readonly key?: kms.Key\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function buildTopic(scope: Construct, id: string, props: BuildTopicProps): BuildTopicResponse {\n  if (!props.existingTopicObj) {\n    // Setup the topic properties\n    const snsTopicProps = consolidateProps(defaultSnsTopicProps, props.topicProps);\n\n    // Set encryption properties\n    if (props.topicProps?.masterKey) {\n      snsTopicProps.masterKey = props.topicProps?.masterKey;\n    } else if (props.encryptionKey) {\n      snsTopicProps.masterKey = props.encryptionKey;\n    } else if (props.encryptionKeyProps || props.enableEncryptionWithCustomerManagedKey === true) {\n      snsTopicProps.masterKey = buildEncryptionKey(scope, id, props.encryptionKeyProps);\n    } else {\n      snsTopicProps.masterKey = kms.Alias.fromAliasName(scope, 'aws-managed-key', 'alias/aws/sns');\n    }\n\n    // Create the SNS Topic\n    // NOSONAR (typescript:S6327) - The masterKey is set in the if statement above, SONAR is\n    // not catching it. Behavior is confirmed in the\n    // 'Test deployment with no properties using AWS Managed KMS Key' unit test\n    const topic: sns.Topic = new sns.Topic(scope, 'SnsTopic', snsTopicProps); // NOSONAR\n\n    applySecureTopicPolicy(topic);\n\n    return { topic, key: snsTopicProps.masterKey };\n  } else {\n    return { topic: props.existingTopicObj, key: props.existingTopicEncryptionKey };\n  }\n}\n\nexport interface SnsProps {\n  readonly topicProps?: sns.TopicProps,\n  readonly existingTopicObj?: sns.Topic,\n  readonly existingTopicObject?: sns.Topic,\n  readonly encryptionKey?: kms.Key,\n  readonly encryptionKeyProps?: kms.KeyProps\n}\n\nexport function CheckSnsProps(propsObject: SnsProps | any) {\n  let errorMessages = '';\n  let errorFound = false;\n\n  // FargateToSns used TopicObject instead of TopicObj - to fix would be a breaking change, so we\n  // must look for both here.\n  if (propsObject.topicProps && (propsObject.existingTopicObj || propsObject.existingTopicObject)) {\n    errorMessages += 'Error - Either provide topicProps or existingTopicObj, but not both.\\n';\n    errorFound = true;\n  }\n\n  if (propsObject.topicProps?.masterKey && propsObject.encryptionKey) {\n    errorMessages += 'Error - Either provide topicProps.masterKey or encryptionKey, but not both.\\n';\n    errorFound = true;\n  }\n\n  if (propsObject.topicProps?.masterKey && propsObject.encryptionKeyProps) {\n    errorMessages += 'Error - Either provide topicProps.masterKey or encryptionKeyProps, but not both.\\n';\n    errorFound = true;\n  }\n\n  if (propsObject.encryptionKey && propsObject.encryptionKeyProps) {\n    errorMessages += 'Error - Either provide encryptionKey or encryptionKeyProps, but not both.\\n';\n    errorFound = true;\n  }\n\n  if (errorFound) {\n    throw new Error(errorMessages);\n  }\n}\n"]}