UNPKG

@aws-cdk/aws-apigateway

Version:

The CDK Construct Library for AWS::ApiGateway

165 lines 29.4 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.Stage = exports.MethodLoggingLevel = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const core_1 = require("@aws-cdk/core"); const access_log_1 = require("./access-log"); const apigateway_generated_1 = require("./apigateway.generated"); const restapi_1 = require("./restapi"); const util_1 = require("./util"); var MethodLoggingLevel; (function (MethodLoggingLevel) { MethodLoggingLevel["OFF"] = "OFF"; MethodLoggingLevel["ERROR"] = "ERROR"; MethodLoggingLevel["INFO"] = "INFO"; })(MethodLoggingLevel = exports.MethodLoggingLevel || (exports.MethodLoggingLevel = {})); class Stage extends core_1.Resource { constructor(scope, id, props) { super(scope, id); try { jsiiDeprecationWarnings._aws_cdk_aws_apigateway_StageProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, Stage); } throw error; } this.enableCacheCluster = props.cacheClusterEnabled; const methodSettings = this.renderMethodSettings(props); // this can mutate `this.cacheClusterEnabled` // custom access logging let accessLogSetting; const accessLogDestination = props.accessLogDestination; const accessLogFormat = props.accessLogFormat; if (!accessLogDestination && !accessLogFormat) { accessLogSetting = undefined; } else { if (accessLogFormat !== undefined && !core_1.Token.isUnresolved(accessLogFormat.toString()) && !/.*\$context.requestId.*/.test(accessLogFormat.toString())) { throw new Error('Access log must include at least `AccessLogFormat.contextRequestId()`'); } if (accessLogFormat !== undefined && accessLogDestination === undefined) { throw new Error('Access log format is specified without a destination'); } accessLogSetting = { destinationArn: accessLogDestination?.bind(this).destinationArn, format: accessLogFormat?.toString() ? accessLogFormat?.toString() : access_log_1.AccessLogFormat.clf().toString(), }; } // enable cache cluster if cacheClusterSize is set if (props.cacheClusterSize !== undefined) { if (this.enableCacheCluster === undefined) { this.enableCacheCluster = true; } else if (this.enableCacheCluster === false) { throw new Error(`Cannot set "cacheClusterSize" to ${props.cacheClusterSize} and "cacheClusterEnabled" to "false"`); } } const cacheClusterSize = this.enableCacheCluster ? (props.cacheClusterSize || '0.5') : undefined; const resource = new apigateway_generated_1.CfnStage(this, 'Resource', { stageName: props.stageName || 'prod', accessLogSetting, cacheClusterEnabled: this.enableCacheCluster, cacheClusterSize, clientCertificateId: props.clientCertificateId, deploymentId: props.deployment.deploymentId, restApiId: props.deployment.api.restApiId, description: props.description, documentationVersion: props.documentationVersion, variables: props.variables, tracingEnabled: props.tracingEnabled, methodSettings, }); this.stageName = resource.ref; this.restApi = props.deployment.api; if (restapi_1.RestApiBase._isRestApiBase(this.restApi)) { this.restApi._attachStage(this); } } /** * Returns the invoke URL for a certain path. * @param path The resource path */ urlForPath(path = '/') { if (!path.startsWith('/')) { throw new Error(`Path must begin with "/": ${path}`); } return `https://${this.restApi.restApiId}.execute-api.${core_1.Stack.of(this).region}.${core_1.Stack.of(this).urlSuffix}/${this.stageName}${path}`; } /** * Returns the resource ARN for this stage: * * arn:aws:apigateway:{region}::/restapis/{restApiId}/stages/{stageName} * * Note that this is separate from the execute-api ARN for methods and resources * within this stage. * * @attribute */ get stageArn() { return core_1.Stack.of(this).formatArn({ arnFormat: core_1.ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME, service: 'apigateway', account: '', resource: 'restapis', resourceName: `${this.restApi.restApiId}/stages/${this.stageName}`, }); } renderMethodSettings(props) { const settings = new Array(); const self = this; // extract common method options from the stage props const commonMethodOptions = { metricsEnabled: props.metricsEnabled, loggingLevel: props.loggingLevel, dataTraceEnabled: props.dataTraceEnabled, throttlingBurstLimit: props.throttlingBurstLimit, throttlingRateLimit: props.throttlingRateLimit, cachingEnabled: props.cachingEnabled, cacheTtl: props.cacheTtl, cacheDataEncrypted: props.cacheDataEncrypted, }; // if any of them are defined, add an entry for '/*/*'. const hasCommonOptions = Object.keys(commonMethodOptions).map(v => commonMethodOptions[v]).filter(x => x).length > 0; if (hasCommonOptions) { settings.push(renderEntry('/*/*', commonMethodOptions)); } if (props.methodOptions) { for (const path of Object.keys(props.methodOptions)) { settings.push(renderEntry(path, props.methodOptions[path])); } } return settings.length === 0 ? undefined : settings; function renderEntry(path, options) { if (options.cachingEnabled) { if (self.enableCacheCluster === undefined) { self.enableCacheCluster = true; } else if (self.enableCacheCluster === false) { throw new Error(`Cannot enable caching for method ${path} since cache cluster is disabled on stage`); } } const { httpMethod, resourcePath } = util_1.parseMethodOptionsPath(path); return { httpMethod, resourcePath, cacheDataEncrypted: options.cacheDataEncrypted, cacheTtlInSeconds: options.cacheTtl && options.cacheTtl.toSeconds(), cachingEnabled: options.cachingEnabled, dataTraceEnabled: options.dataTraceEnabled ?? false, loggingLevel: options.loggingLevel, metricsEnabled: options.metricsEnabled, throttlingBurstLimit: options.throttlingBurstLimit, throttlingRateLimit: options.throttlingRateLimit, }; } } } exports.Stage = Stage; _a = JSII_RTTI_SYMBOL_1; Stage[_a] = { fqn: "@aws-cdk/aws-apigateway.Stage", version: "1.204.0" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stage.js","sourceRoot":"","sources":["stage.ts"],"names":[],"mappings":"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","sourcesContent":["import { ArnFormat, Duration, IResource, Resource, Stack, Token } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { AccessLogFormat, IAccessLogDestination } from './access-log';\nimport { CfnStage } from './apigateway.generated';\nimport { Deployment } from './deployment';\nimport { IRestApi, RestApiBase } from './restapi';\nimport { parseMethodOptionsPath } from './util';\n\n/**\n * Represents an APIGateway Stage.\n */\nexport interface IStage extends IResource {\n  /**\n   * Name of this stage.\n   * @attribute\n   */\n  readonly stageName: string;\n\n  /**\n   * RestApi to which this stage is associated.\n   */\n  readonly restApi: IRestApi;\n}\n\nexport interface StageOptions extends MethodDeploymentOptions {\n  /**\n   * The name of the stage, which API Gateway uses as the first path segment\n   * in the invoked Uniform Resource Identifier (URI).\n   *\n   * @default - \"prod\"\n   */\n  readonly stageName?: string;\n\n  /**\n   * The CloudWatch Logs log group.\n   *\n   * @default - No destination\n   */\n  readonly accessLogDestination?: IAccessLogDestination;\n\n  /**\n   * A single line format of access logs of data, as specified by selected $content variables.\n   * The format must include at least `AccessLogFormat.contextRequestId()`.\n   * @see https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-mapping-template-reference.html#context-variable-reference\n   *\n   * @default - Common Log Format\n   */\n  readonly accessLogFormat?: AccessLogFormat;\n\n  /**\n   * Specifies whether Amazon X-Ray tracing is enabled for this method.\n   *\n   * @default false\n   */\n  readonly tracingEnabled?: boolean;\n\n  /**\n   * Indicates whether cache clustering is enabled for the stage.\n   *\n   * @default - Disabled for the stage.\n   */\n  readonly cacheClusterEnabled?: boolean;\n\n  /**\n   * The stage's cache cluster size.\n   * @default 0.5\n   */\n  readonly cacheClusterSize?: string;\n\n  /**\n   * The identifier of the client certificate that API Gateway uses to call\n   * your integration endpoints in the stage.\n   *\n   * @default - None.\n   */\n  readonly clientCertificateId?: string;\n\n  /**\n   * A description of the purpose of the stage.\n   *\n   * @default - No description.\n   */\n  readonly description?: string;\n\n  /**\n   * The version identifier of the API documentation snapshot.\n   *\n   * @default - No documentation version.\n   */\n  readonly documentationVersion?: string;\n\n  /**\n   * A map that defines the stage variables. Variable names must consist of\n   * alphanumeric characters, and the values must match the following regular\n   * expression: [A-Za-z0-9-._~:/?#&amp;=,]+.\n   *\n   * @default - No stage variables.\n   */\n  readonly variables?: { [key: string]: string };\n\n  /**\n   * Method deployment options for specific resources/methods. These will\n   * override common options defined in `StageOptions#methodOptions`.\n   *\n   * @param path is {resource_path}/{http_method} (i.e. /api/toys/GET) for an\n   * individual method override. You can use `*` for both {resource_path} and {http_method}\n   * to define options for all methods/resources.\n   *\n   * @default - Common options will be used.\n   */\n  readonly methodOptions?: { [path: string]: MethodDeploymentOptions };\n}\n\nexport interface StageProps extends StageOptions {\n  /**\n   * The deployment that this stage points to [disable-awslint:ref-via-interface].\n   */\n  readonly deployment: Deployment;\n}\n\nexport enum MethodLoggingLevel {\n  OFF = 'OFF',\n  ERROR = 'ERROR',\n  INFO = 'INFO'\n}\n\nexport interface MethodDeploymentOptions {\n  /**\n   * Specifies whether Amazon CloudWatch metrics are enabled for this method.\n   *\n   * @default false\n   */\n  readonly metricsEnabled?: boolean;\n\n  /**\n   * Specifies the logging level for this method, which effects the log\n   * entries pushed to Amazon CloudWatch Logs.\n   *\n   * @default - Off\n   */\n  readonly loggingLevel?: MethodLoggingLevel;\n\n  /**\n   * Specifies whether data trace logging is enabled for this method.\n   * When enabled, API gateway will log the full API requests and responses.\n   * This can be useful to troubleshoot APIs, but can result in logging sensitive data.\n   * We recommend that you don't enable this feature for production APIs.\n   *\n   * @default false\n   */\n  readonly dataTraceEnabled?: boolean;\n\n  /**\n   * Specifies the throttling burst limit.\n   * The total rate of all requests in your AWS account is limited to 5,000 requests.\n   * @see https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-request-throttling.html\n   *\n   * @default - No additional restriction.\n   */\n  readonly throttlingBurstLimit?: number;\n\n  /**\n   * Specifies the throttling rate limit.\n   * The total rate of all requests in your AWS account is limited to 10,000 requests per second (rps).\n   * @see https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-request-throttling.html\n   *\n   * @default - No additional restriction.\n   */\n  readonly throttlingRateLimit?: number;\n\n  /**\n   * Specifies whether responses should be cached and returned for requests. A\n   * cache cluster must be enabled on the stage for responses to be cached.\n   *\n   * @default - Caching is Disabled.\n   */\n  readonly cachingEnabled?: boolean;\n\n  /**\n   * Specifies the time to live (TTL), in seconds, for cached responses. The\n   * higher the TTL, the longer the response will be cached.\n   * @see https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-caching.html\n   *\n   * @default Duration.minutes(5)\n   */\n  readonly cacheTtl?: Duration;\n\n  /**\n   * Indicates whether the cached responses are encrypted.\n   *\n   * @default false\n   */\n  readonly cacheDataEncrypted?: boolean;\n}\n\nexport class Stage extends Resource implements IStage {\n  public readonly stageName: string;\n\n  public readonly restApi: IRestApi;\n  private enableCacheCluster?: boolean;\n\n  constructor(scope: Construct, id: string, props: StageProps) {\n    super(scope, id);\n\n    this.enableCacheCluster = props.cacheClusterEnabled;\n\n    const methodSettings = this.renderMethodSettings(props); // this can mutate `this.cacheClusterEnabled`\n\n    // custom access logging\n    let accessLogSetting: CfnStage.AccessLogSettingProperty | undefined;\n    const accessLogDestination = props.accessLogDestination;\n    const accessLogFormat = props.accessLogFormat;\n    if (!accessLogDestination && !accessLogFormat) {\n      accessLogSetting = undefined;\n    } else {\n      if (accessLogFormat !== undefined &&\n        !Token.isUnresolved(accessLogFormat.toString()) &&\n        !/.*\\$context.requestId.*/.test(accessLogFormat.toString())) {\n\n        throw new Error('Access log must include at least `AccessLogFormat.contextRequestId()`');\n      }\n      if (accessLogFormat !== undefined && accessLogDestination === undefined) {\n        throw new Error('Access log format is specified without a destination');\n      }\n\n      accessLogSetting = {\n        destinationArn: accessLogDestination?.bind(this).destinationArn,\n        format: accessLogFormat?.toString() ? accessLogFormat?.toString() : AccessLogFormat.clf().toString(),\n      };\n    }\n\n    // enable cache cluster if cacheClusterSize is set\n    if (props.cacheClusterSize !== undefined) {\n      if (this.enableCacheCluster === undefined) {\n        this.enableCacheCluster = true;\n      } else if (this.enableCacheCluster === false) {\n        throw new Error(`Cannot set \"cacheClusterSize\" to ${props.cacheClusterSize} and \"cacheClusterEnabled\" to \"false\"`);\n      }\n    }\n\n    const cacheClusterSize = this.enableCacheCluster ? (props.cacheClusterSize || '0.5') : undefined;\n    const resource = new CfnStage(this, 'Resource', {\n      stageName: props.stageName || 'prod',\n      accessLogSetting,\n      cacheClusterEnabled: this.enableCacheCluster,\n      cacheClusterSize,\n      clientCertificateId: props.clientCertificateId,\n      deploymentId: props.deployment.deploymentId,\n      restApiId: props.deployment.api.restApiId,\n      description: props.description,\n      documentationVersion: props.documentationVersion,\n      variables: props.variables,\n      tracingEnabled: props.tracingEnabled,\n      methodSettings,\n    });\n\n    this.stageName = resource.ref;\n    this.restApi = props.deployment.api;\n\n    if (RestApiBase._isRestApiBase(this.restApi)) {\n      this.restApi._attachStage(this);\n    }\n  }\n\n  /**\n   * Returns the invoke URL for a certain path.\n   * @param path The resource path\n   */\n  public urlForPath(path: string = '/') {\n    if (!path.startsWith('/')) {\n      throw new Error(`Path must begin with \"/\": ${path}`);\n    }\n    return `https://${this.restApi.restApiId}.execute-api.${Stack.of(this).region}.${Stack.of(this).urlSuffix}/${this.stageName}${path}`;\n  }\n\n  /**\n   * Returns the resource ARN for this stage:\n   *\n   *   arn:aws:apigateway:{region}::/restapis/{restApiId}/stages/{stageName}\n   *\n   * Note that this is separate from the execute-api ARN for methods and resources\n   * within this stage.\n   *\n   * @attribute\n   */\n  public get stageArn() {\n    return Stack.of(this).formatArn({\n      arnFormat: ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME,\n      service: 'apigateway',\n      account: '',\n      resource: 'restapis',\n      resourceName: `${this.restApi.restApiId}/stages/${this.stageName}`,\n    });\n  }\n\n  private renderMethodSettings(props: StageProps): CfnStage.MethodSettingProperty[] | undefined {\n    const settings = new Array<CfnStage.MethodSettingProperty>();\n    const self = this;\n\n    // extract common method options from the stage props\n    const commonMethodOptions: MethodDeploymentOptions = {\n      metricsEnabled: props.metricsEnabled,\n      loggingLevel: props.loggingLevel,\n      dataTraceEnabled: props.dataTraceEnabled,\n      throttlingBurstLimit: props.throttlingBurstLimit,\n      throttlingRateLimit: props.throttlingRateLimit,\n      cachingEnabled: props.cachingEnabled,\n      cacheTtl: props.cacheTtl,\n      cacheDataEncrypted: props.cacheDataEncrypted,\n    };\n\n    // if any of them are defined, add an entry for '/*/*'.\n    const hasCommonOptions = Object.keys(commonMethodOptions).map(v => (commonMethodOptions as any)[v]).filter(x => x).length > 0;\n    if (hasCommonOptions) {\n      settings.push(renderEntry('/*/*', commonMethodOptions));\n    }\n\n    if (props.methodOptions) {\n      for (const path of Object.keys(props.methodOptions)) {\n        settings.push(renderEntry(path, props.methodOptions[path]));\n      }\n    }\n\n    return settings.length === 0 ? undefined : settings;\n\n    function renderEntry(path: string, options: MethodDeploymentOptions): CfnStage.MethodSettingProperty {\n      if (options.cachingEnabled) {\n        if (self.enableCacheCluster === undefined) {\n          self.enableCacheCluster = true;\n        } else if (self.enableCacheCluster === false) {\n          throw new Error(`Cannot enable caching for method ${path} since cache cluster is disabled on stage`);\n        }\n      }\n\n      const { httpMethod, resourcePath } = parseMethodOptionsPath(path);\n\n      return {\n        httpMethod,\n        resourcePath,\n        cacheDataEncrypted: options.cacheDataEncrypted,\n        cacheTtlInSeconds: options.cacheTtl && options.cacheTtl.toSeconds(),\n        cachingEnabled: options.cachingEnabled,\n        dataTraceEnabled: options.dataTraceEnabled ?? false,\n        loggingLevel: options.loggingLevel,\n        metricsEnabled: options.metricsEnabled,\n        throttlingBurstLimit: options.throttlingBurstLimit,\n        throttlingRateLimit: options.throttlingRateLimit,\n      };\n    }\n  }\n}\n"]}