UNPKG

cdk-openapi-to-http-api

Version:

CDK Construct that lets you build AWS Api Gateway Http Api, backed by Lambdas, based on a OpenAPI spec file.

186 lines 30.4 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.RootResource = exports.HttpOpenApi = void 0; const fs = require("fs"); const aws_apigatewayv2_alpha_1 = require("@aws-cdk/aws-apigatewayv2-alpha"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const constructs_1 = require("constructs"); const YAML = require("yaml"); const aws_apigateway_1 = require("aws-cdk-lib/aws-apigateway"); const aws_apigatewayv2_1 = require("aws-cdk-lib/aws-apigatewayv2"); const aws_wafv2_1 = require("aws-cdk-lib/aws-wafv2"); const AUTHORIZER_KEY = 'custom_authorizer'; class HttpOpenApi extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); this.routes = []; this.functions = {}; this.permissions = {}; const file = fs.readFileSync(props.openApiSpec, 'utf8'); const spec = YAML.parse(file); const stack = aws_cdk_lib_1.Stack.of(this); this.methodMappings = this.buildMethodMappings(spec); this.cfnApi = new aws_apigatewayv2_1.CfnApi(this, `${props.functionNamePrefix}-api`, { name: `${props.functionNamePrefix}-api`, corsConfiguration: props.corsConfig, protocolType: 'HTTP' }); this.apiStage = new aws_apigatewayv2_1.CfnStage(this, `${props.functionNamePrefix}-stage`, { apiId: this.cfnApi.attrApiId, stageName: '$default', autoDeploy: true }); if (props.acls) { this.association = new aws_wafv2_1.CfnWebACLAssociation(this, `${props.functionNamePrefix}-waf-association`, { resourceArn: `arn:${stack.partition}:execute-api:${stack.region}:${stack.account}:${this.cfnApi.attrApiId}`, webAclArn: props.acls.attrArn }); } props.integrations.forEach((integration) => { const method = this.methodMappings[integration.operationId]; if (!method) { throw new Error(`There is no path in the Open API Spec matching ${integration.operationId}`); } else { const functionName = `${props.functionNamePrefix}-${integration.operationId}`; const func = new aws_cdk_lib_1.aws_lambda.Function(this, functionName, { ...integration, functionName, code: aws_cdk_lib_1.aws_lambda.AssetCode.fromAsset(integration.sourcePath), logRetention: integration.logRetention ?? 90, timeout: integration.timeout ?? aws_cdk_lib_1.Duration.seconds(3), memorySize: integration.memorySize ?? 128 }); this.functions[integration.operationId] = func; const intgr = new aws_apigatewayv2_1.CfnIntegration(this, `integration-${method.path.replace(/\//g, '-')}-${method.method}`, { apiId: this.cfnApi.attrApiId, integrationType: aws_apigateway_1.IntegrationType.AWS_PROXY, payloadFormatVersion: aws_apigatewayv2_1.PayloadFormatVersion.VERSION_2_0.version, integrationUri: `arn:${stack.partition}:apigateway:${stack.region}:lambda:path/2015-03-31/functions/${func.functionArn}/invocations` }); const route = new aws_apigatewayv2_1.CfnRoute(this, `route-${method.path.replace(/\//g, '-')}-${method.method}`, { apiId: this.cfnApi.attrApiId, routeKey: `${method.method.toUpperCase()} ${method.path}`, target: `integrations/${intgr.attrIntegrationId}` }); this.routes.push(route); if (props.customAuthorizerLambdaArn) { spec.paths[method.path][method.method].security = [ { [AUTHORIZER_KEY]: [] } ]; } } }); // First loop with authorizers to add their configurations to the spec if (props.customAuthorizerLambdaArn) { spec.components.securitySchemes = {}; // https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-swagger-extensions-authorizer.html spec.components.securitySchemes[AUTHORIZER_KEY] = this.toAuthorizerSpec(props.customAuthorizerLambdaArn, stack.region); } // Second loop with Authorizers, in order to add InvokeFunction permission // to the created API. It has to be separated because we need the ref from cfnApi if (props.customAuthorizerLambdaArn) { const permission = new aws_cdk_lib_1.aws_lambda.CfnPermission(this, 'AuthorizerPermission', { action: 'lambda:InvokeFunction', principal: 'apigateway.amazonaws.com', functionName: props.customAuthorizerLambdaArn, sourceArn: `arn:${stack.partition}:execute-api:${stack.region}:${stack.account}:${this.cfnApi.attrApiId}/*/*/*` }); this.permissions[AUTHORIZER_KEY] = permission; } Object.keys(this.functions).forEach((funcKey, idx) => { const func = this.functions[funcKey]; const permission = new aws_cdk_lib_1.aws_lambda.CfnPermission(this, `LambdaPermission_${idx}`, { action: 'lambda:InvokeFunction', principal: 'apigateway.amazonaws.com', functionName: func.functionName, sourceArn: `arn:${stack.partition}:execute-api:${stack.region}:${stack.account}:${this.cfnApi.attrApiId}/*/*` }); this.permissions[funcKey] = permission; }); } /** * Enable custom domain for this API * @param customDomainName - customDomainName to be created in Api Gateway * @param certificateArn Arn of the certificate needed for the creation of custom domain. It must be a regional certificate. */ enableCustomDomain(customDomainName, certificateArn, zoneName) { const certificate = aws_cdk_lib_1.aws_certificatemanager.Certificate.fromCertificateArn(this, 'DomainCertificate', certificateArn); const domainName = new aws_apigatewayv2_alpha_1.DomainName(this, 'CustomDomainName', { domainName: customDomainName, certificate }); const routeConfig = { recordName: customDomainName, zone: aws_cdk_lib_1.aws_route53.HostedZone.fromLookup(this, 'ZoneLookup', { domainName: zoneName }), target: aws_cdk_lib_1.aws_route53.RecordTarget.fromAlias({ bind: () => ({ dnsName: domainName.regionalDomainName, hostedZoneId: domainName.regionalHostedZoneId }) }) }; const aRecord = new aws_cdk_lib_1.aws_route53.ARecord(this, 'CustomDomainARecord', routeConfig); const aaaaRecord = new aws_cdk_lib_1.aws_route53.AaaaRecord(this, 'CustomDomainAAAARecord', routeConfig); const apiMapping = new aws_cdk_lib_1.aws_apigatewayv2.CfnApiMapping(this, 'CustomDomainApiMapping', { apiId: this.cfnApi.attrApiId, domainName: customDomainName, stage: this.apiStage.stageName }); apiMapping.node.addDependency(this.cfnApi); apiMapping.node.addDependency(this.apiStage); apiMapping.node.addDependency(domainName); apiMapping.node.addDependency(aRecord); apiMapping.node.addDependency(aaaaRecord); } /** * Extracts path and method that map to the operationId needed * So finding the right place on the spec is just a matter of accessing the right attribute * @param spec * @returns methods */ // eslint-disable-next-line @typescript-eslint/no-explicit-any buildMethodMappings(spec) { const methods = {}; Object.entries(spec.paths).forEach(([path, pathObj]) => { Object.keys(pathObj).forEach((method) => { methods[pathObj[method]['x-amazon-apigateway-integration'].uri] = { path, method }; }); }); return methods; } toAuthorizerSpec(lambdaAuthorizerArn, region) { const uri = `arn:aws:apigateway:${region}:lambda:path/2015-03-31/functions/${lambdaAuthorizerArn}/invocations`; return { type: 'apiKey', name: 'Authorization', in: 'header', 'x-amazon-apigateway-authorizer': { type: 'request', identitySource: '$request.header.Authorization', // Request parameter mapping expression of the identity source. In this example, it is the 'auth' header. authorizerUri: uri, authorizerPayloadFormatVersion: '2.0', authorizerResultTtlInSeconds: 300 } }; } } exports.HttpOpenApi = HttpOpenApi; class RootResource extends aws_apigateway_1.ResourceBase { constructor(api, resourceId) { super(api, resourceId); this.api = api; this.resourceId = resourceId; this.path = '/'; } } exports.RootResource = RootResource; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http-openapi.js","sourceRoot":"","sources":["../src/http-openapi.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs'\n\nimport { DomainName } from '@aws-cdk/aws-apigatewayv2-alpha'\nimport {\n  aws_certificatemanager as acm,\n  aws_apigatewayv2 as apigwv2, Duration, aws_lambda as lambda, aws_route53 as route53,\n  Stack\n} from 'aws-cdk-lib'\nimport { Construct } from 'constructs'\nimport * as YAML from 'yaml'\n\nimport { CorsOptions, Integration, IntegrationType, IResource, IRestApi, MethodOptions, ResourceBase } from 'aws-cdk-lib/aws-apigateway'\nimport { CfnApi, CfnIntegration, CfnRoute, CfnStage, PayloadFormatVersion } from 'aws-cdk-lib/aws-apigatewayv2'\nimport { CfnWebACLAssociation } from 'aws-cdk-lib/aws-wafv2'\nimport { HttpApiProps, MethodMapping } from './types'\n\nconst AUTHORIZER_KEY = 'custom_authorizer'\n\nexport class HttpOpenApi extends Construct {\n  /**\n   *  Api Resource being created based on openAPI definition\n   */\n  public readonly cfnApi: apigwv2.CfnApi\n\n  /**\n   * Default stage being created & deployed for the API\n   */\n  public readonly apiStage: apigwv2.CfnStage\n\n  /**\n   * Maps operationId to lambda Function that is being created\n   */\n  public readonly functions: Record<string, lambda.Function>\n\n  public readonly permissions: Record<string, lambda.CfnPermission>\n\n  public readonly routes: CfnRoute[] = []\n\n  /**\n   * Maps operationId to http path and method - for routing purposes\n   */\n  public readonly methodMappings: Record<string, MethodMapping>\n\n  public readonly association?: CfnWebACLAssociation\n\n  constructor(scope: Construct, id: string, props: HttpApiProps) {\n    super(scope, id)\n\n    this.functions = {}\n    this.permissions = {}\n\n    const file = fs.readFileSync(props.openApiSpec, 'utf8')\n    const spec = YAML.parse(file)\n    const stack = Stack.of(this)\n\n    this.methodMappings = this.buildMethodMappings(spec)\n\n    this.cfnApi = new CfnApi(this, `${props.functionNamePrefix}-api`, {\n      name: `${props.functionNamePrefix}-api`,\n      corsConfiguration: props.corsConfig,\n      protocolType: 'HTTP'\n    })\n\n    this.apiStage = new CfnStage(this, `${props.functionNamePrefix}-stage`, {\n      apiId: this.cfnApi.attrApiId,\n      stageName: '$default',\n      autoDeploy: true\n    })\n\n    if (props.acls) {\n      this.association = new CfnWebACLAssociation(this, `${props.functionNamePrefix}-waf-association`, {\n        resourceArn: `arn:${stack.partition}:execute-api:${stack.region}:${stack.account}:${this.cfnApi.attrApiId}`,\n        webAclArn: props.acls.attrArn\n      })\n    }\n\n    props.integrations.forEach((integration) => {\n      const method = this.methodMappings[integration.operationId]\n      if (!method) {\n        throw new Error(`There is no path in the Open API Spec matching ${integration.operationId}`)\n      } else {\n        const functionName = `${props.functionNamePrefix}-${integration.operationId}`\n        const func = new lambda.Function(this, functionName, {\n          ...integration,\n          functionName,\n          code: lambda.AssetCode.fromAsset(\n            integration.sourcePath\n          ),\n          logRetention: integration.logRetention ?? 90,\n          timeout: integration.timeout ?? Duration.seconds(3),\n          memorySize: integration.memorySize ?? 128\n        })\n\n        this.functions[integration.operationId] = func\n\n        const intgr = new CfnIntegration(this, `integration-${method.path.replace(/\\//g, '-')}-${method.method}`, {\n          apiId: this.cfnApi.attrApiId,\n          integrationType: IntegrationType.AWS_PROXY,\n          payloadFormatVersion: PayloadFormatVersion.VERSION_2_0.version,\n          integrationUri: `arn:${stack.partition}:apigateway:${stack.region}:lambda:path/2015-03-31/functions/${func.functionArn}/invocations`\n        })\n\n        const route = new CfnRoute(this, `route-${method.path.replace(/\\//g, '-')}-${method.method}`, {\n          apiId: this.cfnApi.attrApiId,\n          routeKey: `${method.method.toUpperCase()} ${method.path}`,\n          target: `integrations/${intgr.attrIntegrationId}`\n        })\n        this.routes.push(route)\n\n        if (props.customAuthorizerLambdaArn) {\n          spec.paths[method.path][method.method].security = [\n            {\n              [AUTHORIZER_KEY]: []\n            }\n          ]\n        }\n      }\n    })\n\n    // First loop with authorizers to add their configurations to the spec\n    if (props.customAuthorizerLambdaArn) {\n      spec.components.securitySchemes = {}\n      // https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-swagger-extensions-authorizer.html\n      spec.components.securitySchemes[AUTHORIZER_KEY] =\n        this.toAuthorizerSpec(props.customAuthorizerLambdaArn, stack.region)\n    }\n\n    // Second loop with Authorizers, in order to add InvokeFunction permission\n    // to the created API. It has to be separated because we need the ref from cfnApi\n    if (props.customAuthorizerLambdaArn) {\n      const permission = new lambda.CfnPermission(this, 'AuthorizerPermission', {\n        action: 'lambda:InvokeFunction',\n        principal: 'apigateway.amazonaws.com',\n        functionName: props.customAuthorizerLambdaArn,\n        sourceArn: `arn:${stack.partition}:execute-api:${stack.region}:${stack.account}:${this.cfnApi.attrApiId}/*/*/*`\n      })\n      this.permissions[AUTHORIZER_KEY] = permission\n    }\n\n    Object.keys(this.functions).forEach((funcKey, idx) => {\n      const func = this.functions[funcKey]\n      const permission = new lambda.CfnPermission(this, `LambdaPermission_${idx}`, {\n        action: 'lambda:InvokeFunction',\n        principal: 'apigateway.amazonaws.com',\n        functionName: func.functionName,\n        sourceArn: `arn:${stack.partition}:execute-api:${stack.region}:${stack.account}:${this.cfnApi.attrApiId}/*/*`\n      })\n      this.permissions[funcKey] = permission\n    })\n  }\n\n  /**\n   * Enable custom domain for this API\n   * @param customDomainName - customDomainName to be created in Api Gateway\n   * @param certificateArn Arn of the certificate needed for the creation of custom domain. It must be a regional certificate.\n   */\n  public enableCustomDomain(\n    customDomainName: string,\n    certificateArn: string,\n    zoneName: string\n  ) {\n    const certificate = acm.Certificate.fromCertificateArn(\n      this,\n      'DomainCertificate',\n      certificateArn\n    )\n\n    const domainName = new DomainName(this, 'CustomDomainName', {\n      domainName: customDomainName,\n      certificate\n    })\n\n    const routeConfig: route53.ARecordProps = {\n      recordName: customDomainName,\n      zone: route53.HostedZone.fromLookup(this, 'ZoneLookup', {\n        domainName: zoneName\n      }),\n      target: route53.RecordTarget.fromAlias({\n        bind: () => ({\n          dnsName: domainName.regionalDomainName,\n          hostedZoneId: domainName.regionalHostedZoneId\n        })\n      })\n    }\n    const aRecord = new route53.ARecord(this, 'CustomDomainARecord', routeConfig)\n    const aaaaRecord = new route53.AaaaRecord(this, 'CustomDomainAAAARecord', routeConfig)\n\n    const apiMapping = new apigwv2.CfnApiMapping(this, 'CustomDomainApiMapping', {\n      apiId: this.cfnApi.attrApiId,\n      domainName: customDomainName,\n      stage: this.apiStage.stageName\n    })\n\n    apiMapping.node.addDependency(this.cfnApi)\n    apiMapping.node.addDependency(this.apiStage)\n    apiMapping.node.addDependency(domainName)\n    apiMapping.node.addDependency(aRecord)\n    apiMapping.node.addDependency(aaaaRecord)\n  }\n\n  /**\n   * Extracts path and method that map to the operationId needed\n   * So finding the right place on the spec is just a matter of accessing the right attribute\n   * @param spec\n   * @returns methods\n   */\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  private buildMethodMappings(spec: any) {\n    const methods = {} as Record<string, MethodMapping>\n\n    Object.entries(spec.paths).forEach(([path, pathObj]: [string, any]) => {\n      Object.keys(pathObj).forEach((method) => {\n        methods[pathObj[method]['x-amazon-apigateway-integration'].uri] = {\n          path,\n          method\n        }\n      })\n    })\n\n    return methods\n  }\n\n  private toAuthorizerSpec(lambdaAuthorizerArn: string, region: string) {\n    const uri = `arn:aws:apigateway:${region}:lambda:path/2015-03-31/functions/${lambdaAuthorizerArn}/invocations`\n\n    return {\n      type: 'apiKey',\n      name: 'Authorization',\n      in: 'header',\n      'x-amazon-apigateway-authorizer': {\n        type: 'request',\n        identitySource: '$request.header.Authorization', // Request parameter mapping expression of the identity source. In this example, it is the 'auth' header.\n        authorizerUri: uri,\n        authorizerPayloadFormatVersion: '2.0',\n        authorizerResultTtlInSeconds: 300\n      }\n    }\n  }\n}\n\nexport class RootResource extends ResourceBase {\n  parentResource?: IResource | undefined\n  api: IRestApi\n  resourceId: string\n  path: string\n  defaultIntegration?: Integration | undefined\n  defaultMethodOptions?: MethodOptions | undefined\n  defaultCorsPreflightOptions?: CorsOptions | undefined\n\n  constructor(api: IRestApi, resourceId: string) {\n    super(api, resourceId)\n    this.api = api\n    this.resourceId = resourceId\n    this.path = '/'\n  }\n}\n"]}