@aws-solutions-constructs/core
Version:
Core CDK Construct for patterns library
143 lines • 23.4 kB
JavaScript
"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.buildWebSocketQueueApi = buildWebSocketQueueApi;
exports.buildWebSocketApiProps = buildWebSocketApiProps;
exports.buildWebSocketQueueRouteOptions = buildWebSocketQueueRouteOptions;
const cdk = require("aws-cdk-lib");
const apigateway = require("aws-cdk-lib/aws-apigateway");
const apigwv2 = require("aws-cdk-lib/aws-apigatewayv2");
const aws_apigatewayv2_integrations_1 = require("aws-cdk-lib/aws-apigatewayv2-integrations");
const iam = require("aws-cdk-lib/aws-iam");
const cloudwatch_log_group_helper_1 = require("./cloudwatch-log-group-helper");
const utils_1 = require("./utils");
const websocket_api_defaults_1 = require("./websocket-api-defaults");
/**
* Builds an AWS API Gateway WebSocket API integrated with an Amazon SQS queue.
*
* @param scope The construct scope where the resources will be created.
* @param id The unique ID for the resources.
* @param props The configuration properties for the WebSocket API and SQS queue integration.
* @returns
*/
function buildWebSocketQueueApi(scope, id, props) {
// Setup the API Gateway role
const apiGatewayRole = new iam.Role(scope, "LambdaRestApiCloudWatchRole", {
assumedBy: new iam.ServicePrincipal("apigateway.amazonaws.com"),
});
props.queue.grantSendMessages(apiGatewayRole);
const finalProps = (0, utils_1.consolidateProps)(buildWebSocketApiProps(apiGatewayRole, props.queue, props.createDefaultRoute, props.defaultRouteRequestTemplate, props.defaultIamAuthorization), props.webSocketApiProps);
const webSocketApi = buildApiGatewayV2WebSocket(scope, id, {
webSocketApiProps: finalProps,
existingWebSocketApi: props.existingWebSocketApi,
});
if (props.customRouteName) {
webSocketApi.addRoute(props.customRouteName, buildWebSocketQueueRouteOptions(apiGatewayRole, props.queue, props.customRouteName, props.defaultRouteRequestTemplate));
}
if (props.existingWebSocketApi === undefined &&
props.defaultIamAuthorization === false &&
finalProps.connectRouteOptions?.authorizer === undefined) {
(0, utils_1.printWarning)("This construct will create a WebSocket with NO Authorizer (defaultIamAuthorization is set to false).");
}
const webSocketStage = new apigwv2.WebSocketStage(scope, "Stage", {
stageName: "prod",
webSocketApi,
autoDeploy: true,
});
(0, utils_1.addCfnGuardSuppressRules)(webSocketStage, ["API_GW_CACHE_ENABLED_AND_ENCRYPTED"]);
const apiGatewayLogGroup = (0, cloudwatch_log_group_helper_1.buildLogGroup)(scope, "LogGroup", props.logGroupProps);
apiGatewayLogGroup.grant(apiGatewayRole, "logs:CreateLogGroup", "logs:CreateLogStream", "logs:DescribeLogGroups", "logs:DescribeLogStreams", "logs:PutLogEvents", "logs:GetLogEvents", "logs:FilterLogEvents");
const cfnStage = webSocketStage.node.defaultChild;
cfnStage.addPropertyOverride("AccessLogSettings", {
DestinationArn: apiGatewayLogGroup.logGroupArn,
Format: apigateway.AccessLogFormat.clf().toString(),
});
cfnStage.addPropertyOverride("DefaultRouteSettings", {
DataTraceEnabled: false,
DetailedMetricsEnabled: true,
LoggingLevel: "ERROR",
});
(0, utils_1.addCfnSuppressRules)(webSocketStage, [
{
id: "AwsSolutions-APIG1",
reason: "Access logging configuration has been provided as per ApiGateway v2 requirements",
},
]);
(0, utils_1.addCfnSuppressRules)(apiGatewayRole.node.tryFindChild("DefaultPolicy")?.node.tryFindChild("Resource"), [
{
id: "AwsSolutions-IAM5",
reason: "The APIGateway requires permissions to KMS so that it can write to an encrypted SQS queue",
},
]);
return {
webSocketApi,
webSocketStage,
apiGatewayRole,
apiGatewayLogGroup,
};
}
/**
* build ApiGateway v2 WebSocket L2 construct. If existing WebSocketApi instance is provided, it returns that instance,
* if not, it creates a new WebSocketApi using the user provided props.
*
* @param scope
* @param props
* @returns
*/
function buildApiGatewayV2WebSocket(scope, id, props) {
if (props.existingWebSocketApi) {
return props.existingWebSocketApi;
}
else {
return new apigwv2.WebSocketApi(scope, `WebSocketApi${id}`, props.webSocketApiProps);
}
}
/**
* @internal This is an internal core function and should not be called directly by Solutions Constructs clients.
*/
function buildWebSocketApiProps(role, sqsQueue, createDefaultRoute, requestTemplate, defaultIamAuthorization) {
if (createDefaultRoute) {
if (!role || !sqsQueue) {
throw new Error("role and sqs must be provided to create a default route");
}
}
// prettier-ignore
// Sonar exception reason: - typescript:S6571 - required because we are not passing all values. Using partial may cause @jsii to not work.
const websocketApiProps = {
defaultRouteOptions: createDefaultRoute ? buildWebSocketQueueRouteOptions(role, sqsQueue, '$default', requestTemplate) : undefined,
connectRouteOptions: (defaultIamAuthorization === undefined || defaultIamAuthorization === true) ? websocket_api_defaults_1.connectRouteOptions : undefined
};
return websocketApiProps;
}
/**
* @internal This is an internal core function and should not be called directly by Solutions Constructs clients.
*/
function buildWebSocketQueueRouteOptions(role, sqsQueue, routeName, requestTemplate) {
return {
integration: new aws_apigatewayv2_integrations_1.WebSocketAwsIntegration(routeName, {
integrationMethod: apigwv2.HttpMethod.POST,
integrationUri: `arn:${cdk.Aws.PARTITION}:apigateway:${cdk.Aws.REGION}:sqs:path/${cdk.Aws.ACCOUNT_ID}/${sqsQueue.queueName}`,
requestTemplates: requestTemplate ?? {
[routeName === "$default" ? "$default" : routeName]: websocket_api_defaults_1.DEFAULT_ROUTE_QUEUE_VTL_CONFIG,
},
templateSelectionExpression: routeName === "$default" ? "\\$default" : routeName,
passthroughBehavior: apigwv2.PassthroughBehavior.NEVER,
credentialsRole: role,
requestParameters: {
"integration.request.header.Content-Type": "'application/x-www-form-urlencoded'",
},
}),
};
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-api-helper.js","sourceRoot":"","sources":["websocket-api-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\nimport * as cdk from \"aws-cdk-lib\";\nimport * as apigateway from \"aws-cdk-lib/aws-apigateway\";\nimport * as apigwv2 from \"aws-cdk-lib/aws-apigatewayv2\";\nimport { WebSocketAwsIntegration } from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as logs from \"aws-cdk-lib/aws-logs\";\nimport * as sqs from \"aws-cdk-lib/aws-sqs\";\nimport { Construct } from \"constructs\";\nimport { buildLogGroup } from \"./cloudwatch-log-group-helper\";\nimport { addCfnGuardSuppressRules, addCfnSuppressRules, consolidateProps, printWarning } from \"./utils\";\nimport { connectRouteOptions, DEFAULT_ROUTE_QUEUE_VTL_CONFIG } from \"./websocket-api-defaults\";\n\nexport interface BuildWebSocketQueueApiResponse {\n  readonly webSocketApi: apigwv2.WebSocketApi;\n  readonly webSocketStage: apigwv2.WebSocketStage;\n  readonly apiGatewayRole: iam.Role;\n  readonly apiGatewayLogGroup: logs.LogGroup;\n}\n\nexport interface BuildWebSocketApiProps {\n  /**\n   * Existing instance of ApiGateway v2 WebSocket\n   */\n  readonly existingWebSocketApi?: apigwv2.WebSocketApi;\n\n  /**\n   * User provided properties of Apigateway v2 WebSocket\n   */\n  readonly webSocketApiProps?: apigwv2.WebSocketApiProps;\n}\n\nexport interface BuildWebSocketQueueApiRequest {\n  readonly queue: sqs.IQueue;\n  readonly defaultRouteRequestTemplate?: { [contentType: string]: string };\n  readonly createDefaultRoute?: boolean;\n  readonly webSocketApiProps?: apigwv2.WebSocketApiProps;\n  readonly existingWebSocketApi?: apigwv2.WebSocketApi;\n  readonly logGroupProps?: logs.LogGroupProps;\n  readonly defaultIamAuthorization?: boolean;\n  readonly customRouteName?: string;\n}\n/**\n * Builds an AWS API Gateway WebSocket API integrated with an Amazon SQS queue.\n *\n * @param scope The construct scope where the resources will be created.\n * @param id The unique ID for the resources.\n * @param props The configuration properties for the WebSocket API and SQS queue integration.\n * @returns\n */\nexport function buildWebSocketQueueApi(\n  scope: Construct,\n  id: string,\n  props: BuildWebSocketQueueApiRequest\n): BuildWebSocketQueueApiResponse {\n  // Setup the API Gateway role\n  const apiGatewayRole = new iam.Role(scope, \"LambdaRestApiCloudWatchRole\", {\n    assumedBy: new iam.ServicePrincipal(\"apigateway.amazonaws.com\"),\n  });\n  props.queue.grantSendMessages(apiGatewayRole);\n  const finalProps = consolidateProps(\n    buildWebSocketApiProps(\n      apiGatewayRole,\n      props.queue,\n      props.createDefaultRoute,\n      props.defaultRouteRequestTemplate,\n      props.defaultIamAuthorization\n    ),\n    props.webSocketApiProps\n  ) as apigwv2.WebSocketApiProps;\n  const webSocketApi = buildApiGatewayV2WebSocket(scope, id, {\n    webSocketApiProps: finalProps,\n    existingWebSocketApi: props.existingWebSocketApi,\n  });\n\n  if (props.customRouteName) {\n    webSocketApi.addRoute(\n      props.customRouteName,\n      buildWebSocketQueueRouteOptions(apiGatewayRole, props.queue, props.customRouteName, props.defaultRouteRequestTemplate)\n    );\n  }\n\n  if (\n    props.existingWebSocketApi === undefined &&\n    props.defaultIamAuthorization === false &&\n    finalProps.connectRouteOptions?.authorizer === undefined\n  ) {\n    printWarning(\n      \"This construct will create a WebSocket with NO Authorizer (defaultIamAuthorization is set to false).\"\n    );\n  }\n\n  const webSocketStage = new apigwv2.WebSocketStage(scope, \"Stage\", {\n    stageName: \"prod\",\n    webSocketApi,\n    autoDeploy: true,\n  });\n\n  addCfnGuardSuppressRules(webSocketStage, [\"API_GW_CACHE_ENABLED_AND_ENCRYPTED\"]);\n\n  const apiGatewayLogGroup = buildLogGroup(scope, \"LogGroup\", props.logGroupProps);\n  apiGatewayLogGroup.grant(\n    apiGatewayRole,\n    \"logs:CreateLogGroup\",\n    \"logs:CreateLogStream\",\n    \"logs:DescribeLogGroups\",\n    \"logs:DescribeLogStreams\",\n    \"logs:PutLogEvents\",\n    \"logs:GetLogEvents\",\n    \"logs:FilterLogEvents\"\n  );\n\n  const cfnStage: apigwv2.CfnStage = webSocketStage.node.defaultChild as apigwv2.CfnStage;\n  cfnStage.addPropertyOverride(\"AccessLogSettings\", {\n    DestinationArn: apiGatewayLogGroup.logGroupArn,\n    Format: apigateway.AccessLogFormat.clf().toString(),\n  });\n  cfnStage.addPropertyOverride(\"DefaultRouteSettings\", {\n    DataTraceEnabled: false,\n    DetailedMetricsEnabled: true,\n    LoggingLevel: \"ERROR\",\n  });\n\n  addCfnSuppressRules(webSocketStage, [\n    {\n      id: \"AwsSolutions-APIG1\",\n      reason: \"Access logging configuration has been provided as per ApiGateway v2 requirements\",\n    },\n  ]);\n\n  addCfnSuppressRules(\n    apiGatewayRole.node.tryFindChild(\"DefaultPolicy\")?.node.tryFindChild(\"Resource\") as cdk.CfnResource,\n    [\n      {\n        id: \"AwsSolutions-IAM5\",\n        reason: \"The APIGateway requires permissions to KMS so that it can write to an encrypted SQS queue\",\n      },\n    ]\n  );\n\n  return {\n    webSocketApi,\n    webSocketStage,\n    apiGatewayRole,\n    apiGatewayLogGroup,\n  };\n}\n\n/**\n * build ApiGateway v2 WebSocket L2 construct. If existing WebSocketApi instance is provided, it returns that instance,\n * if not, it creates a new WebSocketApi using the user provided props.\n *\n * @param scope\n * @param props\n * @returns\n */\nfunction buildApiGatewayV2WebSocket(scope: Construct, id: string, props: BuildWebSocketApiProps): apigwv2.WebSocketApi {\n  if (props.existingWebSocketApi) {\n    return props.existingWebSocketApi;\n  } else {\n    return new apigwv2.WebSocketApi(scope, `WebSocketApi${id}`, props.webSocketApiProps);\n  }\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function buildWebSocketApiProps(\n  role?: iam.Role,\n  sqsQueue?: sqs.IQueue,\n  createDefaultRoute?: boolean,\n  requestTemplate?: { [contentType: string]: string },\n  defaultIamAuthorization?: boolean\n): apigwv2.WebSocketApiProps {\n  if (createDefaultRoute) {\n    if (!role || !sqsQueue) {\n      throw new Error(\"role and sqs must be provided to create a default route\");\n    }\n  }\n\n  // prettier-ignore\n  // Sonar exception reason: - typescript:S6571 - required because we are not passing all values. Using partial may cause @jsii to not work.\n  const websocketApiProps: apigwv2.WebSocketApiProps = { // NOSONAR\n    defaultRouteOptions: createDefaultRoute ? buildWebSocketQueueRouteOptions(role!, sqsQueue!, '$default', requestTemplate) : undefined,\n    connectRouteOptions: (defaultIamAuthorization === undefined || defaultIamAuthorization === true) ? connectRouteOptions : undefined\n  };\n  return websocketApiProps;\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function buildWebSocketQueueRouteOptions(\n  role: iam.Role,\n  sqsQueue: sqs.IQueue,\n  routeName: string,\n  requestTemplate?: { [contentType: string]: string }\n): apigwv2.WebSocketRouteOptions {\n  return {\n    integration: new WebSocketAwsIntegration(routeName, {\n      integrationMethod: apigwv2.HttpMethod.POST,\n      integrationUri: `arn:${cdk.Aws.PARTITION}:apigateway:${cdk.Aws.REGION}:sqs:path/${cdk.Aws.ACCOUNT_ID}/${sqsQueue.queueName}`,\n      requestTemplates: requestTemplate ?? {\n        [routeName === \"$default\" ? \"$default\" : routeName]: DEFAULT_ROUTE_QUEUE_VTL_CONFIG,\n      },\n      templateSelectionExpression: routeName === \"$default\" ? \"\\\\$default\" : routeName,\n      passthroughBehavior: apigwv2.PassthroughBehavior.NEVER,\n      credentialsRole: role,\n      requestParameters: {\n        \"integration.request.header.Content-Type\": \"'application/x-www-form-urlencoded'\",\n      },\n    }),\n  };\n}\n"]}