UNPKG

@aws/pdk

Version:

All documentation is located at: https://aws.github.io/aws-pdk

291 lines 48.7 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.TypeSafeRestApi = 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_apigateway_1 = require("aws-cdk-lib/aws-apigateway"); 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 prepare_spec_1 = require("./prepare-spec-event-handler/prepare-spec"); const api_gateway_auth_1 = require("./spec/api-gateway-auth"); const api_gateway_integrations_1 = require("./spec/api-gateway-integrations"); const open_api_gateway_web_acl_1 = require("./waf/open-api-gateway-web-acl"); /** * A construct for creating an api gateway rest api based on the definition in the OpenAPI spec. */ class TypeSafeRestApi extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); (0, monorepo_1.addMetric)(scope, "type-safe-rest-api"); const { integrations, specPath, operationLookup, defaultAuthorizer, corsOptions, outputSpecBucket, ...options } = props; // Upload the spec to s3 as an asset const inputSpecAsset = new aws_s3_assets_1.Asset(this, "InputSpec", { path: specPath, }); const prepareSpecOutputBucket = outputSpecBucket ?? inputSpecAsset.bucket; // We'll output the prepared spec in the same asset bucket const preparedSpecOutputKeyPrefix = `${inputSpecAsset.s3ObjectKey}-prepared`; const stack = aws_cdk_lib_1.Stack.of(this); // Lambda name prefix is truncated to 48 characters (16 below the max of 64) const lambdaNamePrefix = `${pdk_nag_1.PDKNag.getStackPrefix(stack) .split("/") .join("-") .slice(0, 40)}${this.node.addr.slice(-8).toUpperCase()}`; const prepareSpecLambdaName = `${lambdaNamePrefix}PrepSpec`; const prepareSpecRole = new aws_iam_1.Role(this, "PrepareSpecRole", { 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/${prepareSpecLambdaName}`, `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${prepareSpecLambdaName}:*`, ], }), ], }), s3: new aws_iam_1.PolicyDocument({ statements: [ new aws_iam_1.PolicyStatement({ effect: aws_iam_1.Effect.ALLOW, actions: ["s3:getObject"], resources: [ inputSpecAsset.bucket.arnForObjects(inputSpecAsset.s3ObjectKey), ], }), new aws_iam_1.PolicyStatement({ effect: aws_iam_1.Effect.ALLOW, actions: ["s3:putObject"], resources: [ // The output file will include a hash of the prepared spec, which is not known until deploy time since // tokens must be resolved prepareSpecOutputBucket.arnForObjects(`${preparedSpecOutputKeyPrefix}/*`), ], }), ], }), }, }); ["AwsSolutions-IAM5", "AwsPrototyping-IAMNoWildcardPermissions"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(prepareSpecRole, [ { id: RuleId, reason: "Cloudwatch resources have been scoped down to the LogGroup level, however * is still needed as stream names are created just in time.", appliesTo: [ { regex: `/^Resource::arn:aws:logs:${pdk_nag_1.PDKNag.getStackRegionRegex(stack)}:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:log-group:/aws/lambda/${prepareSpecLambdaName}:\*/g`, }, ], }, { id: RuleId, reason: "S3 resources have been scoped down to the appropriate prefix in the CDK asset bucket, however * is still needed as since the prepared spec hash is not known until deploy time.", appliesTo: [ { regex: `/^Resource::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:s3:.*/${preparedSpecOutputKeyPrefix}/\*/g`, }, ], }, ], true); }); // Create a custom resource for preparing the spec for deployment (adding integrations, authorizers, etc) const prepareSpec = new aws_lambda_1.Function(this, "PrepareSpecHandler", { handler: "index.handler", runtime: aws_lambda_1.Runtime.NODEJS_18_X, code: aws_lambda_1.Code.fromAsset(path.join(__dirname, "./prepare-spec-event-handler")), timeout: aws_cdk_lib_1.Duration.seconds(30), role: prepareSpecRole, functionName: prepareSpecLambdaName, }); const providerFunctionName = `${lambdaNamePrefix}PrepSpecProvider`; 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/${providerFunctionName}`, `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${providerFunctionName}:*`, ], }), ], }), }, }); const provider = new custom_resources_1.Provider(this, "PrepareSpecProvider", { onEventHandler: prepareSpec, role: providerRole, providerFunctionName, }); ["AwsSolutions-IAM5", "AwsPrototyping-IAMNoWildcardPermissions"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(providerRole, [ { id: RuleId, reason: "Cloudwatch resources have been scoped down to the LogGroup level, however * is still needed as stream names are created just in time.", }, ], true); }); ["AwsSolutions-L1", "AwsPrototyping-LambdaLatestVersion"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(provider, [ { id: RuleId, reason: "Latest runtime cannot be configured. CDK will need to upgrade the Provider construct accordingly.", }, ], true); }); const serializedCorsOptions = corsOptions && { allowHeaders: corsOptions.allowHeaders || [ ...aws_apigateway_1.Cors.DEFAULT_HEADERS, "x-amz-content-sha256", ], allowMethods: corsOptions.allowMethods || aws_apigateway_1.Cors.ALL_METHODS, allowOrigins: corsOptions.allowOrigins, statusCode: corsOptions.statusCode || 204, }; const prepareSpecOptions = { defaultAuthorizerReference: (0, api_gateway_auth_1.serializeAsAuthorizerReference)(defaultAuthorizer), integrations: Object.fromEntries(Object.entries(integrations).map(([operationId, integration]) => [ operationId, { integration: integration.integration.render({ operationId, scope: this, ...operationLookup[operationId], corsOptions: serializedCorsOptions, operationLookup, }), methodAuthorizer: (0, api_gateway_auth_1.serializeAsAuthorizerReference)(integration.authorizer), options: integration.options, }, ])), securitySchemes: (0, api_gateway_auth_1.prepareSecuritySchemes)(this, integrations, defaultAuthorizer, options.apiKeyOptions), corsOptions: serializedCorsOptions, operationLookup, apiKeyOptions: options.apiKeyOptions, gatewayResponses: options.gatewayResponses, }; // Spec preparation will happen in a custom resource lambda so that references to lambda integrations etc can be // resolved. However, we also prepare inline to perform some additional validation at synth time. const spec = JSON.parse(fs.readFileSync(specPath, "utf-8")); this.extendedApiSpecification = (0, prepare_spec_1.prepareApiSpec)(spec, prepareSpecOptions); const prepareApiSpecCustomResourceProperties = { inputSpecLocation: { bucket: inputSpecAsset.bucket.bucketName, key: inputSpecAsset.s3ObjectKey, }, outputSpecLocation: { bucket: prepareSpecOutputBucket.bucketName, key: preparedSpecOutputKeyPrefix, }, ...prepareSpecOptions, }; const prepareSpecCustomResource = new aws_cdk_lib_1.CustomResource(this, "PrepareSpecCustomResource", { serviceToken: provider.serviceToken, properties: { options: prepareApiSpecCustomResourceProperties, }, }); // Create the api gateway resources from the spec, augmenting the spec with the properties specific to api gateway // such as integrations or auth types this.api = new aws_apigateway_1.SpecRestApi(this, id, { apiDefinition: this.node.tryGetContext("type-safe-api-local") ? aws_apigateway_1.ApiDefinition.fromInline(this.extendedApiSpecification) : aws_apigateway_1.ApiDefinition.fromBucket(prepareSpecOutputBucket, prepareSpecCustomResource.getAttString("outputSpecKey")), deployOptions: { accessLogDestination: new aws_apigateway_1.LogGroupLogDestination(new aws_logs_1.LogGroup(this, `AccessLogs`)), accessLogFormat: aws_apigateway_1.AccessLogFormat.clf(), loggingLevel: aws_apigateway_1.MethodLoggingLevel.INFO, }, ...options, }); this.api.node.addDependency(prepareSpecCustomResource); // While the api will be updated when the output path from the custom resource changes, CDK still needs to know when // to redeploy the api. This is achieved by including a hash of the spec in the logical id (internalised in the // addToLogicalId method since this is how changes of individual resources/methods etc trigger redeployments in CDK) this.api.latestDeployment?.addToLogicalId(this.extendedApiSpecification); // Grant API Gateway permission to invoke the integrations Object.keys(integrations).forEach((operationId) => { integrations[operationId].integration.grant({ operationId, scope: this, api: this.api, ...operationLookup[operationId], operationLookup, }); }); // Grant API Gateway permission to invoke each custom authorizer lambda (if any) (0, api_gateway_integrations_1.getAuthorizerFunctions)(props).forEach(({ label, function: lambda }) => { new aws_lambda_1.CfnPermission(this, `LambdaPermission-${label}`, { action: "lambda:InvokeFunction", principal: "apigateway.amazonaws.com", functionName: lambda.functionArn, sourceArn: stack.formatArn({ service: "execute-api", resource: this.api.restApiId, resourceName: "*/*", }), }); }); // Create and associate the web acl if not disabled if (!props.webAclOptions?.disable) { const acl = new open_api_gateway_web_acl_1.OpenApiGatewayWebAcl(this, `${id}-Acl`, { ...props.webAclOptions, apiDeploymentStageArn: this.api.deploymentStage.stageArn, }); this.webAcl = acl.webAcl; this.ipSet = acl.ipSet; this.webAclAssociation = acl.webAclAssociation; } ["AwsSolutions-IAM4", "AwsPrototyping-IAMNoManagedPolicies"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(this, [ { id: RuleId, reason: "Cloudwatch Role requires access to create/read groups at the root level.", appliesTo: [ { regex: `/^Policy::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:iam::aws:policy/service-role/AmazonAPIGatewayPushToCloudWatchLogs$/g`, }, ], }, ], true); }); ["AwsSolutions-APIG2", "AwsPrototyping-APIGWRequestValidation"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(this, [ { id: RuleId, reason: "This construct implements fine grained validation via OpenApi.", }, ], true); }); } } exports.TypeSafeRestApi = TypeSafeRestApi; _a = JSII_RTTI_SYMBOL_1; TypeSafeRestApi[_a] = { fqn: "@aws/pdk.type_safe_api.TypeSafeRestApi", version: "0.26.14" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"type-safe-rest-api.js","sourceRoot":"","sources":["type-safe-rest-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, Size, Stack } from \"aws-cdk-lib\";\nimport {\n  AccessLogFormat,\n  ApiDefinition,\n  Cors,\n  GatewayResponseOptions,\n  LogGroupLogDestination,\n  MethodLoggingLevel,\n  RestApiBaseProps,\n  SpecRestApi,\n} from \"aws-cdk-lib/aws-apigateway\";\nimport {\n  Effect,\n  PolicyDocument,\n  PolicyStatement,\n  Role,\n  ServicePrincipal,\n} from \"aws-cdk-lib/aws-iam\";\nimport {\n  CfnPermission,\n  Code,\n  Function as LambdaFunction,\n  Runtime,\n} from \"aws-cdk-lib/aws-lambda\";\nimport { LogGroup } from \"aws-cdk-lib/aws-logs\";\nimport { IBucket } from \"aws-cdk-lib/aws-s3\";\nimport { Asset } from \"aws-cdk-lib/aws-s3-assets\";\nimport {\n  CfnIPSet,\n  CfnWebACL,\n  CfnWebACLAssociation,\n} from \"aws-cdk-lib/aws-wafv2\";\nimport { Provider } from \"aws-cdk-lib/custom-resources\";\nimport { NagSuppressions } from \"cdk-nag\";\nimport { Construct } from \"constructs\";\nimport { PrepareApiSpecCustomResourceProperties } from \"./prepare-spec-event-handler\";\nimport {\n  prepareApiSpec,\n  PrepareApiSpecOptions,\n} from \"./prepare-spec-event-handler/prepare-spec\";\nimport { SerializedCorsOptions, TypeSafeApiOptions } from \"./spec\";\nimport {\n  prepareSecuritySchemes,\n  serializeAsAuthorizerReference,\n} from \"./spec/api-gateway-auth\";\nimport { getAuthorizerFunctions } from \"./spec/api-gateway-integrations\";\nimport { OpenApiGatewayWebAcl } from \"./waf/open-api-gateway-web-acl\";\nimport { TypeSafeApiWebAclOptions } from \"./waf/types\";\n\n/**\n * Configuration for the TypeSafeRestApi construct\n */\nexport interface TypeSafeRestApiProps\n  extends RestApiBaseProps,\n    TypeSafeApiOptions {\n  /**\n   * Path to the JSON open api spec\n   */\n  readonly specPath: string;\n  /**\n   * Options for the AWS WAF v2 WebACL associated with the api. By default, a Web ACL with the AWS default managed\n   * rule set will be associated with the API. These options may disable or override the defaults.\n   */\n  readonly webAclOptions?: TypeSafeApiWebAclOptions;\n\n  /**\n   * A Size(in bytes, kibibytes, mebibytes etc) that is used to enable compression (with non-negative\n   * between 0 and 10485760 (10M) bytes, inclusive) or disable compression\n   * (when undefined) on an API. When compression is enabled, compression or\n   * decompression is not applied on the payload if the payload size is\n   * smaller than this value. Setting it to zero allows compression for any\n   * payload size.\n   *\n   * @default - Compression is disabled.\n   */\n  readonly minCompressionSize?: Size;\n\n  /**\n   * By default, the spec is prepared and outputted into the CDK assets bucket. If this is undesired,\n   * use this option to specify the output bucket.\n   */\n  readonly outputSpecBucket?: IBucket;\n\n  /**\n   * Optional gateway responses for the API.\n   *\n   * Note that Type Safe API automatically configures request validation for you, and defines a\n   * default BAD_REQUEST_BODY gateway response which returns the validation error message. You can\n   * use this property to override this gateway response if desired.\n   *\n   * @see https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-gatewayResponse-definition.html\n   */\n  readonly gatewayResponses?: GatewayResponseOptions[];\n}\n\n/**\n * A construct for creating an api gateway rest api based on the definition in the OpenAPI spec.\n */\nexport class TypeSafeRestApi extends Construct {\n  /**\n   * Underlying API Gateway API construct\n   */\n  public readonly api: SpecRestApi;\n  /**\n   * The OpenAPI specification with applied API gateway extensions\n   */\n  readonly extendedApiSpecification: any;\n  /**\n   * Reference to the webacl, if created\n   */\n  readonly webAcl?: CfnWebACL;\n  /**\n   * Reference to the IP set if created\n   */\n  readonly ipSet?: CfnIPSet;\n  /**\n   * Reference to the web acl association if created\n   */\n  readonly webAclAssociation?: CfnWebACLAssociation;\n\n  constructor(scope: Construct, id: string, props: TypeSafeRestApiProps) {\n    super(scope, id);\n\n    addMetric(scope, \"type-safe-rest-api\");\n\n    const {\n      integrations,\n      specPath,\n      operationLookup,\n      defaultAuthorizer,\n      corsOptions,\n      outputSpecBucket,\n      ...options\n    } = props;\n\n    // Upload the spec to s3 as an asset\n    const inputSpecAsset = new Asset(this, \"InputSpec\", {\n      path: specPath,\n    });\n\n    const prepareSpecOutputBucket = outputSpecBucket ?? inputSpecAsset.bucket;\n    // We'll output the prepared spec in the same asset bucket\n    const preparedSpecOutputKeyPrefix = `${inputSpecAsset.s3ObjectKey}-prepared`;\n\n    const stack = Stack.of(this);\n\n    // Lambda name prefix is truncated to 48 characters (16 below the max of 64)\n    const lambdaNamePrefix = `${PDKNag.getStackPrefix(stack)\n      .split(\"/\")\n      .join(\"-\")\n      .slice(0, 40)}${this.node.addr.slice(-8).toUpperCase()}`;\n    const prepareSpecLambdaName = `${lambdaNamePrefix}PrepSpec`;\n    const prepareSpecRole = new Role(this, \"PrepareSpecRole\", {\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/${prepareSpecLambdaName}`,\n                `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${prepareSpecLambdaName}:*`,\n              ],\n            }),\n          ],\n        }),\n        s3: new PolicyDocument({\n          statements: [\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: [\"s3:getObject\"],\n              resources: [\n                inputSpecAsset.bucket.arnForObjects(inputSpecAsset.s3ObjectKey),\n              ],\n            }),\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: [\"s3:putObject\"],\n              resources: [\n                // The output file will include a hash of the prepared spec, which is not known until deploy time since\n                // tokens must be resolved\n                prepareSpecOutputBucket.arnForObjects(\n                  `${preparedSpecOutputKeyPrefix}/*`\n                ),\n              ],\n            }),\n          ],\n        }),\n      },\n    });\n\n    [\"AwsSolutions-IAM5\", \"AwsPrototyping-IAMNoWildcardPermissions\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          prepareSpecRole,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Cloudwatch resources have been scoped down to the LogGroup level, however * is still needed as stream names are created just in time.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:aws:logs:${PDKNag.getStackRegionRegex(\n                    stack\n                  )}:${PDKNag.getStackAccountRegex(\n                    stack\n                  )}:log-group:/aws/lambda/${prepareSpecLambdaName}:\\*/g`,\n                },\n              ],\n            },\n            {\n              id: RuleId,\n              reason:\n                \"S3 resources have been scoped down to the appropriate prefix in the CDK asset bucket, however * is still needed as since the prepared spec hash is not known until deploy time.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:s3:.*/${preparedSpecOutputKeyPrefix}/\\*/g`,\n                },\n              ],\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    // Create a custom resource for preparing the spec for deployment (adding integrations, authorizers, etc)\n    const prepareSpec = new LambdaFunction(this, \"PrepareSpecHandler\", {\n      handler: \"index.handler\",\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(\n        path.join(__dirname, \"./prepare-spec-event-handler\")\n      ),\n      timeout: Duration.seconds(30),\n      role: prepareSpecRole,\n      functionName: prepareSpecLambdaName,\n    });\n\n    const providerFunctionName = `${lambdaNamePrefix}PrepSpecProvider`;\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/${providerFunctionName}`,\n                `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${providerFunctionName}:*`,\n              ],\n            }),\n          ],\n        }),\n      },\n    });\n\n    const provider = new Provider(this, \"PrepareSpecProvider\", {\n      onEventHandler: prepareSpec,\n      role: providerRole,\n      providerFunctionName,\n    });\n\n    [\"AwsSolutions-IAM5\", \"AwsPrototyping-IAMNoWildcardPermissions\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          providerRole,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Cloudwatch resources have been scoped down to the LogGroup level, however * is still needed as stream names are created just in time.\",\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    [\"AwsSolutions-L1\", \"AwsPrototyping-LambdaLatestVersion\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          provider,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Latest runtime cannot be configured. CDK will need to upgrade the Provider construct accordingly.\",\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    const serializedCorsOptions: SerializedCorsOptions | undefined =\n      corsOptions && {\n        allowHeaders: corsOptions.allowHeaders || [\n          ...Cors.DEFAULT_HEADERS,\n          \"x-amz-content-sha256\",\n        ],\n        allowMethods: corsOptions.allowMethods || Cors.ALL_METHODS,\n        allowOrigins: corsOptions.allowOrigins,\n        statusCode: corsOptions.statusCode || 204,\n      };\n\n    const prepareSpecOptions: PrepareApiSpecOptions = {\n      defaultAuthorizerReference:\n        serializeAsAuthorizerReference(defaultAuthorizer),\n      integrations: Object.fromEntries(\n        Object.entries(integrations).map(([operationId, integration]) => [\n          operationId,\n          {\n            integration: integration.integration.render({\n              operationId,\n              scope: this,\n              ...operationLookup[operationId],\n              corsOptions: serializedCorsOptions,\n              operationLookup,\n            }),\n            methodAuthorizer: serializeAsAuthorizerReference(\n              integration.authorizer\n            ),\n            options: integration.options,\n          },\n        ])\n      ),\n      securitySchemes: prepareSecuritySchemes(\n        this,\n        integrations,\n        defaultAuthorizer,\n        options.apiKeyOptions\n      ),\n      corsOptions: serializedCorsOptions,\n      operationLookup,\n      apiKeyOptions: options.apiKeyOptions,\n      gatewayResponses: options.gatewayResponses,\n    };\n\n    // Spec preparation will happen in a custom resource lambda so that references to lambda integrations etc can be\n    // resolved. However, we also prepare inline to perform some additional validation at synth time.\n    const spec = JSON.parse(fs.readFileSync(specPath, \"utf-8\"));\n    this.extendedApiSpecification = prepareApiSpec(spec, prepareSpecOptions);\n\n    const prepareApiSpecCustomResourceProperties: PrepareApiSpecCustomResourceProperties =\n      {\n        inputSpecLocation: {\n          bucket: inputSpecAsset.bucket.bucketName,\n          key: inputSpecAsset.s3ObjectKey,\n        },\n        outputSpecLocation: {\n          bucket: prepareSpecOutputBucket.bucketName,\n          key: preparedSpecOutputKeyPrefix,\n        },\n        ...prepareSpecOptions,\n      };\n\n    const prepareSpecCustomResource = new CustomResource(\n      this,\n      \"PrepareSpecCustomResource\",\n      {\n        serviceToken: provider.serviceToken,\n        properties: {\n          options: prepareApiSpecCustomResourceProperties,\n        },\n      }\n    );\n\n    // Create the api gateway resources from the spec, augmenting the spec with the properties specific to api gateway\n    // such as integrations or auth types\n    this.api = new SpecRestApi(this, id, {\n      apiDefinition: this.node.tryGetContext(\"type-safe-api-local\")\n        ? ApiDefinition.fromInline(this.extendedApiSpecification)\n        : ApiDefinition.fromBucket(\n            prepareSpecOutputBucket,\n            prepareSpecCustomResource.getAttString(\"outputSpecKey\")\n          ),\n      deployOptions: {\n        accessLogDestination: new LogGroupLogDestination(\n          new LogGroup(this, `AccessLogs`)\n        ),\n        accessLogFormat: AccessLogFormat.clf(),\n        loggingLevel: MethodLoggingLevel.INFO,\n      },\n      ...options,\n    });\n\n    this.api.node.addDependency(prepareSpecCustomResource);\n\n    // While the api will be updated when the output path from the custom resource changes, CDK still needs to know when\n    // to redeploy the api. This is achieved by including a hash of the spec in the logical id (internalised in the\n    // addToLogicalId method since this is how changes of individual resources/methods etc trigger redeployments in CDK)\n    this.api.latestDeployment?.addToLogicalId(this.extendedApiSpecification);\n\n    // Grant API Gateway permission to invoke the integrations\n    Object.keys(integrations).forEach((operationId) => {\n      integrations[operationId].integration.grant({\n        operationId,\n        scope: this,\n        api: this.api,\n        ...operationLookup[operationId],\n        operationLookup,\n      });\n    });\n\n    // Grant API Gateway permission to invoke each custom authorizer lambda (if any)\n    getAuthorizerFunctions(props).forEach(({ label, function: lambda }) => {\n      new CfnPermission(this, `LambdaPermission-${label}`, {\n        action: \"lambda:InvokeFunction\",\n        principal: \"apigateway.amazonaws.com\",\n        functionName: lambda.functionArn,\n        sourceArn: stack.formatArn({\n          service: \"execute-api\",\n          resource: this.api.restApiId,\n          resourceName: \"*/*\",\n        }),\n      });\n    });\n\n    // Create and associate the web acl if not disabled\n    if (!props.webAclOptions?.disable) {\n      const acl = new OpenApiGatewayWebAcl(this, `${id}-Acl`, {\n        ...props.webAclOptions,\n        apiDeploymentStageArn: this.api.deploymentStage.stageArn,\n      });\n\n      this.webAcl = acl.webAcl;\n      this.ipSet = acl.ipSet;\n      this.webAclAssociation = acl.webAclAssociation;\n    }\n\n    [\"AwsSolutions-IAM4\", \"AwsPrototyping-IAMNoManagedPolicies\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          this,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Cloudwatch Role requires access to create/read groups at the root level.\",\n              appliesTo: [\n                {\n                  regex: `/^Policy::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:iam::aws:policy/service-role/AmazonAPIGatewayPushToCloudWatchLogs$/g`,\n                },\n              ],\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    [\"AwsSolutions-APIG2\", \"AwsPrototyping-APIGWRequestValidation\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          this,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"This construct implements fine grained validation via OpenApi.\",\n            },\n          ],\n          true\n        );\n      }\n    );\n  }\n}\n"]}