@umccr/htsget-lambda 0.9.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,363 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.HtsgetLambda = void 0;
7
+ const fs_1 = require("fs");
8
+ const node_path_1 = require("node:path");
9
+ const os_1 = require("os");
10
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
11
+ const constructs_1 = require("constructs");
12
+ const aws_cognito_1 = require("aws-cdk-lib/aws-cognito");
13
+ const aws_iam_1 = require("aws-cdk-lib/aws-iam");
14
+ const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
15
+ const aws_certificatemanager_1 = require("aws-cdk-lib/aws-certificatemanager");
16
+ const aws_route53_1 = require("aws-cdk-lib/aws-route53");
17
+ const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
18
+ const cargo_lambda_cdk_1 = require("cargo-lambda-cdk");
19
+ const path_1 = __importDefault(require("path"));
20
+ const aws_apigatewayv2_integrations_1 = require("aws-cdk-lib/aws-apigatewayv2-integrations");
21
+ const aws_apigatewayv2_1 = require("aws-cdk-lib/aws-apigatewayv2");
22
+ const aws_apigatewayv2_authorizers_1 = require("aws-cdk-lib/aws-apigatewayv2-authorizers");
23
+ const aws_s3_1 = require("aws-cdk-lib/aws-s3");
24
+ const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
25
+ const aws_secretsmanager_1 = require("aws-cdk-lib/aws-secretsmanager");
26
+ const util_1 = require("cargo-lambda-cdk/lib/util");
27
+ /**
28
+ * @ignore
29
+ * Construct used to deploy htsget-lambda.
30
+ */
31
+ class HtsgetLambda extends constructs_1.Construct {
32
+ constructor(scope, id, props) {
33
+ super(scope, id);
34
+ props.htsgetConfig ??= {
35
+ locations: [],
36
+ };
37
+ let httpApi;
38
+ if (props.httpApi !== undefined) {
39
+ httpApi = props.httpApi;
40
+ }
41
+ else {
42
+ if (props.domain === undefined) {
43
+ throw Error("domain must be defined if httpApi is not specified");
44
+ }
45
+ httpApi = this.createHttpApi(props.domain, props.jwt, props.cors, props.subDomain, props.hostedZone, props.certificateArn);
46
+ }
47
+ let lambdaRole;
48
+ if (props.role == undefined) {
49
+ lambdaRole = HtsgetLambda.createRole(this, id, props.roleName);
50
+ }
51
+ props.buildEnvironment ??= {};
52
+ const htsgetLambda = new cargo_lambda_cdk_1.RustFunction(this, "Function", {
53
+ gitRemote: "https://github.com/umccr/htsget-rs",
54
+ gitForceClone: props.gitForceClone,
55
+ gitReference: props.gitReference,
56
+ functionName: props.functionName,
57
+ binaryName: "htsget-lambda",
58
+ bundling: {
59
+ environment: {
60
+ RUSTFLAGS: "-C target-cpu=neoverse-n1",
61
+ CARGO_PROFILE_RELEASE_LTO: "true",
62
+ CARGO_PROFILE_RELEASE_CODEGEN_UNITS: "1",
63
+ AWS_LAMBDA_HTTP_IGNORE_STAGE_IN_PATH: "true",
64
+ ...props.buildEnvironment,
65
+ },
66
+ cargoLambdaFlags: props.cargoLambdaFlags ?? [
67
+ HtsgetLambda.resolveFeatures(props.htsgetConfig, props.copyTestData ?? false),
68
+ ],
69
+ },
70
+ memorySize: 128,
71
+ timeout: aws_cdk_lib_1.Duration.seconds(28),
72
+ architecture: aws_lambda_1.Architecture.ARM_64,
73
+ role: lambdaRole ?? props.role,
74
+ vpc: props.vpc,
75
+ });
76
+ let bucket = undefined;
77
+ let privateKey = undefined;
78
+ let publicKey = undefined;
79
+ if (props.copyTestData) {
80
+ [bucket, privateKey, publicKey] = this.setupTestData(props.gitReference, props.bucketName);
81
+ }
82
+ if (lambdaRole !== undefined) {
83
+ HtsgetLambda.setPermissions(lambdaRole, props.htsgetConfig, bucket, privateKey, publicKey);
84
+ }
85
+ const env = HtsgetLambda.configToEnv(props.htsgetConfig, props.cors, bucket, privateKey, publicKey);
86
+ for (const key in env) {
87
+ htsgetLambda.addEnvironment(key, env[key]);
88
+ }
89
+ htsgetLambda.addEnvironment("RUST_LOG", "trace");
90
+ const httpIntegration = new aws_apigatewayv2_integrations_1.HttpLambdaIntegration("Integration", htsgetLambda);
91
+ [aws_apigatewayv2_1.HttpMethod.GET, aws_apigatewayv2_1.HttpMethod.POST].map((method) => {
92
+ const path = "/{proxy+}";
93
+ new aws_apigatewayv2_1.HttpRoute(this, `${method}${path}`, {
94
+ httpApi: httpApi,
95
+ routeKey: aws_apigatewayv2_1.HttpRouteKey.with(path, method),
96
+ integration: httpIntegration,
97
+ });
98
+ });
99
+ if (httpApi.defaultAuthorizer === undefined) {
100
+ console.warn("This will create an instance of htsget-rs that is public! Anyone will be able to query the server without authorization.");
101
+ }
102
+ }
103
+ /**
104
+ * Determine the correct features based on the locations.
105
+ */
106
+ static resolveFeatures(config, bucketSetup) {
107
+ const features = [];
108
+ if (config.locations?.some((location) => location.location.startsWith("s3://")) ||
109
+ bucketSetup) {
110
+ features.push("aws");
111
+ }
112
+ if (config.locations?.some((location) => location.location.startsWith("http://") ||
113
+ location.location.startsWith("https://"))) {
114
+ features.push("url");
115
+ }
116
+ if (config.locations?.some((location) => location.private_key !== undefined ||
117
+ location.public_key !== undefined) ||
118
+ bucketSetup) {
119
+ features.push("experimental");
120
+ }
121
+ return features.length === 0
122
+ ? "--all-features"
123
+ : `--features ${features.join(",")}`;
124
+ }
125
+ /**
126
+ * Create a bucket and copy test data if configured.
127
+ */
128
+ setupTestData(gitReference, bucketName) {
129
+ const gitRemote = "https://github.com/umccr/htsget-rs";
130
+ const latestCommit = (0, util_1.exec)("git", [
131
+ "ls-remote",
132
+ gitRemote,
133
+ gitReference ?? "HEAD",
134
+ ])
135
+ .stdout.toString()
136
+ .split(/(\s+)/)[0];
137
+ const localPath = (0, node_path_1.join)((0, os_1.tmpdir)(), latestCommit);
138
+ const bucket = new aws_s3_1.Bucket(this, "Bucket", {
139
+ blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL,
140
+ encryption: aws_s3_1.BucketEncryption.S3_MANAGED,
141
+ enforceSSL: true,
142
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.RETAIN,
143
+ bucketName,
144
+ });
145
+ // Copy data from upstream htsget repo
146
+ const localDataPath = path_1.default.join(localPath, "data");
147
+ new aws_s3_deployment_1.BucketDeployment(this, "DeployFiles", {
148
+ sources: [
149
+ aws_s3_deployment_1.Source.asset(path_1.default.join(localDataPath, "c4gh")),
150
+ aws_s3_deployment_1.Source.asset(path_1.default.join(localDataPath, "bam")),
151
+ aws_s3_deployment_1.Source.asset(path_1.default.join(localDataPath, "cram")),
152
+ aws_s3_deployment_1.Source.asset(path_1.default.join(localDataPath, "vcf")),
153
+ aws_s3_deployment_1.Source.asset(path_1.default.join(localDataPath, "bcf")),
154
+ ],
155
+ destinationBucket: bucket,
156
+ });
157
+ const keyDir = path_1.default.join(localPath, "data", "c4gh", "keys");
158
+ const privateKey = new aws_secretsmanager_1.Secret(this, "PrivateKey", {
159
+ secretStringValue: aws_cdk_lib_1.SecretValue.unsafePlainText((0, fs_1.readFileSync)(path_1.default.join(keyDir, "bob.sec")).toString()),
160
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
161
+ });
162
+ const publicKey = new aws_secretsmanager_1.Secret(this, "PublicKey", {
163
+ secretStringValue: aws_cdk_lib_1.SecretValue.unsafePlainText((0, fs_1.readFileSync)(path_1.default.join(keyDir, "alice.pub")).toString()),
164
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
165
+ });
166
+ new aws_cdk_lib_1.CfnOutput(this, "BucketName", { value: bucket.bucketName });
167
+ return [bucket, privateKey, publicKey];
168
+ }
169
+ /**
170
+ * Set permissions for the Lambda role.
171
+ */
172
+ static setPermissions(role, config, bucket, privateKey, publicKey) {
173
+ role.addManagedPolicy(aws_iam_1.ManagedPolicy.fromAwsManagedPolicyName("service-role/AWSLambdaBasicExecutionRole"));
174
+ const locations = config.locations ?? [];
175
+ // Add any "s3://" locations to policy.
176
+ const buckets = locations.flatMap((location) => {
177
+ if (location.location.startsWith("s3://")) {
178
+ return [location.location.split("/")[2]];
179
+ }
180
+ else {
181
+ return [];
182
+ }
183
+ });
184
+ if (bucket !== undefined) {
185
+ buckets.push(bucket.bucketName);
186
+ }
187
+ const bucketPolicy = new aws_iam_1.PolicyStatement({
188
+ actions: ["s3:GetObject"],
189
+ resources: buckets.map((bucket) => `arn:aws:s3:::${bucket}/*`),
190
+ });
191
+ if (bucketPolicy.resources.length !== 0) {
192
+ role.addToPolicy(bucketPolicy);
193
+ }
194
+ // Add any keys from the locations.
195
+ const keys = locations.flatMap((location) => {
196
+ const keys = [];
197
+ if (location.private_key !== undefined) {
198
+ keys.push([false, location.private_key]);
199
+ }
200
+ if (location.public_key !== undefined) {
201
+ keys.push([false, location.public_key]);
202
+ }
203
+ return keys;
204
+ });
205
+ if (privateKey !== undefined) {
206
+ keys.push([true, privateKey.secretArn]);
207
+ }
208
+ if (publicKey !== undefined) {
209
+ keys.push([true, publicKey.secretArn]);
210
+ }
211
+ const secretPolicy = new aws_iam_1.PolicyStatement({
212
+ actions: ["secretsmanager:GetSecretValue"],
213
+ resources: keys.map(([arn, key]) => {
214
+ if (arn) {
215
+ return key;
216
+ }
217
+ else {
218
+ return `arn:aws:secretsmanager:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:secret:${key}-*`;
219
+ }
220
+ }),
221
+ });
222
+ if (secretPolicy.resources.length !== 0) {
223
+ role.addToPolicy(secretPolicy);
224
+ }
225
+ }
226
+ /**
227
+ * Creates a lambda role with the configured permissions.
228
+ */
229
+ static createRole(scope, id, roleName) {
230
+ return new aws_iam_1.Role(scope, "Role", {
231
+ assumedBy: new aws_iam_1.ServicePrincipal("lambda.amazonaws.com"),
232
+ description: "Lambda execution role for " + id,
233
+ roleName,
234
+ });
235
+ }
236
+ /**
237
+ * Create stateful config related to the httpApi and the API itself.
238
+ */
239
+ createHttpApi(domain, jwtAuthorizer, config, subDomain, hostedZone, certificateArn) {
240
+ // Add an authorizer if auth is required.
241
+ let authorizer = undefined;
242
+ if (jwtAuthorizer !== undefined) {
243
+ // If the cog user pool id is not specified, create a new one.
244
+ if (jwtAuthorizer.cogUserPoolId === undefined) {
245
+ const pool = new aws_cognito_1.UserPool(this, "UserPool");
246
+ jwtAuthorizer.cogUserPoolId = pool.userPoolId;
247
+ }
248
+ authorizer = new aws_apigatewayv2_authorizers_1.HttpJwtAuthorizer("HtsgetAuthorizer", `https://cognito-idp.${aws_cdk_lib_1.Stack.of(this).region}.amazonaws.com/${jwtAuthorizer.cogUserPoolId}`, {
249
+ identitySource: ["$request.header.Authorization"],
250
+ jwtAudience: jwtAuthorizer.audience ?? [],
251
+ });
252
+ }
253
+ let zone;
254
+ if (hostedZone === undefined) {
255
+ zone = aws_route53_1.HostedZone.fromLookup(this, "HostedZone", {
256
+ domainName: domain,
257
+ });
258
+ }
259
+ else {
260
+ zone = hostedZone;
261
+ }
262
+ const url = `${subDomain ?? "htsget"}.${domain}`;
263
+ let certificate;
264
+ if (certificateArn !== undefined) {
265
+ certificate = aws_certificatemanager_1.Certificate.fromCertificateArn(this, "Certificate", certificateArn);
266
+ }
267
+ else {
268
+ certificate = new aws_certificatemanager_1.Certificate(this, "Certificate", {
269
+ domainName: url,
270
+ validation: aws_certificatemanager_1.CertificateValidation.fromDns(hostedZone),
271
+ certificateName: url,
272
+ });
273
+ }
274
+ const domainName = new aws_apigatewayv2_1.DomainName(this, "DomainName", {
275
+ certificate: certificate,
276
+ domainName: url,
277
+ });
278
+ new aws_route53_1.ARecord(this, "ARecord", {
279
+ zone,
280
+ recordName: subDomain ?? "htsget",
281
+ target: aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.ApiGatewayv2DomainProperties(domainName.regionalDomainName, domainName.regionalHostedZoneId)),
282
+ });
283
+ return new aws_apigatewayv2_1.HttpApi(this, "HtsGetApiGateway", {
284
+ defaultAuthorizer: authorizer,
285
+ defaultDomainMapping: {
286
+ domainName: domainName,
287
+ },
288
+ corsPreflight: {
289
+ allowCredentials: config?.allowCredentials ?? false,
290
+ allowHeaders: config?.allowHeaders ?? ["*"],
291
+ allowMethods: config?.allowMethods ?? [aws_apigatewayv2_1.CorsHttpMethod.ANY],
292
+ allowOrigins: config?.allowOrigins ?? ["*"],
293
+ exposeHeaders: config?.exposeHeaders ?? ["*"],
294
+ maxAge: config?.maxAge ?? aws_cdk_lib_1.Duration.days(30),
295
+ },
296
+ });
297
+ }
298
+ /**
299
+ * Convert JSON config to htsget-rs env representation.
300
+ */
301
+ static configToEnv(config, corsConfig, bucket, privateKey, publicKey) {
302
+ const toHtsgetEnv = (value) => {
303
+ return JSON.stringify(value)
304
+ .replaceAll(new RegExp(/"( )*:( )*/g), "=")
305
+ .replaceAll('"', "");
306
+ };
307
+ const out = {};
308
+ const locations = config.locations ?? [];
309
+ if (bucket !== undefined) {
310
+ locations.push({
311
+ location: `s3://${bucket.bucketName}`,
312
+ private_key: privateKey?.secretArn,
313
+ public_key: publicKey?.secretArn,
314
+ });
315
+ }
316
+ let locationsEnv = locations
317
+ .map((location) => {
318
+ return toHtsgetEnv({
319
+ location: location.location,
320
+ ...(location.private_key !== undefined &&
321
+ location.public_key !== undefined && {
322
+ keys: {
323
+ kind: "SecretsManager",
324
+ private: location.private_key,
325
+ public: location.public_key,
326
+ },
327
+ }),
328
+ });
329
+ })
330
+ .join(",");
331
+ locationsEnv = `[${locationsEnv}]`;
332
+ if (locationsEnv == "[]" &&
333
+ config.environment_override?.HTSGET_LOCATIONS === undefined) {
334
+ locationsEnv = undefined;
335
+ }
336
+ out.HTSGET_LOCATIONS = locationsEnv;
337
+ out.HTSGET_TICKET_SERVER_CORS_ALLOW_CREDENTIALS =
338
+ corsConfig?.allowCredentials?.toString();
339
+ // eslint-disable-next-line @typescript-eslint/non-nullable-type-assertion-style
340
+ out.HTSGET_TICKET_SERVER_CORS_ALLOW_HEADERS = `[${corsConfig?.allowHeaders?.join(",")}]`;
341
+ // eslint-disable-next-line @typescript-eslint/non-nullable-type-assertion-style
342
+ out.HTSGET_TICKET_SERVER_CORS_ALLOW_METHODS = `[${corsConfig?.allowMethods?.join(",")}]`;
343
+ // eslint-disable-next-line @typescript-eslint/non-nullable-type-assertion-style
344
+ out.HTSGET_TICKET_SERVER_CORS_ALLOW_ORIGINS = `[${corsConfig?.allowOrigins?.join(",")}]`;
345
+ // eslint-disable-next-line @typescript-eslint/non-nullable-type-assertion-style
346
+ out.HTSGET_TICKET_SERVER_CORS_EXPOSE_HEADERS = `[${corsConfig?.exposeHeaders?.join(",")}]`;
347
+ out.HTSGET_TICKET_SERVER_CORS_MAX_AGE = corsConfig?.maxAge
348
+ ?.toSeconds()
349
+ .toString();
350
+ for (const key in config.service_info) {
351
+ out[`HTSGET_SERVICE_INFO_${key.toUpperCase()}`] = toHtsgetEnv(config.service_info[key]);
352
+ }
353
+ for (const key in config.environment_override) {
354
+ out[key] = toHtsgetEnv(config.environment_override[key]);
355
+ }
356
+ Object.keys(out).forEach((key) =>
357
+ // eslint-disable-next-line @typescript-eslint/no-dynamic-delete
358
+ (out[key] == `[undefined]` || out[key] == "[]") && delete out[key]);
359
+ return out;
360
+ }
361
+ }
362
+ exports.HtsgetLambda = HtsgetLambda;
363
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"htsget-lambda.js","sourceRoot":"","sources":["htsget-lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { readFileSync } from \"fs\";\nimport { join } from \"node:path\";\nimport { tmpdir } from \"os\";\n\nimport {\n  Aws,\n  CfnOutput,\n  Duration,\n  RemovalPolicy,\n  SecretValue,\n  Stack,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\n\nimport { UserPool } from \"aws-cdk-lib/aws-cognito\";\nimport {\n  ManagedPolicy,\n  PolicyStatement,\n  Role,\n  ServicePrincipal,\n} from \"aws-cdk-lib/aws-iam\";\nimport { Architecture } from \"aws-cdk-lib/aws-lambda\";\nimport {\n  Certificate,\n  CertificateValidation,\n  ICertificate,\n} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n  ARecord,\n  HostedZone,\n  IHostedZone,\n  RecordTarget,\n} from \"aws-cdk-lib/aws-route53\";\nimport { ApiGatewayv2DomainProperties } from \"aws-cdk-lib/aws-route53-targets\";\nimport { RustFunction } from \"cargo-lambda-cdk\";\nimport path from \"path\";\nimport { HttpLambdaIntegration } from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport {\n  CorsHttpMethod,\n  DomainName,\n  HttpApi,\n  HttpMethod,\n  HttpRoute,\n  HttpRouteKey,\n  IHttpApi,\n} from \"aws-cdk-lib/aws-apigatewayv2\";\nimport { HttpJwtAuthorizer } from \"aws-cdk-lib/aws-apigatewayv2-authorizers\";\nimport {\n  BlockPublicAccess,\n  Bucket,\n  BucketEncryption,\n} from \"aws-cdk-lib/aws-s3\";\nimport { BucketDeployment, Source } from \"aws-cdk-lib/aws-s3-deployment\";\nimport { Secret } from \"aws-cdk-lib/aws-secretsmanager\";\nimport {\n  CorsConifg,\n  HtsgetConfig,\n  HtsgetLambdaProps,\n  JwtConfig,\n} from \"./config\";\nimport { exec } from \"cargo-lambda-cdk/lib/util\";\n\n/**\n * @ignore\n * Construct used to deploy htsget-lambda.\n */\nexport class HtsgetLambda extends Construct {\n  constructor(scope: Construct, id: string, props: HtsgetLambdaProps) {\n    super(scope, id);\n\n    props.htsgetConfig ??= {\n      locations: [],\n    };\n\n    let httpApi: IHttpApi;\n    if (props.httpApi !== undefined) {\n      httpApi = props.httpApi;\n    } else {\n      if (props.domain === undefined) {\n        throw Error(\"domain must be defined if httpApi is not specified\");\n      }\n\n      httpApi = this.createHttpApi(\n        props.domain,\n        props.jwt,\n        props.cors,\n        props.subDomain,\n        props.hostedZone,\n        props.certificateArn,\n      );\n    }\n\n    let lambdaRole: Role | undefined;\n    if (props.role == undefined) {\n      lambdaRole = HtsgetLambda.createRole(this, id, props.roleName);\n    }\n\n    props.buildEnvironment ??= {};\n\n    const htsgetLambda = new RustFunction(this, \"Function\", {\n      gitRemote: \"https://github.com/umccr/htsget-rs\",\n      gitForceClone: props.gitForceClone,\n      gitReference: props.gitReference,\n      functionName: props.functionName,\n      binaryName: \"htsget-lambda\",\n      bundling: {\n        environment: {\n          RUSTFLAGS: \"-C target-cpu=neoverse-n1\",\n          CARGO_PROFILE_RELEASE_LTO: \"true\",\n          CARGO_PROFILE_RELEASE_CODEGEN_UNITS: \"1\",\n          AWS_LAMBDA_HTTP_IGNORE_STAGE_IN_PATH: \"true\",\n          ...props.buildEnvironment,\n        },\n        cargoLambdaFlags: props.cargoLambdaFlags ?? [\n          HtsgetLambda.resolveFeatures(\n            props.htsgetConfig,\n            props.copyTestData ?? false,\n          ),\n        ],\n      },\n      memorySize: 128,\n      timeout: Duration.seconds(28),\n      architecture: Architecture.ARM_64,\n      role: lambdaRole ?? props.role,\n      vpc: props.vpc,\n    });\n\n    let bucket: Bucket | undefined = undefined;\n    let privateKey: Secret | undefined = undefined;\n    let publicKey: Secret | undefined = undefined;\n    if (props.copyTestData) {\n      [bucket, privateKey, publicKey] = this.setupTestData(\n        props.gitReference,\n        props.bucketName,\n      );\n    }\n\n    if (lambdaRole !== undefined) {\n      HtsgetLambda.setPermissions(\n        lambdaRole,\n        props.htsgetConfig,\n        bucket,\n        privateKey,\n        publicKey,\n      );\n    }\n\n    const env = HtsgetLambda.configToEnv(\n      props.htsgetConfig,\n      props.cors,\n      bucket,\n      privateKey,\n      publicKey,\n    );\n    for (const key in env) {\n      htsgetLambda.addEnvironment(key, env[key]);\n    }\n    htsgetLambda.addEnvironment(\"RUST_LOG\", \"trace\");\n\n    const httpIntegration = new HttpLambdaIntegration(\n      \"Integration\",\n      htsgetLambda,\n    );\n\n    [HttpMethod.GET, HttpMethod.POST].map((method) => {\n      const path = \"/{proxy+}\";\n      new HttpRoute(this, `${method}${path}`, {\n        httpApi: httpApi,\n        routeKey: HttpRouteKey.with(path, method),\n        integration: httpIntegration,\n      });\n    });\n\n    if (httpApi.defaultAuthorizer === undefined) {\n      console.warn(\n        \"This will create an instance of htsget-rs that is public! Anyone will be able to query the server without authorization.\",\n      );\n    }\n  }\n\n  /**\n   * Determine the correct features based on the locations.\n   */\n  public static resolveFeatures(\n    config: HtsgetConfig,\n    bucketSetup: boolean,\n  ): string {\n    const features = [];\n\n    if (\n      config.locations?.some((location) =>\n        location.location.startsWith(\"s3://\"),\n      ) ||\n      bucketSetup\n    ) {\n      features.push(\"aws\");\n    }\n    if (\n      config.locations?.some(\n        (location) =>\n          location.location.startsWith(\"http://\") ||\n          location.location.startsWith(\"https://\"),\n      )\n    ) {\n      features.push(\"url\");\n    }\n    if (\n      config.locations?.some(\n        (location) =>\n          location.private_key !== undefined ||\n          location.public_key !== undefined,\n      ) ||\n      bucketSetup\n    ) {\n      features.push(\"experimental\");\n    }\n\n    return features.length === 0\n      ? \"--all-features\"\n      : `--features ${features.join(\",\")}`;\n  }\n\n  /**\n   * Create a bucket and copy test data if configured.\n   */\n  private setupTestData(\n    gitReference?: string,\n    bucketName?: string,\n  ): [Bucket, Secret, Secret] {\n    const gitRemote = \"https://github.com/umccr/htsget-rs\";\n    const latestCommit = exec(\"git\", [\n      \"ls-remote\",\n      gitRemote,\n      gitReference ?? \"HEAD\",\n    ])\n      .stdout.toString()\n      .split(/(\\s+)/)[0];\n    const localPath = join(tmpdir(), latestCommit);\n\n    const bucket = new Bucket(this, \"Bucket\", {\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n      encryption: BucketEncryption.S3_MANAGED,\n      enforceSSL: true,\n      removalPolicy: RemovalPolicy.RETAIN,\n      bucketName,\n    });\n\n    // Copy data from upstream htsget repo\n    const localDataPath = path.join(localPath, \"data\");\n    new BucketDeployment(this, \"DeployFiles\", {\n      sources: [\n        Source.asset(path.join(localDataPath, \"c4gh\")),\n        Source.asset(path.join(localDataPath, \"bam\")),\n        Source.asset(path.join(localDataPath, \"cram\")),\n        Source.asset(path.join(localDataPath, \"vcf\")),\n        Source.asset(path.join(localDataPath, \"bcf\")),\n      ],\n      destinationBucket: bucket,\n    });\n\n    const keyDir = path.join(localPath, \"data\", \"c4gh\", \"keys\");\n    const privateKey = new Secret(this, \"PrivateKey\", {\n      secretStringValue: SecretValue.unsafePlainText(\n        readFileSync(path.join(keyDir, \"bob.sec\")).toString(),\n      ),\n      removalPolicy: RemovalPolicy.DESTROY,\n    });\n    const publicKey = new Secret(this, \"PublicKey\", {\n      secretStringValue: SecretValue.unsafePlainText(\n        readFileSync(path.join(keyDir, \"alice.pub\")).toString(),\n      ),\n      removalPolicy: RemovalPolicy.DESTROY,\n    });\n\n    new CfnOutput(this, \"BucketName\", { value: bucket.bucketName });\n\n    return [bucket, privateKey, publicKey];\n  }\n\n  /**\n   * Set permissions for the Lambda role.\n   */\n  public static setPermissions(\n    role: Role,\n    config: HtsgetConfig,\n    bucket?: Bucket,\n    privateKey?: Secret,\n    publicKey?: Secret,\n  ) {\n    role.addManagedPolicy(\n      ManagedPolicy.fromAwsManagedPolicyName(\n        \"service-role/AWSLambdaBasicExecutionRole\",\n      ),\n    );\n\n    const locations = config.locations ?? [];\n    // Add any \"s3://\" locations to policy.\n    const buckets = locations.flatMap((location) => {\n      if (location.location.startsWith(\"s3://\")) {\n        return [location.location.split(\"/\")[2]];\n      } else {\n        return [];\n      }\n    });\n    if (bucket !== undefined) {\n      buckets.push(bucket.bucketName);\n    }\n\n    const bucketPolicy = new PolicyStatement({\n      actions: [\"s3:GetObject\"],\n      resources: buckets.map((bucket) => `arn:aws:s3:::${bucket}/*`),\n    });\n    if (bucketPolicy.resources.length !== 0) {\n      role.addToPolicy(bucketPolicy);\n    }\n\n    // Add any keys from the locations.\n    const keys = locations.flatMap((location) => {\n      const keys: [boolean, string][] = [];\n      if (location.private_key !== undefined) {\n        keys.push([false, location.private_key]);\n      }\n      if (location.public_key !== undefined) {\n        keys.push([false, location.public_key]);\n      }\n      return keys;\n    });\n    if (privateKey !== undefined) {\n      keys.push([true, privateKey.secretArn]);\n    }\n    if (publicKey !== undefined) {\n      keys.push([true, publicKey.secretArn]);\n    }\n\n    const secretPolicy = new PolicyStatement({\n      actions: [\"secretsmanager:GetSecretValue\"],\n      resources: keys.map(([arn, key]) => {\n        if (arn) {\n          return key;\n        } else {\n          return `arn:aws:secretsmanager:${Aws.REGION}:${Aws.ACCOUNT_ID}:secret:${key}-*`;\n        }\n      }),\n    });\n    if (secretPolicy.resources.length !== 0) {\n      role.addToPolicy(secretPolicy);\n    }\n  }\n\n  /**\n   * Creates a lambda role with the configured permissions.\n   */\n  public static createRole(\n    scope: Construct,\n    id: string,\n    roleName?: string,\n  ): Role {\n    return new Role(scope, \"Role\", {\n      assumedBy: new ServicePrincipal(\"lambda.amazonaws.com\"),\n      description: \"Lambda execution role for \" + id,\n      roleName,\n    });\n  }\n\n  /**\n   * Create stateful config related to the httpApi and the API itself.\n   */\n  private createHttpApi(\n    domain: string,\n    jwtAuthorizer?: JwtConfig,\n    config?: CorsConifg,\n    subDomain?: string,\n    hostedZone?: IHostedZone,\n    certificateArn?: string,\n  ): HttpApi {\n    // Add an authorizer if auth is required.\n    let authorizer: HttpJwtAuthorizer | undefined = undefined;\n    if (jwtAuthorizer !== undefined) {\n      // If the cog user pool id is not specified, create a new one.\n      if (jwtAuthorizer.cogUserPoolId === undefined) {\n        const pool = new UserPool(this, \"UserPool\");\n        jwtAuthorizer.cogUserPoolId = pool.userPoolId;\n      }\n\n      authorizer = new HttpJwtAuthorizer(\n        \"HtsgetAuthorizer\",\n        `https://cognito-idp.${Stack.of(this).region}.amazonaws.com/${jwtAuthorizer.cogUserPoolId}`,\n        {\n          identitySource: [\"$request.header.Authorization\"],\n          jwtAudience: jwtAuthorizer.audience ?? [],\n        },\n      );\n    }\n\n    let zone: IHostedZone;\n    if (hostedZone === undefined) {\n      zone = HostedZone.fromLookup(this, \"HostedZone\", {\n        domainName: domain,\n      });\n    } else {\n      zone = hostedZone;\n    }\n\n    const url = `${subDomain ?? \"htsget\"}.${domain}`;\n    let certificate: ICertificate;\n    if (certificateArn !== undefined) {\n      certificate = Certificate.fromCertificateArn(\n        this,\n        \"Certificate\",\n        certificateArn,\n      );\n    } else {\n      certificate = new Certificate(this, \"Certificate\", {\n        domainName: url,\n        validation: CertificateValidation.fromDns(hostedZone),\n        certificateName: url,\n      });\n    }\n\n    const domainName = new DomainName(this, \"DomainName\", {\n      certificate: certificate,\n      domainName: url,\n    });\n\n    new ARecord(this, \"ARecord\", {\n      zone,\n      recordName: subDomain ?? \"htsget\",\n      target: RecordTarget.fromAlias(\n        new ApiGatewayv2DomainProperties(\n          domainName.regionalDomainName,\n          domainName.regionalHostedZoneId,\n        ),\n      ),\n    });\n\n    return new HttpApi(this, \"HtsGetApiGateway\", {\n      defaultAuthorizer: authorizer,\n      defaultDomainMapping: {\n        domainName: domainName,\n      },\n      corsPreflight: {\n        allowCredentials: config?.allowCredentials ?? false,\n        allowHeaders: config?.allowHeaders ?? [\"*\"],\n        allowMethods: config?.allowMethods ?? [CorsHttpMethod.ANY],\n        allowOrigins: config?.allowOrigins ?? [\"*\"],\n        exposeHeaders: config?.exposeHeaders ?? [\"*\"],\n        maxAge: config?.maxAge ?? Duration.days(30),\n      },\n    });\n  }\n\n  /**\n   * Convert JSON config to htsget-rs env representation.\n   */\n  public static configToEnv(\n    config: HtsgetConfig,\n    corsConfig?: CorsConifg,\n    bucket?: Bucket,\n    privateKey?: Secret,\n    publicKey?: Secret,\n  ): Record<string, string> {\n    const toHtsgetEnv = (value: unknown) => {\n      return JSON.stringify(value)\n        .replaceAll(new RegExp(/\"( )*:( )*/g), \"=\")\n        .replaceAll('\"', \"\");\n    };\n\n    const out: Record<string, string | undefined> = {};\n    const locations = config.locations ?? [];\n\n    if (bucket !== undefined) {\n      locations.push({\n        location: `s3://${bucket.bucketName}`,\n        private_key: privateKey?.secretArn,\n        public_key: publicKey?.secretArn,\n      });\n    }\n\n    let locationsEnv: string | undefined = locations\n      .map((location) => {\n        return toHtsgetEnv({\n          location: location.location,\n          ...(location.private_key !== undefined &&\n            location.public_key !== undefined && {\n              keys: {\n                kind: \"SecretsManager\",\n                private: location.private_key,\n                public: location.public_key,\n              },\n            }),\n        });\n      })\n      .join(\",\");\n    locationsEnv = `[${locationsEnv}]`;\n\n    if (\n      locationsEnv == \"[]\" &&\n      config.environment_override?.HTSGET_LOCATIONS === undefined\n    ) {\n      locationsEnv = undefined;\n    }\n\n    out.HTSGET_LOCATIONS = locationsEnv;\n    out.HTSGET_TICKET_SERVER_CORS_ALLOW_CREDENTIALS =\n      corsConfig?.allowCredentials?.toString();\n    // eslint-disable-next-line @typescript-eslint/non-nullable-type-assertion-style\n    out.HTSGET_TICKET_SERVER_CORS_ALLOW_HEADERS = `[${corsConfig?.allowHeaders?.join(\",\") as string}]`;\n    // eslint-disable-next-line @typescript-eslint/non-nullable-type-assertion-style\n    out.HTSGET_TICKET_SERVER_CORS_ALLOW_METHODS = `[${corsConfig?.allowMethods?.join(\",\") as string}]`;\n    // eslint-disable-next-line @typescript-eslint/non-nullable-type-assertion-style\n    out.HTSGET_TICKET_SERVER_CORS_ALLOW_ORIGINS = `[${corsConfig?.allowOrigins?.join(\",\") as string}]`;\n    // eslint-disable-next-line @typescript-eslint/non-nullable-type-assertion-style\n    out.HTSGET_TICKET_SERVER_CORS_EXPOSE_HEADERS = `[${corsConfig?.exposeHeaders?.join(\",\") as string}]`;\n    out.HTSGET_TICKET_SERVER_CORS_MAX_AGE = corsConfig?.maxAge\n      ?.toSeconds()\n      .toString();\n\n    for (const key in config.service_info) {\n      out[`HTSGET_SERVICE_INFO_${key.toUpperCase()}`] = toHtsgetEnv(\n        config.service_info[key],\n      );\n    }\n    for (const key in config.environment_override) {\n      out[key] = toHtsgetEnv(config.environment_override[key]);\n    }\n\n    Object.keys(out).forEach(\n      (key) =>\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n        (out[key] == `[undefined]` || out[key] == \"[]\") && delete out[key],\n    );\n    return out as Record<string, string>;\n  }\n}\n"]}