@aws-solutions-constructs/core
Version:
Core CDK Construct for patterns library
138 lines • 20.1 kB
JavaScript
/**
* 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"]}
;