@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
95 lines • 15.4 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.S3Integration = 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 aws_iam_1 = require("aws-cdk-lib/aws-iam");
const integration_1 = require("./integration");
const integration_response_sets_1 = require("./integration-response-sets");
const utils_1 = require("../spec/utils");
/**
* An S3 integration
* @see https://docs.aws.amazon.com/apigateway/latest/developerguide/integrating-api-with-aws-services-s3.html
*/
class S3Integration extends integration_1.Integration {
constructor(props) {
super();
this.executionRoleId = "S3IntegrationsExecutionRole";
this.bucket = props.bucket;
this.method = props.method;
this.path = props.path;
this.integrationResponseSet = props.integrationResponseSet;
this.additionalRequestParameters = Object.fromEntries([
...(props.queryStringRequestParameters ?? []).map((param) => [
`integration.request.path.${param}`,
`method.request.querystring.${param}`,
]),
...(props.headerRequestParameters ?? []).map((header) => [
`integration.request.path.${header}`,
`method.request.header.${header}`,
]),
]);
}
isRole(construct) {
return "roleArn" in construct && "grantPrincipal" in construct;
}
executionRole(scope) {
// Retrieve or create the shared S3 execution role
const existingExecutionRole = scope.node.tryFindChild(this.executionRoleId);
if (existingExecutionRole) {
if (this.isRole(existingExecutionRole)) {
return existingExecutionRole;
}
throw new Error(`Found construct with ID ${this.executionRoleId} in API scope which was not a role`);
}
return new aws_iam_1.Role(scope, this.executionRoleId, {
assumedBy: new aws_iam_1.ServicePrincipal("apigateway.amazonaws.com"),
});
}
/**
* Render the S3 integration as a snippet of OpenAPI
*/
render(props) {
return {
type: "AWS",
httpMethod: (this.method ?? props.method).toUpperCase(),
uri: (0, utils_1.bucketInvocationUri)(this.bucket, this.path ?? props.path),
credentials: this.executionRole(props.scope).roleArn,
requestParameters: {
// Add every path parameter to the integration request
...Object.fromEntries([...props.path.matchAll(/\{([^\}]*)\}/g)]
.map((m) => m[1])
.map((param) => [
`integration.request.path.${param}`,
`method.request.path.${param}`,
])),
...this.additionalRequestParameters,
},
responses: {
...(this.integrationResponseSet ??
integration_response_sets_1.IntegrationResponseSets.composite(integration_response_sets_1.IntegrationResponseSets.defaultPassthrough(), integration_response_sets_1.IntegrationResponseSets.s3JsonErrorMessage())).render(props),
},
};
}
/**
* Grant API Gateway permissions to invoke the S3 bucket
*/
grant({ scope, method, path }) {
const executionRole = this.executionRole(scope);
// Replace path parameters with * to grant access to arbitrary values for path parameters
const permissionPath = (this.path ?? path).replace(/{[^\}]*\}/g, "*");
// Grant read access for GET/HEAD/OPTIONS/TRACE, otherwise write
if (["get", "head", "options", "trace"].includes((this.method ?? method).toLowerCase())) {
this.bucket.grantRead(executionRole, permissionPath);
}
else {
this.bucket.grantWrite(executionRole, permissionPath);
}
}
}
exports.S3Integration = S3Integration;
_a = JSII_RTTI_SYMBOL_1;
S3Integration[_a] = { fqn: "@aws/pdk.type_safe_api.S3Integration", version: "0.26.14" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"s3.js","sourceRoot":"","sources":["s3.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 { IRole, Role, ServicePrincipal } from \"aws-cdk-lib/aws-iam\";\nimport { IBucket } from \"aws-cdk-lib/aws-s3\";\nimport { IConstruct } from \"constructs\";\nimport {\n  ApiGatewayIntegration,\n  Integration,\n  IntegrationGrantProps,\n  IntegrationRenderProps,\n} from \"./integration\";\nimport { IntegrationResponseSet } from \"./integration-response-set\";\nimport { IntegrationResponseSets } from \"./integration-response-sets\";\nimport { Method } from \"../spec\";\nimport { bucketInvocationUri } from \"../spec/utils\";\n\n/**\n * Options for S3Integration\n */\nexport interface S3IntegrationProps {\n  /**\n   * The S3 bucket to be invoked on integration\n   */\n  readonly bucket: IBucket;\n\n  /**\n   * The HTTP method to use when invoking the S3 bucket\n   * @default - integration method is used\n   */\n  readonly method?: Method;\n\n  /**\n   * The path override to use when invoking the S3 bucket\n   * @default - integration path is used\n   */\n  readonly path?: string;\n\n  /**\n   * Override the integration response set for the S3 integration\n   * @default - a combination of IntegrationResponseSets.defaultPassthrough() and IntegrationResponseSets.s3JsonErrorMessage()\n   */\n  readonly integrationResponseSet?: IntegrationResponseSet;\n\n  /**\n   * Specifies additional query string request parameters to be passed to the integration request.\n   * @default - no additional query string request parameters\n   */\n  readonly queryStringRequestParameters?: string[];\n\n  /**\n   * Specifies additional header request parameters to be passed to the integration request.\n   *  @default - no additional header request parameters\n   */\n  readonly headerRequestParameters?: string[];\n}\n\n/**\n * An S3 integration\n * @see https://docs.aws.amazon.com/apigateway/latest/developerguide/integrating-api-with-aws-services-s3.html\n */\nexport class S3Integration extends Integration {\n  private readonly bucket: IBucket;\n  private readonly method?: Method;\n  private readonly path?: string;\n  private readonly integrationResponseSet?: IntegrationResponseSet;\n  private readonly additionalRequestParameters?: {\n    [property: string]: string;\n  };\n\n  private readonly executionRoleId = \"S3IntegrationsExecutionRole\";\n\n  constructor(props: S3IntegrationProps) {\n    super();\n\n    this.bucket = props.bucket;\n    this.method = props.method;\n    this.path = props.path;\n    this.integrationResponseSet = props.integrationResponseSet;\n    this.additionalRequestParameters = Object.fromEntries([\n      ...(props.queryStringRequestParameters ?? []).map((param) => [\n        `integration.request.path.${param}`,\n        `method.request.querystring.${param}`,\n      ]),\n      ...(props.headerRequestParameters ?? []).map((header) => [\n        `integration.request.path.${header}`,\n        `method.request.header.${header}`,\n      ]),\n    ]);\n  }\n\n  private isRole(construct: IConstruct): construct is IRole {\n    return \"roleArn\" in construct && \"grantPrincipal\" in construct;\n  }\n\n  private executionRole(scope: IConstruct): IRole {\n    // Retrieve or create the shared S3 execution role\n    const existingExecutionRole = scope.node.tryFindChild(this.executionRoleId);\n    if (existingExecutionRole) {\n      if (this.isRole(existingExecutionRole)) {\n        return existingExecutionRole;\n      }\n      throw new Error(\n        `Found construct with ID ${this.executionRoleId} in API scope which was not a role`\n      );\n    }\n    return new Role(scope, this.executionRoleId, {\n      assumedBy: new ServicePrincipal(\"apigateway.amazonaws.com\"),\n    });\n  }\n\n  /**\n   * Render the S3 integration as a snippet of OpenAPI\n   */\n  public render(props: IntegrationRenderProps): ApiGatewayIntegration {\n    return {\n      type: \"AWS\",\n      httpMethod: (this.method ?? props.method).toUpperCase(),\n      uri: bucketInvocationUri(this.bucket, this.path ?? props.path),\n      credentials: this.executionRole(props.scope).roleArn,\n      requestParameters: {\n        // Add every path parameter to the integration request\n        ...Object.fromEntries(\n          [...props.path.matchAll(/\\{([^\\}]*)\\}/g)]\n            .map((m) => m[1])\n            .map((param) => [\n              `integration.request.path.${param}`,\n              `method.request.path.${param}`,\n            ])\n        ),\n        ...this.additionalRequestParameters,\n      },\n      responses: {\n        ...(\n          this.integrationResponseSet ??\n          IntegrationResponseSets.composite(\n            IntegrationResponseSets.defaultPassthrough(),\n            IntegrationResponseSets.s3JsonErrorMessage()\n          )\n        ).render(props),\n      },\n    };\n  }\n\n  /**\n   * Grant API Gateway permissions to invoke the S3 bucket\n   */\n  public grant({ scope, method, path }: IntegrationGrantProps) {\n    const executionRole = this.executionRole(scope);\n\n    // Replace path parameters with * to grant access to arbitrary values for path parameters\n    const permissionPath = (this.path ?? path).replace(/{[^\\}]*\\}/g, \"*\");\n\n    // Grant read access for GET/HEAD/OPTIONS/TRACE, otherwise write\n    if (\n      [\"get\", \"head\", \"options\", \"trace\"].includes(\n        (this.method ?? method).toLowerCase()\n      )\n    ) {\n      this.bucket.grantRead(executionRole, permissionPath);\n    } else {\n      this.bucket.grantWrite(executionRole, permissionPath);\n    }\n  }\n}\n"]}