@softchef/cdk-restapi
Version:
Easy to manage Rest-API
168 lines • 23.2 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.RestApi = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const aws_apigateway_1 = require("aws-cdk-lib/aws-apigateway");
const constructs_1 = require("constructs");
class RestApi extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
this.resources = {};
let restApiProps = {};
if (props.enableCors !== false) {
restApiProps.defaultCorsPreflightOptions = {
allowMethods: aws_apigateway_1.Cors.ALL_METHODS,
allowOrigins: aws_apigateway_1.Cors.ALL_ORIGINS,
allowHeaders: [
'Origin',
'Content-Type',
'Fetch-Mode',
'Accept',
'X-Amz-Date',
'Accept-Encoding',
'Authorization',
'cache-control',
'SyncCenter',
'X-Api-Key',
'X-Amz-Security-Token',
'X-Amz-User-Agent',
],
};
}
if (props.authorizationType) {
this.globalAuthorizationType = props.authorizationType;
}
if (props.authorizer) {
this.globalAuthorizer = props.authorizer;
}
// Use custom or create new
this.awsRestApi = props.restApi ?? new aws_apigateway_1.RestApi(this, this.node.id, {
...props,
...restApiProps,
});
// Define resources
this.addResources(props.resources);
}
get restApi() {
return this.awsRestApi;
}
get restApiId() {
return this.awsRestApi.restApiId;
}
get url() {
return this.awsRestApi.url;
}
addResources(resources) {
for (const resource of resources) {
this.addResource(resource);
}
return this;
}
addResource(resource) {
const path = `/${resource.path.replace(/^\/{1}/, '')}`.replace(/\?.*/, '').split('/');
const lastPath = path.reduce((previous, current, index) => {
const part = `${previous}/${current}`;
if (!this.resources[part]) {
if (index === 1) {
if (current === '') {
this.resources[part] = this.awsRestApi.root;
}
else {
this.resources[part] = this.awsRestApi.root.addResource(current);
}
}
else {
this.resources[part] = this.resources[previous].addResource(current);
}
}
return part;
});
const authorizationType = resource.authorizationType
?? this.globalAuthorizationType
?? aws_apigateway_1.AuthorizationType.NONE;
let integration;
let methodOptions = {};
if (resource.lambdaFunction) {
integration = new aws_apigateway_1.LambdaIntegration(resource.lambdaFunction);
}
else if (resource.networkLoadBalancer) {
const pathParameters = resource.path.match(/\/\{[\w]*\}/g) ?? [];
const queryParameters = resource.path.match(/[\w]*=\{[\w]*\}/g) ?? [];
const methodRequestParameters = {};
const integrationRequestParameters = {};
for (const pathParameter of pathParameters) {
const [, key] = pathParameter.match(/\/\{([\w]*)\}/) ?? [];
methodRequestParameters[`method.request.path.${key}`] = true;
integrationRequestParameters[`integration.request.path.${key}`] = `method.request.path.${key}`;
}
for (const queryParameter of queryParameters) {
const [, key] = queryParameter.match(/[\w]*=\{([\w]*[\?]?)\}/) ?? [];
const keyName = key.replace(/\?$/, '');
methodRequestParameters[`method.request.querystring.${keyName}`] = /\?$/.test(key);
integrationRequestParameters[`integration.request.querystring.${keyName}`] = `method.request.querystring.${keyName}`;
}
methodOptions.requestParameters = {
...methodRequestParameters,
...resource.methodOptions?.requestParameters,
};
integration = new aws_apigateway_1.Integration({
type: resource.vpcLink ? aws_apigateway_1.IntegrationType.HTTP : aws_apigateway_1.IntegrationType.HTTP_PROXY,
integrationHttpMethod: resource.httpMethod.toString(),
uri: `http://${resource.networkLoadBalancer.loadBalancerDnsName}${resource.path.replace(/\?.*/, '')}`,
options: {
connectionType: aws_apigateway_1.ConnectionType.VPC_LINK,
vpcLink: resource.vpcLink ?? resource.vpcLinkProxy,
integrationResponses: [{
statusCode: '200',
}],
requestParameters: integrationRequestParameters,
...resource.vpcLinkIntegrationOptions,
},
});
methodOptions.methodResponses = [
{
statusCode: '200',
},
...(resource.methodOptions?.methodResponses ? resource.methodOptions?.methodResponses : []),
];
}
else if (resource.integration) {
integration = resource.integration;
}
else {
throw new Error('Must provide integration props, support lambdaFunction, API Gateway Integration types');
}
switch (authorizationType) {
case aws_apigateway_1.AuthorizationType.COGNITO:
case aws_apigateway_1.AuthorizationType.CUSTOM:
let authorizer;
if (resource.authorizer) {
authorizer = resource.authorizer;
}
else if (this.globalAuthorizer) {
authorizer = this.globalAuthorizer;
}
else {
throw new Error('You specify authorization type is COGNITO, but not specify authorizer.');
}
methodOptions.authorizationType = aws_apigateway_1.AuthorizationType.COGNITO;
methodOptions.authorizer = authorizer;
break;
case aws_apigateway_1.AuthorizationType.IAM:
methodOptions.authorizationType = aws_apigateway_1.AuthorizationType.IAM;
break;
case aws_apigateway_1.AuthorizationType.NONE:
default:
}
this.resources[lastPath].addMethod(resource.httpMethod.toString(), integration, {
...resource.methodOptions,
...methodOptions,
});
return this;
}
}
exports.RestApi = RestApi;
_a = JSII_RTTI_SYMBOL_1;
RestApi[_a] = { fqn: "@softchef/cdk-restapi.RestApi", version: "2.0.183" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rest-api.js","sourceRoot":"","sources":["../src/rest-api.ts"],"names":[],"mappings":";;;;;AAAA,+DAYoC;AACpC,2CAEoB;AA4BpB,MAAa,OAAQ,SAAQ,sBAAS;IAYpC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAmB;QAC3D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QALX,cAAS,GAEb,EAAE,CAAC;QAKL,IAAI,YAAY,GAEZ,EAAE,CAAC;QAEP,IAAI,KAAK,CAAC,UAAU,KAAK,KAAK,EAAE;YAC9B,YAAY,CAAC,2BAA2B,GAAG;gBACzC,YAAY,EAAE,qBAAI,CAAC,WAAW;gBAC9B,YAAY,EAAE,qBAAI,CAAC,WAAW;gBAC9B,YAAY,EAAE;oBACZ,QAAQ;oBACR,cAAc;oBACd,YAAY;oBACZ,QAAQ;oBACR,YAAY;oBACZ,iBAAiB;oBACjB,eAAe;oBACf,eAAe;oBACf,YAAY;oBACZ,WAAW;oBACX,sBAAsB;oBACtB,kBAAkB;iBACnB;aACF,CAAC;SACH;QAED,IAAI,KAAK,CAAC,iBAAiB,EAAE;YAC3B,IAAI,CAAC,uBAAuB,GAAG,KAAK,CAAC,iBAAiB,CAAC;SACxD;QAED,IAAI,KAAK,CAAC,UAAU,EAAE;YACpB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC,UAAU,CAAC;SAC1C;QACD,2BAA2B;QAC3B,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,OAAO,IAAI,IAAI,wBAAU,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE;YACpE,GAAG,KAAK;YACR,GAAG,YAAY;SAChB,CAAC,CAAC;QACH,mBAAmB;QACnB,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IACrC,CAAC;IAED,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC;IACnC,CAAC;IAED,IAAI,GAAG;QACL,OAAO,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;IAC7B,CAAC;IAEM,YAAY,CAAC,SAAiC;QACnD,KAAK,MAAM,QAAQ,IAAI,SAAS,EAAE;YAChC,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;SAC5B;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAEM,WAAW,CAAC,QAA8B;QAC/C,MAAM,IAAI,GAAa,IAAI,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,EAAE,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAChG,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;YACxD,MAAM,IAAI,GAAW,GAAG,QAAQ,IAAI,OAAO,EAAE,CAAC;YAC9C,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE;gBACzB,IAAI,KAAK,KAAK,CAAC,EAAE;oBACf,IAAI,OAAO,KAAK,EAAE,EAAE;wBAClB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAc,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;qBACxD;yBAAM;wBACL,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;qBAClE;iBACF;qBAAM;oBACL,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;iBACtE;aACF;YACD,OAAO,IAAI,CAAC;QACd,CAAC,CAAC,CAAC;QACH,MAAM,iBAAiB,GAAsB,QAAQ,CAAC,iBAAiB;eAClE,IAAI,CAAC,uBAAuB;eAC5B,kCAAiB,CAAC,IAAI,CAAC;QAC5B,IAAI,WAAwB,CAAC;QAC7B,IAAI,aAAa,GAEb,EAAE,CAAC;QACP,IAAI,QAAQ,CAAC,cAAc,EAAE;YAC3B,WAAW,GAAG,IAAI,kCAAiB,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;SAC9D;aAAM,IAAI,QAAQ,CAAC,mBAAmB,EAAE;YACvC,MAAM,cAAc,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC;YACjE,MAAM,eAAe,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,kBAAkB,CAAC,IAAI,EAAE,CAAC;YACtE,MAAM,uBAAuB,GAEzB,EAAE,CAAC;YACP,MAAM,4BAA4B,GAE9B,EAAE,CAAC;YACP,KAAK,MAAM,aAAa,IAAI,cAAc,EAAE;gBAC1C,MAAM,CAAC,EAAE,GAAG,CAAC,GAAG,aAAa,CAAC,KAAK,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC;gBAC3D,uBAAuB,CAAC,uBAAuB,GAAG,EAAE,CAAC,GAAG,IAAI,CAAC;gBAC7D,4BAA4B,CAAC,4BAA4B,GAAG,EAAE,CAAC,GAAG,uBAAuB,GAAG,EAAE,CAAC;aAChG;YACD,KAAK,MAAM,cAAc,IAAI,eAAe,EAAE;gBAC5C,MAAM,CAAC,EAAE,GAAG,CAAC,GAAG,cAAc,CAAC,KAAK,CAAC,wBAAwB,CAAC,IAAI,EAAE,CAAC;gBACrE,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;gBACvC,uBAAuB,CAAC,8BAA8B,OAAO,EAAE,CAAC,GAAG,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBACnF,4BAA4B,CAAC,mCAAmC,OAAO,EAAE,CAAC,GAAG,8BAA8B,OAAO,EAAE,CAAC;aACtH;YACD,aAAa,CAAC,iBAAiB,GAAG;gBAChC,GAAG,uBAAuB;gBAC1B,GAAG,QAAQ,CAAC,aAAa,EAAE,iBAAiB;aAC7C,CAAC;YACF,WAAW,GAAG,IAAI,4BAAW,CAAC;gBAC5B,IAAI,EAAE,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,gCAAe,CAAC,IAAI,CAAC,CAAC,CAAC,gCAAe,CAAC,UAAU;gBAC1E,qBAAqB,EAAE,QAAQ,CAAC,UAAU,CAAC,QAAQ,EAAE;gBACrD,GAAG,EAAE,UAAU,QAAQ,CAAC,mBAAmB,CAAC,mBAAmB,GAAG,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE;gBACrG,OAAO,EAAE;oBACP,cAAc,EAAE,+BAAc,CAAC,QAAQ;oBACvC,OAAO,EAAE,QAAQ,CAAC,OAAO,IAAI,QAAQ,CAAC,YAAY;oBAClD,oBAAoB,EAAE,CAAC;4BACrB,UAAU,EAAE,KAAK;yBAClB,CAAC;oBACF,iBAAiB,EAAE,4BAA4B;oBAC/C,GAAG,QAAQ,CAAC,yBAAyB;iBACtC;aACF,CAAC,CAAC;YACH,aAAa,CAAC,eAAe,GAAG;gBAC9B;oBACE,UAAU,EAAE,KAAK;iBAClB;gBACD,GAAG,CAAC,QAAQ,CAAC,aAAa,EAAE,eAAe,CAAC,CAAC,CAAC,QAAQ,CAAC,aAAa,EAAE,eAAe,CAAC,CAAC,CAAC,EAAE,CAAC;aAC5F,CAAC;SACH;aAAM,IAAI,QAAQ,CAAC,WAAW,EAAE;YAC/B,WAAW,GAAG,QAAQ,CAAC,WAAW,CAAC;SACpC;aAAM;YACL,MAAM,IAAI,KAAK,CAAC,uFAAuF,CAAC,CAAC;SAC1G;QACD,QAAQ,iBAAiB,EAAE;YACzB,KAAK,kCAAiB,CAAC,OAAO,CAAC;YAC/B,KAAK,kCAAiB,CAAC,MAAM;gBAC3B,IAAI,UAAuB,CAAC;gBAC5B,IAAI,QAAQ,CAAC,UAAU,EAAE;oBACvB,UAAU,GAAG,QAAQ,CAAC,UAAU,CAAC;iBAClC;qBAAM,IAAI,IAAI,CAAC,gBAAgB,EAAE;oBAChC,UAAU,GAAG,IAAI,CAAC,gBAAgB,CAAC;iBACpC;qBAAM;oBACL,MAAM,IAAI,KAAK,CAAC,wEAAwE,CAAC,CAAC;iBAC3F;gBACD,aAAa,CAAC,iBAAiB,GAAG,kCAAiB,CAAC,OAAO,CAAC;gBAC5D,aAAa,CAAC,UAAU,GAAG,UAAU,CAAC;gBACtC,MAAM;YACR,KAAK,kCAAiB,CAAC,GAAG;gBACxB,aAAa,CAAC,iBAAiB,GAAG,kCAAiB,CAAC,GAAG,CAAC;gBACxD,MAAM;YACR,KAAK,kCAAiB,CAAC,IAAI,CAAC;YAC5B,QAAQ;SACT;QACD,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,SAAS,CAChC,QAAQ,CAAC,UAAU,CAAC,QAAQ,EAAE,EAC9B,WAAW,EACX;YACE,GAAG,QAAQ,CAAC,aAAa;YACzB,GAAmB,aAAa;SACjC,CACF,CAAC;QACF,OAAO,IAAI,CAAC;IACd,CAAC;;AAnLH,0BAoLC","sourcesContent":["import {\n  AuthorizationType,\n  ConnectionType,\n  Cors,\n  IAuthorizer,\n  Integration,\n  IntegrationType,\n  LambdaIntegration,\n  MethodOptions,\n  Resource,\n  RestApi as AwsRestApi,\n  RestApiProps as AwsRestApiProps,\n} from 'aws-cdk-lib/aws-apigateway';\nimport {\n  Construct,\n} from 'constructs';\nimport {\n  RestApiResourceProps,\n} from './resource';\n\nexport interface RestApiProps extends AwsRestApiProps {\n  /**\n   * Custom RestApi\n   */\n  readonly restApi?: AwsRestApi;\n  /**\n   * Define Rest API resources\n   */\n  readonly resources: RestApiResourceProps[];\n  /**\n   * Enable cors, default is true\n   */\n  readonly enableCors?: boolean;\n  /**\n   * Specify globally AuthorizationType by aws-apigateway.AuthorizationType, default is NONE\n   */\n  readonly authorizationType?: AuthorizationType;\n  /**\n   * Specify globally Authorizer by aws-Authorizer, default is null\n   */\n  readonly authorizer?: IAuthorizer | undefined;\n}\n\nexport class RestApi extends Construct {\n\n  private globalAuthorizationType?: AuthorizationType | undefined;\n\n  private globalAuthorizer?: IAuthorizer | undefined;\n\n  private awsRestApi: AwsRestApi;\n\n  private resources: {\n    [key: string]: Resource;\n  } = {};\n\n  constructor(scope: Construct, id: string, props: RestApiProps) {\n    super(scope, id);\n\n    let restApiProps: {\n      [key: string]: any;\n    } = {};\n\n    if (props.enableCors !== false) {\n      restApiProps.defaultCorsPreflightOptions = {\n        allowMethods: Cors.ALL_METHODS,\n        allowOrigins: Cors.ALL_ORIGINS,\n        allowHeaders: [\n          'Origin',\n          'Content-Type',\n          'Fetch-Mode',\n          'Accept',\n          'X-Amz-Date',\n          'Accept-Encoding',\n          'Authorization',\n          'cache-control',\n          'SyncCenter',\n          'X-Api-Key',\n          'X-Amz-Security-Token',\n          'X-Amz-User-Agent',\n        ],\n      };\n    }\n\n    if (props.authorizationType) {\n      this.globalAuthorizationType = props.authorizationType;\n    }\n\n    if (props.authorizer) {\n      this.globalAuthorizer = props.authorizer;\n    }\n    // Use custom or create new\n    this.awsRestApi = props.restApi ?? new AwsRestApi(this, this.node.id, {\n      ...props,\n      ...restApiProps,\n    });\n    // Define resources\n    this.addResources(props.resources);\n  }\n\n  get restApi(): AwsRestApi {\n    return this.awsRestApi;\n  }\n\n  get restApiId(): string {\n    return this.awsRestApi.restApiId;\n  }\n\n  get url(): string {\n    return this.awsRestApi.url;\n  }\n\n  public addResources(resources: RestApiResourceProps[]): this {\n    for (const resource of resources) {\n      this.addResource(resource);\n    }\n    return this;\n  }\n\n  public addResource(resource: RestApiResourceProps): this {\n    const path: string[] = `/${resource.path.replace(/^\\/{1}/, '')}`.replace(/\\?.*/, '').split('/');\n    const lastPath = path.reduce((previous, current, index) => {\n      const part: string = `${previous}/${current}`;\n      if (!this.resources[part]) {\n        if (index === 1) {\n          if (current === '') {\n            this.resources[part] = <Resource> this.awsRestApi.root;\n          } else {\n            this.resources[part] = this.awsRestApi.root.addResource(current);\n          }\n        } else {\n          this.resources[part] = this.resources[previous].addResource(current);\n        }\n      }\n      return part;\n    });\n    const authorizationType: AuthorizationType = resource.authorizationType\n      ?? this.globalAuthorizationType\n      ?? AuthorizationType.NONE;\n    let integration: Integration;\n    let methodOptions: {\n      [key: string]: any;\n    } = {};\n    if (resource.lambdaFunction) {\n      integration = new LambdaIntegration(resource.lambdaFunction);\n    } else if (resource.networkLoadBalancer) {\n      const pathParameters = resource.path.match(/\\/\\{[\\w]*\\}/g) ?? [];\n      const queryParameters = resource.path.match(/[\\w]*=\\{[\\w]*\\}/g) ?? [];\n      const methodRequestParameters: {\n        [key: string]: boolean;\n      } = {};\n      const integrationRequestParameters: {\n        [key: string]: string;\n      } = {};\n      for (const pathParameter of pathParameters) {\n        const [, key] = pathParameter.match(/\\/\\{([\\w]*)\\}/) ?? [];\n        methodRequestParameters[`method.request.path.${key}`] = true;\n        integrationRequestParameters[`integration.request.path.${key}`] = `method.request.path.${key}`;\n      }\n      for (const queryParameter of queryParameters) {\n        const [, key] = queryParameter.match(/[\\w]*=\\{([\\w]*[\\?]?)\\}/) ?? [];\n        const keyName = key.replace(/\\?$/, '');\n        methodRequestParameters[`method.request.querystring.${keyName}`] = /\\?$/.test(key);\n        integrationRequestParameters[`integration.request.querystring.${keyName}`] = `method.request.querystring.${keyName}`;\n      }\n      methodOptions.requestParameters = {\n        ...methodRequestParameters,\n        ...resource.methodOptions?.requestParameters,\n      };\n      integration = new Integration({\n        type: resource.vpcLink ? IntegrationType.HTTP : IntegrationType.HTTP_PROXY,\n        integrationHttpMethod: resource.httpMethod.toString(),\n        uri: `http://${resource.networkLoadBalancer.loadBalancerDnsName}${resource.path.replace(/\\?.*/, '')}`,\n        options: {\n          connectionType: ConnectionType.VPC_LINK,\n          vpcLink: resource.vpcLink ?? resource.vpcLinkProxy,\n          integrationResponses: [{\n            statusCode: '200',\n          }],\n          requestParameters: integrationRequestParameters,\n          ...resource.vpcLinkIntegrationOptions,\n        },\n      });\n      methodOptions.methodResponses = [\n        {\n          statusCode: '200',\n        },\n        ...(resource.methodOptions?.methodResponses ? resource.methodOptions?.methodResponses : []),\n      ];\n    } else if (resource.integration) {\n      integration = resource.integration;\n    } else {\n      throw new Error('Must provide integration props, support lambdaFunction, API Gateway Integration types');\n    }\n    switch (authorizationType) {\n      case AuthorizationType.COGNITO:\n      case AuthorizationType.CUSTOM:\n        let authorizer: IAuthorizer;\n        if (resource.authorizer) {\n          authorizer = resource.authorizer;\n        } else if (this.globalAuthorizer) {\n          authorizer = this.globalAuthorizer;\n        } else {\n          throw new Error('You specify authorization type is COGNITO, but not specify authorizer.');\n        }\n        methodOptions.authorizationType = AuthorizationType.COGNITO;\n        methodOptions.authorizer = authorizer;\n        break;\n      case AuthorizationType.IAM:\n        methodOptions.authorizationType = AuthorizationType.IAM;\n        break;\n      case AuthorizationType.NONE:\n      default:\n    }\n    this.resources[lastPath].addMethod(\n      resource.httpMethod.toString(),\n      integration,\n      {\n        ...resource.methodOptions,\n        ...<MethodOptions> methodOptions,\n      },\n    );\n    return this;\n  }\n}"]}