@liflig/cdk 3.2.0 → 3.3.0
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.
- package/lib/api-gateway/authorizer-lambdas/basic-auth-authorizer.d.ts +19 -1
- package/lib/api-gateway/authorizer-lambdas/basic-auth-authorizer.js +47 -17
- package/lib/api-gateway/authorizer-lambdas/cognito-user-pool-authorizer.d.ts +7 -6
- package/lib/api-gateway/authorizer-lambdas/cognito-user-pool-authorizer.js +3 -3
- package/lib/api-gateway/authorizer-lambdas/cognito-user-pool-or-basic-auth-authorizer.d.ts +17 -8
- package/lib/api-gateway/authorizer-lambdas/cognito-user-pool-or-basic-auth-authorizer.js +47 -21
- package/lib/api-gateway/http-api-gateway.d.ts +7 -7
- package/lib/api-gateway/http-api-gateway.js +5 -6
- package/package.json +1 -1
|
@@ -10,9 +10,27 @@
|
|
|
10
10
|
*/
|
|
11
11
|
import type { APIGatewayRequestAuthorizerEventV2, APIGatewaySimpleAuthorizerResult } from "aws-lambda";
|
|
12
12
|
import { SecretsManager } from "@aws-sdk/client-secrets-manager";
|
|
13
|
-
|
|
13
|
+
type AuthorizerResult = APIGatewaySimpleAuthorizerResult & {
|
|
14
|
+
/**
|
|
15
|
+
* Returning a context object from our authorizer allows our API Gateway to access these variables
|
|
16
|
+
* via `${context.authorizer.<property>}`.
|
|
17
|
+
* https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html
|
|
18
|
+
*/
|
|
19
|
+
context?: {
|
|
20
|
+
/**
|
|
21
|
+
* If the request's credentials are verified, we return the username that was used in this
|
|
22
|
+
* context variable (named `authorizer.username`). We use this to include the requesting user in
|
|
23
|
+
* the API Gateway access logs (see `defaultAccessLogFormat` in our `ApiGateway` construct). You
|
|
24
|
+
* can also use this when mapping parameters to the backend integration (see
|
|
25
|
+
* `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).
|
|
26
|
+
*/
|
|
27
|
+
username: string;
|
|
28
|
+
};
|
|
29
|
+
};
|
|
30
|
+
export declare const handler: (event: APIGatewayRequestAuthorizerEventV2) => Promise<AuthorizerResult>;
|
|
14
31
|
/** For overriding dependency creation in tests. */
|
|
15
32
|
export declare const dependencies: {
|
|
16
33
|
createSecretsManager: () => SecretsManager;
|
|
17
34
|
};
|
|
18
35
|
export declare function clearCache(): void;
|
|
36
|
+
export {};
|
|
@@ -14,37 +14,42 @@ export const handler = async (event) => {
|
|
|
14
14
|
if (!authHeader || !authHeader.startsWith("Basic ")) {
|
|
15
15
|
return { isAuthorized: false };
|
|
16
16
|
}
|
|
17
|
-
const
|
|
18
|
-
for (const
|
|
19
|
-
if (authHeader ===
|
|
20
|
-
return {
|
|
17
|
+
const expectedCredentials = await getExpectedBasicAuthCredentials();
|
|
18
|
+
for (const expected of expectedCredentials) {
|
|
19
|
+
if (authHeader === expected.basicAuthHeader) {
|
|
20
|
+
return {
|
|
21
|
+
isAuthorized: true,
|
|
22
|
+
context: {
|
|
23
|
+
username: expected.username,
|
|
24
|
+
},
|
|
25
|
+
};
|
|
21
26
|
}
|
|
22
27
|
}
|
|
23
28
|
return { isAuthorized: false };
|
|
24
29
|
};
|
|
25
30
|
/** Cache this value, so that subsequent lambda invocations don't have to refetch. */
|
|
26
|
-
let
|
|
31
|
+
let cachedBasicAuthCredentials = undefined;
|
|
27
32
|
/**
|
|
28
|
-
* Returns an array
|
|
29
|
-
*
|
|
33
|
+
* Returns an array, to support credential secrets with multiple values (see
|
|
34
|
+
* `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).
|
|
30
35
|
*/
|
|
31
|
-
async function
|
|
32
|
-
if (
|
|
36
|
+
async function getExpectedBasicAuthCredentials() {
|
|
37
|
+
if (cachedBasicAuthCredentials === undefined) {
|
|
33
38
|
const secretName = process.env["CREDENTIALS_SECRET_NAME"];
|
|
34
39
|
if (!secretName) {
|
|
35
40
|
console.error("CREDENTIALS_SECRET_NAME env variable is not defined");
|
|
36
41
|
throw new Error();
|
|
37
42
|
}
|
|
38
|
-
|
|
43
|
+
cachedBasicAuthCredentials = await getBasicAuthCredentialsSecret(secretName);
|
|
39
44
|
}
|
|
40
|
-
return
|
|
45
|
+
return cachedBasicAuthCredentials;
|
|
41
46
|
}
|
|
42
|
-
async function
|
|
47
|
+
async function getBasicAuthCredentialsSecret(secretName) {
|
|
43
48
|
const secret = await getSecretValue(secretName);
|
|
44
49
|
if (isSingleUsernameAndPassword(secret)) {
|
|
45
50
|
const header = "Basic " +
|
|
46
51
|
Buffer.from(`${secret.username}:${secret.password}`).toString("base64");
|
|
47
|
-
return [header];
|
|
52
|
+
return [{ basicAuthHeader: header, username: secret.username }];
|
|
48
53
|
}
|
|
49
54
|
// See `BasicAuthAuthorizerProps` on the `ApiGateway` construct for an explanation of the formats
|
|
50
55
|
// we parse here
|
|
@@ -58,7 +63,7 @@ async function getSecretAsBasicAuthHeaders(secretName) {
|
|
|
58
63
|
throw new Error();
|
|
59
64
|
}
|
|
60
65
|
if (isStringArray(credentialsArray)) {
|
|
61
|
-
return credentialsArray.map(
|
|
66
|
+
return credentialsArray.map(parseEncodedBasicAuthCredentials);
|
|
62
67
|
}
|
|
63
68
|
}
|
|
64
69
|
console.error(`Basic auth credentials secret did not follow any expected format (secret name: '${secretName}')`);
|
|
@@ -79,7 +84,7 @@ async function getSecretValue(secretName) {
|
|
|
79
84
|
return JSON.parse(secret.SecretString);
|
|
80
85
|
}
|
|
81
86
|
catch (e) {
|
|
82
|
-
console.error(`Failed to parse secret '${secretName}' as JSON
|
|
87
|
+
console.error(`Failed to parse secret '${secretName}' as JSON:`, e);
|
|
83
88
|
throw new Error();
|
|
84
89
|
}
|
|
85
90
|
}
|
|
@@ -108,7 +113,32 @@ function isStringArray(value) {
|
|
|
108
113
|
}
|
|
109
114
|
return true;
|
|
110
115
|
}
|
|
116
|
+
/**
|
|
117
|
+
* We want to return the requesting username as a context variable in
|
|
118
|
+
* {@link AuthorizerResult.context}, for API Gateway access logs and parameter mapping. So if the
|
|
119
|
+
* basic auth credentials secret is stored as pre-encoded base64 strings, we need to parse them to
|
|
120
|
+
* get the username.
|
|
121
|
+
*/
|
|
122
|
+
function parseEncodedBasicAuthCredentials(encodedCredentials) {
|
|
123
|
+
let decodedCredentials;
|
|
124
|
+
try {
|
|
125
|
+
decodedCredentials = Buffer.from(encodedCredentials, "base64").toString();
|
|
126
|
+
}
|
|
127
|
+
catch (e) {
|
|
128
|
+
console.error("Basic auth credentials secret could not be decoded as base64:", e);
|
|
129
|
+
throw new Error();
|
|
130
|
+
}
|
|
131
|
+
const usernameAndPassword = decodedCredentials.split(":", 2);
|
|
132
|
+
if (usernameAndPassword.length !== 2) {
|
|
133
|
+
console.error("Basic auth credentials secret could not be decoded as 'username:password'");
|
|
134
|
+
throw new Error();
|
|
135
|
+
}
|
|
136
|
+
return {
|
|
137
|
+
basicAuthHeader: `Basic ${encodedCredentials}`,
|
|
138
|
+
username: usernameAndPassword[0],
|
|
139
|
+
};
|
|
140
|
+
}
|
|
111
141
|
export function clearCache() {
|
|
112
|
-
|
|
142
|
+
cachedBasicAuthCredentials = undefined;
|
|
113
143
|
}
|
|
114
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"basic-auth-authorizer.js","sourceRoot":"","sources":["../../../src/api-gateway/authorizer-lambdas/basic-auth-authorizer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This lambda verifies authorization header against static basic auth credentials saved in Secret\n * Manager.\n *\n * Expects the following environment variables:\n * - CREDENTIALS_SECRET_NAME\n *   - Secret value should follow this format: `{\"username\":\"<username>\",\"password\":\"<password>\"}`.\n *     A different format with an array of pre-encoded credentials is also supported - see docs for\n *     the `BasicAuthAuthorizerProps` on the `ApiGateway` construct.\n */\n\nimport type {\n  APIGatewayRequestAuthorizerEventV2,\n  APIGatewaySimpleAuthorizerResult,\n} from \"aws-lambda\"\nimport { SecretsManager } from \"@aws-sdk/client-secrets-manager\"\n\nexport const handler = async (\n  event: APIGatewayRequestAuthorizerEventV2,\n): Promise<APIGatewaySimpleAuthorizerResult> => {\n  const authHeader = event.headers?.authorization\n  if (!authHeader || !authHeader.startsWith(\"Basic \")) {\n    return { isAuthorized: false }\n  }\n\n  const expectedAuthHeaders = await getExpectedBasicAuthHeaders()\n\n  for (const expectedHeader of expectedAuthHeaders) {\n    if (authHeader === expectedHeader) {\n      return { isAuthorized: true }\n    }\n  }\n\n  return { isAuthorized: false }\n}\n\n/** Cache this value, so that subsequent lambda invocations don't have to refetch. */\nlet cachedBasicAuthHeaders: string[] | undefined = undefined\n\n/**\n * Returns an array of allowed basic auth headers, to support credential secrets with multiple\n * values (see `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).\n */\nasync function getExpectedBasicAuthHeaders(): Promise<string[]> {\n  if (cachedBasicAuthHeaders === undefined) {\n    const secretName: string | undefined =\n      process.env[\"CREDENTIALS_SECRET_NAME\"]\n    if (!secretName) {\n      console.error(\"CREDENTIALS_SECRET_NAME env variable is not defined\")\n      throw new Error()\n    }\n\n    cachedBasicAuthHeaders = await getSecretAsBasicAuthHeaders(secretName)\n  }\n\n  return cachedBasicAuthHeaders\n}\n\nasync function getSecretAsBasicAuthHeaders(\n  secretName: string,\n): Promise<string[]> {\n  const secret = await getSecretValue(secretName)\n\n  if (isSingleUsernameAndPassword(secret)) {\n    const header =\n      \"Basic \" +\n      Buffer.from(`${secret.username}:${secret.password}`).toString(\"base64\")\n    return [header]\n  }\n\n  // See `BasicAuthAuthorizerProps` on the `ApiGateway` construct for an explanation of the formats\n  // we parse here\n  if (hasCredentialsKeyWithStringValue(secret)) {\n    let credentialsArray: unknown\n    try {\n      credentialsArray = JSON.parse(secret.credentials)\n    } catch (e) {\n      console.error(\n        `Failed to parse credentials array in secret '${secretName}' as JSON`,\n        e,\n      )\n      throw new Error()\n    }\n\n    if (isStringArray(credentialsArray)) {\n      return credentialsArray.map(\n        (encodedCredential) => `Basic ${encodedCredential}`,\n      )\n    }\n  }\n\n  console.error(\n    `Basic auth credentials secret did not follow any expected format (secret name: '${secretName}')`,\n  )\n  throw new Error()\n}\n\n/** For overriding dependency creation in tests. */\nexport const dependencies = {\n  createSecretsManager: () => new SecretsManager(),\n}\n\nasync function getSecretValue(secretName: string): Promise<unknown> {\n  const client = dependencies.createSecretsManager()\n  const secret = await client.getSecretValue({ SecretId: secretName })\n\n  if (!secret.SecretString) {\n    console.error(`Secret value not found for '${secretName}'`)\n    throw new Error()\n  }\n\n  try {\n    return JSON.parse(secret.SecretString)\n  } catch (e) {\n    console.error(`Failed to parse secret '${secretName}' as JSON`, e)\n    throw new Error()\n  }\n}\n\nfunction isSingleUsernameAndPassword(\n  value: unknown,\n): value is { username: string; password: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"username\" in value &&\n    typeof value.username === \"string\" &&\n    \"password\" in value &&\n    typeof value.password === \"string\"\n  )\n}\n\nfunction hasCredentialsKeyWithStringValue(\n  value: unknown,\n): value is { credentials: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"credentials\" in value &&\n    typeof value.credentials === \"string\"\n  )\n}\n\nfunction isStringArray(value: unknown): value is string[] {\n  if (!Array.isArray(value)) {\n    return false\n  }\n\n  for (const element of value) {\n    if (typeof element !== \"string\") {\n      return false\n    }\n  }\n\n  return true\n}\n\nexport function clearCache() {\n  cachedBasicAuthHeaders = undefined\n}\n"]}
|
|
144
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"basic-auth-authorizer.js","sourceRoot":"","sources":["../../../src/api-gateway/authorizer-lambdas/basic-auth-authorizer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This lambda verifies authorization header against static basic auth credentials saved in Secret\n * Manager.\n *\n * Expects the following environment variables:\n * - CREDENTIALS_SECRET_NAME\n *   - Secret value should follow this format: `{\"username\":\"<username>\",\"password\":\"<password>\"}`.\n *     A different format with an array of pre-encoded credentials is also supported - see docs for\n *     the `BasicAuthAuthorizerProps` on the `ApiGateway` construct.\n */\n\nimport type {\n  APIGatewayRequestAuthorizerEventV2,\n  APIGatewaySimpleAuthorizerResult,\n} from \"aws-lambda\"\nimport { SecretsManager } from \"@aws-sdk/client-secrets-manager\"\n\ntype AuthorizerResult = APIGatewaySimpleAuthorizerResult & {\n  /**\n   * Returning a context object from our authorizer allows our API Gateway to access these variables\n   * via `${context.authorizer.<property>}`.\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html\n   */\n  context?: {\n    /**\n     * If the request's credentials are verified, we return the username that was used in this\n     * context variable (named `authorizer.username`). We use this to include the requesting user in\n     * the API Gateway access logs (see `defaultAccessLogFormat` in our `ApiGateway` construct). You\n     * can also use this when mapping parameters to the backend integration (see\n     * `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).\n     */\n    username: string\n  }\n}\n\nexport const handler = async (\n  event: APIGatewayRequestAuthorizerEventV2,\n): Promise<AuthorizerResult> => {\n  const authHeader = event.headers?.authorization\n  if (!authHeader || !authHeader.startsWith(\"Basic \")) {\n    return { isAuthorized: false }\n  }\n\n  const expectedCredentials = await getExpectedBasicAuthCredentials()\n\n  for (const expected of expectedCredentials) {\n    if (authHeader === expected.basicAuthHeader) {\n      return {\n        isAuthorized: true,\n        context: {\n          username: expected.username,\n        },\n      }\n    }\n  }\n\n  return { isAuthorized: false }\n}\n\ntype ExpectedBasicAuthCredentials = {\n  basicAuthHeader: string\n  username: string\n}\n\n/** Cache this value, so that subsequent lambda invocations don't have to refetch. */\nlet cachedBasicAuthCredentials: ExpectedBasicAuthCredentials[] | undefined =\n  undefined\n\n/**\n * Returns an array, to support credential secrets with multiple values (see\n * `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).\n */\nasync function getExpectedBasicAuthCredentials(): Promise<\n  ExpectedBasicAuthCredentials[]\n> {\n  if (cachedBasicAuthCredentials === undefined) {\n    const secretName: string | undefined =\n      process.env[\"CREDENTIALS_SECRET_NAME\"]\n    if (!secretName) {\n      console.error(\"CREDENTIALS_SECRET_NAME env variable is not defined\")\n      throw new Error()\n    }\n\n    cachedBasicAuthCredentials = await getBasicAuthCredentialsSecret(secretName)\n  }\n\n  return cachedBasicAuthCredentials\n}\n\nasync function getBasicAuthCredentialsSecret(\n  secretName: string,\n): Promise<ExpectedBasicAuthCredentials[]> {\n  const secret = await getSecretValue(secretName)\n\n  if (isSingleUsernameAndPassword(secret)) {\n    const header =\n      \"Basic \" +\n      Buffer.from(`${secret.username}:${secret.password}`).toString(\"base64\")\n    return [{ basicAuthHeader: header, username: secret.username }]\n  }\n\n  // See `BasicAuthAuthorizerProps` on the `ApiGateway` construct for an explanation of the formats\n  // we parse here\n  if (hasCredentialsKeyWithStringValue(secret)) {\n    let credentialsArray: unknown\n    try {\n      credentialsArray = JSON.parse(secret.credentials)\n    } catch (e) {\n      console.error(\n        `Failed to parse credentials array in secret '${secretName}' as JSON`,\n        e,\n      )\n      throw new Error()\n    }\n\n    if (isStringArray(credentialsArray)) {\n      return credentialsArray.map(parseEncodedBasicAuthCredentials)\n    }\n  }\n\n  console.error(\n    `Basic auth credentials secret did not follow any expected format (secret name: '${secretName}')`,\n  )\n  throw new Error()\n}\n\n/** For overriding dependency creation in tests. */\nexport const dependencies = {\n  createSecretsManager: () => new SecretsManager(),\n}\n\nasync function getSecretValue(secretName: string): Promise<unknown> {\n  const client = dependencies.createSecretsManager()\n  const secret = await client.getSecretValue({ SecretId: secretName })\n\n  if (!secret.SecretString) {\n    console.error(`Secret value not found for '${secretName}'`)\n    throw new Error()\n  }\n\n  try {\n    return JSON.parse(secret.SecretString)\n  } catch (e) {\n    console.error(`Failed to parse secret '${secretName}' as JSON:`, e)\n    throw new Error()\n  }\n}\n\nfunction isSingleUsernameAndPassword(\n  value: unknown,\n): value is { username: string; password: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"username\" in value &&\n    typeof value.username === \"string\" &&\n    \"password\" in value &&\n    typeof value.password === \"string\"\n  )\n}\n\nfunction hasCredentialsKeyWithStringValue(\n  value: unknown,\n): value is { credentials: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"credentials\" in value &&\n    typeof value.credentials === \"string\"\n  )\n}\n\nfunction isStringArray(value: unknown): value is string[] {\n  if (!Array.isArray(value)) {\n    return false\n  }\n\n  for (const element of value) {\n    if (typeof element !== \"string\") {\n      return false\n    }\n  }\n\n  return true\n}\n\n/**\n * We want to return the requesting username as a context variable in\n * {@link AuthorizerResult.context}, for API Gateway access logs and parameter mapping. So if the\n * basic auth credentials secret is stored as pre-encoded base64 strings, we need to parse them to\n * get the username.\n */\nfunction parseEncodedBasicAuthCredentials(\n  encodedCredentials: string,\n): ExpectedBasicAuthCredentials {\n  let decodedCredentials: string\n  try {\n    decodedCredentials = Buffer.from(encodedCredentials, \"base64\").toString()\n  } catch (e) {\n    console.error(\n      \"Basic auth credentials secret could not be decoded as base64:\",\n      e,\n    )\n    throw new Error()\n  }\n\n  const usernameAndPassword = decodedCredentials.split(\":\", 2)\n  if (usernameAndPassword.length !== 2) {\n    console.error(\n      \"Basic auth credentials secret could not be decoded as 'username:password'\",\n    )\n    throw new Error()\n  }\n\n  return {\n    basicAuthHeader: `Basic ${encodedCredentials}`,\n    username: usernameAndPassword[0],\n  }\n}\n\nexport function clearCache() {\n  cachedBasicAuthCredentials = undefined\n}\n"]}
|
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* This lambda verifies
|
|
2
|
+
* This lambda verifies access token in Bearer authorization header using Cognito.
|
|
3
3
|
*
|
|
4
4
|
* Expects the following environment variables:
|
|
5
5
|
* - USER_POOL_ID
|
|
6
6
|
* - REQUIRED_SCOPE (optional)
|
|
7
|
-
* - Set this to require that the
|
|
7
|
+
* - Set this to require that the access token payload contains the given scope
|
|
8
8
|
* - CREDENTIALS_FOR_INTERNAL_AUTHORIZATION (optional)
|
|
9
9
|
* - Secret name from which to get basic auth credentials that should be forwarded to backend
|
|
10
10
|
* integration if authentication succeeds
|
|
@@ -17,14 +17,15 @@ type AuthorizerResult = APIGatewaySimpleAuthorizerResult & {
|
|
|
17
17
|
/**
|
|
18
18
|
* Returning a context object from our authorizer allows our API Gateway to access these variables
|
|
19
19
|
* via `${context.authorizer.<property>}`.
|
|
20
|
-
* https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-
|
|
20
|
+
* https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html
|
|
21
21
|
*/
|
|
22
22
|
context?: {
|
|
23
23
|
/**
|
|
24
24
|
* If the token is verified, we return the auth client ID from the token's claims as a context
|
|
25
|
-
* variable (named `authorizer.clientId`).
|
|
26
|
-
* API Gateway (see `
|
|
27
|
-
*
|
|
25
|
+
* variable (named `authorizer.clientId`). We use this to include the requesting client in the
|
|
26
|
+
* API Gateway access logs (see `defaultAccessLogFormat` in our `ApiGateway` construct). You can
|
|
27
|
+
* also use this when mapping parameters to the backend integration (see
|
|
28
|
+
* `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).
|
|
28
29
|
*/
|
|
29
30
|
clientId: string;
|
|
30
31
|
/**
|
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* This lambda verifies
|
|
2
|
+
* This lambda verifies access token in Bearer authorization header using Cognito.
|
|
3
3
|
*
|
|
4
4
|
* Expects the following environment variables:
|
|
5
5
|
* - USER_POOL_ID
|
|
6
6
|
* - REQUIRED_SCOPE (optional)
|
|
7
|
-
* - Set this to require that the
|
|
7
|
+
* - Set this to require that the access token payload contains the given scope
|
|
8
8
|
* - CREDENTIALS_FOR_INTERNAL_AUTHORIZATION (optional)
|
|
9
9
|
* - Secret name from which to get basic auth credentials that should be forwarded to backend
|
|
10
10
|
* integration if authentication succeeds
|
|
@@ -130,4 +130,4 @@ export function clearCache() {
|
|
|
130
130
|
cachedTokenVerifier = undefined;
|
|
131
131
|
cachedInternalAuthorizationHeader = undefined;
|
|
132
132
|
}
|
|
133
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cognito-user-pool-authorizer.js","sourceRoot":"","sources":["../../../src/api-gateway/authorizer-lambdas/cognito-user-pool-authorizer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This lambda verifies bearer token in authorization header using Cognito.\n *\n * Expects the following environment variables:\n * - USER_POOL_ID\n * - REQUIRED_SCOPE (optional)\n *   - Set this to require that the bearer token payload contains the given scope\n * - CREDENTIALS_FOR_INTERNAL_AUTHORIZATION (optional)\n *   - Secret name from which to get basic auth credentials that should be forwarded to backend\n *     integration if authentication succeeds\n *   - Secret value should follow this format: `{\"username\":\"<username>\",\"password\":\"<password>\"}`\n */\n\nimport type {\n  APIGatewayRequestAuthorizerEventV2,\n  APIGatewaySimpleAuthorizerResult,\n} from \"aws-lambda\"\nimport { SecretsManager } from \"@aws-sdk/client-secrets-manager\"\nimport { CognitoJwtVerifier } from \"aws-jwt-verify\"\nimport type { CognitoAccessTokenPayload } from \"aws-jwt-verify/jwt-model\"\n\ntype AuthorizerResult = APIGatewaySimpleAuthorizerResult & {\n  /**\n   * Returning a context object from our authorizer allows our API Gateway to access these variables\n   * via `${context.authorizer.<property>}`.\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-logging-variables.html\n   */\n  context?: {\n    /**\n     * If the token is verified, we return the auth client ID from the token's claims as a context\n     * variable (named `authorizer.clientId`). You can then use this for parameter mapping on the\n     * API Gateway (see `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct), if for\n     * example you want to forward this to the backend integration.\n     */\n    clientId: string\n    /**\n     * If `CREDENTIALS_FOR_INTERNAL_AUTHORIZATION` is provided, we want to forward basic auth\n     * credentials to our backend, as an additional authentication layer. See the docstring on\n     * `CognitoUserPoolAuthorizerProps.basicAuthForInternalAuthorization` in the `ApiGateway`\n     * construct for more on this.\n     */\n    internalAuthorizationHeader?: string\n  }\n}\n\nexport const handler = async (\n  event: APIGatewayRequestAuthorizerEventV2,\n): Promise<AuthorizerResult> => {\n  const authHeader = event.headers?.authorization\n  if (!authHeader || !authHeader.startsWith(\"Bearer \")) {\n    return { isAuthorized: false }\n  }\n\n  const result = await verifyAccessToken(authHeader.substring(7)) // substring(7) == after 'Bearer '\n  switch (result) {\n    case \"INVALID\":\n      return { isAuthorized: false }\n    case \"EXPIRED\":\n      // We want to return 401 Unauthorized for expired tokens, so the client knows to refresh\n      // their token when receiving this status code. API Gateway authorizer lambdas return\n      // 403 Forbidden for {isAuthorized: false}, but there is a way to return 401: throwing an\n      // error with this exact string. https://stackoverflow.com/a/71965890\n      throw new Error(\"Unauthorized\")\n    default: {\n      return {\n        isAuthorized: true,\n        context: {\n          clientId: result.client_id,\n          internalAuthorizationHeader: await getInternalAuthorizationHeader(),\n        },\n      }\n    }\n  }\n}\n\n/** Decodes and verifies the given token against Cognito. */\nasync function verifyAccessToken(\n  token: string,\n): Promise<CognitoAccessTokenPayload | \"EXPIRED\" | \"INVALID\"> {\n  try {\n    const tokenVerifier = getTokenVerifier()\n    // Must await here instead of returning the promise directly, so that errors can be caught in\n    // this function\n    return await tokenVerifier.verify(token)\n  } catch (e) {\n    // If the JWT has expired, aws-jwt-verify throws this error:\n    // https://github.com/awslabs/aws-jwt-verify/blob/8d8f714d7281913ecd660147f5c30311479601c1/src/jwt.ts#L197\n    // We can't check instanceof on that error class, since it's not exported, so this is the next\n    // best thing.\n    if (e instanceof Error && e.message?.includes(\"Token expired\")) {\n      return \"EXPIRED\"\n    } else {\n      return \"INVALID\"\n    }\n  }\n}\n\nexport type TokenVerifier = {\n  verify: (accessToken: string) => Promise<CognitoAccessTokenPayload>\n}\n\n/**\n * We cache the verifier in this global variable, so that subsequent invocations of a hot lambda\n * will re-use this.\n */\nlet cachedTokenVerifier: TokenVerifier | undefined = undefined\n\nfunction getTokenVerifier(): TokenVerifier {\n  if (cachedTokenVerifier === undefined) {\n    cachedTokenVerifier = dependencies.createTokenVerifier()\n  }\n  return cachedTokenVerifier\n}\n\n/** For overriding dependency creation in tests. */\nexport const dependencies = {\n  createTokenVerifier: (): TokenVerifier => {\n    const userPoolId = process.env[\"USER_POOL_ID\"]\n    if (!userPoolId) {\n      console.error(\"USER_POOL_ID env variable is not defined\")\n      throw new Error()\n    }\n\n    return CognitoJwtVerifier.create({\n      userPoolId,\n      tokenUse: \"access\",\n      clientId: null,\n      scope: process.env.REQUIRED_SCOPE || undefined, // `|| undefined` to discard empty string\n    })\n  },\n  createSecretsManager: () => new SecretsManager(),\n}\n\n/** Cache this value, so that subsequent lambda invocations don't have to refetch. */\nlet cachedInternalAuthorizationHeader: string | undefined = undefined\n\nasync function getInternalAuthorizationHeader(): Promise<string | undefined> {\n  if (cachedInternalAuthorizationHeader === undefined) {\n    const secretName: string | undefined =\n      process.env[\"CREDENTIALS_FOR_INTERNAL_AUTHORIZATION\"]\n    if (!secretName) {\n      return undefined\n    }\n\n    cachedInternalAuthorizationHeader =\n      await getSecretAsBasicAuthHeader(secretName)\n  }\n\n  return cachedInternalAuthorizationHeader\n}\n\nasync function getSecretAsBasicAuthHeader(secretName: string): Promise<string> {\n  const credentials = await getSecretValue(secretName)\n  if (!secretHasExpectedFormat(credentials)) {\n    console.error(\n      `Basic auth credentials secret did not follow expected format (secret name: '${secretName}')`,\n    )\n    throw new Error()\n  }\n\n  return (\n    \"Basic \" +\n    Buffer.from(`${credentials.username}:${credentials.password}`).toString(\n      \"base64\",\n    )\n  )\n}\n\nasync function getSecretValue(secretName: string): Promise<unknown> {\n  const client = dependencies.createSecretsManager()\n  const secret = await client.getSecretValue({ SecretId: secretName })\n\n  if (!secret.SecretString) {\n    console.error(`Secret value not found for '${secretName}'`)\n    throw new Error()\n  }\n\n  return JSON.parse(secret.SecretString)\n}\n\nfunction secretHasExpectedFormat(\n  value: unknown,\n): value is { username: string; password: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"username\" in value &&\n    typeof value.username === \"string\" &&\n    \"password\" in value &&\n    typeof value.password === \"string\"\n  )\n}\n\nexport function clearCache() {\n  cachedTokenVerifier = undefined\n  cachedInternalAuthorizationHeader = undefined\n}\n"]}
|
|
133
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cognito-user-pool-authorizer.js","sourceRoot":"","sources":["../../../src/api-gateway/authorizer-lambdas/cognito-user-pool-authorizer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This lambda verifies access token in Bearer authorization header using Cognito.\n *\n * Expects the following environment variables:\n * - USER_POOL_ID\n * - REQUIRED_SCOPE (optional)\n *   - Set this to require that the access token payload contains the given scope\n * - CREDENTIALS_FOR_INTERNAL_AUTHORIZATION (optional)\n *   - Secret name from which to get basic auth credentials that should be forwarded to backend\n *     integration if authentication succeeds\n *   - Secret value should follow this format: `{\"username\":\"<username>\",\"password\":\"<password>\"}`\n */\n\nimport type {\n  APIGatewayRequestAuthorizerEventV2,\n  APIGatewaySimpleAuthorizerResult,\n} from \"aws-lambda\"\nimport { SecretsManager } from \"@aws-sdk/client-secrets-manager\"\nimport { CognitoJwtVerifier } from \"aws-jwt-verify\"\nimport type { CognitoAccessTokenPayload } from \"aws-jwt-verify/jwt-model\"\n\ntype AuthorizerResult = APIGatewaySimpleAuthorizerResult & {\n  /**\n   * Returning a context object from our authorizer allows our API Gateway to access these variables\n   * via `${context.authorizer.<property>}`.\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html\n   */\n  context?: {\n    /**\n     * If the token is verified, we return the auth client ID from the token's claims as a context\n     * variable (named `authorizer.clientId`). We use this to include the requesting client in the\n     * API Gateway access logs (see `defaultAccessLogFormat` in our `ApiGateway` construct). You can\n     * also use this when mapping parameters to the backend integration (see\n     * `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).\n     */\n    clientId: string\n    /**\n     * If `CREDENTIALS_FOR_INTERNAL_AUTHORIZATION` is provided, we want to forward basic auth\n     * credentials to our backend, as an additional authentication layer. See the docstring on\n     * `CognitoUserPoolAuthorizerProps.basicAuthForInternalAuthorization` in the `ApiGateway`\n     * construct for more on this.\n     */\n    internalAuthorizationHeader?: string\n  }\n}\n\nexport const handler = async (\n  event: APIGatewayRequestAuthorizerEventV2,\n): Promise<AuthorizerResult> => {\n  const authHeader = event.headers?.authorization\n  if (!authHeader || !authHeader.startsWith(\"Bearer \")) {\n    return { isAuthorized: false }\n  }\n\n  const result = await verifyAccessToken(authHeader.substring(7)) // substring(7) == after 'Bearer '\n  switch (result) {\n    case \"INVALID\":\n      return { isAuthorized: false }\n    case \"EXPIRED\":\n      // We want to return 401 Unauthorized for expired tokens, so the client knows to refresh\n      // their token when receiving this status code. API Gateway authorizer lambdas return\n      // 403 Forbidden for {isAuthorized: false}, but there is a way to return 401: throwing an\n      // error with this exact string. https://stackoverflow.com/a/71965890\n      throw new Error(\"Unauthorized\")\n    default: {\n      return {\n        isAuthorized: true,\n        context: {\n          clientId: result.client_id,\n          internalAuthorizationHeader: await getInternalAuthorizationHeader(),\n        },\n      }\n    }\n  }\n}\n\n/** Decodes and verifies the given token against Cognito. */\nasync function verifyAccessToken(\n  token: string,\n): Promise<CognitoAccessTokenPayload | \"EXPIRED\" | \"INVALID\"> {\n  try {\n    const tokenVerifier = getTokenVerifier()\n    // Must await here instead of returning the promise directly, so that errors can be caught in\n    // this function\n    return await tokenVerifier.verify(token)\n  } catch (e) {\n    // If the JWT has expired, aws-jwt-verify throws this error:\n    // https://github.com/awslabs/aws-jwt-verify/blob/8d8f714d7281913ecd660147f5c30311479601c1/src/jwt.ts#L197\n    // We can't check instanceof on that error class, since it's not exported, so this is the next\n    // best thing.\n    if (e instanceof Error && e.message?.includes(\"Token expired\")) {\n      return \"EXPIRED\"\n    } else {\n      return \"INVALID\"\n    }\n  }\n}\n\nexport type TokenVerifier = {\n  verify: (accessToken: string) => Promise<CognitoAccessTokenPayload>\n}\n\n/**\n * We cache the verifier in this global variable, so that subsequent invocations of a hot lambda\n * will re-use this.\n */\nlet cachedTokenVerifier: TokenVerifier | undefined = undefined\n\nfunction getTokenVerifier(): TokenVerifier {\n  if (cachedTokenVerifier === undefined) {\n    cachedTokenVerifier = dependencies.createTokenVerifier()\n  }\n  return cachedTokenVerifier\n}\n\n/** For overriding dependency creation in tests. */\nexport const dependencies = {\n  createTokenVerifier: (): TokenVerifier => {\n    const userPoolId = process.env[\"USER_POOL_ID\"]\n    if (!userPoolId) {\n      console.error(\"USER_POOL_ID env variable is not defined\")\n      throw new Error()\n    }\n\n    return CognitoJwtVerifier.create({\n      userPoolId,\n      tokenUse: \"access\",\n      clientId: null,\n      scope: process.env.REQUIRED_SCOPE || undefined, // `|| undefined` to discard empty string\n    })\n  },\n  createSecretsManager: () => new SecretsManager(),\n}\n\n/** Cache this value, so that subsequent lambda invocations don't have to refetch. */\nlet cachedInternalAuthorizationHeader: string | undefined = undefined\n\nasync function getInternalAuthorizationHeader(): Promise<string | undefined> {\n  if (cachedInternalAuthorizationHeader === undefined) {\n    const secretName: string | undefined =\n      process.env[\"CREDENTIALS_FOR_INTERNAL_AUTHORIZATION\"]\n    if (!secretName) {\n      return undefined\n    }\n\n    cachedInternalAuthorizationHeader =\n      await getSecretAsBasicAuthHeader(secretName)\n  }\n\n  return cachedInternalAuthorizationHeader\n}\n\nasync function getSecretAsBasicAuthHeader(secretName: string): Promise<string> {\n  const credentials = await getSecretValue(secretName)\n  if (!secretHasExpectedFormat(credentials)) {\n    console.error(\n      `Basic auth credentials secret did not follow expected format (secret name: '${secretName}')`,\n    )\n    throw new Error()\n  }\n\n  return (\n    \"Basic \" +\n    Buffer.from(`${credentials.username}:${credentials.password}`).toString(\n      \"base64\",\n    )\n  )\n}\n\nasync function getSecretValue(secretName: string): Promise<unknown> {\n  const client = dependencies.createSecretsManager()\n  const secret = await client.getSecretValue({ SecretId: secretName })\n\n  if (!secret.SecretString) {\n    console.error(`Secret value not found for '${secretName}'`)\n    throw new Error()\n  }\n\n  return JSON.parse(secret.SecretString)\n}\n\nfunction secretHasExpectedFormat(\n  value: unknown,\n): value is { username: string; password: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"username\" in value &&\n    typeof value.username === \"string\" &&\n    \"password\" in value &&\n    typeof value.password === \"string\"\n  )\n}\n\nexport function clearCache() {\n  cachedTokenVerifier = undefined\n  cachedInternalAuthorizationHeader = undefined\n}\n"]}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* This lambda verifies credentials:
|
|
3
|
-
* - Against Cognito user pool if request uses Bearer
|
|
3
|
+
* - Against Cognito user pool if request uses access token in Bearer authorization header
|
|
4
4
|
* - Against credentials saved in Secret Manager if request uses basic auth (and if secret exists)
|
|
5
5
|
*
|
|
6
6
|
* Expects the following environment variables
|
|
@@ -10,7 +10,7 @@
|
|
|
10
10
|
* A different format with an array of pre-encoded credentials is also supported - see docs for
|
|
11
11
|
* the `CognitoUserPoolOrBasicAuthAuthorizerProps` on the `ApiGateway` construct.
|
|
12
12
|
* - REQUIRED_SCOPE (optional)
|
|
13
|
-
* - Set this to require that the
|
|
13
|
+
* - Set this to require that the access token payload contains the given scope
|
|
14
14
|
*/
|
|
15
15
|
import type { APIGatewayRequestAuthorizerEventV2, APIGatewaySimpleAuthorizerResult } from "aws-lambda";
|
|
16
16
|
import { SecretsManager } from "@aws-sdk/client-secrets-manager";
|
|
@@ -19,18 +19,27 @@ type AuthorizerResult = APIGatewaySimpleAuthorizerResult & {
|
|
|
19
19
|
/**
|
|
20
20
|
* Returning a context object from our authorizer allows our API Gateway to access these variables
|
|
21
21
|
* via `${context.authorizer.<property>}`.
|
|
22
|
-
* https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-
|
|
22
|
+
* https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html
|
|
23
23
|
*/
|
|
24
24
|
context?: {
|
|
25
25
|
/**
|
|
26
|
-
* If the token
|
|
27
|
-
* variable (named `authorizer.clientId`).
|
|
28
|
-
*
|
|
29
|
-
*
|
|
26
|
+
* If the request used an access token, and the token was verified, we return the auth client ID
|
|
27
|
+
* from the token's claims in this context variable (named `authorizer.clientId`). We use this
|
|
28
|
+
* to include the requesting client in the API Gateway access logs (see `defaultAccessLogFormat`
|
|
29
|
+
* in our `ApiGateway` construct). You can also use this when mapping parameters to the backend
|
|
30
|
+
* integration (see `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).
|
|
30
31
|
*/
|
|
31
32
|
clientId?: string;
|
|
32
33
|
/**
|
|
33
|
-
*
|
|
34
|
+
* If the request used Basic Auth, and the credentials were verified, we return the username
|
|
35
|
+
* that was used in this context variable (named `authorizer.username`). We use this to include
|
|
36
|
+
* the requesting user in the API Gateway access logs (see `defaultAccessLogFormat` in our
|
|
37
|
+
* `ApiGateway` construct). You can also use this when mapping parameters to the backend
|
|
38
|
+
* integration (see `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).
|
|
39
|
+
*/
|
|
40
|
+
username?: string;
|
|
41
|
+
/**
|
|
42
|
+
* See `CognitoUserPoolAuthorizerProps.basicAuthForInternalAuthorization` on the `ApiGateway`
|
|
34
43
|
* construct (we provide the same context variable here as in the Cognito User Pool authorizer,
|
|
35
44
|
* using the credentials from BASIC_AUTH_CREDENTIALS_SECRET_NAME).
|
|
36
45
|
*/
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* This lambda verifies credentials:
|
|
3
|
-
* - Against Cognito user pool if request uses Bearer
|
|
3
|
+
* - Against Cognito user pool if request uses access token in Bearer authorization header
|
|
4
4
|
* - Against credentials saved in Secret Manager if request uses basic auth (and if secret exists)
|
|
5
5
|
*
|
|
6
6
|
* Expects the following environment variables
|
|
@@ -10,7 +10,7 @@
|
|
|
10
10
|
* A different format with an array of pre-encoded credentials is also supported - see docs for
|
|
11
11
|
* the `CognitoUserPoolOrBasicAuthAuthorizerProps` on the `ApiGateway` construct.
|
|
12
12
|
* - REQUIRED_SCOPE (optional)
|
|
13
|
-
* - Set this to require that the
|
|
13
|
+
* - Set this to require that the access token payload contains the given scope
|
|
14
14
|
*/
|
|
15
15
|
import { SecretsManager } from "@aws-sdk/client-secrets-manager";
|
|
16
16
|
import { CognitoJwtVerifier } from "aws-jwt-verify";
|
|
@@ -19,7 +19,7 @@ export const handler = async (event) => {
|
|
|
19
19
|
if (!authHeader) {
|
|
20
20
|
return { isAuthorized: false };
|
|
21
21
|
}
|
|
22
|
-
const
|
|
22
|
+
const expectedBasicAuthCredentials = await getExpectedBasicAuthCredentials();
|
|
23
23
|
if (authHeader.startsWith("Bearer ")) {
|
|
24
24
|
const result = await verifyAccessToken(authHeader.substring(7)); // substring(7) == after 'Bearer '
|
|
25
25
|
switch (result) {
|
|
@@ -36,19 +36,20 @@ export const handler = async (event) => {
|
|
|
36
36
|
isAuthorized: true,
|
|
37
37
|
context: {
|
|
38
38
|
clientId: result.client_id,
|
|
39
|
-
internalAuthorizationHeader:
|
|
39
|
+
internalAuthorizationHeader: expectedBasicAuthCredentials?.[0]?.basicAuthHeader,
|
|
40
40
|
},
|
|
41
41
|
};
|
|
42
42
|
}
|
|
43
43
|
}
|
|
44
44
|
else if (authHeader.startsWith("Basic ") &&
|
|
45
|
-
|
|
46
|
-
for (const
|
|
47
|
-
if (authHeader ===
|
|
45
|
+
expectedBasicAuthCredentials !== undefined) {
|
|
46
|
+
for (const expected of expectedBasicAuthCredentials) {
|
|
47
|
+
if (authHeader === expected.basicAuthHeader) {
|
|
48
48
|
return {
|
|
49
49
|
isAuthorized: true,
|
|
50
50
|
context: {
|
|
51
|
-
|
|
51
|
+
username: expected.username,
|
|
52
|
+
internalAuthorizationHeader: expected.basicAuthHeader,
|
|
52
53
|
},
|
|
53
54
|
};
|
|
54
55
|
}
|
|
@@ -109,27 +110,27 @@ export const dependencies = {
|
|
|
109
110
|
createSecretsManager: () => new SecretsManager(),
|
|
110
111
|
};
|
|
111
112
|
/** Cache this value, so that subsequent lambda invocations don't have to refetch. */
|
|
112
|
-
let
|
|
113
|
+
let cachedBasicAuthCredentials = undefined;
|
|
113
114
|
/**
|
|
114
|
-
* Returns an array
|
|
115
|
-
*
|
|
115
|
+
* Returns an array, to support credential secrets with multiple values (see
|
|
116
|
+
* `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).
|
|
116
117
|
*/
|
|
117
|
-
async function
|
|
118
|
-
if (
|
|
118
|
+
async function getExpectedBasicAuthCredentials() {
|
|
119
|
+
if (cachedBasicAuthCredentials === undefined) {
|
|
119
120
|
const secretName = process.env["BASIC_AUTH_CREDENTIALS_SECRET_NAME"];
|
|
120
121
|
if (!secretName) {
|
|
121
122
|
return undefined;
|
|
122
123
|
}
|
|
123
|
-
|
|
124
|
+
cachedBasicAuthCredentials = await getBasicAuthCredentialsSecret(secretName);
|
|
124
125
|
}
|
|
125
|
-
return
|
|
126
|
+
return cachedBasicAuthCredentials;
|
|
126
127
|
}
|
|
127
|
-
async function
|
|
128
|
+
async function getBasicAuthCredentialsSecret(secretName) {
|
|
128
129
|
const secret = await getSecretValue(secretName);
|
|
129
130
|
if (isSingleUsernameAndPassword(secret)) {
|
|
130
131
|
const header = "Basic " +
|
|
131
132
|
Buffer.from(`${secret.username}:${secret.password}`).toString("base64");
|
|
132
|
-
return [header];
|
|
133
|
+
return [{ basicAuthHeader: header, username: secret.username }];
|
|
133
134
|
}
|
|
134
135
|
// See `BasicAuthAuthorizerProps` on the `ApiGateway` construct for an explanation of the formats
|
|
135
136
|
// we parse here
|
|
@@ -143,7 +144,7 @@ async function getSecretAsBasicAuthHeaders(secretName) {
|
|
|
143
144
|
throw new Error();
|
|
144
145
|
}
|
|
145
146
|
if (isStringArray(credentialsArray)) {
|
|
146
|
-
return credentialsArray.map(
|
|
147
|
+
return credentialsArray.map(parseEncodedBasicAuthCredentials);
|
|
147
148
|
}
|
|
148
149
|
}
|
|
149
150
|
console.error(`Basic auth credentials secret did not follow any expected format (secret name: '${secretName}')`);
|
|
@@ -160,7 +161,7 @@ async function getSecretValue(secretName) {
|
|
|
160
161
|
return JSON.parse(secret.SecretString);
|
|
161
162
|
}
|
|
162
163
|
catch (e) {
|
|
163
|
-
console.error(`Failed to parse secret '${secretName}' as JSON
|
|
164
|
+
console.error(`Failed to parse secret '${secretName}' as JSON:`, e);
|
|
164
165
|
throw new Error();
|
|
165
166
|
}
|
|
166
167
|
}
|
|
@@ -189,8 +190,33 @@ function isStringArray(value) {
|
|
|
189
190
|
}
|
|
190
191
|
return true;
|
|
191
192
|
}
|
|
193
|
+
/**
|
|
194
|
+
* We want to return the requesting username as a context variable in
|
|
195
|
+
* {@link AuthorizerResult.context}, for API Gateway access logs and parameter mapping. So if the
|
|
196
|
+
* basic auth credentials secret is stored as pre-encoded base64 strings, we need to parse them to
|
|
197
|
+
* get the username.
|
|
198
|
+
*/
|
|
199
|
+
function parseEncodedBasicAuthCredentials(encodedCredentials) {
|
|
200
|
+
let decodedCredentials;
|
|
201
|
+
try {
|
|
202
|
+
decodedCredentials = Buffer.from(encodedCredentials, "base64").toString();
|
|
203
|
+
}
|
|
204
|
+
catch (e) {
|
|
205
|
+
console.error("Basic auth credentials secret could not be decoded as base64:", e);
|
|
206
|
+
throw new Error();
|
|
207
|
+
}
|
|
208
|
+
const usernameAndPassword = decodedCredentials.split(":", 2);
|
|
209
|
+
if (usernameAndPassword.length !== 2) {
|
|
210
|
+
console.error("Basic auth credentials secret could not be decoded as 'username:password'");
|
|
211
|
+
throw new Error();
|
|
212
|
+
}
|
|
213
|
+
return {
|
|
214
|
+
basicAuthHeader: `Basic ${encodedCredentials}`,
|
|
215
|
+
username: usernameAndPassword[0],
|
|
216
|
+
};
|
|
217
|
+
}
|
|
192
218
|
export function clearCache() {
|
|
193
219
|
cachedTokenVerifier = undefined;
|
|
194
|
-
|
|
220
|
+
cachedBasicAuthCredentials = undefined;
|
|
195
221
|
}
|
|
196
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cognito-user-pool-or-basic-auth-authorizer.js","sourceRoot":"","sources":["../../../src/api-gateway/authorizer-lambdas/cognito-user-pool-or-basic-auth-authorizer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This lambda verifies credentials:\n * - Against Cognito user pool if request uses Bearer token\n * - Against credentials saved in Secret Manager if request uses basic auth (and if secret exists)\n *\n * Expects the following environment variables\n * - USER_POOL_ID\n * - BASIC_AUTH_CREDENTIALS_SECRET_NAME (optional)\n *   - Secret value should follow this format: `{\"username\":\"<username>\",\"password\":\"<password>\"}`.\n *     A different format with an array of pre-encoded credentials is also supported - see docs for\n *     the `CognitoUserPoolOrBasicAuthAuthorizerProps` on the `ApiGateway` construct.\n * - REQUIRED_SCOPE (optional)\n *   - Set this to require that the bearer token payload contains the given scope\n */\n\nimport type {\n  APIGatewayRequestAuthorizerEventV2,\n  APIGatewaySimpleAuthorizerResult,\n} from \"aws-lambda\"\nimport { SecretsManager } from \"@aws-sdk/client-secrets-manager\"\nimport { CognitoJwtVerifier } from \"aws-jwt-verify\"\nimport type { CognitoAccessTokenPayload } from \"aws-jwt-verify/jwt-model\"\n\ntype AuthorizerResult = APIGatewaySimpleAuthorizerResult & {\n  /**\n   * Returning a context object from our authorizer allows our API Gateway to access these variables\n   * via `${context.authorizer.<property>}`.\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-logging-variables.html\n   */\n  context?: {\n    /**\n     * If the token is verified, we return the auth client ID from the token's claims as a context\n     * variable (named `authorizer.clientId`). You can then use this for parameter mapping on the\n     * API Gateway (see `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct), if for\n     * example you want to forward this to the backend integration.\n     */\n    clientId?: string\n    /**\n     * See `CognitoUserPoolAuthorizerProps.basicAuthForInternalAuthorization` in the `ApiGateway`\n     * construct (we provide the same context variable here as in the Cognito User Pool authorizer,\n     * using the credentials from BASIC_AUTH_CREDENTIALS_SECRET_NAME).\n     */\n    internalAuthorizationHeader?: string\n  }\n}\n\nexport const handler = async (\n  event: APIGatewayRequestAuthorizerEventV2,\n): Promise<AuthorizerResult> => {\n  const authHeader = event.headers?.authorization\n  if (!authHeader) {\n    return { isAuthorized: false }\n  }\n\n  const expectedBasicAuthHeaders = await getExpectedBasicAuthHeaders()\n\n  if (authHeader.startsWith(\"Bearer \")) {\n    const result = await verifyAccessToken(authHeader.substring(7)) // substring(7) == after 'Bearer '\n    switch (result) {\n      case \"INVALID\":\n        return { isAuthorized: false }\n      case \"EXPIRED\":\n        // We want to return 401 Unauthorized for expired tokens, so the client knows to refresh\n        // their token when receiving this status code. API Gateway authorizer lambdas return\n        // 403 Forbidden for {isAuthorized: false}, but there is a way to return 401: throwing an\n        // error with this exact string. https://stackoverflow.com/a/71965890\n        throw new Error(\"Unauthorized\")\n      default:\n        return {\n          isAuthorized: true,\n          context: {\n            clientId: result.client_id,\n            internalAuthorizationHeader: expectedBasicAuthHeaders?.[0],\n          },\n        }\n    }\n  } else if (\n    authHeader.startsWith(\"Basic \") &&\n    expectedBasicAuthHeaders !== undefined\n  ) {\n    for (const expectedHeader of expectedBasicAuthHeaders) {\n      if (authHeader === expectedHeader) {\n        return {\n          isAuthorized: true,\n          context: {\n            internalAuthorizationHeader: expectedHeader,\n          },\n        }\n      }\n    }\n    return { isAuthorized: false }\n  } else {\n    return { isAuthorized: false }\n  }\n}\n\n/** Decodes and verifies the given token against Cognito. */\nasync function verifyAccessToken(\n  token: string,\n): Promise<CognitoAccessTokenPayload | \"EXPIRED\" | \"INVALID\"> {\n  try {\n    const tokenVerifier = getTokenVerifier()\n    // Must await here instead of returning the promise directly, so that errors can be caught in\n    // this function\n    return await tokenVerifier.verify(token)\n  } catch (e) {\n    // If the JWT has expired, aws-jwt-verify throws this error:\n    // https://github.com/awslabs/aws-jwt-verify/blob/8d8f714d7281913ecd660147f5c30311479601c1/src/jwt.ts#L197\n    // We can't check instanceof on that error class, since it's not exported, so this is the next\n    // best thing.\n    if (e instanceof Error && e.message?.includes(\"Token expired\")) {\n      return \"EXPIRED\"\n    } else {\n      return \"INVALID\"\n    }\n  }\n}\n\nexport type TokenVerifier = {\n  verify: (accessToken: string) => Promise<CognitoAccessTokenPayload>\n}\n\n/**\n * We cache the verifier in this global variable, so that subsequent invocations of a hot lambda\n * will re-use this.\n */\nlet cachedTokenVerifier: TokenVerifier | undefined = undefined\n\nfunction getTokenVerifier(): TokenVerifier {\n  if (cachedTokenVerifier === undefined) {\n    cachedTokenVerifier = dependencies.createTokenVerifier()\n  }\n  return cachedTokenVerifier\n}\n\n/** For overriding dependency creation in tests. */\nexport const dependencies = {\n  createTokenVerifier: (): TokenVerifier => {\n    const userPoolId = process.env[\"USER_POOL_ID\"]\n    if (!userPoolId) {\n      console.error(\"USER_POOL_ID env variable is not defined\")\n      throw new Error()\n    }\n\n    return CognitoJwtVerifier.create({\n      userPoolId,\n      tokenUse: \"access\",\n      clientId: null,\n      scope: process.env.REQUIRED_SCOPE || undefined, // `|| undefined` to discard empty string\n    })\n  },\n  createSecretsManager: () => new SecretsManager(),\n}\n\n/** Cache this value, so that subsequent lambda invocations don't have to refetch. */\nlet cachedBasicAuthHeaders: string[] | undefined = undefined\n\n/**\n * Returns an array of allowed basic auth headers, to support credential secrets with multiple\n * values (see `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).\n */\nasync function getExpectedBasicAuthHeaders(): Promise<string[] | undefined> {\n  if (cachedBasicAuthHeaders === undefined) {\n    const secretName: string | undefined =\n      process.env[\"BASIC_AUTH_CREDENTIALS_SECRET_NAME\"]\n    if (!secretName) {\n      return undefined\n    }\n\n    cachedBasicAuthHeaders = await getSecretAsBasicAuthHeaders(secretName)\n  }\n\n  return cachedBasicAuthHeaders\n}\n\nasync function getSecretAsBasicAuthHeaders(\n  secretName: string,\n): Promise<string[]> {\n  const secret = await getSecretValue(secretName)\n\n  if (isSingleUsernameAndPassword(secret)) {\n    const header =\n      \"Basic \" +\n      Buffer.from(`${secret.username}:${secret.password}`).toString(\"base64\")\n    return [header]\n  }\n\n  // See `BasicAuthAuthorizerProps` on the `ApiGateway` construct for an explanation of the formats\n  // we parse here\n  if (hasCredentialsKeyWithStringValue(secret)) {\n    let credentialsArray: unknown\n    try {\n      credentialsArray = JSON.parse(secret.credentials)\n    } catch (e) {\n      console.error(\n        `Failed to parse credentials array in secret '${secretName}' as JSON`,\n        e,\n      )\n      throw new Error()\n    }\n\n    if (isStringArray(credentialsArray)) {\n      return credentialsArray.map(\n        (encodedCredential) => `Basic ${encodedCredential}`,\n      )\n    }\n  }\n\n  console.error(\n    `Basic auth credentials secret did not follow any expected format (secret name: '${secretName}')`,\n  )\n  throw new Error()\n}\n\nasync function getSecretValue(secretName: string): Promise<unknown> {\n  const client = dependencies.createSecretsManager()\n  const secret = await client.getSecretValue({ SecretId: secretName })\n\n  if (!secret.SecretString) {\n    console.error(`Secret value not found for '${secretName}'`)\n    throw new Error()\n  }\n\n  try {\n    return JSON.parse(secret.SecretString)\n  } catch (e) {\n    console.error(`Failed to parse secret '${secretName}' as JSON`, e)\n    throw new Error()\n  }\n}\n\nfunction isSingleUsernameAndPassword(\n  value: unknown,\n): value is { username: string; password: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"username\" in value &&\n    typeof value.username === \"string\" &&\n    \"password\" in value &&\n    typeof value.password === \"string\"\n  )\n}\n\nfunction hasCredentialsKeyWithStringValue(\n  value: unknown,\n): value is { credentials: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"credentials\" in value &&\n    typeof value.credentials === \"string\"\n  )\n}\n\nfunction isStringArray(value: unknown): value is string[] {\n  if (!Array.isArray(value)) {\n    return false\n  }\n\n  for (const element of value) {\n    if (typeof element !== \"string\") {\n      return false\n    }\n  }\n\n  return true\n}\n\nexport function clearCache() {\n  cachedTokenVerifier = undefined\n  cachedBasicAuthHeaders = undefined\n}\n"]}
|
|
222
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cognito-user-pool-or-basic-auth-authorizer.js","sourceRoot":"","sources":["../../../src/api-gateway/authorizer-lambdas/cognito-user-pool-or-basic-auth-authorizer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This lambda verifies credentials:\n * - Against Cognito user pool if request uses access token in Bearer authorization header\n * - Against credentials saved in Secret Manager if request uses basic auth (and if secret exists)\n *\n * Expects the following environment variables\n * - USER_POOL_ID\n * - BASIC_AUTH_CREDENTIALS_SECRET_NAME (optional)\n *   - Secret value should follow this format: `{\"username\":\"<username>\",\"password\":\"<password>\"}`.\n *     A different format with an array of pre-encoded credentials is also supported - see docs for\n *     the `CognitoUserPoolOrBasicAuthAuthorizerProps` on the `ApiGateway` construct.\n * - REQUIRED_SCOPE (optional)\n *   - Set this to require that the access token payload contains the given scope\n */\n\nimport type {\n  APIGatewayRequestAuthorizerEventV2,\n  APIGatewaySimpleAuthorizerResult,\n} from \"aws-lambda\"\nimport { SecretsManager } from \"@aws-sdk/client-secrets-manager\"\nimport { CognitoJwtVerifier } from \"aws-jwt-verify\"\nimport type { CognitoAccessTokenPayload } from \"aws-jwt-verify/jwt-model\"\n\ntype AuthorizerResult = APIGatewaySimpleAuthorizerResult & {\n  /**\n   * Returning a context object from our authorizer allows our API Gateway to access these variables\n   * via `${context.authorizer.<property>}`.\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html\n   */\n  context?: {\n    /**\n     * If the request used an access token, and the token was verified, we return the auth client ID\n     * from the token's claims in this context variable (named `authorizer.clientId`). We use this\n     * to include the requesting client in the API Gateway access logs (see `defaultAccessLogFormat`\n     * in our `ApiGateway` construct). You can also use this when mapping parameters to the backend\n     * integration (see `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).\n     */\n    clientId?: string\n    /**\n     * If the request used Basic Auth, and the credentials were verified, we return the username\n     * that was used in this context variable (named `authorizer.username`). We use this to include\n     * the requesting user in the API Gateway access logs (see `defaultAccessLogFormat` in our\n     * `ApiGateway` construct). You can also use this when mapping parameters to the backend\n     * integration (see `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).\n     */\n    username?: string\n    /**\n     * See `CognitoUserPoolAuthorizerProps.basicAuthForInternalAuthorization` on the `ApiGateway`\n     * construct (we provide the same context variable here as in the Cognito User Pool authorizer,\n     * using the credentials from BASIC_AUTH_CREDENTIALS_SECRET_NAME).\n     */\n    internalAuthorizationHeader?: string\n  }\n}\n\nexport const handler = async (\n  event: APIGatewayRequestAuthorizerEventV2,\n): Promise<AuthorizerResult> => {\n  const authHeader = event.headers?.authorization\n  if (!authHeader) {\n    return { isAuthorized: false }\n  }\n\n  const expectedBasicAuthCredentials = await getExpectedBasicAuthCredentials()\n\n  if (authHeader.startsWith(\"Bearer \")) {\n    const result = await verifyAccessToken(authHeader.substring(7)) // substring(7) == after 'Bearer '\n    switch (result) {\n      case \"INVALID\":\n        return { isAuthorized: false }\n      case \"EXPIRED\":\n        // We want to return 401 Unauthorized for expired tokens, so the client knows to refresh\n        // their token when receiving this status code. API Gateway authorizer lambdas return\n        // 403 Forbidden for {isAuthorized: false}, but there is a way to return 401: throwing an\n        // error with this exact string. https://stackoverflow.com/a/71965890\n        throw new Error(\"Unauthorized\")\n      default:\n        return {\n          isAuthorized: true,\n          context: {\n            clientId: result.client_id,\n            internalAuthorizationHeader:\n              expectedBasicAuthCredentials?.[0]?.basicAuthHeader,\n          },\n        }\n    }\n  } else if (\n    authHeader.startsWith(\"Basic \") &&\n    expectedBasicAuthCredentials !== undefined\n  ) {\n    for (const expected of expectedBasicAuthCredentials) {\n      if (authHeader === expected.basicAuthHeader) {\n        return {\n          isAuthorized: true,\n          context: {\n            username: expected.username,\n            internalAuthorizationHeader: expected.basicAuthHeader,\n          },\n        }\n      }\n    }\n    return { isAuthorized: false }\n  } else {\n    return { isAuthorized: false }\n  }\n}\n\n/** Decodes and verifies the given token against Cognito. */\nasync function verifyAccessToken(\n  token: string,\n): Promise<CognitoAccessTokenPayload | \"EXPIRED\" | \"INVALID\"> {\n  try {\n    const tokenVerifier = getTokenVerifier()\n    // Must await here instead of returning the promise directly, so that errors can be caught in\n    // this function\n    return await tokenVerifier.verify(token)\n  } catch (e) {\n    // If the JWT has expired, aws-jwt-verify throws this error:\n    // https://github.com/awslabs/aws-jwt-verify/blob/8d8f714d7281913ecd660147f5c30311479601c1/src/jwt.ts#L197\n    // We can't check instanceof on that error class, since it's not exported, so this is the next\n    // best thing.\n    if (e instanceof Error && e.message?.includes(\"Token expired\")) {\n      return \"EXPIRED\"\n    } else {\n      return \"INVALID\"\n    }\n  }\n}\n\nexport type TokenVerifier = {\n  verify: (accessToken: string) => Promise<CognitoAccessTokenPayload>\n}\n\n/**\n * We cache the verifier in this global variable, so that subsequent invocations of a hot lambda\n * will re-use this.\n */\nlet cachedTokenVerifier: TokenVerifier | undefined = undefined\n\nfunction getTokenVerifier(): TokenVerifier {\n  if (cachedTokenVerifier === undefined) {\n    cachedTokenVerifier = dependencies.createTokenVerifier()\n  }\n  return cachedTokenVerifier\n}\n\n/** For overriding dependency creation in tests. */\nexport const dependencies = {\n  createTokenVerifier: (): TokenVerifier => {\n    const userPoolId = process.env[\"USER_POOL_ID\"]\n    if (!userPoolId) {\n      console.error(\"USER_POOL_ID env variable is not defined\")\n      throw new Error()\n    }\n\n    return CognitoJwtVerifier.create({\n      userPoolId,\n      tokenUse: \"access\",\n      clientId: null,\n      scope: process.env.REQUIRED_SCOPE || undefined, // `|| undefined` to discard empty string\n    })\n  },\n  createSecretsManager: () => new SecretsManager(),\n}\n\ntype ExpectedBasicAuthCredentials = {\n  basicAuthHeader: string\n  username: string\n}\n\n/** Cache this value, so that subsequent lambda invocations don't have to refetch. */\nlet cachedBasicAuthCredentials: ExpectedBasicAuthCredentials[] | undefined =\n  undefined\n\n/**\n * Returns an array, to support credential secrets with multiple values (see\n * `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).\n */\nasync function getExpectedBasicAuthCredentials(): Promise<\n  ExpectedBasicAuthCredentials[] | undefined\n> {\n  if (cachedBasicAuthCredentials === undefined) {\n    const secretName: string | undefined =\n      process.env[\"BASIC_AUTH_CREDENTIALS_SECRET_NAME\"]\n    if (!secretName) {\n      return undefined\n    }\n\n    cachedBasicAuthCredentials = await getBasicAuthCredentialsSecret(secretName)\n  }\n\n  return cachedBasicAuthCredentials\n}\n\nasync function getBasicAuthCredentialsSecret(\n  secretName: string,\n): Promise<ExpectedBasicAuthCredentials[]> {\n  const secret = await getSecretValue(secretName)\n\n  if (isSingleUsernameAndPassword(secret)) {\n    const header =\n      \"Basic \" +\n      Buffer.from(`${secret.username}:${secret.password}`).toString(\"base64\")\n    return [{ basicAuthHeader: header, username: secret.username }]\n  }\n\n  // See `BasicAuthAuthorizerProps` on the `ApiGateway` construct for an explanation of the formats\n  // we parse here\n  if (hasCredentialsKeyWithStringValue(secret)) {\n    let credentialsArray: unknown\n    try {\n      credentialsArray = JSON.parse(secret.credentials)\n    } catch (e) {\n      console.error(\n        `Failed to parse credentials array in secret '${secretName}' as JSON`,\n        e,\n      )\n      throw new Error()\n    }\n\n    if (isStringArray(credentialsArray)) {\n      return credentialsArray.map(parseEncodedBasicAuthCredentials)\n    }\n  }\n\n  console.error(\n    `Basic auth credentials secret did not follow any expected format (secret name: '${secretName}')`,\n  )\n  throw new Error()\n}\n\nasync function getSecretValue(secretName: string): Promise<unknown> {\n  const client = dependencies.createSecretsManager()\n  const secret = await client.getSecretValue({ SecretId: secretName })\n\n  if (!secret.SecretString) {\n    console.error(`Secret value not found for '${secretName}'`)\n    throw new Error()\n  }\n\n  try {\n    return JSON.parse(secret.SecretString)\n  } catch (e) {\n    console.error(`Failed to parse secret '${secretName}' as JSON:`, e)\n    throw new Error()\n  }\n}\n\nfunction isSingleUsernameAndPassword(\n  value: unknown,\n): value is { username: string; password: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"username\" in value &&\n    typeof value.username === \"string\" &&\n    \"password\" in value &&\n    typeof value.password === \"string\"\n  )\n}\n\nfunction hasCredentialsKeyWithStringValue(\n  value: unknown,\n): value is { credentials: string } {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"credentials\" in value &&\n    typeof value.credentials === \"string\"\n  )\n}\n\nfunction isStringArray(value: unknown): value is string[] {\n  if (!Array.isArray(value)) {\n    return false\n  }\n\n  for (const element of value) {\n    if (typeof element !== \"string\") {\n      return false\n    }\n  }\n\n  return true\n}\n\n/**\n * We want to return the requesting username as a context variable in\n * {@link AuthorizerResult.context}, for API Gateway access logs and parameter mapping. So if the\n * basic auth credentials secret is stored as pre-encoded base64 strings, we need to parse them to\n * get the username.\n */\nfunction parseEncodedBasicAuthCredentials(\n  encodedCredentials: string,\n): ExpectedBasicAuthCredentials {\n  let decodedCredentials: string\n  try {\n    decodedCredentials = Buffer.from(encodedCredentials, \"base64\").toString()\n  } catch (e) {\n    console.error(\n      \"Basic auth credentials secret could not be decoded as base64:\",\n      e,\n    )\n    throw new Error()\n  }\n\n  const usernameAndPassword = decodedCredentials.split(\":\", 2)\n  if (usernameAndPassword.length !== 2) {\n    console.error(\n      \"Basic auth credentials secret could not be decoded as 'username:password'\",\n    )\n    throw new Error()\n  }\n\n  return {\n    basicAuthHeader: `Basic ${encodedCredentials}`,\n    username: usernameAndPassword[0],\n  }\n}\n\nexport function clearCache() {\n  cachedTokenVerifier = undefined\n  cachedBasicAuthCredentials = undefined\n}\n"]}
|
|
@@ -280,8 +280,8 @@ export type AuthorizationProps<AuthScopesT extends string = string> =
|
|
|
280
280
|
type: "IAM";
|
|
281
281
|
}
|
|
282
282
|
/**
|
|
283
|
-
* Creates a custom authorizer lambda which reads `Authorization: Bearer <token>` header
|
|
284
|
-
* verifies the token against a Cognito user pool.
|
|
283
|
+
* Creates a custom authorizer lambda which reads `Authorization: Bearer <access token>` header
|
|
284
|
+
* and verifies the token against a Cognito user pool.
|
|
285
285
|
*/
|
|
286
286
|
| ({
|
|
287
287
|
type: "COGNITO_USER_POOL";
|
|
@@ -295,8 +295,8 @@ export type AuthorizationProps<AuthScopesT extends string = string> =
|
|
|
295
295
|
} & BasicAuthAuthorizerProps)
|
|
296
296
|
/**
|
|
297
297
|
* Creates a custom authorizer lambda which allows both:
|
|
298
|
-
* - `Authorization: Bearer <token>` header, for which the token is checked against the
|
|
299
|
-
* Cognito user pool
|
|
298
|
+
* - `Authorization: Bearer <access token>` header, for which the token is checked against the
|
|
299
|
+
* given Cognito user pool
|
|
300
300
|
* - `Authorization: Basic <base64-encoded credentials>` header, for which the credentials are
|
|
301
301
|
* checked against the credentials from the given basic auth secret name
|
|
302
302
|
*
|
|
@@ -308,7 +308,7 @@ export type AuthorizationProps<AuthScopesT extends string = string> =
|
|
|
308
308
|
export type CognitoUserPoolAuthorizerProps<AuthScopesT extends string = string> = {
|
|
309
309
|
userPool: IUserPool;
|
|
310
310
|
/**
|
|
311
|
-
* Verifies that token claims contain the given scope.
|
|
311
|
+
* Verifies that access token claims contain the given scope.
|
|
312
312
|
*
|
|
313
313
|
* When defined as part of a resource server, scopes are on the format:
|
|
314
314
|
* `{resource server identifier}/{scope name}`, e.g. `external/view_users`.
|
|
@@ -400,8 +400,8 @@ export type CognitoUserPoolOrBasicAuthAuthorizerProps<AuthScopesT extends string
|
|
|
400
400
|
*/
|
|
401
401
|
basicAuthCredentialsSecretName?: string;
|
|
402
402
|
/**
|
|
403
|
-
* Verifies that token claims contain the given scope. Only applicable for
|
|
404
|
-
*
|
|
403
|
+
* Verifies that access token claims contain the given scope. Only applicable for requests that
|
|
404
|
+
* use `Authorization: Bearer <access token>` (not applicable for basic auth).
|
|
405
405
|
*
|
|
406
406
|
* When defined as part of a resource server, scopes are on the format:
|
|
407
407
|
* `{resource server identifier}/{scope name}`, e.g. `external/view_users`.
|
|
@@ -406,7 +406,7 @@ const authorizerFileExtension = __dirname.endsWith("src/api-gateway")
|
|
|
406
406
|
? "ts"
|
|
407
407
|
: "js";
|
|
408
408
|
/**
|
|
409
|
-
* Creates a custom authorizer lambda which reads `Authorization: Bearer <token>` header and
|
|
409
|
+
* Creates a custom authorizer lambda which reads `Authorization: Bearer <access token>` header and
|
|
410
410
|
* verifies the token against a Cognito user pool.
|
|
411
411
|
*/
|
|
412
412
|
class CognitoUserPoolAuthorizer extends constructs.Construct {
|
|
@@ -466,7 +466,7 @@ class BasicAuthAuthorizer extends constructs.Construct {
|
|
|
466
466
|
}
|
|
467
467
|
/**
|
|
468
468
|
* Creates a custom authorizer lambda which allows both:
|
|
469
|
-
* - `Authorization: Bearer <token>` header, for which the token is checked against the given
|
|
469
|
+
* - `Authorization: Bearer <access token>` header, for which the token is checked against the given
|
|
470
470
|
* Cognito user pool
|
|
471
471
|
* - `Authorization: Basic <base64-encoded credentials>` header, for which the credentials are
|
|
472
472
|
* checked against the credentials from the given basic auth secret name
|
|
@@ -520,7 +520,6 @@ const defaultAccessLogFormat = {
|
|
|
520
520
|
responseLength: "$context.responseLength",
|
|
521
521
|
responseLatency: "$context.responseLatency",
|
|
522
522
|
domainName: "$context.domainName",
|
|
523
|
-
// hostHeaderOverride: "$context.requestOverride.header.Host", //Mapping template overrides cannot be used with proxy integration endpoints, which lack data mappings https://docs.aws.amazon.com/apigateway/latest/developerguide/apigateway-override-request-response-parameters.html#:~:text=Mapping%20template%20overrides%20cannot%20be%20used%20with%20proxy%20integration%20endpoints%2C%20which%20lack%20data%20mappings
|
|
524
523
|
error: {
|
|
525
524
|
type: "$context.error.responseType",
|
|
526
525
|
gatewayError: "$context.error.message",
|
|
@@ -539,13 +538,13 @@ const defaultAccessLogFormat = {
|
|
|
539
538
|
awsPrincipal: "$context.identity.caller",
|
|
540
539
|
awsPrincipalOrg: "$context.identity.principalOrgId",
|
|
541
540
|
},
|
|
542
|
-
basic: {
|
|
541
|
+
basic: { username: "$context.authorizer.username" },
|
|
542
|
+
cognito: { clientId: "$context.authorizer.clientId" },
|
|
543
543
|
},
|
|
544
544
|
awsEndpointRequest: {
|
|
545
545
|
id: "$context.awsEndpointRequestId",
|
|
546
546
|
id2: "$context.awsEndpointRequestId2",
|
|
547
547
|
},
|
|
548
|
-
// For datadog
|
|
549
548
|
message: "$context.identity.sourceIp - $context.httpMethod $context.domainName $context.path ($context.routeKey) - $context.status [$context.responseLatency ms]",
|
|
550
549
|
};
|
|
551
550
|
/**
|
|
@@ -586,4 +585,4 @@ function shortHash(str) {
|
|
|
586
585
|
// which is fine
|
|
587
586
|
return createHash("sha1").update(str).digest("hex").substring(0, 10);
|
|
588
587
|
}
|
|
589
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http-api-gateway.js","sourceRoot":"","sources":["../../src/api-gateway/http-api-gateway.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as cdk from \"aws-cdk-lib\"\nimport type * as elb from \"aws-cdk-lib/aws-elasticloadbalancingv2\"\nimport type * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport type * as sqs from \"aws-cdk-lib/aws-sqs\"\nimport * as apigw from \"aws-cdk-lib/aws-apigatewayv2\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as secretsmanager from \"aws-cdk-lib/aws-secretsmanager\"\nimport * as integrations from \"aws-cdk-lib/aws-apigatewayv2-integrations\"\nimport * as authorizers from \"aws-cdk-lib/aws-apigatewayv2-authorizers\"\nimport type { IUserPool } from \"aws-cdk-lib/aws-cognito\"\nimport * as lambdaNodejs from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport { createHash } from \"crypto\"\nimport * as route53 from \"aws-cdk-lib/aws-route53\"\nimport * as route53Targets from \"aws-cdk-lib/aws-route53-targets\"\nimport * as acm from \"aws-cdk-lib/aws-certificatemanager\"\nimport { tagResources } from \"../tags\"\nimport * as path from \"path\"\nimport { fileURLToPath } from \"url\"\n\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\n/**\n * Props for the {@link ApiGateway} construct.\n *\n * @author Kristian Rekstad <kre@capraconsulting.no>\n * @author Hermann Mørkrid <hem@liflig.no>\n */\nexport type ApiGatewayProps<AuthScopesT extends string = string> = {\n  /** Settings for the external-facing part of the API-GW. */\n  dns: ApiGatewayDnsProps\n\n  /**\n   * If no integration is specified for a route, this integration is used.\n   *\n   * See {@link IntegrationProps} for the available options.\n   */\n  defaultIntegration?: IntegrationProps\n\n  /**\n   * If no authorization is specified for a route, this authorization is used.\n   *\n   * See {@link AuthorizationProps} for the available options.\n   */\n  defaultAuthorization?: AuthorizationProps<AuthScopesT>\n\n  routes: ApiGatewayRoute<AuthScopesT>[]\n\n  /**\n   * The API-GW access logs for the `$default` stage are kept.\n   * This has options for the logs.\n   */\n  accessLogs?: ApiGatewayAccessLogsProps\n\n  /**\n   * Set to false to disable route-level metrics.\n   * This can increase CloudWatch costs when not disabled.\n   *\n   * See [AWS: Working with metrics for HTTP APIs](https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-metrics.html?icmpid=apigateway_console_help#:~:text=and%20stage%20ID.-,ApiId%2C%20Stage%2C%20Route,-Filters%20API%20Gateway)\n   * for more info.\n   *\n   * @default true\n   */\n  detailedMetrics?: boolean\n\n  /**\n   * Throttling of requests.\n   * If not set, the AWS default of 5000 burst and 10000 rate is used.\n   *\n   * The default throttling is per-account, and counts all APIs in the account and region.\n   * If you have another API in this region getting 10_000 request rate, it may impact this API as well.\n   */\n  throttling?: {\n    /**\n     * Going over `5_000` may require you to contact AWS Support - they are account bound.\n     */\n    burst?: number\n    /**\n     * Going over `10_000` may require you to contact AWS Support - they are account bound.\n     */\n    rate?: number\n  }\n\n  /**\n   * Sets CORS headers on responses from the API Gateway to allow all origins, headers and methods.\n   * Useful if your API should be accessed by any browser.\n   */\n  corsAllowAll?: boolean\n\n  /**\n   * If some settings in this construct do not work for you, this is an escape hatch mechanism to\n   * override anything.\n   */\n  propsOverride?: {\n    /**\n     * Override settings for the {@link apigw.HttpApi} (accessible in {@link ApiGateway.httpApi}).\n     *\n     * For example, if you have a frontend accessing this API, you might want to set\n     * [CORS preflight](https://docs.aws.amazon.com/cdk/api/v1/docs/aws-apigatewayv2-readme.html#cross-origin-resource-sharing-cors)\n     * settings.\n     */\n    httpApi?: Partial<apigw.HttpApiProps>\n  }\n}\n\nexport type ApiGatewayDnsProps = {\n  /**\n   * Only the subdomain prefix, which should be the name of the service.\n   * Example: Subdomain `product` would give the end result of an API-GW with\n   * `product.platform.example.no`.\n   */\n  subdomain: string\n\n  /**\n   * Hosted Zone for the external facing domain.\n   * This is where routes will be created, to redirect consumers to the API-GW.\n   * For example a HZ for `platform.example.no`.\n   */\n  hostedZone: route53.IHostedZone\n\n  /**\n   * The Time To Live (TTL) for the public DNS A record that will expose the API-GW.\n   * This is how long DNS servers will cache the record.\n   *\n   * A long TTL (hours) is beneficial to DNS servers, but makes developers (you) wait longer when\n   * doing changes.\n   *\n   * @default 5 minutes\n   */\n  ttl?: cdk.Duration\n}\n\nexport type ApiGatewayRoute<AuthScopesT extends string = string> = {\n  /** The path of the route to expose through the API Gateway. Use \"/\" for the root route. */\n  path: string\n\n  /**\n   * By default, we only forward requests that match the route's path exactly. So for a route with\n   * path `/api/users`, a request to `/api/users` will be forwarded, but a request to\n   * `/api/users/admin` will not. If you want to forward requests to all sub-paths under the route's\n   * path, you can set this to true.\n   *\n   * @default false\n   */\n  includeSubpaths?: boolean\n\n  /**\n   * The HTTP method to expose. `ANY` exposes all HTTP methods on the path.\n   *\n   * @default \"ANY\"\n   */\n  method?: HttpMethod\n\n  /**\n   * The integration that the route will forward to. See {@link IntegrationProps} for the available\n   * options.\n   *\n   * If undefined, uses the {@link ApiGatewayProps.defaultIntegration}.\n   */\n  integration?: IntegrationProps\n\n  /**\n   * How requests on the route are authenticated. See {@link AuthorizationProps} for the available\n   * options.\n   *\n   * If undefined, uses the {@link ApiGatewayProps.defaultAuthorization}.\n   */\n  authorization?: AuthorizationProps<AuthScopesT>\n}\n\nexport type HttpMethod =\n  | \"ANY\"\n  | \"GET\"\n  | \"POST\"\n  | \"PUT\"\n  | \"PATCH\"\n  | \"DELETE\"\n  | \"OPTIONS\"\n  | \"HEAD\"\n\nexport type IntegrationProps =\n  /** Use this when connecting the route to an ALB (Application Load Balancer). */\n  | ({ type: \"ALB\" } & AlbIntegrationProps)\n  /** Use this when connecting route to a Lambda. */\n  | ({ type: \"Lambda\" } & LambdaIntegrationProps)\n  /** Use this when connecting a route to send to an SQS queue. */\n  | ({ type: \"SQS\" } & SqsIntegrationProps)\n\n/**\n * Props for the API-GW -> ALB (Application Load Balancer) integration.\n *\n * See the note on {@link ApiGateway} about the load balancer security group.\n */\nexport type AlbIntegrationProps = {\n  /**\n   * A listener on e.g. port 443 (HTTPS).\n   *\n   * See the note on {@link ApiGateway} about the load balancer security group.\n   */\n  loadBalancerListener: elb.IApplicationListener\n\n  /**\n   * The host name (domain name) of the backend service that we want to reach through the ALB.\n   *\n   * This is used to:\n   * - Verify the HTTPS certificate of the backend service, so that the request forwarded from\n   *   API-GW can use TLS\n   * - Set the `Host` header on the request when forwarding to the ALB, so that requests can be\n   *   routed to the correct Target Group\n   *\n   * Example value: `<service>.staging.my-project.liflig.io` (not prefixed by `https://`).\n   */\n  hostName: string\n\n  /**\n   * The VPC used by the ALB. The API-GW integration will connect to the ALB using a VPC Link for\n   * this VPC.\n   *\n   * See the note on {@link ApiGateway} about the load balancer security group.\n   */\n  vpc: ec2.IVpc\n\n  /**\n   * A security group (SG) that allows incoming traffic to the ALB. Will be used by the VPC link, so\n   * the API-GW integration can connect.\n   *\n   * This is usually the same SG as the ALB uses, because they have a rule that allows traffic from\n   * others in the same SG.\n   *\n   * See the note on {@link ApiGateway} about the load balancer security group.\n   */\n  securityGroup: ec2.ISecurityGroup\n\n  /**\n   * Map request parameters (add/overwrite path/headers/query params) before forwarding to the\n   * backend.\n   *\n   * See {@link https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html}\n   * for more on this. Read the 'Reserved headers' section for which headers cannot be overridden.\n   * In addition to the AWS-reserved headers, you should not override the 'Host' header either, as\n   * that's used for routing the request to the correct service behind the load balancer.\n   *\n   * ### Example:\n   *\n   * Adding a header:\n   * ```\n   * mapParameters: (parameters) => parameters.overwriteHeader(\n   *    \"X-My-Custom-Header\",\n   *    apigw.MappingValue.custom(\"my-custom-value\"),\n   * )\n   * ```\n   *\n   * Overwriting the path (if, for example, you configure a `/users` route on the API Gateway that\n   * you want to forward to `/api/users` on the backend):\n   * ```\n   * mapParameters: (parameters) => parameters.overwritePath(\"/api/users\")\n   * ```\n   */\n  mapParameters?: (parameters: apigw.ParameterMapping) => void\n}\n\nexport type LambdaIntegrationProps = {\n  /**\n   * The Lambda integration uses the V2 payload format:\n   * {@link https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-develop-integrations-lambda.html}\n   *\n   * If writing the Lambda in TypeScript, this means you should use `APIGatewayProxyEventV2` as the\n   * request type, and `APIGatewayProxyResultV2` as the response type.\n   */\n  lambda: lambda.IFunction\n}\n\nexport type SqsIntegrationProps = {\n  queue: sqs.IQueue\n\n  /**\n   * Message attributes to pass on to SQS. The keys in this object are the names of the attributes.\n   * Each attribute has a DataType field, and either a StringValue or BinaryValue field depending on\n   * its type. See AWS docs:\n   * https://docs.aws.amazon.com/AWSSimpleQueueService/latest/APIReference/API_MessageAttributeValue.html\n   *\n   * In the StringValue field, you can do API Gateway parameter mapping:\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html\n   *\n   * Example:\n   * ```\n   * messageAttributes: {\n   *   clientId: {\n   *     DataType: \"String\",\n   *     StringValue: \"${context.authorizer.clientId}\",\n   *   },\n   * },\n   * ```\n   */\n  messageAttributes?: {\n    [attributeName: string]:\n      | { DataType: \"String\" | \"Number\"; StringValue: string }\n      | {\n          DataType: \"Binary\"\n          /** Base64-encoded binary data object. */\n          BinaryValue: string\n        }\n  }\n}\n\nexport type AuthorizationProps<AuthScopesT extends string = string> =\n  /**\n   * No authentication, for when you want a fully public route (or handle authentication in the\n   * backend integration).\n   */\n  | { type: \"NONE\" }\n  /**\n   * AWS IAM authorization.\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-access-control-iam.html\n   */\n  | { type: \"IAM\" }\n  /**\n   * Creates a custom authorizer lambda which reads `Authorization: Bearer <token>` header and\n   * verifies the token against a Cognito user pool.\n   */\n  | ({\n      type: \"COGNITO_USER_POOL\"\n    } & CognitoUserPoolAuthorizerProps<AuthScopesT>)\n  /**\n   * Creates a custom authorizer lambda which reads `Authorization: Basic <base64-encoded credentials>`\n   * header and verifies the credentials against a given secret.\n   */\n  | ({ type: \"BASIC_AUTH\" } & BasicAuthAuthorizerProps)\n  /**\n   * Creates a custom authorizer lambda which allows both:\n   * - `Authorization: Bearer <token>` header, for which the token is checked against the given\n   *   Cognito user pool\n   * - `Authorization: Basic <base64-encoded credentials>` header, for which the credentials are\n   *   checked against the credentials from the given basic auth secret name\n   *\n   * If either of these are given and valid, the request is authenticated.\n   */\n  | ({\n      type: \"COGNITO_USER_POOL_OR_BASIC_AUTH\"\n    } & CognitoUserPoolOrBasicAuthAuthorizerProps<AuthScopesT>)\n\nexport type CognitoUserPoolAuthorizerProps<\n  AuthScopesT extends string = string,\n> = {\n  userPool: IUserPool\n\n  /**\n   * Verifies that token claims contain the given scope.\n   *\n   * When defined as part of a resource server, scopes are on the format:\n   * `{resource server identifier}/{scope name}`, e.g. `external/view_users`.\n   *\n   * To get more type safety on this parameter, see the docs for the `AuthScopesT` type parameter on\n   * {@link ApiGateway}.\n   */\n  requiredScope?: AuthScopesT\n\n  /**\n   * Name of secret in AWS Secrets Manager that stores basic auth credentials for the backend\n   * service, to be forwarded to the backend if Cognito user pool authentication succeeded.\n   *\n   * The secret value must follow this format:\n   * ```json\n   * { \"username\": \"<username>\", \"password\": \"<password>\" }\n   * ```\n   *\n   * This prop solves the following use-case:\n   * - You want to do Cognito user pool authentication in the API Gateway\n   * - You want an additional auth check in the backend, but you don't want to deal with Cognito\n   *   there\n   * - The backend uses basic auth\n   *\n   * This prop solves this by letting you specify credentials to pass to the backend after API-GW\n   * authentication succeeds. You can pass the encoded credentials through\n   * {@link AlbIntegrationProps.mapParameters}, using the `authorizer.internalAuthorizationHeader`\n   * context variable, like so:\n   * ```\n   * mapParameters: (parameters) => parameters.overwriteHeader(\n   *   // 'Authorization' header cannot be overridden, so we use a custom header\n   *   \"X-Internal-Authorization\",\n   *   apigw.MappingValue.contextVariable(\"authorizer.internalAuthorizationHeader\"),\n   * )\n   * ```\n   * The backend can then check the `X-Internal-Authorization` header.\n   */\n  credentialsForInternalAuthorization?: string\n}\n\nexport type BasicAuthAuthorizerProps = {\n  /**\n   * Name of secret in AWS Secrets Manager that stores basic auth credentials. The secret value\n   * should follow this format:\n   * ```json\n   * { \"username\": \"<username>\", \"password\": \"<password>\" }\n   * ```\n   *\n   * The following format is also supported:\n   * ```json\n   * { \"credentials\": \"[\\\"<encoded-credential-1>\\\",\\\"<encoded-credential-2>\\\"]\" }\n   * ```\n   * ...consisting of:\n   * - A single key, `credentials`\n   * - With a _string_ value\n   * - Which is a stringified, escaped JSON array of base64-encoded credentials\n   * - In which each element is encoded from `<username>:<password>`\n   *\n   * If the secret is on this format, the authorizer will match the request's Authorization header\n   * against any one of these encoded credentials.\n   *\n   * The reason that this second format stores stringified JSON _inside_ JSON, is due to a\n   * limitation in Liflig's `load-secrets` library, which only allows storing strings values.\n   */\n  credentialsSecretName: string\n}\n\nexport type CognitoUserPoolOrBasicAuthAuthorizerProps<\n  AuthScopesT extends string = string,\n> = {\n  userPool: IUserPool\n\n  /**\n   * Name of secret in AWS Secrets Manager that stores basic auth credentials. The secret value\n   * should follow this format:\n   * ```json\n   * { \"username\": \"<username>\", \"password\": \"<password>\" }\n   * ```\n   *\n   * The following format is also supported:\n   * ```json\n   * { \"credentials\": \"[\\\"<encoded-credential-1>\\\",\\\"<encoded-credential-2>\\\"]\" }\n   * ```\n   * ...consisting of:\n   * - A single key, `credentials`\n   * - With a _string_ value\n   * - Which is a stringified, escaped JSON array of base64-encoded credentials\n   * - In which each element is encoded from `<username>:<password>`\n   *\n   * If the secret is on this format, the authorizer will match the request's Authorization header\n   * against any one of these encoded credentials.\n   *\n   * The reason that this second format stores stringified JSON _inside_ JSON, is due to a\n   * limitation in Liflig's `load-secrets` library, which only allows storing strings values.\n   */\n  basicAuthCredentialsSecretName?: string\n\n  /**\n   * Verifies that token claims contain the given scope. Only applicable for `Bearer` token requests\n   * checked against the Cognito User Pool (not applicable for basic auth).\n   *\n   * When defined as part of a resource server, scopes are on the format:\n   * `{resource server identifier}/{scope name}`, e.g. `external/view_users`.\n   *\n   * To get more type safety on this parameter, see the docs for the `AuthScopesT` type parameter on\n   * {@link ApiGateway}.\n   */\n  requiredScope?: AuthScopesT\n}\n\nexport type ApiGatewayAccessLogsProps = {\n  /**\n   * Delete the access logs if this construct is deleted?\n   *\n   * Maybe you want to DESTROY instead. Or for legal reasons, retain for audit.\n   *\n   * @default RemovalPolicy.RETAIN\n   */\n  removalPolicy?: cdk.RemovalPolicy\n\n  /**\n   * How long to keep the logs. If undefined, uses the same default as new AWS log groups.\n   *\n   * @default RetentionDays.TWO_YEARS\n   */\n  retention?: logs.RetentionDays\n\n  /**\n   * A custom JSON log format, which uses variables from `\"$context\"`.\n   *\n   * See [AWS: CloudWatch log formats for API Gateway](https://docs.aws.amazon.com/apigateway/latest/developerguide/set-up-logging.html#apigateway-cloudwatch-log-formats)\n   * for formats and rules. It is possible to use other formats like CLF and XML, but this construct\n   * only supports JSON for now.\n   *\n   * For a list of all possible variables to log, see\n   * [AWS: $context Variables for data models, authorizers, mapping templates, and CloudWatch access logging](https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-mapping-template-reference.html#context-variable-reference)\n   * and\n   * [AWS: $context Variables for access logging only](https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-mapping-template-reference.html#context-variable-reference-access-logging-only) .\n   *\n   * @default {@link defaultAccessLogFormat}\n   */\n  accessLogFormat?: Record<string, string>\n}\n\n/**\n * This construct tries to simplify the creation of an API Gateway for a service, by collecting most\n * of the common setup here.\n *\n * The approach followed in this construct is:\n * 1. One API-GW per service\n * 2. One subdomain per API-GW / service\n * 3. Use HTTP API, not REST\n * 4. Use a $default stage with autodeploy\n * 5. Support multiple routes (with possible `/{proxy+}` to let all sub-paths through)\n * 6. Allow custom integration/authorizer for each route, or defaults for the whole gateway\n *\n * The route integration is one of these:\n * - ALB private integration with VPC Link using HTTPS to the ALB\n * - Lambda integration\n * - SQS integration\n *\n * ### Load Balancer Security Group\n *\n * Note that the load balancer used in an {@link AlbIntegrationProps} must allow outbound HTTPS\n * traffic to its SecurityGroup. Otherwise, the VPC Link used by the API-GW can't get traffic from\n * the ALB.\n *\n * ```\n * const loadBalancerSecurityGroup = new ec2.SecurityGroup(..., {\n *   allowAllOutbound: false,\n * })\n *\n * loadBalancerSecurityGroup.addEgressRule(\n *   loadBalancerSecurityGroup,\n *   ec2.Port.tcp(443),\n *   \"Outbound to self for ALB to API-GW VPC-Link\",\n * )\n *\n * const loadBalancer = new lifligLoadBalancer.LoadBalancer(...,\n *   {\n *     overrideLoadBalancerProps: {\n *       securityGroup: loadBalancerSecurityGroup,\n *     },\n *   },\n * )\n * ```\n *\n * @template AuthScopesT This type parameter allows you to improve type safety on the\n * `requiredScope` field on {@link CognitoUserPoolOrBasicAuthAuthorizerProps}, by narrowing the type\n * to specific strings. You can then extend the `ApiGateway` with this type to enforce those scopes\n * across the application. Remember that auth scopes must be on the format\n * `{resource server identifier}/{scope name}`.\n *\n * Example:\n * ```\n * type AuthScopes = \"external/read_users\" | \"internal/create_users\"\n *\n * export class MyProjectApiGateway extends ApiGateway<AuthScopes> {}\n * ```\n * TypeScript will then enforce that `requiredScope` is one of `AuthScopes`, and provide\n * auto-complete.\n *\n * @author Kristian Rekstad <kre@capraconsulting.no>\n * @author Hermann Mørkrid <hem@liflig.no>\n */\nexport class ApiGateway<\n  AuthScopesT extends string = string,\n> extends constructs.Construct {\n  /** The API Gateway HTTP API. This is the main construct for API-GW. */\n  public readonly httpApi: apigw.HttpApi\n\n  /** The routes which connect the {@link httpApi} to the backend integration(s). */\n  public readonly routes: apigw.HttpRoute[] = []\n\n  /** The domain which consumers must use.*/\n  public readonly domain: string\n\n  /** Access log group. */\n  public readonly logGroup: logs.LogGroup\n\n  private readonly props: ApiGatewayProps<AuthScopesT>\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ApiGatewayProps<AuthScopesT>,\n  ) {\n    super(scope, id)\n    this.props = props\n\n    ApiGateway.validateProps(props, id, cdk.Stack.of(this))\n\n    const customDomain = new CustomDomain(this, \"CustomDomain\", props.dns)\n    this.domain = customDomain.customDomainName\n\n    let corsOptions: apigw.CorsPreflightOptions | undefined\n    if (props.corsAllowAll) {\n      corsOptions = {\n        allowOrigins: [\"*\"],\n        // \"The Authorization header can't be wildcarded and always needs to be listed explicitly\"\n        // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Access-Control-Allow-Headers\n        // Content-Type may also require explicit whitelisting: https://stackoverflow.com/a/63567647\n        allowHeaders: [\"Authorization\", \"Content-Type\", \"*\"],\n        // Not using '*' here, because \"In requests with credentials, it is treated as the literal\n        // method name '*' without special semantics\"\n        // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Access-Control-Allow-Methods\n        allowMethods: [\n          apigw.CorsHttpMethod.GET,\n          apigw.CorsHttpMethod.POST,\n          apigw.CorsHttpMethod.PUT,\n          apigw.CorsHttpMethod.PATCH,\n          apigw.CorsHttpMethod.DELETE,\n          apigw.CorsHttpMethod.OPTIONS,\n          apigw.CorsHttpMethod.HEAD,\n        ],\n        maxAge: cdk.Duration.days(1),\n      }\n    }\n\n    // The actual API. This holds the routes, authorizers and integrations.\n    const api = new apigw.HttpApi(this, \"HttpApi-\" + props.dns.subdomain, {\n      // defaultIntegration: defaultIntegration, // This is for a catch-all $default route\n      description: `An HTTP API for ${props.dns.subdomain}.${props.dns.hostedZone.zoneName}.`,\n      disableExecuteApiEndpoint: true, // Force externals to go through custom domain. MUST be true when using Mutual TLS, for security reasons\n      createDefaultStage: true,\n      defaultDomainMapping: { domainName: customDomain.apiGwCustomDomain },\n      corsPreflight: corsOptions,\n      ...props?.propsOverride?.httpApi,\n    })\n    this.httpApi = api\n\n    const stage = api.defaultStage?.node.defaultChild as apigw.CfnStage\n\n    const logs = new ApiGatewayAccessLogs(\n      this,\n      \"AccessLogs\",\n      stage,\n      props.accessLogs,\n      defaultAccessLogFormat,\n    )\n    this.logGroup = logs.logGroup\n\n    stage.defaultRouteSettings = {\n      ...stage.defaultRouteSettings,\n      detailedMetricsEnabled: props.detailedMetrics ?? true,\n      throttlingBurstLimit: props.throttling?.burst, // Default for account is 5_000\n      throttlingRateLimit: props.throttling?.rate, // Default for account is 10_000\n    }\n\n    const defaultIntegration = props.defaultIntegration\n      ? this.createIntegration(props.defaultIntegration, props.dns)\n      : undefined\n\n    const defaultAuthorizer = props.defaultAuthorization\n      ? this.createAuthorizer(\"DefaultAuthorizer\", props.defaultAuthorization)\n      : undefined\n\n    for (const route of props.routes) {\n      let integration: apigw.HttpRouteIntegration\n      if (route.integration) {\n        integration = this.createIntegration(route.integration, props.dns)\n      } else {\n        integration = defaultIntegration! // Verified in validateProps\n      }\n\n      let authorizer: apigw.IHttpRouteAuthorizer | undefined\n      if (route.authorization) {\n        authorizer = this.createAuthorizer(\n          // Using the route path here caused CloudFormation to fail due to resource names being too\n          // long. Therefore, we now hash the route path to get a shorter name.\n          `RouteAuthorizer${shortHash(`${route.method ?? \"\"}${route.path}`)}`,\n          route.authorization,\n        )\n      } else {\n        authorizer = defaultAuthorizer\n      }\n\n      const routePaths = [route.path]\n      if (route.includeSubpaths === true) {\n        // If we include sub-paths, we add an additional route with /{proxy+} (the + means it will\n        // match all subroutes). We want both this route and the normal route without /{proxy+},\n        // since /{proxy+} will only match the base path with trailing slash.\n        routePaths.push(\n          route.path + (route.path === \"/\" ? \"\" : \"/\") + \"{proxy+}\",\n        )\n      }\n\n      for (const routePath of routePaths) {\n        // The previous version of the API Gateway construct had a single route with the ID\n        // 'DefaultProxyRoute', and all gateways we used exposed the route /{proxy+}. When trying to\n        // change to this new version of the construct, deployment failed for gateways exposing\n        // /{proxy+}, because \"a route with that key [i.e. path + method] already exists\". We think\n        // this may be because we used the same route path, but with a new route ID, causing\n        // confusion for CloudFormation. So we keep the old route ID here for /{proxy+}.\n        const routeId =\n          routePath === \"/{proxy+}\" ? \"DefaultProxyRoute\" : `Route-${routePath}`\n\n        this.routes.push(\n          new apigw.HttpRoute(this, routeId, {\n            httpApi: api,\n            integration: integration,\n            authorizer: authorizer,\n            routeKey: apigw.HttpRouteKey.with(\n              routePath,\n              route.method\n                ? apigw.HttpMethod[route.method]\n                : apigw.HttpMethod.ANY,\n            ),\n          }),\n        )\n      }\n    }\n\n    tagResources(this, () => ({ service: props.dns.subdomain }))\n  }\n\n  /** @throws Error */\n  private static validateProps(\n    props: ApiGatewayProps,\n    id: string,\n    stack: cdk.Stack,\n  ) {\n    for (const route of props.routes) {\n      if (!route.integration && !props.defaultIntegration) {\n        throw new Error(\n          `No integration defined for route '${route.path}', and no default integration specified for the gateway`,\n        )\n      }\n      if (!route.authorization && !props.defaultAuthorization) {\n        throw new Error(\n          `No authorization defined for route '${route.path}', and no default authorization specified for the gateway`,\n        )\n      }\n      if (!route.path.startsWith(\"/\")) {\n        throw new Error(\n          `Invalid path '${route.path}': paths must begin with '/' (use '/' for root path)`,\n        )\n      }\n      if (route.path !== \"/\" && route.path.endsWith(\"/\")) {\n        throw new Error(\n          `Invalid path '${route.path}': paths cannot end with '/' (except root path)`,\n        )\n      }\n      if (\n        route.integration?.type === \"ALB\" &&\n        route.integration.hostName.startsWith(\"https:\")\n      ) {\n        throw new Error(\n          `The albIntegration.hostHttpsName should not include a protocol: ${route.integration.hostName}`,\n        )\n      }\n    }\n    if (props.dns.subdomain === \"\" || props.dns.subdomain.includes(\" \")) {\n      throw new Error(\n        `Subdomain must be set, and not contain spaces: ${props.dns.subdomain}`,\n      )\n    }\n    if (props.throttling?.burst && props.throttling.burst > 5_000) {\n      console.warn(\n        `⚠️ Your throttling burst limit '${props.throttling.burst}' is higher than the AWS Account limit. Make sure your account has upgraded this quota! `,\n        stack,\n        id,\n      )\n    }\n    if (props.throttling?.rate && props.throttling.rate > 10_000) {\n      console.warn(\n        `⚠️ Your throttling rate limit '${props.throttling.rate}' is higher than the AWS Account limit. Make sure your account has upgraded this quota! `,\n        stack,\n        id,\n      )\n    }\n  }\n\n  /**\n   * The authorizer only accepts requests from external users that are authorized.\n   * Unauthorized users are stopped in the API-GW, and not forwarded to the integration.\n   */\n  private createAuthorizer<AuthScopesT extends string>(\n    id: string,\n    authorization: AuthorizationProps<AuthScopesT>,\n  ): apigw.IHttpRouteAuthorizer | undefined {\n    switch (authorization.type) {\n      case \"NONE\": {\n        return undefined\n      }\n      case \"IAM\": {\n        // API Gateway invokes your API route only if the client has execute-api permission for the route.\n        // The client has to use AWS SigV4 to identify themselves in the request.\n        // Read this page for help with IAM and API-GW https://docs.aws.amazon.com/apigateway/latest/developerguide/security_iam_service-with-iam.html\n        // Note that [resource policies are not supported yet for HTTP](https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-access-control-iam.html)\n        return new authorizers.HttpIamAuthorizer()\n      }\n      case \"COGNITO_USER_POOL\": {\n        // We use a custom lambda authorizer here instead of the `HttpUserPoolAuthorizer` provided\n        // by CDK, in order to support `requiredScope` and setting of custom context variables.\n        const authorizer = new CognitoUserPoolAuthorizer(\n          this,\n          id + \"Lambda\",\n          authorization,\n        )\n\n        return new authorizers.HttpLambdaAuthorizer(id, authorizer.lambda, {\n          responseTypes: authorizer.responseTypes,\n          // Max 1h, or disabled (0s). The value only matters when invalidating/dynamic credentials\n          resultsCacheTtl: cdk.Duration.hours(1),\n        })\n      }\n      case \"BASIC_AUTH\": {\n        const authorizer = new BasicAuthAuthorizer(\n          this,\n          id + \"Lambda\",\n          authorization,\n        )\n\n        return new authorizers.HttpLambdaAuthorizer(id, authorizer.lambda, {\n          responseTypes: authorizer.responseTypes,\n          // Max 1h, or disabled (0s). The value only matters when invalidating/dynamic credentials\n          resultsCacheTtl: cdk.Duration.minutes(30),\n        })\n      }\n      case \"COGNITO_USER_POOL_OR_BASIC_AUTH\": {\n        const authorizer = new CognitoUserPoolOrBasicAuthAuthorizer(\n          this,\n          id + \"Lambda\",\n          authorization,\n        )\n\n        return new authorizers.HttpLambdaAuthorizer(id, authorizer.lambda, {\n          responseTypes: authorizer.responseTypes,\n          // Max 1h, or disabled (0s). The value only matters when invalidating/dynamic credentials\n          resultsCacheTtl: cdk.Duration.hours(1),\n        })\n      }\n    }\n  }\n\n  private createIntegration(\n    integration: IntegrationProps,\n    dns: ApiGatewayProps[\"dns\"],\n  ): apigw.HttpRouteIntegration {\n    switch (integration.type) {\n      case \"ALB\": {\n        // The VPCLink connects the integration into the ALB's VPC. Must be manually created when\n        // the VPC is imported, which is the case when using CorePlatformConsumer.\n        const vpcLink = new apigw.VpcLink(this, \"AlbVpcLink\", {\n          vpc: integration.vpc,\n          securityGroups: [integration.securityGroup],\n          subnets: integration.vpc.selectSubnets(), // This correctly selects the private subnets\n        })\n\n        const parameterMapping = new apigw.ParameterMapping()\n          /** See {@link AlbIntegrationProps.hostName} */\n          .overwriteHeader(\n            \"Host\",\n            // The Host header can NOT use dynamic mapping, like $context.domainName etc.\n            apigw.MappingValue.custom(integration.hostName),\n          )\n        if (integration.mapParameters !== undefined) {\n          integration.mapParameters(parameterMapping)\n        }\n\n        return new integrations.HttpAlbIntegration(\n          \"AlbIntegration-\" + dns.subdomain,\n          integration.loadBalancerListener,\n          {\n            secureServerName: integration.hostName,\n            vpcLink: vpcLink,\n            method: apigw.HttpMethod.ANY,\n            parameterMapping,\n          },\n        )\n      }\n      case \"Lambda\": {\n        return new integrations.HttpLambdaIntegration(\n          \"LambdaIntegration\",\n          integration.lambda,\n          {\n            payloadFormatVersion: apigw.PayloadFormatVersion.VERSION_2_0,\n            parameterMapping: undefined,\n          },\n        )\n      }\n      case \"SQS\": {\n        // API-GW does not have access to SQS by default\n        const role = new iam.Role(\n          this,\n          `ApiGwTo${integration.queue.node.id}ServiceRole`,\n          {\n            description:\n              \"Allows API-GW to add messages to \" + integration.queue.queueArn,\n            assumedBy: new iam.ServicePrincipal(\"apigateway.amazonaws.com\"),\n          },\n        )\n        integration.queue.grantSendMessages(role)\n\n        let parameterMapping = new apigw.ParameterMapping()\n          // https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-develop-integrations-aws-services-reference.html#SQS-SendMessage\n          .custom(\"QueueUrl\", integration.queue.queueUrl)\n          .custom(\"MessageBody\", \"$request.body\")\n          .custom(\"Region\", \"eu-west-1\") // Change this if the SQS queue is in another region!\n\n        if (integration.messageAttributes) {\n          parameterMapping = parameterMapping.custom(\n            \"MessageAttributes\",\n            JSON.stringify(integration.messageAttributes),\n          )\n        }\n\n        return new SqsRouteIntegration(\"SqsIntegration\", {\n          type: apigw.HttpIntegrationType.AWS_PROXY,\n          subtype: apigw.HttpIntegrationSubtype.SQS_SEND_MESSAGE,\n          credentials: apigw.IntegrationCredentials.fromRole(role),\n          parameterMapping: parameterMapping,\n          payloadFormatVersion: apigw.PayloadFormatVersion.VERSION_1_0,\n        })\n      }\n    }\n  }\n\n  /**\n   * Allows a grantable target (role, user etc.) permission to invoke the API.\n   * Only works when using `IAM` as {@link ApiGatewayRoute.authorization}.\n   *\n   * @param target A grantable, like {@link iam.Role}\n   */\n  public grantInvoke(target: iam.IGrantable) {\n    for (const routeProps of this.props.routes) {\n      const authType =\n        routeProps.authorization?.type ?? this.props.defaultAuthorization?.type\n\n      if (authType !== \"IAM\") {\n        throw new Error(\n          `Cannot grant invoke for an API Gateway when not using IAM auth (found auth '${authType}' on route '${routeProps.path}') [${cdk.Stack.of(this).stackName}]`,\n        )\n      }\n    }\n\n    for (const route of this.routes) {\n      route.grantInvoke(target)\n    }\n  }\n}\n\n/**\n * Creates a custom domain for the API-Gateway, a Route53 record and an HTTPS cert.\n * @author Kristian Rekstad <kre@capraconsulting.no>\n */\nclass CustomDomain extends constructs.Construct {\n  public readonly apiGwCustomDomain: apigw.DomainName\n\n  /** The Fully Qualified Domain Name (FQDN) like `product.platform.example.no`. */\n  public readonly customDomainName: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ApiGatewayDnsProps,\n  ) {\n    super(scope, id)\n    this.customDomainName = `${props.subdomain}.${props.hostedZone.zoneName}`\n\n    // Can also use wildcard certs instead! Cheaper\n    /** Allows external users to connect with HTTPS. */\n    const customDomainCert = new acm.Certificate(this, \"HttpsCertificate\", {\n      domainName: this.customDomainName,\n      validation: acm.CertificateValidation.fromDns(props.hostedZone),\n    })\n\n    // Note that API-GW can also support wildcard domains! https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-custom-domain-names.html#http-wildcard-custom-domain-names\n    // But this will not work when AWS account X has CustomDomain `staging.platform.example.no` and account Y has CustomDomain `*.platform.example.no`.\n    // Not sure how sub-subdomains are affected: `myservice.staging.platform.example.no` and `*.platform.example.no`.\n    this.apiGwCustomDomain = new apigw.DomainName(\n      this,\n      \"DomainName-\" + props.subdomain,\n      {\n        domainName: this.customDomainName,\n        certificate: customDomainCert,\n        endpointType: apigw.EndpointType.REGIONAL,\n        securityPolicy: apigw.SecurityPolicy.TLS_1_2,\n      },\n    )\n\n    // This makes the API-GW publicly available on the custom domain name.\n    new route53.ARecord(this, \"Route53ARecordApigwAlias\", {\n      recordName: props.subdomain,\n      zone: props.hostedZone,\n      target: route53.RecordTarget.fromAlias(\n        new route53Targets.ApiGatewayv2DomainProperties(\n          this.apiGwCustomDomain.regionalDomainName,\n          this.apiGwCustomDomain.regionalHostedZoneId,\n        ),\n      ),\n      ttl: props.ttl ?? cdk.Duration.minutes(5), // Low TTL makes it easier to do changes\n    })\n  }\n}\n\n/** Acts as glue (between the integration props and the HttpApi) when creating an SqsIntegration. */\nclass SqsRouteIntegration extends apigw.HttpRouteIntegration {\n  /**\n   * @param id The id used in the {@link apigw.HttpIntegration} construct\n   *    created internally by {@link apigw.HttpRouteIntegration._bindToRoute}.\n   *    [Source code](https://github.com/aws/aws-cdk/blob/b5ae37782bc3cb637eeef9fbb1fbe2c5efdfc068/packages/%40aws-cdk/aws-apigatewayv2/lib/http/integration.ts#L321)\n   * @param integrationProps The props to pass to the {@link apigw.HttpIntegration} construct.\n   */\n  constructor(\n    id: string,\n    private integrationProps: apigw.HttpRouteIntegrationConfig,\n  ) {\n    super(id)\n  }\n\n  /**\n   * This sends the properties needed for creating a {@link apigw.HttpIntegration} to the\n   * {@link apigw.HttpRouteIntegration}.\n   */\n  bind(): apigw.HttpRouteIntegrationConfig {\n    // This method is called by:\n    // https://github.com/aws/aws-cdk/blob/b5ae37782bc3cb637eeef9fbb1fbe2c5efdfc068/packages/%40aws-cdk/aws-apigatewayv2/lib/http/integration.ts#L319\n    return this.integrationProps\n  }\n}\n\n/**\n * When using `NodejsFunction` with `entry`, we point CDK to a file with our lambda code. This can\n * either be a TypeScript or JavaScript file. When creating constructs in tests inside this library,\n * the entry will point to the TypeScript file in the source code. But when a library consumer uses\n * this, it will instead point to the transpiled JavaScript file. So to set the correct file\n * extension, we must check if we're being run in the context of the TypeScript source code\n * (src/api-gateway), otherwise we're being run by a consumer as transpiled JavaScript.\n */\nconst authorizerFileExtension = __dirname.endsWith(\"src/api-gateway\")\n  ? \"ts\"\n  : \"js\"\n\n/**\n * Creates a custom authorizer lambda which reads `Authorization: Bearer <token>` header and\n * verifies the token against a Cognito user pool.\n */\nclass CognitoUserPoolAuthorizer<\n  AuthScopesT extends string,\n> extends constructs.Construct {\n  public readonly lambda: lambda.IFunction\n\n  /**\n   * Use simple response type (`{ isAuthorized: true/false }`), as opposed to returning an IAM\n   * Policy document.\n   */\n  public readonly responseTypes: authorizers.HttpLambdaResponseType[] = [\n    authorizers.HttpLambdaResponseType.SIMPLE,\n  ]\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: CognitoUserPoolAuthorizerProps<AuthScopesT>,\n  ) {\n    super(scope, id)\n\n    this.lambda = new lambdaNodejs.NodejsFunction(this, \"AuthorizerFunction\", {\n      entry: path.join(\n        __dirname,\n        `authorizer-lambdas/cognito-user-pool-authorizer.${authorizerFileExtension}`,\n      ),\n      runtime: lambda.Runtime.NODEJS_22_X,\n      timeout: cdk.Duration.seconds(5),\n      environment: {\n        [\"USER_POOL_ID\"]: props.userPool.userPoolId,\n        [\"REQUIRED_SCOPE\"]: props.requiredScope ?? \"\",\n        [\"CREDENTIALS_FOR_INTERNAL_AUTHORIZATION\"]:\n          props.credentialsForInternalAuthorization\n            ? props.credentialsForInternalAuthorization\n            : \"\",\n      },\n    })\n\n    if (props.credentialsForInternalAuthorization) {\n      secretsmanager.Secret.fromSecretNameV2(\n        scope,\n        id + \"BasicAuthSecret\",\n        props.credentialsForInternalAuthorization,\n      ).grantRead(this.lambda)\n    }\n  }\n}\n\n/**\n * Creates a custom authorizer lambda which reads `Authorization: Basic <base64-encoded credentials>`\n * header and verifies the credentials against a given secret.\n */\nclass BasicAuthAuthorizer extends constructs.Construct {\n  public readonly lambda: lambda.IFunction\n\n  // Simple is `{ isAuthorized: true/false }`, as opposed to returning an IAM Policy document\n  public readonly responseTypes: authorizers.HttpLambdaResponseType[] = [\n    authorizers.HttpLambdaResponseType.SIMPLE,\n  ]\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: BasicAuthAuthorizerProps,\n  ) {\n    super(scope, id)\n\n    this.lambda = new lambdaNodejs.NodejsFunction(this, \"BasicAuthLambda\", {\n      entry: path.join(\n        __dirname,\n        `authorizer-lambdas/basic-auth-authorizer.${authorizerFileExtension}`,\n      ),\n      description:\n        \"An authorizer for API-Gateway that checks Basic Auth credentials on requests\",\n      runtime: lambda.Runtime.NODEJS_22_X,\n      environment: {\n        [\"CREDENTIALS_SECRET_NAME\"]: props.credentialsSecretName\n          ? props.credentialsSecretName\n          : \"\",\n      },\n    })\n\n    if (props.credentialsSecretName) {\n      secretsmanager.Secret.fromSecretNameV2(\n        scope,\n        id + \"BasicAuthSecret\",\n        props.credentialsSecretName,\n      ).grantRead(this.lambda)\n    }\n  }\n}\n\n/**\n * Creates a custom authorizer lambda which allows both:\n * - `Authorization: Bearer <token>` header, for which the token is checked against the given\n *   Cognito user pool\n * - `Authorization: Basic <base64-encoded credentials>` header, for which the credentials are\n *   checked against the credentials from the given basic auth secret name\n *\n * If either of these are given and valid, the request is authenticated.\n */\nclass CognitoUserPoolOrBasicAuthAuthorizer<\n  AuthScopesT extends string,\n> extends constructs.Construct {\n  public readonly lambda: lambda.IFunction\n\n  /**\n   * Use simple response type (`{ isAuthorized: true/false }`), as opposed to returning an IAM\n   * Policy document.\n   */\n  public readonly responseTypes: authorizers.HttpLambdaResponseType[] = [\n    authorizers.HttpLambdaResponseType.SIMPLE,\n  ]\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: CognitoUserPoolOrBasicAuthAuthorizerProps<AuthScopesT>,\n  ) {\n    super(scope, id)\n\n    this.lambda = new lambdaNodejs.NodejsFunction(this, \"AuthorizerFunction\", {\n      entry: path.join(\n        __dirname,\n        `authorizer-lambdas/cognito-user-pool-or-basic-auth-authorizer.${authorizerFileExtension}`,\n      ),\n      runtime: lambda.Runtime.NODEJS_22_X,\n      timeout: cdk.Duration.seconds(5),\n      environment: {\n        [\"USER_POOL_ID\"]: props.userPool ? props.userPool.userPoolId : \"\",\n        [\"REQUIRED_SCOPE\"]: props.requiredScope ?? \"\",\n        [\"BASIC_AUTH_CREDENTIALS_SECRET_NAME\"]:\n          props.basicAuthCredentialsSecretName\n            ? props.basicAuthCredentialsSecretName\n            : \"\",\n      },\n    })\n\n    if (props.basicAuthCredentialsSecretName) {\n      secretsmanager.Secret.fromSecretNameV2(\n        scope,\n        id + \"BasicAuthSecret\",\n        props.basicAuthCredentialsSecretName,\n      ).grantRead(this.lambda)\n    }\n  }\n}\n\n/**\n * A slightly extended version of the [default JSON format suggested by AWS](https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-logging.html#http-api-enable-logging.examples).\n */\nconst defaultAccessLogFormat = {\n  requestId: \"$context.requestId\",\n  userAgent: \"$context.identity.userAgent\",\n  ip: \"$context.identity.sourceIp\",\n  /** CLF format: `dd/MMM/yyyy:HH:mm:ss +-hhmm` */\n  requestTime: \"$context.requestTime\",\n  requestTimeEpoch: \"$context.requestTimeEpoch\",\n  dataProcessed: \"$context.dataProcessed\",\n  httpMethod: \"$context.httpMethod\",\n  path: \"$context.path\",\n  routeKey: \"$context.routeKey\",\n  status: \"$context.status\",\n  protocol: \"$context.protocol\",\n  responseLength: \"$context.responseLength\",\n  responseLatency: \"$context.responseLatency\",\n  domainName: \"$context.domainName\",\n  //  hostHeaderOverride: \"$context.requestOverride.header.Host\", //Mapping template overrides cannot be used with proxy integration endpoints, which lack data mappings https://docs.aws.amazon.com/apigateway/latest/developerguide/apigateway-override-request-response-parameters.html#:~:text=Mapping%20template%20overrides%20cannot%20be%20used%20with%20proxy%20integration%20endpoints%2C%20which%20lack%20data%20mappings\n  error: {\n    type: \"$context.error.responseType\",\n    gatewayError: \"$context.error.message\",\n    integrationError: \"$context.integration.error\",\n    authorizerError: \"$context.authorizer.error\",\n  },\n  integration: {\n    latency: \"$context.integration.latency\",\n    requestId: \"$context.integration.requestId\",\n    responseStatus: \"$context.integration.status\",\n  },\n  auth: {\n    iam: {\n      userArn: \"$context.identity.userArn\",\n      awsAccount: \"$context.identity.accountId\",\n      awsPrincipal: \"$context.identity.caller\",\n      awsPrincipalOrg: \"$context.identity.principalOrgId\",\n    },\n    basic: { user: \"$context.authorizer.user\" },\n  },\n  awsEndpointRequest: {\n    id: \"$context.awsEndpointRequestId\",\n    id2: \"$context.awsEndpointRequestId2\",\n  },\n  // For datadog\n  message:\n    \"$context.identity.sourceIp - $context.httpMethod $context.domainName $context.path ($context.routeKey) - $context.status [$context.responseLatency ms]\",\n}\n\n/**\n * Enables access logs on the API-Gateway.\n *\n * @author Kristian Rekstad <kre@capraconsulting.no>\n */\nclass ApiGatewayAccessLogs extends constructs.Construct {\n  public readonly logGroup: logs.LogGroup\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    stage: apigw.CfnStage,\n    props: ApiGatewayAccessLogsProps | undefined,\n    defaultAccessLogFormat: Record<string, unknown>,\n  ) {\n    super(scope, id)\n\n    // logGroup is set up with help from: https://github.com/aws/aws-cdk/issues/11100#issuecomment-904627081\n    // Not sure if HTTP API actually needs the service role with managed policy: https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-logging.html\n    const accessLogs = new logs.LogGroup(this, \"AccessLogGroup\", {\n      retention: props?.retention,\n      removalPolicy: props?.removalPolicy ?? cdk.RemovalPolicy.RETAIN,\n      // Always use the default encryption key. Otherwise, the key needs special policies:\n      // https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/encrypt-log-data-kms.html#cmk-permissions\n      encryptionKey: undefined,\n    })\n    this.logGroup = accessLogs\n\n    stage.accessLogSettings = {\n      destinationArn: accessLogs.logGroupArn,\n      format: JSON.stringify(props?.accessLogFormat ?? defaultAccessLogFormat),\n    }\n\n    const apiGwLogsRole = new iam.Role(\n      this,\n      \"ApiGatewayPushToCloudwatchLogsRole\",\n      {\n        assumedBy: new iam.ServicePrincipal(\"apigateway.amazonaws.com\"),\n        managedPolicies: [\n          iam.ManagedPolicy.fromAwsManagedPolicyName(\n            \"service-role/AmazonAPIGatewayPushToCloudWatchLogs\",\n          ),\n        ],\n      },\n    )\n\n    accessLogs.grantWrite(apiGwLogsRole)\n  }\n}\n\n/** Returns a short semi-unique hash of the given string. */\nfunction shortHash(str: string): string {\n  // SHA-1 is no-no when we need cryptographic security, but here we just it for shortening a name,\n  // which is fine\n  return createHash(\"sha1\").update(str).digest(\"hex\").substring(0, 10)\n}\n"]}
|
|
588
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http-api-gateway.js","sourceRoot":"","sources":["../../src/api-gateway/http-api-gateway.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as cdk from \"aws-cdk-lib\"\nimport type * as elb from \"aws-cdk-lib/aws-elasticloadbalancingv2\"\nimport type * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport type * as sqs from \"aws-cdk-lib/aws-sqs\"\nimport * as apigw from \"aws-cdk-lib/aws-apigatewayv2\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as secretsmanager from \"aws-cdk-lib/aws-secretsmanager\"\nimport * as integrations from \"aws-cdk-lib/aws-apigatewayv2-integrations\"\nimport * as authorizers from \"aws-cdk-lib/aws-apigatewayv2-authorizers\"\nimport type { IUserPool } from \"aws-cdk-lib/aws-cognito\"\nimport * as lambdaNodejs from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport { createHash } from \"crypto\"\nimport * as route53 from \"aws-cdk-lib/aws-route53\"\nimport * as route53Targets from \"aws-cdk-lib/aws-route53-targets\"\nimport * as acm from \"aws-cdk-lib/aws-certificatemanager\"\nimport { tagResources } from \"../tags\"\nimport * as path from \"path\"\nimport { fileURLToPath } from \"url\"\n\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\n/**\n * Props for the {@link ApiGateway} construct.\n *\n * @author Kristian Rekstad <kre@capraconsulting.no>\n * @author Hermann Mørkrid <hem@liflig.no>\n */\nexport type ApiGatewayProps<AuthScopesT extends string = string> = {\n  /** Settings for the external-facing part of the API-GW. */\n  dns: ApiGatewayDnsProps\n\n  /**\n   * If no integration is specified for a route, this integration is used.\n   *\n   * See {@link IntegrationProps} for the available options.\n   */\n  defaultIntegration?: IntegrationProps\n\n  /**\n   * If no authorization is specified for a route, this authorization is used.\n   *\n   * See {@link AuthorizationProps} for the available options.\n   */\n  defaultAuthorization?: AuthorizationProps<AuthScopesT>\n\n  routes: ApiGatewayRoute<AuthScopesT>[]\n\n  /**\n   * The API-GW access logs for the `$default` stage are kept.\n   * This has options for the logs.\n   */\n  accessLogs?: ApiGatewayAccessLogsProps\n\n  /**\n   * Set to false to disable route-level metrics.\n   * This can increase CloudWatch costs when not disabled.\n   *\n   * See [AWS: Working with metrics for HTTP APIs](https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-metrics.html?icmpid=apigateway_console_help#:~:text=and%20stage%20ID.-,ApiId%2C%20Stage%2C%20Route,-Filters%20API%20Gateway)\n   * for more info.\n   *\n   * @default true\n   */\n  detailedMetrics?: boolean\n\n  /**\n   * Throttling of requests.\n   * If not set, the AWS default of 5000 burst and 10000 rate is used.\n   *\n   * The default throttling is per-account, and counts all APIs in the account and region.\n   * If you have another API in this region getting 10_000 request rate, it may impact this API as well.\n   */\n  throttling?: {\n    /**\n     * Going over `5_000` may require you to contact AWS Support - they are account bound.\n     */\n    burst?: number\n    /**\n     * Going over `10_000` may require you to contact AWS Support - they are account bound.\n     */\n    rate?: number\n  }\n\n  /**\n   * Sets CORS headers on responses from the API Gateway to allow all origins, headers and methods.\n   * Useful if your API should be accessed by any browser.\n   */\n  corsAllowAll?: boolean\n\n  /**\n   * If some settings in this construct do not work for you, this is an escape hatch mechanism to\n   * override anything.\n   */\n  propsOverride?: {\n    /**\n     * Override settings for the {@link apigw.HttpApi} (accessible in {@link ApiGateway.httpApi}).\n     *\n     * For example, if you have a frontend accessing this API, you might want to set\n     * [CORS preflight](https://docs.aws.amazon.com/cdk/api/v1/docs/aws-apigatewayv2-readme.html#cross-origin-resource-sharing-cors)\n     * settings.\n     */\n    httpApi?: Partial<apigw.HttpApiProps>\n  }\n}\n\nexport type ApiGatewayDnsProps = {\n  /**\n   * Only the subdomain prefix, which should be the name of the service.\n   * Example: Subdomain `product` would give the end result of an API-GW with\n   * `product.platform.example.no`.\n   */\n  subdomain: string\n\n  /**\n   * Hosted Zone for the external facing domain.\n   * This is where routes will be created, to redirect consumers to the API-GW.\n   * For example a HZ for `platform.example.no`.\n   */\n  hostedZone: route53.IHostedZone\n\n  /**\n   * The Time To Live (TTL) for the public DNS A record that will expose the API-GW.\n   * This is how long DNS servers will cache the record.\n   *\n   * A long TTL (hours) is beneficial to DNS servers, but makes developers (you) wait longer when\n   * doing changes.\n   *\n   * @default 5 minutes\n   */\n  ttl?: cdk.Duration\n}\n\nexport type ApiGatewayRoute<AuthScopesT extends string = string> = {\n  /** The path of the route to expose through the API Gateway. Use \"/\" for the root route. */\n  path: string\n\n  /**\n   * By default, we only forward requests that match the route's path exactly. So for a route with\n   * path `/api/users`, a request to `/api/users` will be forwarded, but a request to\n   * `/api/users/admin` will not. If you want to forward requests to all sub-paths under the route's\n   * path, you can set this to true.\n   *\n   * @default false\n   */\n  includeSubpaths?: boolean\n\n  /**\n   * The HTTP method to expose. `ANY` exposes all HTTP methods on the path.\n   *\n   * @default \"ANY\"\n   */\n  method?: HttpMethod\n\n  /**\n   * The integration that the route will forward to. See {@link IntegrationProps} for the available\n   * options.\n   *\n   * If undefined, uses the {@link ApiGatewayProps.defaultIntegration}.\n   */\n  integration?: IntegrationProps\n\n  /**\n   * How requests on the route are authenticated. See {@link AuthorizationProps} for the available\n   * options.\n   *\n   * If undefined, uses the {@link ApiGatewayProps.defaultAuthorization}.\n   */\n  authorization?: AuthorizationProps<AuthScopesT>\n}\n\nexport type HttpMethod =\n  | \"ANY\"\n  | \"GET\"\n  | \"POST\"\n  | \"PUT\"\n  | \"PATCH\"\n  | \"DELETE\"\n  | \"OPTIONS\"\n  | \"HEAD\"\n\nexport type IntegrationProps =\n  /** Use this when connecting the route to an ALB (Application Load Balancer). */\n  | ({ type: \"ALB\" } & AlbIntegrationProps)\n  /** Use this when connecting route to a Lambda. */\n  | ({ type: \"Lambda\" } & LambdaIntegrationProps)\n  /** Use this when connecting a route to send to an SQS queue. */\n  | ({ type: \"SQS\" } & SqsIntegrationProps)\n\n/**\n * Props for the API-GW -> ALB (Application Load Balancer) integration.\n *\n * See the note on {@link ApiGateway} about the load balancer security group.\n */\nexport type AlbIntegrationProps = {\n  /**\n   * A listener on e.g. port 443 (HTTPS).\n   *\n   * See the note on {@link ApiGateway} about the load balancer security group.\n   */\n  loadBalancerListener: elb.IApplicationListener\n\n  /**\n   * The host name (domain name) of the backend service that we want to reach through the ALB.\n   *\n   * This is used to:\n   * - Verify the HTTPS certificate of the backend service, so that the request forwarded from\n   *   API-GW can use TLS\n   * - Set the `Host` header on the request when forwarding to the ALB, so that requests can be\n   *   routed to the correct Target Group\n   *\n   * Example value: `<service>.staging.my-project.liflig.io` (not prefixed by `https://`).\n   */\n  hostName: string\n\n  /**\n   * The VPC used by the ALB. The API-GW integration will connect to the ALB using a VPC Link for\n   * this VPC.\n   *\n   * See the note on {@link ApiGateway} about the load balancer security group.\n   */\n  vpc: ec2.IVpc\n\n  /**\n   * A security group (SG) that allows incoming traffic to the ALB. Will be used by the VPC link, so\n   * the API-GW integration can connect.\n   *\n   * This is usually the same SG as the ALB uses, because they have a rule that allows traffic from\n   * others in the same SG.\n   *\n   * See the note on {@link ApiGateway} about the load balancer security group.\n   */\n  securityGroup: ec2.ISecurityGroup\n\n  /**\n   * Map request parameters (add/overwrite path/headers/query params) before forwarding to the\n   * backend.\n   *\n   * See {@link https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html}\n   * for more on this. Read the 'Reserved headers' section for which headers cannot be overridden.\n   * In addition to the AWS-reserved headers, you should not override the 'Host' header either, as\n   * that's used for routing the request to the correct service behind the load balancer.\n   *\n   * ### Example:\n   *\n   * Adding a header:\n   * ```\n   * mapParameters: (parameters) => parameters.overwriteHeader(\n   *    \"X-My-Custom-Header\",\n   *    apigw.MappingValue.custom(\"my-custom-value\"),\n   * )\n   * ```\n   *\n   * Overwriting the path (if, for example, you configure a `/users` route on the API Gateway that\n   * you want to forward to `/api/users` on the backend):\n   * ```\n   * mapParameters: (parameters) => parameters.overwritePath(\"/api/users\")\n   * ```\n   */\n  mapParameters?: (parameters: apigw.ParameterMapping) => void\n}\n\nexport type LambdaIntegrationProps = {\n  /**\n   * The Lambda integration uses the V2 payload format:\n   * {@link https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-develop-integrations-lambda.html}\n   *\n   * If writing the Lambda in TypeScript, this means you should use `APIGatewayProxyEventV2` as the\n   * request type, and `APIGatewayProxyResultV2` as the response type.\n   */\n  lambda: lambda.IFunction\n}\n\nexport type SqsIntegrationProps = {\n  queue: sqs.IQueue\n\n  /**\n   * Message attributes to pass on to SQS. The keys in this object are the names of the attributes.\n   * Each attribute has a DataType field, and either a StringValue or BinaryValue field depending on\n   * its type. See AWS docs:\n   * https://docs.aws.amazon.com/AWSSimpleQueueService/latest/APIReference/API_MessageAttributeValue.html\n   *\n   * In the StringValue field, you can do API Gateway parameter mapping:\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html\n   *\n   * Example:\n   * ```\n   * messageAttributes: {\n   *   clientId: {\n   *     DataType: \"String\",\n   *     StringValue: \"${context.authorizer.clientId}\",\n   *   },\n   * },\n   * ```\n   */\n  messageAttributes?: {\n    [attributeName: string]:\n      | { DataType: \"String\" | \"Number\"; StringValue: string }\n      | {\n          DataType: \"Binary\"\n          /** Base64-encoded binary data object. */\n          BinaryValue: string\n        }\n  }\n}\n\nexport type AuthorizationProps<AuthScopesT extends string = string> =\n  /**\n   * No authentication, for when you want a fully public route (or handle authentication in the\n   * backend integration).\n   */\n  | { type: \"NONE\" }\n  /**\n   * AWS IAM authorization.\n   * https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-access-control-iam.html\n   */\n  | { type: \"IAM\" }\n  /**\n   * Creates a custom authorizer lambda which reads `Authorization: Bearer <access token>` header\n   * and verifies the token against a Cognito user pool.\n   */\n  | ({\n      type: \"COGNITO_USER_POOL\"\n    } & CognitoUserPoolAuthorizerProps<AuthScopesT>)\n  /**\n   * Creates a custom authorizer lambda which reads `Authorization: Basic <base64-encoded credentials>`\n   * header and verifies the credentials against a given secret.\n   */\n  | ({ type: \"BASIC_AUTH\" } & BasicAuthAuthorizerProps)\n  /**\n   * Creates a custom authorizer lambda which allows both:\n   * - `Authorization: Bearer <access token>` header, for which the token is checked against the\n   *   given Cognito user pool\n   * - `Authorization: Basic <base64-encoded credentials>` header, for which the credentials are\n   *   checked against the credentials from the given basic auth secret name\n   *\n   * If either of these are given and valid, the request is authenticated.\n   */\n  | ({\n      type: \"COGNITO_USER_POOL_OR_BASIC_AUTH\"\n    } & CognitoUserPoolOrBasicAuthAuthorizerProps<AuthScopesT>)\n\nexport type CognitoUserPoolAuthorizerProps<\n  AuthScopesT extends string = string,\n> = {\n  userPool: IUserPool\n\n  /**\n   * Verifies that access token claims contain the given scope.\n   *\n   * When defined as part of a resource server, scopes are on the format:\n   * `{resource server identifier}/{scope name}`, e.g. `external/view_users`.\n   *\n   * To get more type safety on this parameter, see the docs for the `AuthScopesT` type parameter on\n   * {@link ApiGateway}.\n   */\n  requiredScope?: AuthScopesT\n\n  /**\n   * Name of secret in AWS Secrets Manager that stores basic auth credentials for the backend\n   * service, to be forwarded to the backend if Cognito user pool authentication succeeded.\n   *\n   * The secret value must follow this format:\n   * ```json\n   * { \"username\": \"<username>\", \"password\": \"<password>\" }\n   * ```\n   *\n   * This prop solves the following use-case:\n   * - You want to do Cognito user pool authentication in the API Gateway\n   * - You want an additional auth check in the backend, but you don't want to deal with Cognito\n   *   there\n   * - The backend uses basic auth\n   *\n   * This prop solves this by letting you specify credentials to pass to the backend after API-GW\n   * authentication succeeds. You can pass the encoded credentials through\n   * {@link AlbIntegrationProps.mapParameters}, using the `authorizer.internalAuthorizationHeader`\n   * context variable, like so:\n   * ```\n   * mapParameters: (parameters) => parameters.overwriteHeader(\n   *   // 'Authorization' header cannot be overridden, so we use a custom header\n   *   \"X-Internal-Authorization\",\n   *   apigw.MappingValue.contextVariable(\"authorizer.internalAuthorizationHeader\"),\n   * )\n   * ```\n   * The backend can then check the `X-Internal-Authorization` header.\n   */\n  credentialsForInternalAuthorization?: string\n}\n\nexport type BasicAuthAuthorizerProps = {\n  /**\n   * Name of secret in AWS Secrets Manager that stores basic auth credentials. The secret value\n   * should follow this format:\n   * ```json\n   * { \"username\": \"<username>\", \"password\": \"<password>\" }\n   * ```\n   *\n   * The following format is also supported:\n   * ```json\n   * { \"credentials\": \"[\\\"<encoded-credential-1>\\\",\\\"<encoded-credential-2>\\\"]\" }\n   * ```\n   * ...consisting of:\n   * - A single key, `credentials`\n   * - With a _string_ value\n   * - Which is a stringified, escaped JSON array of base64-encoded credentials\n   * - In which each element is encoded from `<username>:<password>`\n   *\n   * If the secret is on this format, the authorizer will match the request's Authorization header\n   * against any one of these encoded credentials.\n   *\n   * The reason that this second format stores stringified JSON _inside_ JSON, is due to a\n   * limitation in Liflig's `load-secrets` library, which only allows storing strings values.\n   */\n  credentialsSecretName: string\n}\n\nexport type CognitoUserPoolOrBasicAuthAuthorizerProps<\n  AuthScopesT extends string = string,\n> = {\n  userPool: IUserPool\n\n  /**\n   * Name of secret in AWS Secrets Manager that stores basic auth credentials. The secret value\n   * should follow this format:\n   * ```json\n   * { \"username\": \"<username>\", \"password\": \"<password>\" }\n   * ```\n   *\n   * The following format is also supported:\n   * ```json\n   * { \"credentials\": \"[\\\"<encoded-credential-1>\\\",\\\"<encoded-credential-2>\\\"]\" }\n   * ```\n   * ...consisting of:\n   * - A single key, `credentials`\n   * - With a _string_ value\n   * - Which is a stringified, escaped JSON array of base64-encoded credentials\n   * - In which each element is encoded from `<username>:<password>`\n   *\n   * If the secret is on this format, the authorizer will match the request's Authorization header\n   * against any one of these encoded credentials.\n   *\n   * The reason that this second format stores stringified JSON _inside_ JSON, is due to a\n   * limitation in Liflig's `load-secrets` library, which only allows storing strings values.\n   */\n  basicAuthCredentialsSecretName?: string\n\n  /**\n   * Verifies that access token claims contain the given scope. Only applicable for requests that\n   * use `Authorization: Bearer <access token>` (not applicable for basic auth).\n   *\n   * When defined as part of a resource server, scopes are on the format:\n   * `{resource server identifier}/{scope name}`, e.g. `external/view_users`.\n   *\n   * To get more type safety on this parameter, see the docs for the `AuthScopesT` type parameter on\n   * {@link ApiGateway}.\n   */\n  requiredScope?: AuthScopesT\n}\n\nexport type ApiGatewayAccessLogsProps = {\n  /**\n   * Delete the access logs if this construct is deleted?\n   *\n   * Maybe you want to DESTROY instead. Or for legal reasons, retain for audit.\n   *\n   * @default RemovalPolicy.RETAIN\n   */\n  removalPolicy?: cdk.RemovalPolicy\n\n  /**\n   * How long to keep the logs. If undefined, uses the same default as new AWS log groups.\n   *\n   * @default RetentionDays.TWO_YEARS\n   */\n  retention?: logs.RetentionDays\n\n  /**\n   * A custom JSON log format, which uses variables from `\"$context\"`.\n   *\n   * See [AWS: CloudWatch log formats for API Gateway](https://docs.aws.amazon.com/apigateway/latest/developerguide/set-up-logging.html#apigateway-cloudwatch-log-formats)\n   * for formats and rules. It is possible to use other formats like CLF and XML, but this construct\n   * only supports JSON for now.\n   *\n   * For a list of all possible variables to log, see\n   * [AWS: $context Variables for data models, authorizers, mapping templates, and CloudWatch access logging](https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-mapping-template-reference.html#context-variable-reference)\n   * and\n   * [AWS: $context Variables for access logging only](https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-mapping-template-reference.html#context-variable-reference-access-logging-only) .\n   *\n   * @default {@link defaultAccessLogFormat}\n   */\n  accessLogFormat?: Record<string, string>\n}\n\n/**\n * This construct tries to simplify the creation of an API Gateway for a service, by collecting most\n * of the common setup here.\n *\n * The approach followed in this construct is:\n * 1. One API-GW per service\n * 2. One subdomain per API-GW / service\n * 3. Use HTTP API, not REST\n * 4. Use a $default stage with autodeploy\n * 5. Support multiple routes (with possible `/{proxy+}` to let all sub-paths through)\n * 6. Allow custom integration/authorizer for each route, or defaults for the whole gateway\n *\n * The route integration is one of these:\n * - ALB private integration with VPC Link using HTTPS to the ALB\n * - Lambda integration\n * - SQS integration\n *\n * ### Load Balancer Security Group\n *\n * Note that the load balancer used in an {@link AlbIntegrationProps} must allow outbound HTTPS\n * traffic to its SecurityGroup. Otherwise, the VPC Link used by the API-GW can't get traffic from\n * the ALB.\n *\n * ```\n * const loadBalancerSecurityGroup = new ec2.SecurityGroup(..., {\n *   allowAllOutbound: false,\n * })\n *\n * loadBalancerSecurityGroup.addEgressRule(\n *   loadBalancerSecurityGroup,\n *   ec2.Port.tcp(443),\n *   \"Outbound to self for ALB to API-GW VPC-Link\",\n * )\n *\n * const loadBalancer = new lifligLoadBalancer.LoadBalancer(...,\n *   {\n *     overrideLoadBalancerProps: {\n *       securityGroup: loadBalancerSecurityGroup,\n *     },\n *   },\n * )\n * ```\n *\n * @template AuthScopesT This type parameter allows you to improve type safety on the\n * `requiredScope` field on {@link CognitoUserPoolOrBasicAuthAuthorizerProps}, by narrowing the type\n * to specific strings. You can then extend the `ApiGateway` with this type to enforce those scopes\n * across the application. Remember that auth scopes must be on the format\n * `{resource server identifier}/{scope name}`.\n *\n * Example:\n * ```\n * type AuthScopes = \"external/read_users\" | \"internal/create_users\"\n *\n * export class MyProjectApiGateway extends ApiGateway<AuthScopes> {}\n * ```\n * TypeScript will then enforce that `requiredScope` is one of `AuthScopes`, and provide\n * auto-complete.\n *\n * @author Kristian Rekstad <kre@capraconsulting.no>\n * @author Hermann Mørkrid <hem@liflig.no>\n */\nexport class ApiGateway<\n  AuthScopesT extends string = string,\n> extends constructs.Construct {\n  /** The API Gateway HTTP API. This is the main construct for API-GW. */\n  public readonly httpApi: apigw.HttpApi\n\n  /** The routes which connect the {@link httpApi} to the backend integration(s). */\n  public readonly routes: apigw.HttpRoute[] = []\n\n  /** The domain which consumers must use.*/\n  public readonly domain: string\n\n  /** Access log group. */\n  public readonly logGroup: logs.LogGroup\n\n  private readonly props: ApiGatewayProps<AuthScopesT>\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ApiGatewayProps<AuthScopesT>,\n  ) {\n    super(scope, id)\n    this.props = props\n\n    ApiGateway.validateProps(props, id, cdk.Stack.of(this))\n\n    const customDomain = new CustomDomain(this, \"CustomDomain\", props.dns)\n    this.domain = customDomain.customDomainName\n\n    let corsOptions: apigw.CorsPreflightOptions | undefined\n    if (props.corsAllowAll) {\n      corsOptions = {\n        allowOrigins: [\"*\"],\n        // \"The Authorization header can't be wildcarded and always needs to be listed explicitly\"\n        // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Access-Control-Allow-Headers\n        // Content-Type may also require explicit whitelisting: https://stackoverflow.com/a/63567647\n        allowHeaders: [\"Authorization\", \"Content-Type\", \"*\"],\n        // Not using '*' here, because \"In requests with credentials, it is treated as the literal\n        // method name '*' without special semantics\"\n        // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Access-Control-Allow-Methods\n        allowMethods: [\n          apigw.CorsHttpMethod.GET,\n          apigw.CorsHttpMethod.POST,\n          apigw.CorsHttpMethod.PUT,\n          apigw.CorsHttpMethod.PATCH,\n          apigw.CorsHttpMethod.DELETE,\n          apigw.CorsHttpMethod.OPTIONS,\n          apigw.CorsHttpMethod.HEAD,\n        ],\n        maxAge: cdk.Duration.days(1),\n      }\n    }\n\n    // The actual API. This holds the routes, authorizers and integrations.\n    const api = new apigw.HttpApi(this, \"HttpApi-\" + props.dns.subdomain, {\n      // defaultIntegration: defaultIntegration, // This is for a catch-all $default route\n      description: `An HTTP API for ${props.dns.subdomain}.${props.dns.hostedZone.zoneName}.`,\n      disableExecuteApiEndpoint: true, // Force externals to go through custom domain. MUST be true when using Mutual TLS, for security reasons\n      createDefaultStage: true,\n      defaultDomainMapping: { domainName: customDomain.apiGwCustomDomain },\n      corsPreflight: corsOptions,\n      ...props?.propsOverride?.httpApi,\n    })\n    this.httpApi = api\n\n    const stage = api.defaultStage?.node.defaultChild as apigw.CfnStage\n\n    const logs = new ApiGatewayAccessLogs(\n      this,\n      \"AccessLogs\",\n      stage,\n      props.accessLogs,\n      defaultAccessLogFormat,\n    )\n    this.logGroup = logs.logGroup\n\n    stage.defaultRouteSettings = {\n      ...stage.defaultRouteSettings,\n      detailedMetricsEnabled: props.detailedMetrics ?? true,\n      throttlingBurstLimit: props.throttling?.burst, // Default for account is 5_000\n      throttlingRateLimit: props.throttling?.rate, // Default for account is 10_000\n    }\n\n    const defaultIntegration = props.defaultIntegration\n      ? this.createIntegration(props.defaultIntegration, props.dns)\n      : undefined\n\n    const defaultAuthorizer = props.defaultAuthorization\n      ? this.createAuthorizer(\"DefaultAuthorizer\", props.defaultAuthorization)\n      : undefined\n\n    for (const route of props.routes) {\n      let integration: apigw.HttpRouteIntegration\n      if (route.integration) {\n        integration = this.createIntegration(route.integration, props.dns)\n      } else {\n        integration = defaultIntegration! // Verified in validateProps\n      }\n\n      let authorizer: apigw.IHttpRouteAuthorizer | undefined\n      if (route.authorization) {\n        authorizer = this.createAuthorizer(\n          // Using the route path here caused CloudFormation to fail due to resource names being too\n          // long. Therefore, we now hash the route path to get a shorter name.\n          `RouteAuthorizer${shortHash(`${route.method ?? \"\"}${route.path}`)}`,\n          route.authorization,\n        )\n      } else {\n        authorizer = defaultAuthorizer\n      }\n\n      const routePaths = [route.path]\n      if (route.includeSubpaths === true) {\n        // If we include sub-paths, we add an additional route with /{proxy+} (the + means it will\n        // match all subroutes). We want both this route and the normal route without /{proxy+},\n        // since /{proxy+} will only match the base path with trailing slash.\n        routePaths.push(\n          route.path + (route.path === \"/\" ? \"\" : \"/\") + \"{proxy+}\",\n        )\n      }\n\n      for (const routePath of routePaths) {\n        // The previous version of the API Gateway construct had a single route with the ID\n        // 'DefaultProxyRoute', and all gateways we used exposed the route /{proxy+}. When trying to\n        // change to this new version of the construct, deployment failed for gateways exposing\n        // /{proxy+}, because \"a route with that key [i.e. path + method] already exists\". We think\n        // this may be because we used the same route path, but with a new route ID, causing\n        // confusion for CloudFormation. So we keep the old route ID here for /{proxy+}.\n        const routeId =\n          routePath === \"/{proxy+}\" ? \"DefaultProxyRoute\" : `Route-${routePath}`\n\n        this.routes.push(\n          new apigw.HttpRoute(this, routeId, {\n            httpApi: api,\n            integration: integration,\n            authorizer: authorizer,\n            routeKey: apigw.HttpRouteKey.with(\n              routePath,\n              route.method\n                ? apigw.HttpMethod[route.method]\n                : apigw.HttpMethod.ANY,\n            ),\n          }),\n        )\n      }\n    }\n\n    tagResources(this, () => ({ service: props.dns.subdomain }))\n  }\n\n  /** @throws Error */\n  private static validateProps(\n    props: ApiGatewayProps,\n    id: string,\n    stack: cdk.Stack,\n  ) {\n    for (const route of props.routes) {\n      if (!route.integration && !props.defaultIntegration) {\n        throw new Error(\n          `No integration defined for route '${route.path}', and no default integration specified for the gateway`,\n        )\n      }\n      if (!route.authorization && !props.defaultAuthorization) {\n        throw new Error(\n          `No authorization defined for route '${route.path}', and no default authorization specified for the gateway`,\n        )\n      }\n      if (!route.path.startsWith(\"/\")) {\n        throw new Error(\n          `Invalid path '${route.path}': paths must begin with '/' (use '/' for root path)`,\n        )\n      }\n      if (route.path !== \"/\" && route.path.endsWith(\"/\")) {\n        throw new Error(\n          `Invalid path '${route.path}': paths cannot end with '/' (except root path)`,\n        )\n      }\n      if (\n        route.integration?.type === \"ALB\" &&\n        route.integration.hostName.startsWith(\"https:\")\n      ) {\n        throw new Error(\n          `The albIntegration.hostHttpsName should not include a protocol: ${route.integration.hostName}`,\n        )\n      }\n    }\n    if (props.dns.subdomain === \"\" || props.dns.subdomain.includes(\" \")) {\n      throw new Error(\n        `Subdomain must be set, and not contain spaces: ${props.dns.subdomain}`,\n      )\n    }\n    if (props.throttling?.burst && props.throttling.burst > 5_000) {\n      console.warn(\n        `⚠️ Your throttling burst limit '${props.throttling.burst}' is higher than the AWS Account limit. Make sure your account has upgraded this quota! `,\n        stack,\n        id,\n      )\n    }\n    if (props.throttling?.rate && props.throttling.rate > 10_000) {\n      console.warn(\n        `⚠️ Your throttling rate limit '${props.throttling.rate}' is higher than the AWS Account limit. Make sure your account has upgraded this quota! `,\n        stack,\n        id,\n      )\n    }\n  }\n\n  /**\n   * The authorizer only accepts requests from external users that are authorized.\n   * Unauthorized users are stopped in the API-GW, and not forwarded to the integration.\n   */\n  private createAuthorizer<AuthScopesT extends string>(\n    id: string,\n    authorization: AuthorizationProps<AuthScopesT>,\n  ): apigw.IHttpRouteAuthorizer | undefined {\n    switch (authorization.type) {\n      case \"NONE\": {\n        return undefined\n      }\n      case \"IAM\": {\n        // API Gateway invokes your API route only if the client has execute-api permission for the route.\n        // The client has to use AWS SigV4 to identify themselves in the request.\n        // Read this page for help with IAM and API-GW https://docs.aws.amazon.com/apigateway/latest/developerguide/security_iam_service-with-iam.html\n        // Note that [resource policies are not supported yet for HTTP](https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-access-control-iam.html)\n        return new authorizers.HttpIamAuthorizer()\n      }\n      case \"COGNITO_USER_POOL\": {\n        // We use a custom lambda authorizer here instead of the `HttpUserPoolAuthorizer` provided\n        // by CDK, in order to support `requiredScope` and setting of custom context variables.\n        const authorizer = new CognitoUserPoolAuthorizer(\n          this,\n          id + \"Lambda\",\n          authorization,\n        )\n\n        return new authorizers.HttpLambdaAuthorizer(id, authorizer.lambda, {\n          responseTypes: authorizer.responseTypes,\n          // Max 1h, or disabled (0s). The value only matters when invalidating/dynamic credentials\n          resultsCacheTtl: cdk.Duration.hours(1),\n        })\n      }\n      case \"BASIC_AUTH\": {\n        const authorizer = new BasicAuthAuthorizer(\n          this,\n          id + \"Lambda\",\n          authorization,\n        )\n\n        return new authorizers.HttpLambdaAuthorizer(id, authorizer.lambda, {\n          responseTypes: authorizer.responseTypes,\n          // Max 1h, or disabled (0s). The value only matters when invalidating/dynamic credentials\n          resultsCacheTtl: cdk.Duration.minutes(30),\n        })\n      }\n      case \"COGNITO_USER_POOL_OR_BASIC_AUTH\": {\n        const authorizer = new CognitoUserPoolOrBasicAuthAuthorizer(\n          this,\n          id + \"Lambda\",\n          authorization,\n        )\n\n        return new authorizers.HttpLambdaAuthorizer(id, authorizer.lambda, {\n          responseTypes: authorizer.responseTypes,\n          // Max 1h, or disabled (0s). The value only matters when invalidating/dynamic credentials\n          resultsCacheTtl: cdk.Duration.hours(1),\n        })\n      }\n    }\n  }\n\n  private createIntegration(\n    integration: IntegrationProps,\n    dns: ApiGatewayProps[\"dns\"],\n  ): apigw.HttpRouteIntegration {\n    switch (integration.type) {\n      case \"ALB\": {\n        // The VPCLink connects the integration into the ALB's VPC. Must be manually created when\n        // the VPC is imported, which is the case when using CorePlatformConsumer.\n        const vpcLink = new apigw.VpcLink(this, \"AlbVpcLink\", {\n          vpc: integration.vpc,\n          securityGroups: [integration.securityGroup],\n          subnets: integration.vpc.selectSubnets(), // This correctly selects the private subnets\n        })\n\n        const parameterMapping = new apigw.ParameterMapping()\n          /** See {@link AlbIntegrationProps.hostName} */\n          .overwriteHeader(\n            \"Host\",\n            // The Host header can NOT use dynamic mapping, like $context.domainName etc.\n            apigw.MappingValue.custom(integration.hostName),\n          )\n        if (integration.mapParameters !== undefined) {\n          integration.mapParameters(parameterMapping)\n        }\n\n        return new integrations.HttpAlbIntegration(\n          \"AlbIntegration-\" + dns.subdomain,\n          integration.loadBalancerListener,\n          {\n            secureServerName: integration.hostName,\n            vpcLink: vpcLink,\n            method: apigw.HttpMethod.ANY,\n            parameterMapping,\n          },\n        )\n      }\n      case \"Lambda\": {\n        return new integrations.HttpLambdaIntegration(\n          \"LambdaIntegration\",\n          integration.lambda,\n          {\n            payloadFormatVersion: apigw.PayloadFormatVersion.VERSION_2_0,\n            parameterMapping: undefined,\n          },\n        )\n      }\n      case \"SQS\": {\n        // API-GW does not have access to SQS by default\n        const role = new iam.Role(\n          this,\n          `ApiGwTo${integration.queue.node.id}ServiceRole`,\n          {\n            description:\n              \"Allows API-GW to add messages to \" + integration.queue.queueArn,\n            assumedBy: new iam.ServicePrincipal(\"apigateway.amazonaws.com\"),\n          },\n        )\n        integration.queue.grantSendMessages(role)\n\n        let parameterMapping = new apigw.ParameterMapping()\n          // https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-develop-integrations-aws-services-reference.html#SQS-SendMessage\n          .custom(\"QueueUrl\", integration.queue.queueUrl)\n          .custom(\"MessageBody\", \"$request.body\")\n          .custom(\"Region\", \"eu-west-1\") // Change this if the SQS queue is in another region!\n\n        if (integration.messageAttributes) {\n          parameterMapping = parameterMapping.custom(\n            \"MessageAttributes\",\n            JSON.stringify(integration.messageAttributes),\n          )\n        }\n\n        return new SqsRouteIntegration(\"SqsIntegration\", {\n          type: apigw.HttpIntegrationType.AWS_PROXY,\n          subtype: apigw.HttpIntegrationSubtype.SQS_SEND_MESSAGE,\n          credentials: apigw.IntegrationCredentials.fromRole(role),\n          parameterMapping: parameterMapping,\n          payloadFormatVersion: apigw.PayloadFormatVersion.VERSION_1_0,\n        })\n      }\n    }\n  }\n\n  /**\n   * Allows a grantable target (role, user etc.) permission to invoke the API.\n   * Only works when using `IAM` as {@link ApiGatewayRoute.authorization}.\n   *\n   * @param target A grantable, like {@link iam.Role}\n   */\n  public grantInvoke(target: iam.IGrantable) {\n    for (const routeProps of this.props.routes) {\n      const authType =\n        routeProps.authorization?.type ?? this.props.defaultAuthorization?.type\n\n      if (authType !== \"IAM\") {\n        throw new Error(\n          `Cannot grant invoke for an API Gateway when not using IAM auth (found auth '${authType}' on route '${routeProps.path}') [${cdk.Stack.of(this).stackName}]`,\n        )\n      }\n    }\n\n    for (const route of this.routes) {\n      route.grantInvoke(target)\n    }\n  }\n}\n\n/**\n * Creates a custom domain for the API-Gateway, a Route53 record and an HTTPS cert.\n * @author Kristian Rekstad <kre@capraconsulting.no>\n */\nclass CustomDomain extends constructs.Construct {\n  public readonly apiGwCustomDomain: apigw.DomainName\n\n  /** The Fully Qualified Domain Name (FQDN) like `product.platform.example.no`. */\n  public readonly customDomainName: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ApiGatewayDnsProps,\n  ) {\n    super(scope, id)\n    this.customDomainName = `${props.subdomain}.${props.hostedZone.zoneName}`\n\n    // Can also use wildcard certs instead! Cheaper\n    /** Allows external users to connect with HTTPS. */\n    const customDomainCert = new acm.Certificate(this, \"HttpsCertificate\", {\n      domainName: this.customDomainName,\n      validation: acm.CertificateValidation.fromDns(props.hostedZone),\n    })\n\n    // Note that API-GW can also support wildcard domains! https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-custom-domain-names.html#http-wildcard-custom-domain-names\n    // But this will not work when AWS account X has CustomDomain `staging.platform.example.no` and account Y has CustomDomain `*.platform.example.no`.\n    // Not sure how sub-subdomains are affected: `myservice.staging.platform.example.no` and `*.platform.example.no`.\n    this.apiGwCustomDomain = new apigw.DomainName(\n      this,\n      \"DomainName-\" + props.subdomain,\n      {\n        domainName: this.customDomainName,\n        certificate: customDomainCert,\n        endpointType: apigw.EndpointType.REGIONAL,\n        securityPolicy: apigw.SecurityPolicy.TLS_1_2,\n      },\n    )\n\n    // This makes the API-GW publicly available on the custom domain name.\n    new route53.ARecord(this, \"Route53ARecordApigwAlias\", {\n      recordName: props.subdomain,\n      zone: props.hostedZone,\n      target: route53.RecordTarget.fromAlias(\n        new route53Targets.ApiGatewayv2DomainProperties(\n          this.apiGwCustomDomain.regionalDomainName,\n          this.apiGwCustomDomain.regionalHostedZoneId,\n        ),\n      ),\n      ttl: props.ttl ?? cdk.Duration.minutes(5), // Low TTL makes it easier to do changes\n    })\n  }\n}\n\n/** Acts as glue (between the integration props and the HttpApi) when creating an SqsIntegration. */\nclass SqsRouteIntegration extends apigw.HttpRouteIntegration {\n  /**\n   * @param id The id used in the {@link apigw.HttpIntegration} construct\n   *    created internally by {@link apigw.HttpRouteIntegration._bindToRoute}.\n   *    [Source code](https://github.com/aws/aws-cdk/blob/b5ae37782bc3cb637eeef9fbb1fbe2c5efdfc068/packages/%40aws-cdk/aws-apigatewayv2/lib/http/integration.ts#L321)\n   * @param integrationProps The props to pass to the {@link apigw.HttpIntegration} construct.\n   */\n  constructor(\n    id: string,\n    private integrationProps: apigw.HttpRouteIntegrationConfig,\n  ) {\n    super(id)\n  }\n\n  /**\n   * This sends the properties needed for creating a {@link apigw.HttpIntegration} to the\n   * {@link apigw.HttpRouteIntegration}.\n   */\n  bind(): apigw.HttpRouteIntegrationConfig {\n    // This method is called by:\n    // https://github.com/aws/aws-cdk/blob/b5ae37782bc3cb637eeef9fbb1fbe2c5efdfc068/packages/%40aws-cdk/aws-apigatewayv2/lib/http/integration.ts#L319\n    return this.integrationProps\n  }\n}\n\n/**\n * When using `NodejsFunction` with `entry`, we point CDK to a file with our lambda code. This can\n * either be a TypeScript or JavaScript file. When creating constructs in tests inside this library,\n * the entry will point to the TypeScript file in the source code. But when a library consumer uses\n * this, it will instead point to the transpiled JavaScript file. So to set the correct file\n * extension, we must check if we're being run in the context of the TypeScript source code\n * (src/api-gateway), otherwise we're being run by a consumer as transpiled JavaScript.\n */\nconst authorizerFileExtension = __dirname.endsWith(\"src/api-gateway\")\n  ? \"ts\"\n  : \"js\"\n\n/**\n * Creates a custom authorizer lambda which reads `Authorization: Bearer <access token>` header and\n * verifies the token against a Cognito user pool.\n */\nclass CognitoUserPoolAuthorizer<\n  AuthScopesT extends string,\n> extends constructs.Construct {\n  public readonly lambda: lambda.IFunction\n\n  /**\n   * Use simple response type (`{ isAuthorized: true/false }`), as opposed to returning an IAM\n   * Policy document.\n   */\n  public readonly responseTypes: authorizers.HttpLambdaResponseType[] = [\n    authorizers.HttpLambdaResponseType.SIMPLE,\n  ]\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: CognitoUserPoolAuthorizerProps<AuthScopesT>,\n  ) {\n    super(scope, id)\n\n    this.lambda = new lambdaNodejs.NodejsFunction(this, \"AuthorizerFunction\", {\n      entry: path.join(\n        __dirname,\n        `authorizer-lambdas/cognito-user-pool-authorizer.${authorizerFileExtension}`,\n      ),\n      runtime: lambda.Runtime.NODEJS_22_X,\n      timeout: cdk.Duration.seconds(5),\n      environment: {\n        [\"USER_POOL_ID\"]: props.userPool.userPoolId,\n        [\"REQUIRED_SCOPE\"]: props.requiredScope ?? \"\",\n        [\"CREDENTIALS_FOR_INTERNAL_AUTHORIZATION\"]:\n          props.credentialsForInternalAuthorization\n            ? props.credentialsForInternalAuthorization\n            : \"\",\n      },\n    })\n\n    if (props.credentialsForInternalAuthorization) {\n      secretsmanager.Secret.fromSecretNameV2(\n        scope,\n        id + \"BasicAuthSecret\",\n        props.credentialsForInternalAuthorization,\n      ).grantRead(this.lambda)\n    }\n  }\n}\n\n/**\n * Creates a custom authorizer lambda which reads `Authorization: Basic <base64-encoded credentials>`\n * header and verifies the credentials against a given secret.\n */\nclass BasicAuthAuthorizer extends constructs.Construct {\n  public readonly lambda: lambda.IFunction\n\n  // Simple is `{ isAuthorized: true/false }`, as opposed to returning an IAM Policy document\n  public readonly responseTypes: authorizers.HttpLambdaResponseType[] = [\n    authorizers.HttpLambdaResponseType.SIMPLE,\n  ]\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: BasicAuthAuthorizerProps,\n  ) {\n    super(scope, id)\n\n    this.lambda = new lambdaNodejs.NodejsFunction(this, \"BasicAuthLambda\", {\n      entry: path.join(\n        __dirname,\n        `authorizer-lambdas/basic-auth-authorizer.${authorizerFileExtension}`,\n      ),\n      description:\n        \"An authorizer for API-Gateway that checks Basic Auth credentials on requests\",\n      runtime: lambda.Runtime.NODEJS_22_X,\n      environment: {\n        [\"CREDENTIALS_SECRET_NAME\"]: props.credentialsSecretName\n          ? props.credentialsSecretName\n          : \"\",\n      },\n    })\n\n    if (props.credentialsSecretName) {\n      secretsmanager.Secret.fromSecretNameV2(\n        scope,\n        id + \"BasicAuthSecret\",\n        props.credentialsSecretName,\n      ).grantRead(this.lambda)\n    }\n  }\n}\n\n/**\n * Creates a custom authorizer lambda which allows both:\n * - `Authorization: Bearer <access token>` header, for which the token is checked against the given\n *   Cognito user pool\n * - `Authorization: Basic <base64-encoded credentials>` header, for which the credentials are\n *   checked against the credentials from the given basic auth secret name\n *\n * If either of these are given and valid, the request is authenticated.\n */\nclass CognitoUserPoolOrBasicAuthAuthorizer<\n  AuthScopesT extends string,\n> extends constructs.Construct {\n  public readonly lambda: lambda.IFunction\n\n  /**\n   * Use simple response type (`{ isAuthorized: true/false }`), as opposed to returning an IAM\n   * Policy document.\n   */\n  public readonly responseTypes: authorizers.HttpLambdaResponseType[] = [\n    authorizers.HttpLambdaResponseType.SIMPLE,\n  ]\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: CognitoUserPoolOrBasicAuthAuthorizerProps<AuthScopesT>,\n  ) {\n    super(scope, id)\n\n    this.lambda = new lambdaNodejs.NodejsFunction(this, \"AuthorizerFunction\", {\n      entry: path.join(\n        __dirname,\n        `authorizer-lambdas/cognito-user-pool-or-basic-auth-authorizer.${authorizerFileExtension}`,\n      ),\n      runtime: lambda.Runtime.NODEJS_22_X,\n      timeout: cdk.Duration.seconds(5),\n      environment: {\n        [\"USER_POOL_ID\"]: props.userPool ? props.userPool.userPoolId : \"\",\n        [\"REQUIRED_SCOPE\"]: props.requiredScope ?? \"\",\n        [\"BASIC_AUTH_CREDENTIALS_SECRET_NAME\"]:\n          props.basicAuthCredentialsSecretName\n            ? props.basicAuthCredentialsSecretName\n            : \"\",\n      },\n    })\n\n    if (props.basicAuthCredentialsSecretName) {\n      secretsmanager.Secret.fromSecretNameV2(\n        scope,\n        id + \"BasicAuthSecret\",\n        props.basicAuthCredentialsSecretName,\n      ).grantRead(this.lambda)\n    }\n  }\n}\n\n/**\n * A slightly extended version of the [default JSON format suggested by AWS](https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-logging.html#http-api-enable-logging.examples).\n */\nconst defaultAccessLogFormat = {\n  requestId: \"$context.requestId\",\n  userAgent: \"$context.identity.userAgent\",\n  ip: \"$context.identity.sourceIp\",\n  /** CLF format: `dd/MMM/yyyy:HH:mm:ss +-hhmm` */\n  requestTime: \"$context.requestTime\",\n  requestTimeEpoch: \"$context.requestTimeEpoch\",\n  dataProcessed: \"$context.dataProcessed\",\n  httpMethod: \"$context.httpMethod\",\n  path: \"$context.path\",\n  routeKey: \"$context.routeKey\",\n  status: \"$context.status\",\n  protocol: \"$context.protocol\",\n  responseLength: \"$context.responseLength\",\n  responseLatency: \"$context.responseLatency\",\n  domainName: \"$context.domainName\",\n  error: {\n    type: \"$context.error.responseType\",\n    gatewayError: \"$context.error.message\",\n    integrationError: \"$context.integration.error\",\n    authorizerError: \"$context.authorizer.error\",\n  },\n  integration: {\n    latency: \"$context.integration.latency\",\n    requestId: \"$context.integration.requestId\",\n    responseStatus: \"$context.integration.status\",\n  },\n  auth: {\n    iam: {\n      userArn: \"$context.identity.userArn\",\n      awsAccount: \"$context.identity.accountId\",\n      awsPrincipal: \"$context.identity.caller\",\n      awsPrincipalOrg: \"$context.identity.principalOrgId\",\n    },\n    basic: { username: \"$context.authorizer.username\" },\n    cognito: { clientId: \"$context.authorizer.clientId\" },\n  },\n  awsEndpointRequest: {\n    id: \"$context.awsEndpointRequestId\",\n    id2: \"$context.awsEndpointRequestId2\",\n  },\n  message:\n    \"$context.identity.sourceIp - $context.httpMethod $context.domainName $context.path ($context.routeKey) - $context.status [$context.responseLatency ms]\",\n}\n\n/**\n * Enables access logs on the API-Gateway.\n *\n * @author Kristian Rekstad <kre@capraconsulting.no>\n */\nclass ApiGatewayAccessLogs extends constructs.Construct {\n  public readonly logGroup: logs.LogGroup\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    stage: apigw.CfnStage,\n    props: ApiGatewayAccessLogsProps | undefined,\n    defaultAccessLogFormat: Record<string, unknown>,\n  ) {\n    super(scope, id)\n\n    // logGroup is set up with help from: https://github.com/aws/aws-cdk/issues/11100#issuecomment-904627081\n    // Not sure if HTTP API actually needs the service role with managed policy: https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-logging.html\n    const accessLogs = new logs.LogGroup(this, \"AccessLogGroup\", {\n      retention: props?.retention,\n      removalPolicy: props?.removalPolicy ?? cdk.RemovalPolicy.RETAIN,\n      // Always use the default encryption key. Otherwise, the key needs special policies:\n      // https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/encrypt-log-data-kms.html#cmk-permissions\n      encryptionKey: undefined,\n    })\n    this.logGroup = accessLogs\n\n    stage.accessLogSettings = {\n      destinationArn: accessLogs.logGroupArn,\n      format: JSON.stringify(props?.accessLogFormat ?? defaultAccessLogFormat),\n    }\n\n    const apiGwLogsRole = new iam.Role(\n      this,\n      \"ApiGatewayPushToCloudwatchLogsRole\",\n      {\n        assumedBy: new iam.ServicePrincipal(\"apigateway.amazonaws.com\"),\n        managedPolicies: [\n          iam.ManagedPolicy.fromAwsManagedPolicyName(\n            \"service-role/AmazonAPIGatewayPushToCloudWatchLogs\",\n          ),\n        ],\n      },\n    )\n\n    accessLogs.grantWrite(apiGwLogsRole)\n  }\n}\n\n/** Returns a short semi-unique hash of the given string. */\nfunction shortHash(str: string): string {\n  // SHA-1 is no-no when we need cryptographic security, but here we just it for shortening a name,\n  // which is fine\n  return createHash(\"sha1\").update(str).digest(\"hex\").substring(0, 10)\n}\n"]}
|