@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
122 lines • 18.8 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.ensurePrimitiveTypes = void 0;
/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0 */
const crypto = require("crypto");
const client_s3_1 = require("@aws-sdk/client-s3");
const prepare_spec_1 = require("./prepare-spec");
const s3 = new client_s3_1.S3Client({
customUserAgent: `aws-pdk/type-safe-api/prepare-spec`,
});
/**
* Prepare the api spec for API Gateway
* @param inputSpecLocation location of the specification to prepare
* @param outputSpecLocation location to write the prepared spec to
* @param options integrations, authorizers etc to apply
* @return the output location of the prepared spec
*/
const prepare = async ({ inputSpecLocation, outputSpecLocation, ...options }) => {
// Read the spec from the s3 input location
const inputSpec = JSON.parse(await (await s3.send(new client_s3_1.GetObjectCommand({
Bucket: inputSpecLocation.bucket,
Key: inputSpecLocation.key,
}))).Body.transformToString("utf-8"));
// Prepare the spec
const preparedSpec = (0, prepare_spec_1.prepareApiSpec)(inputSpec, options);
const preparedSpecHash = crypto
.createHash("sha256")
.update(JSON.stringify(preparedSpec))
.digest("hex");
const outputLocation = {
bucket: outputSpecLocation.bucket,
key: `${outputSpecLocation.key}/${preparedSpecHash}.json`,
};
// Write the spec to the s3 output location
await s3.send(new client_s3_1.PutObjectCommand({
Bucket: outputLocation.bucket,
Key: outputLocation.key,
Body: JSON.stringify(preparedSpec),
}));
return outputLocation;
};
/**
* Due to a bug in cloudformation, primitive types are coerced into strings! Coerce them back here.
* @see https://github.com/aws-cloudformation/cloudformation-coverage-roadmap/issues/1037
*/
const ensurePrimitiveTypes = (options) => {
const result = JSON.parse(JSON.stringify(options));
// Handle apiKeyOptions.requiredByDefault (boolean)
if (result.apiKeyOptions?.requiredByDefault !== undefined) {
if (result.apiKeyOptions.requiredByDefault === "true") {
result.apiKeyOptions.requiredByDefault = true;
}
else if (result.apiKeyOptions.requiredByDefault === "false") {
result.apiKeyOptions.requiredByDefault = false;
}
}
// Handle corsOptions.statusCode (number)
if (result.corsOptions?.statusCode !== undefined) {
const statusCode = Number(result.corsOptions.statusCode);
if (!isNaN(statusCode)) {
result.corsOptions.statusCode = statusCode;
}
}
if (result.integrations) {
for (const operationId in result.integrations) {
const integration = result.integrations[operationId];
// Handle integration options.apiKeyRequired (boolean)
if (integration.options?.apiKeyRequired !== undefined) {
if (integration.options.apiKeyRequired === "true") {
integration.options.apiKeyRequired = true;
}
else if (integration.options.apiKeyRequired === "false") {
integration.options.apiKeyRequired = false;
}
}
// Handle timeoutInMillis (number)
if (integration.integration?.timeoutInMillis !== undefined) {
const timeoutInMillis = Number(integration.integration.timeoutInMillis);
if (!isNaN(timeoutInMillis)) {
integration.integration.timeoutInMillis = timeoutInMillis;
}
}
// Handle tlsConfig.insecureSkipVerification (boolean)
if (integration.integration?.tlsConfig?.insecureSkipVerification !==
undefined) {
if (integration.integration.tlsConfig.insecureSkipVerification === "true") {
integration.integration.tlsConfig.insecureSkipVerification = true;
}
else if (integration.integration.tlsConfig.insecureSkipVerification === "false") {
integration.integration.tlsConfig.insecureSkipVerification = false;
}
}
}
}
return result;
};
exports.ensurePrimitiveTypes = ensurePrimitiveTypes;
exports.handler = async (event) => {
switch (event.RequestType) {
case "Create":
case "Update":
// Prepare the spec on create
const outputLocation = await prepare((0, exports.ensurePrimitiveTypes)(event.ResourceProperties.options));
return {
PhysicalResourceId: outputLocation.key,
Status: "SUCCESS",
Data: {
outputSpecKey: outputLocation.key,
},
};
case "Delete":
// Nothing to do for delete
default:
break;
}
return {
PhysicalResourceId: event.PhysicalResourceId,
Status: "SUCCESS",
};
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.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 crypto from \"crypto\";\nimport { // eslint-disable-line\n  S3Client,\n  GetObjectCommand,\n  PutObjectCommand,\n} from \"@aws-sdk/client-s3\";\nimport { prepareApiSpec, PrepareApiSpecOptions } from \"./prepare-spec\";\n\n/**\n * Represents an object location in an s3 bucket\n */\nexport interface S3Location {\n  /**\n   * The bucket in which the object resides\n   */\n  readonly bucket: string;\n  /**\n   * The object key\n   */\n  readonly key: string;\n}\n\n/**\n * Properties required to prepare the api specification with the given integrations, authorizers, etc\n */\nexport interface PrepareApiSpecCustomResourceProperties\n  extends PrepareApiSpecOptions {\n  /**\n   * The location from which to read the spec to prepare\n   */\n  readonly inputSpecLocation: S3Location;\n  /**\n   * The location to write the prepared spec. Note that the key is used as a prefix and the output location will\n   * include a hash.\n   */\n  readonly outputSpecLocation: S3Location;\n}\n\n/**\n * Cloudformation event type for custom resource\n */\ninterface OnEventRequest {\n  /**\n   * The type of cloudformation request\n   */\n  readonly RequestType: \"Create\" | \"Update\" | \"Delete\";\n  /**\n   * Physical resource id of the custom resource\n   */\n  readonly PhysicalResourceId?: string;\n  /**\n   * Properties for preparing the api\n   */\n  readonly ResourceProperties: {\n    options: PrepareApiSpecCustomResourceProperties;\n  };\n}\n\n/**\n * Custom resource response\n */\ninterface OnEventResponse {\n  /**\n   * Physical resource id of the custom resource\n   */\n  readonly PhysicalResourceId: string;\n  /**\n   * Status of the custom resource\n   */\n  readonly Status: \"SUCCESS\" | \"FAILED\";\n  /**\n   * Data returned by the custom resource\n   */\n  readonly Data?: {\n    /**\n     * The key for the output spec in the output bucket\n     */\n    readonly outputSpecKey: string;\n  };\n}\n\nconst s3 = new S3Client({\n  customUserAgent: `aws-pdk/type-safe-api/prepare-spec`,\n});\n\n/**\n * Prepare the api spec for API Gateway\n * @param inputSpecLocation location of the specification to prepare\n * @param outputSpecLocation location to write the prepared spec to\n * @param options integrations, authorizers etc to apply\n * @return the output location of the prepared spec\n */\nconst prepare = async ({\n  inputSpecLocation,\n  outputSpecLocation,\n  ...options\n}: PrepareApiSpecCustomResourceProperties): Promise<S3Location> => {\n  // Read the spec from the s3 input location\n  const inputSpec = JSON.parse(\n    await (\n      await s3.send(\n        new GetObjectCommand({\n          Bucket: inputSpecLocation.bucket,\n          Key: inputSpecLocation.key,\n        })\n      )\n    ).Body!.transformToString(\"utf-8\")\n  );\n\n  // Prepare the spec\n  const preparedSpec = prepareApiSpec(inputSpec, options);\n  const preparedSpecHash = crypto\n    .createHash(\"sha256\")\n    .update(JSON.stringify(preparedSpec))\n    .digest(\"hex\");\n\n  const outputLocation: S3Location = {\n    bucket: outputSpecLocation.bucket,\n    key: `${outputSpecLocation.key}/${preparedSpecHash}.json`,\n  };\n\n  // Write the spec to the s3 output location\n  await s3.send(\n    new PutObjectCommand({\n      Bucket: outputLocation.bucket,\n      Key: outputLocation.key,\n      Body: JSON.stringify(preparedSpec),\n    })\n  );\n\n  return outputLocation;\n};\n\n/**\n * Due to a bug in cloudformation, primitive types are coerced into strings! Coerce them back here.\n * @see https://github.com/aws-cloudformation/cloudformation-coverage-roadmap/issues/1037\n */\nexport const ensurePrimitiveTypes = (\n  options: PrepareApiSpecCustomResourceProperties\n): PrepareApiSpecCustomResourceProperties => {\n  const result = JSON.parse(JSON.stringify(options));\n\n  // Handle apiKeyOptions.requiredByDefault (boolean)\n  if (result.apiKeyOptions?.requiredByDefault !== undefined) {\n    if (result.apiKeyOptions.requiredByDefault === \"true\") {\n      result.apiKeyOptions.requiredByDefault = true;\n    } else if (result.apiKeyOptions.requiredByDefault === \"false\") {\n      result.apiKeyOptions.requiredByDefault = false;\n    }\n  }\n\n  // Handle corsOptions.statusCode (number)\n  if (result.corsOptions?.statusCode !== undefined) {\n    const statusCode = Number(result.corsOptions.statusCode);\n    if (!isNaN(statusCode)) {\n      result.corsOptions.statusCode = statusCode;\n    }\n  }\n\n  if (result.integrations) {\n    for (const operationId in result.integrations) {\n      const integration = result.integrations[operationId];\n      // Handle integration options.apiKeyRequired (boolean)\n      if (integration.options?.apiKeyRequired !== undefined) {\n        if (integration.options.apiKeyRequired === \"true\") {\n          integration.options.apiKeyRequired = true;\n        } else if (integration.options.apiKeyRequired === \"false\") {\n          integration.options.apiKeyRequired = false;\n        }\n      }\n\n      // Handle timeoutInMillis (number)\n      if (integration.integration?.timeoutInMillis !== undefined) {\n        const timeoutInMillis = Number(integration.integration.timeoutInMillis);\n        if (!isNaN(timeoutInMillis)) {\n          integration.integration.timeoutInMillis = timeoutInMillis;\n        }\n      }\n\n      // Handle tlsConfig.insecureSkipVerification (boolean)\n      if (\n        integration.integration?.tlsConfig?.insecureSkipVerification !==\n        undefined\n      ) {\n        if (\n          integration.integration.tlsConfig.insecureSkipVerification === \"true\"\n        ) {\n          integration.integration.tlsConfig.insecureSkipVerification = true;\n        } else if (\n          integration.integration.tlsConfig.insecureSkipVerification === \"false\"\n        ) {\n          integration.integration.tlsConfig.insecureSkipVerification = false;\n        }\n      }\n    }\n  }\n\n  return result;\n};\n\nexports.handler = async (event: OnEventRequest): Promise<OnEventResponse> => {\n  switch (event.RequestType) {\n    case \"Create\":\n    case \"Update\":\n      // Prepare the spec on create\n      const outputLocation = await prepare(\n        ensurePrimitiveTypes(event.ResourceProperties.options)\n      );\n      return {\n        PhysicalResourceId: outputLocation.key,\n        Status: \"SUCCESS\",\n        Data: {\n          outputSpecKey: outputLocation.key,\n        },\n      };\n    case \"Delete\":\n    // Nothing to do for delete\n    default:\n      break;\n  }\n\n  return {\n    PhysicalResourceId: event.PhysicalResourceId!,\n    Status: \"SUCCESS\",\n  };\n};\n"]}