@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
294 lines • 48.6 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.TypeSafeWebsocketApi = 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 fs = require("fs");
const path = require("path");
const monorepo_1 = require("../../monorepo");
const pdk_nag_1 = require("../../pdk-nag");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const aws_apigatewayv2_1 = require("aws-cdk-lib/aws-apigatewayv2");
const aws_apigatewayv2_integrations_1 = require("aws-cdk-lib/aws-apigatewayv2-integrations");
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
const custom_resources_1 = require("aws-cdk-lib/custom-resources");
const cdk_nag_1 = require("cdk-nag");
const constructs_1 = require("constructs");
const websocket_schema_1 = require("./prepare-spec-event-handler/websocket-schema");
/**
* A construct for creating a websocket API, based on the provided spec and integrations
*/
class TypeSafeWebsocketApi extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
(0, monorepo_1.addMetric)(scope, "type-safe-websocket-api");
this._props = props;
// Create the WebSocket API
this.api = new aws_apigatewayv2_1.WebSocketApi(this, id, {
...props,
routeSelectionExpression: "$request.body.route",
});
// Add the connect/disconnect routes
this.addRoute("$connect", {
integration: props.connect?.integration ??
new aws_apigatewayv2_integrations_1.WebSocketMockIntegration("ConnectIntegration"),
authorizer: props.authorizer,
});
const disconnectRoute = this.addRoute("$disconnect", {
integration: props.connect?.integration ??
new aws_apigatewayv2_integrations_1.WebSocketMockIntegration("DisconnectIntegration"),
});
cdk_nag_1.NagSuppressions.addResourceSuppressions(disconnectRoute, ["AwsPrototyping-APIGWAuthorization", "AwsSolutions-APIG4"].map((ruleId) => ({
id: ruleId,
reason: `Authorizers only apply to the $connect route`,
})), true);
// Create a default stage
this.defaultStage = new aws_apigatewayv2_1.WebSocketStage(this, "default", {
webSocketApi: this.api,
autoDeploy: true,
stageName: "default",
...props.stageProps,
});
// Enable execution logs by default
this.defaultStage.node.defaultChild.defaultRouteSettings = {
loggingLevel: "INFO",
dataTraceEnabled: false,
};
// Enable access logging by default
if (!props.disableAccessLogging) {
const logGroup = new aws_logs_1.LogGroup(this, `AccessLogs`);
this.defaultStage.node.defaultChild.accessLogSettings = {
destinationArn: logGroup.logGroupArn,
format: `$context.identity.sourceIp - - [$context.requestTime] "$context.httpMethod $context.routeKey $context.protocol" $context.status $context.responseLength $context.requestId`,
};
}
const lambdaHandlers = [
...Object.values(props.integrations),
props.connect,
props.disconnect,
].flatMap((integration) => integration?.integration instanceof aws_apigatewayv2_integrations_1.WebSocketLambdaIntegration &&
integration.integration.handler?.grantPrincipal
? [integration.integration.handler]
: []);
const stack = aws_cdk_lib_1.Stack.of(this);
// By default, grant lambda handlers access to the management api
if (!props.disableGrantManagementAccessToLambdas) {
lambdaHandlers.forEach((fn) => {
this.defaultStage.grantManagementApiAccess(fn);
cdk_nag_1.NagSuppressions.addResourceSuppressions(fn, ["AwsPrototyping-IAMNoWildcardPermissions", "AwsSolutions-IAM5"].map((ruleId) => ({
id: ruleId,
reason: "WebSocket handlers are granted permissions to manage arbitrary connections",
appliesTo: [
{
regex: `/^Resource::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:execute-api:${pdk_nag_1.PDKNag.getStackRegionRegex(stack)}:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:.*\\/${this.defaultStage.stageName}\\/\\*\\/@connections\\/\\*$/g`,
},
],
})), true);
});
}
// Where the same function is used for multiple routes, grant permission for API gateway to invoke
// the lambda for all routes
const uniqueLambdaHandlers = new Set();
const duplicateLambdaHandlers = new Set();
lambdaHandlers.forEach((fn) => {
if (uniqueLambdaHandlers.has(fn)) {
duplicateLambdaHandlers.add(fn);
}
uniqueLambdaHandlers.add(fn);
});
[...duplicateLambdaHandlers].forEach((fn, i) => {
new aws_lambda_1.CfnPermission(this, `GrantRouteInvoke${i}`, {
action: "lambda:InvokeFunction",
principal: "apigateway.amazonaws.com",
functionName: fn.functionArn,
sourceArn: stack.formatArn({
service: "execute-api",
resource: this.api.apiId,
resourceName: "*",
}),
});
});
// Read and parse the spec
const spec = JSON.parse(fs.readFileSync(props.specPath, "utf-8"));
// Map of route key to paths
const serverOperationPaths = Object.fromEntries(Object.values(props.operationLookup).map((details) => [
details.path.replace(/\//g, ""),
details.path,
]));
// Locally check that we can extract the schema for every operation
const schemas = (0, websocket_schema_1.extractWebSocketSchemas)(Object.keys(serverOperationPaths), serverOperationPaths, spec);
// Check that every operation has an integration
const missingIntegrations = Object.keys(props.operationLookup).filter((operationId) => !props.integrations[operationId]);
if (missingIntegrations.length > 0) {
throw new Error(`Missing integrations for operations ${missingIntegrations.join(", ")}`);
}
// Create an asset for the spec, which we'll read from the custom resource
const inputSpec = new aws_s3_assets_1.Asset(this, "InputSpec", {
path: props.specPath,
});
// Function for managing schemas/models associated with routes
const schemaHandler = new aws_lambda_1.Function(this, "SchemaHandler", {
handler: "websocket-schema-handler.handler",
runtime: aws_lambda_1.Runtime.NODEJS_20_X,
code: aws_lambda_1.Code.fromAsset(path.join(__dirname, "./prepare-spec-event-handler")),
timeout: aws_cdk_lib_1.Duration.minutes(1),
});
cdk_nag_1.NagSuppressions.addResourceSuppressions(schemaHandler, ["AwsPrototyping-IAMNoManagedPolicies", "AwsSolutions-IAM4"].map((ruleId) => ({
id: ruleId,
reason: `AWSLambdaBasicExecutionRole grants minimal permissions required for lambda execution`,
})), true);
schemaHandler.addToRolePolicy(new aws_iam_1.PolicyStatement({
actions: ["s3:GetObject"],
resources: [inputSpec.bucket.arnForObjects(inputSpec.s3ObjectKey)],
}));
schemaHandler.addToRolePolicy(new aws_iam_1.PolicyStatement({
actions: [
"apigateway:DELETE",
"apigateway:PATCH",
"apigateway:POST",
"apigateway:GET",
],
resources: [
stack.formatArn({
service: "apigateway",
account: "",
resource: `/apis/${this.api.apiId}/models`,
}),
stack.formatArn({
service: "apigateway",
account: "",
resource: `/apis/${this.api.apiId}/models/*`,
}),
],
}));
schemaHandler.addToRolePolicy(new aws_iam_1.PolicyStatement({
actions: ["apigateway:PATCH", "apigateway:GET"],
resources: [
stack.formatArn({
service: "apigateway",
account: "",
resource: `/apis/${this.api.apiId}/routes`,
}),
stack.formatArn({
service: "apigateway",
account: "",
resource: `/apis/${this.api.apiId}/routes/*`,
}),
],
}));
cdk_nag_1.NagSuppressions.addResourceSuppressions(schemaHandler, ["AwsPrototyping-IAMNoWildcardPermissions", "AwsSolutions-IAM5"].map((ruleId) => ({
id: ruleId,
reason: `Schema custom resource manages all routes and models`,
})), true);
const providerRole = new aws_iam_1.Role(this, "PrepareSpecProviderRole", {
assumedBy: new aws_iam_1.ServicePrincipal("lambda.amazonaws.com"),
inlinePolicies: {
logs: new aws_iam_1.PolicyDocument({
statements: [
new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: [
"logs:CreateLogGroup",
"logs:CreateLogStream",
"logs:PutLogEvents",
],
resources: [
`arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/*`,
],
}),
],
}),
},
});
const provider = new custom_resources_1.Provider(this, "SchemaProvider", {
onEventHandler: schemaHandler,
role: providerRole,
});
cdk_nag_1.NagSuppressions.addResourceSuppressions(providerRole, ["AwsPrototyping-IAMNoWildcardPermissions", "AwsSolutions-IAM5"].map((ruleId) => ({
id: ruleId,
reason: `Custom resource provider may invoke arbitrary lambda versions`,
})), true);
cdk_nag_1.NagSuppressions.addResourceSuppressions(provider, ["AwsPrototyping-LambdaLatestVersion", "AwsSolutions-L1"].map((ruleId) => ({
id: ruleId,
reason: `Provider framework lambda is managed by CDK`,
})), true);
const schemaCustomResourceProperties = {
apiId: this.api.apiId,
inputSpecLocation: {
bucket: inputSpec.s3BucketName,
key: inputSpec.s3ObjectKey,
},
serverOperationPaths,
};
const schemaCustomResource = new aws_cdk_lib_1.CustomResource(this, "SchemaCustomResource", {
serviceToken: provider.serviceToken,
properties: schemaCustomResourceProperties,
});
// Add a route for every integration
Object.entries(props.integrations).forEach(([operationId, integration]) => {
const op = props.operationLookup[operationId];
if (!op) {
throw new Error(`Integration not found in operation lookup for operation ${operationId}`);
}
// Add the route
const routeKey = op.path.replace(/\//g, "");
const route = this.addRoute(routeKey, {
integration: integration.integration,
});
cdk_nag_1.NagSuppressions.addResourceSuppressions(route, ["AwsPrototyping-APIGWAuthorization", "AwsSolutions-APIG4"].map((ruleId) => ({
id: ruleId,
reason: `Authorizers only apply to the $connect route`,
})), true);
// Associate the route with its corresponding schema (which is created by the custom resource)
if (schemas[routeKey]) {
route.node.defaultChild.requestModels = {
model: routeKey,
};
route.node.defaultChild.modelSelectionExpression =
"model";
}
route.node.addDependency(schemaCustomResource);
});
}
/**
* Add a route to the websocket api
*/
addRoute(routeKey, options) {
// Unless disableMockIntegrationResponses is true, we automatically configure the integration requests and responses
// required to successfully mock the route, when the integration is a mock integration
const shouldAddMockResponse = !this._props.disableMockIntegrationResponses &&
options.integration instanceof aws_apigatewayv2_integrations_1.WebSocketMockIntegration;
const route = this.api.addRoute(routeKey, {
...options,
returnResponse: shouldAddMockResponse,
});
if (shouldAddMockResponse &&
options.integration?.integration
?.integrationId) {
const integration = options.integration
?.integration;
integration.node.defaultChild.requestTemplates = {
"application/json": '{"statusCode":200}',
};
new aws_apigatewayv2_1.CfnIntegrationResponse(this, `${routeKey}IntegRes`, {
apiId: this.api.apiId,
integrationId: integration.integrationId,
integrationResponseKey: "/2\\d\\d/",
templateSelectionExpression: "/2\\d\\d/",
responseTemplates: {
"200": '{"statusCode":200}',
},
});
}
return route;
}
}
exports.TypeSafeWebsocketApi = TypeSafeWebsocketApi;
_a = JSII_RTTI_SYMBOL_1;
TypeSafeWebsocketApi[_a] = { fqn: "@aws/pdk.type_safe_api.TypeSafeWebsocketApi", version: "0.26.14" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"type-safe-websocket-api.js","sourceRoot":"","sources":["type-safe-websocket-api.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 * as fs from \"fs\";\nimport * as path from \"path\";\nimport { addMetric } from \"@aws/monorepo\";\nimport { PDKNag } from \"@aws/pdk-nag\";\nimport { CustomResource, Duration, Stack } from \"aws-cdk-lib\";\nimport {\n  CfnIntegration,\n  CfnIntegrationResponse,\n  CfnRoute,\n  CfnStage,\n  IWebSocketRouteAuthorizer,\n  WebSocketApi,\n  WebSocketIntegration,\n  WebSocketRouteIntegration,\n  WebSocketRouteOptions,\n  WebSocketStage,\n} from \"aws-cdk-lib/aws-apigatewayv2\";\nimport {\n  WebSocketLambdaIntegration,\n  WebSocketMockIntegration,\n} from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport {\n  Effect,\n  PolicyDocument,\n  PolicyStatement,\n  Role,\n  ServicePrincipal,\n} from \"aws-cdk-lib/aws-iam\";\nimport {\n  CfnPermission,\n  Code,\n  IFunction,\n  Function as LambdaFunction,\n  Runtime,\n} from \"aws-cdk-lib/aws-lambda\";\nimport { LogGroup } from \"aws-cdk-lib/aws-logs\";\nimport { Asset } from \"aws-cdk-lib/aws-s3-assets\";\nimport { Provider } from \"aws-cdk-lib/custom-resources\";\nimport { NagSuppressions } from \"cdk-nag\";\nimport { Construct } from \"constructs\";\nimport { OpenAPIV3 } from \"openapi-types\";\nimport { extractWebSocketSchemas } from \"./prepare-spec-event-handler/websocket-schema\";\nimport { WebSocketSchemaResourceProperties } from \"./prepare-spec-event-handler/websocket-schema-handler\";\nimport { WebSocketApiProps } from \"./websocket/websocket-api-props\";\nimport { WebSocketStageProps } from \"./websocket/websocket-stage-props\";\n\n/**\n * Represents an integration for a route\n */\nexport interface TypeSafeWebsocketApiIntegration {\n  /**\n   * The integration to service the route\n   */\n  readonly integration: WebSocketRouteIntegration;\n}\n\nexport interface WebsocketOperationDetails {\n  /**\n   * Path in the OpenAPI spec for the operation\n   */\n  readonly path: string;\n}\n\nexport type WebsocketOperationLookup = {\n  [operationId: string]: WebsocketOperationDetails;\n};\n\n/**\n * Properties for a Type Safe WebSocket API\n */\nexport interface TypeSafeWebsocketApiProps extends WebSocketApiProps {\n  /**\n   * Path to the websocket api specification json file\n   */\n  readonly specPath: string;\n  /**\n   * Details about each operation\n   */\n  readonly operationLookup: WebsocketOperationLookup;\n  /**\n   * WebSocket routes and their corresponding integrations\n   */\n  readonly integrations: {\n    [operationId: string]: TypeSafeWebsocketApiIntegration;\n  };\n  /**\n   * Integration for the $connect route (invoked when a new client connects)\n   * @default mocked\n   */\n  readonly connect?: TypeSafeWebsocketApiIntegration;\n  /**\n   * Integration for the $disconnect route (invoked when a client disconnects)\n   * @default mocked\n   */\n  readonly disconnect?: TypeSafeWebsocketApiIntegration;\n  /**\n   * Authorizer to use for the API (applied to the $connect route)\n   * @default NONE\n   */\n  readonly authorizer?: IWebSocketRouteAuthorizer;\n  /**\n   * Options for the default stage\n   */\n  readonly stageProps?: WebSocketStageProps;\n  /**\n   * By default, all lambda integrations are granted management API access for the websocket API to send messages, disconnect clients, etc.\n   * Set to true if you would like to manage these permissions manually.\n   * @default false\n   */\n  readonly disableGrantManagementAccessToLambdas?: boolean;\n  /**\n   * By default, all mock integrations will automatically be configured with integration responses such that the integration is considered\n   * successful. Set to true to disable this (mock integrations will respond with errors)\n   * @default false\n   */\n  readonly disableMockIntegrationResponses?: boolean;\n  /**\n   * Disable access logging\n   * @default false\n   */\n  readonly disableAccessLogging?: boolean;\n}\n\n/**\n * A construct for creating a websocket API, based on the provided spec and integrations\n */\nexport class TypeSafeWebsocketApi extends Construct {\n  /**\n   * Reference to the websocket API\n   */\n  public readonly api: WebSocketApi;\n  /**\n   * Reference to the default deploy stage\n   */\n  public readonly defaultStage: WebSocketStage;\n\n  private readonly _props: TypeSafeWebsocketApiProps;\n\n  constructor(scope: Construct, id: string, props: TypeSafeWebsocketApiProps) {\n    super(scope, id);\n\n    addMetric(scope, \"type-safe-websocket-api\");\n\n    this._props = props;\n\n    // Create the WebSocket API\n    this.api = new WebSocketApi(this, id, {\n      ...props,\n      routeSelectionExpression: \"$request.body.route\",\n    });\n\n    // Add the connect/disconnect routes\n    this.addRoute(\"$connect\", {\n      integration:\n        props.connect?.integration ??\n        new WebSocketMockIntegration(\"ConnectIntegration\"),\n      authorizer: props.authorizer,\n    });\n    const disconnectRoute = this.addRoute(\"$disconnect\", {\n      integration:\n        props.connect?.integration ??\n        new WebSocketMockIntegration(\"DisconnectIntegration\"),\n    });\n    NagSuppressions.addResourceSuppressions(\n      disconnectRoute,\n      [\"AwsPrototyping-APIGWAuthorization\", \"AwsSolutions-APIG4\"].map(\n        (ruleId) => ({\n          id: ruleId,\n          reason: `Authorizers only apply to the $connect route`,\n        })\n      ),\n      true\n    );\n\n    // Create a default stage\n    this.defaultStage = new WebSocketStage(this, \"default\", {\n      webSocketApi: this.api,\n      autoDeploy: true,\n      stageName: \"default\",\n      ...props.stageProps,\n    });\n\n    // Enable execution logs by default\n    (this.defaultStage.node.defaultChild as CfnStage).defaultRouteSettings = {\n      loggingLevel: \"INFO\",\n      dataTraceEnabled: false,\n    };\n\n    // Enable access logging by default\n    if (!props.disableAccessLogging) {\n      const logGroup = new LogGroup(this, `AccessLogs`);\n      (this.defaultStage.node.defaultChild as CfnStage).accessLogSettings = {\n        destinationArn: logGroup.logGroupArn,\n        format: `$context.identity.sourceIp - - [$context.requestTime] \"$context.httpMethod $context.routeKey $context.protocol\" $context.status $context.responseLength $context.requestId`,\n      };\n    }\n\n    const lambdaHandlers: IFunction[] = [\n      ...Object.values(props.integrations),\n      props.connect,\n      props.disconnect,\n    ].flatMap((integration) =>\n      integration?.integration instanceof WebSocketLambdaIntegration &&\n      (integration.integration as any).handler?.grantPrincipal\n        ? [(integration.integration as any).handler]\n        : []\n    );\n\n    const stack = Stack.of(this);\n\n    // By default, grant lambda handlers access to the management api\n    if (!props.disableGrantManagementAccessToLambdas) {\n      lambdaHandlers.forEach((fn) => {\n        this.defaultStage.grantManagementApiAccess(fn);\n        NagSuppressions.addResourceSuppressions(\n          fn,\n          [\"AwsPrototyping-IAMNoWildcardPermissions\", \"AwsSolutions-IAM5\"].map(\n            (ruleId) => ({\n              id: ruleId,\n              reason:\n                \"WebSocket handlers are granted permissions to manage arbitrary connections\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:execute-api:${PDKNag.getStackRegionRegex(\n                    stack\n                  )}:${PDKNag.getStackAccountRegex(stack)}:.*\\\\/${\n                    this.defaultStage.stageName\n                  }\\\\/\\\\*\\\\/@connections\\\\/\\\\*$/g`,\n                },\n              ],\n            })\n          ),\n          true\n        );\n      });\n    }\n\n    // Where the same function is used for multiple routes, grant permission for API gateway to invoke\n    // the lambda for all routes\n    const uniqueLambdaHandlers = new Set<IFunction>();\n    const duplicateLambdaHandlers = new Set<IFunction>();\n    lambdaHandlers.forEach((fn) => {\n      if (uniqueLambdaHandlers.has(fn)) {\n        duplicateLambdaHandlers.add(fn);\n      }\n      uniqueLambdaHandlers.add(fn);\n    });\n    [...duplicateLambdaHandlers].forEach((fn, i) => {\n      new CfnPermission(this, `GrantRouteInvoke${i}`, {\n        action: \"lambda:InvokeFunction\",\n        principal: \"apigateway.amazonaws.com\",\n        functionName: fn.functionArn,\n        sourceArn: stack.formatArn({\n          service: \"execute-api\",\n          resource: this.api.apiId,\n          resourceName: \"*\",\n        }),\n      });\n    });\n\n    // Read and parse the spec\n    const spec = JSON.parse(\n      fs.readFileSync(props.specPath, \"utf-8\")\n    ) as OpenAPIV3.Document;\n\n    // Map of route key to paths\n    const serverOperationPaths = Object.fromEntries(\n      Object.values(props.operationLookup).map((details) => [\n        details.path.replace(/\\//g, \"\"),\n        details.path,\n      ])\n    );\n\n    // Locally check that we can extract the schema for every operation\n    const schemas = extractWebSocketSchemas(\n      Object.keys(serverOperationPaths),\n      serverOperationPaths,\n      spec\n    );\n\n    // Check that every operation has an integration\n    const missingIntegrations = Object.keys(props.operationLookup).filter(\n      (operationId) => !props.integrations[operationId]\n    );\n    if (missingIntegrations.length > 0) {\n      throw new Error(\n        `Missing integrations for operations ${missingIntegrations.join(\", \")}`\n      );\n    }\n\n    // Create an asset for the spec, which we'll read from the custom resource\n    const inputSpec = new Asset(this, \"InputSpec\", {\n      path: props.specPath,\n    });\n\n    // Function for managing schemas/models associated with routes\n    const schemaHandler = new LambdaFunction(this, \"SchemaHandler\", {\n      handler: \"websocket-schema-handler.handler\",\n      runtime: Runtime.NODEJS_20_X,\n      code: Code.fromAsset(\n        path.join(__dirname, \"./prepare-spec-event-handler\")\n      ),\n      timeout: Duration.minutes(1),\n    });\n\n    NagSuppressions.addResourceSuppressions(\n      schemaHandler,\n      [\"AwsPrototyping-IAMNoManagedPolicies\", \"AwsSolutions-IAM4\"].map(\n        (ruleId) => ({\n          id: ruleId,\n          reason: `AWSLambdaBasicExecutionRole grants minimal permissions required for lambda execution`,\n        })\n      ),\n      true\n    );\n\n    schemaHandler.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\"s3:GetObject\"],\n        resources: [inputSpec.bucket.arnForObjects(inputSpec.s3ObjectKey)],\n      })\n    );\n\n    schemaHandler.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\n          \"apigateway:DELETE\",\n          \"apigateway:PATCH\",\n          \"apigateway:POST\",\n          \"apigateway:GET\",\n        ],\n        resources: [\n          stack.formatArn({\n            service: \"apigateway\",\n            account: \"\",\n            resource: `/apis/${this.api.apiId}/models`,\n          }),\n          stack.formatArn({\n            service: \"apigateway\",\n            account: \"\",\n            resource: `/apis/${this.api.apiId}/models/*`,\n          }),\n        ],\n      })\n    );\n    schemaHandler.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\"apigateway:PATCH\", \"apigateway:GET\"],\n        resources: [\n          stack.formatArn({\n            service: \"apigateway\",\n            account: \"\",\n            resource: `/apis/${this.api.apiId}/routes`,\n          }),\n          stack.formatArn({\n            service: \"apigateway\",\n            account: \"\",\n            resource: `/apis/${this.api.apiId}/routes/*`,\n          }),\n        ],\n      })\n    );\n\n    NagSuppressions.addResourceSuppressions(\n      schemaHandler,\n      [\"AwsPrototyping-IAMNoWildcardPermissions\", \"AwsSolutions-IAM5\"].map(\n        (ruleId) => ({\n          id: ruleId,\n          reason: `Schema custom resource manages all routes and models`,\n        })\n      ),\n      true\n    );\n\n    const providerRole = new Role(this, \"PrepareSpecProviderRole\", {\n      assumedBy: new ServicePrincipal(\"lambda.amazonaws.com\"),\n      inlinePolicies: {\n        logs: new PolicyDocument({\n          statements: [\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: [\n                \"logs:CreateLogGroup\",\n                \"logs:CreateLogStream\",\n                \"logs:PutLogEvents\",\n              ],\n              resources: [\n                `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/*`,\n              ],\n            }),\n          ],\n        }),\n      },\n    });\n\n    const provider = new Provider(this, \"SchemaProvider\", {\n      onEventHandler: schemaHandler,\n      role: providerRole,\n    });\n\n    NagSuppressions.addResourceSuppressions(\n      providerRole,\n      [\"AwsPrototyping-IAMNoWildcardPermissions\", \"AwsSolutions-IAM5\"].map(\n        (ruleId) => ({\n          id: ruleId,\n          reason: `Custom resource provider may invoke arbitrary lambda versions`,\n        })\n      ),\n      true\n    );\n    NagSuppressions.addResourceSuppressions(\n      provider,\n      [\"AwsPrototyping-LambdaLatestVersion\", \"AwsSolutions-L1\"].map(\n        (ruleId) => ({\n          id: ruleId,\n          reason: `Provider framework lambda is managed by CDK`,\n        })\n      ),\n      true\n    );\n\n    const schemaCustomResourceProperties: WebSocketSchemaResourceProperties = {\n      apiId: this.api.apiId,\n      inputSpecLocation: {\n        bucket: inputSpec.s3BucketName,\n        key: inputSpec.s3ObjectKey,\n      },\n      serverOperationPaths,\n    };\n\n    const schemaCustomResource = new CustomResource(\n      this,\n      \"SchemaCustomResource\",\n      {\n        serviceToken: provider.serviceToken,\n        properties: schemaCustomResourceProperties,\n      }\n    );\n\n    // Add a route for every integration\n    Object.entries(props.integrations).forEach(([operationId, integration]) => {\n      const op = props.operationLookup[operationId];\n      if (!op) {\n        throw new Error(\n          `Integration not found in operation lookup for operation ${operationId}`\n        );\n      }\n\n      // Add the route\n      const routeKey = op.path.replace(/\\//g, \"\");\n      const route = this.addRoute(routeKey, {\n        integration: integration.integration,\n      });\n      NagSuppressions.addResourceSuppressions(\n        route,\n        [\"AwsPrototyping-APIGWAuthorization\", \"AwsSolutions-APIG4\"].map(\n          (ruleId) => ({\n            id: ruleId,\n            reason: `Authorizers only apply to the $connect route`,\n          })\n        ),\n        true\n      );\n\n      // Associate the route with its corresponding schema (which is created by the custom resource)\n      if (schemas[routeKey]) {\n        (route.node.defaultChild as CfnRoute).requestModels = {\n          model: routeKey,\n        };\n        (route.node.defaultChild as CfnRoute).modelSelectionExpression =\n          \"model\";\n      }\n      route.node.addDependency(schemaCustomResource);\n    });\n  }\n\n  /**\n   * Add a route to the websocket api\n   */\n  private addRoute(routeKey: string, options: WebSocketRouteOptions) {\n    // Unless disableMockIntegrationResponses is true, we automatically configure the integration requests and responses\n    // required to successfully mock the route, when the integration is a mock integration\n    const shouldAddMockResponse =\n      !this._props.disableMockIntegrationResponses &&\n      options.integration instanceof WebSocketMockIntegration;\n\n    const route = this.api.addRoute(routeKey, {\n      ...options,\n      returnResponse: shouldAddMockResponse,\n    });\n\n    if (\n      shouldAddMockResponse &&\n      ((options.integration as any)?.integration as WebSocketIntegration)\n        ?.integrationId\n    ) {\n      const integration = (options.integration as any)\n        ?.integration as WebSocketIntegration;\n      (integration.node.defaultChild as CfnIntegration).requestTemplates = {\n        \"application/json\": '{\"statusCode\":200}',\n      };\n\n      new CfnIntegrationResponse(this, `${routeKey}IntegRes`, {\n        apiId: this.api.apiId,\n        integrationId: integration.integrationId,\n        integrationResponseKey: \"/2\\\\d\\\\d/\",\n        templateSelectionExpression: \"/2\\\\d\\\\d/\",\n        responseTemplates: {\n          \"200\": '{\"statusCode\":200}',\n        },\n      });\n    }\n\n    return route;\n  }\n}\n"]}