@liflig/cdk 3.2.0 → 3.4.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.
@@ -1,16 +1,15 @@
1
1
  /**
2
2
  * This lambda verifies credentials:
3
- * - Against Cognito user pool if request uses Bearer token
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
- * Expects the following environment variables
6
+ * Expects the following environment variables:
7
7
  * - USER_POOL_ID
8
8
  * - BASIC_AUTH_CREDENTIALS_SECRET_NAME (optional)
9
- * - Secret value should follow this format: `{"username":"<username>","password":"<password>"}`.
10
- * A different format with an array of pre-encoded credentials is also supported - see docs for
11
- * the `CognitoUserPoolOrBasicAuthAuthorizerProps` on the `ApiGateway` construct.
9
+ * - Name of secret in AWS Secrets Manager that stores basic auth credentials. See
10
+ * `BasicAuthAuthorizerProps` on the `ApiGateway` construct for the supported formats.
12
11
  * - REQUIRED_SCOPE (optional)
13
- * - Set this to require that the bearer token payload contains the given scope
12
+ * - Set this to require that the access token payload contains the given scope
14
13
  */
15
14
  import { SecretsManager } from "@aws-sdk/client-secrets-manager";
16
15
  import { CognitoJwtVerifier } from "aws-jwt-verify";
@@ -19,7 +18,7 @@ export const handler = async (event) => {
19
18
  if (!authHeader) {
20
19
  return { isAuthorized: false };
21
20
  }
22
- const expectedBasicAuthHeaders = await getExpectedBasicAuthHeaders();
21
+ const expectedBasicAuthCredentials = await getExpectedBasicAuthCredentials();
23
22
  if (authHeader.startsWith("Bearer ")) {
24
23
  const result = await verifyAccessToken(authHeader.substring(7)); // substring(7) == after 'Bearer '
25
24
  switch (result) {
@@ -36,19 +35,20 @@ export const handler = async (event) => {
36
35
  isAuthorized: true,
37
36
  context: {
38
37
  clientId: result.client_id,
39
- internalAuthorizationHeader: expectedBasicAuthHeaders?.[0],
38
+ internalAuthorizationHeader: expectedBasicAuthCredentials?.[0]?.basicAuthHeader,
40
39
  },
41
40
  };
42
41
  }
43
42
  }
44
43
  else if (authHeader.startsWith("Basic ") &&
45
- expectedBasicAuthHeaders !== undefined) {
46
- for (const expectedHeader of expectedBasicAuthHeaders) {
47
- if (authHeader === expectedHeader) {
44
+ expectedBasicAuthCredentials !== undefined) {
45
+ for (const expected of expectedBasicAuthCredentials) {
46
+ if (authHeader === expected.basicAuthHeader) {
48
47
  return {
49
48
  isAuthorized: true,
50
49
  context: {
51
- internalAuthorizationHeader: expectedHeader,
50
+ username: expected.username,
51
+ internalAuthorizationHeader: expected.basicAuthHeader,
52
52
  },
53
53
  };
54
54
  }
@@ -109,27 +109,25 @@ export const dependencies = {
109
109
  createSecretsManager: () => new SecretsManager(),
110
110
  };
111
111
  /** Cache this value, so that subsequent lambda invocations don't have to refetch. */
112
- let cachedBasicAuthHeaders = undefined;
112
+ let cachedBasicAuthCredentials = undefined;
113
113
  /**
114
- * Returns an array of allowed basic auth headers, to support credential secrets with multiple
115
- * values (see `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).
114
+ * Returns an array, to support credential secrets with multiple values (see
115
+ * `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).
116
116
  */
117
- async function getExpectedBasicAuthHeaders() {
118
- if (cachedBasicAuthHeaders === undefined) {
117
+ async function getExpectedBasicAuthCredentials() {
118
+ if (cachedBasicAuthCredentials === undefined) {
119
119
  const secretName = process.env["BASIC_AUTH_CREDENTIALS_SECRET_NAME"];
120
120
  if (!secretName) {
121
121
  return undefined;
122
122
  }
123
- cachedBasicAuthHeaders = await getSecretAsBasicAuthHeaders(secretName);
123
+ cachedBasicAuthCredentials = await getBasicAuthCredentialsSecret(secretName);
124
124
  }
125
- return cachedBasicAuthHeaders;
125
+ return cachedBasicAuthCredentials;
126
126
  }
127
- async function getSecretAsBasicAuthHeaders(secretName) {
127
+ async function getBasicAuthCredentialsSecret(secretName) {
128
128
  const secret = await getSecretValue(secretName);
129
- if (isSingleUsernameAndPassword(secret)) {
130
- const header = "Basic " +
131
- Buffer.from(`${secret.username}:${secret.password}`).toString("base64");
132
- return [header];
129
+ if (isUsernameAndPasswordObject(secret)) {
130
+ return [encodeBasicAuthCredentials(secret)];
133
131
  }
134
132
  // See `BasicAuthAuthorizerProps` on the `ApiGateway` construct for an explanation of the formats
135
133
  // we parse here
@@ -142,8 +140,11 @@ async function getSecretAsBasicAuthHeaders(secretName) {
142
140
  console.error(`Failed to parse credentials array in secret '${secretName}' as JSON`, e);
143
141
  throw new Error();
144
142
  }
143
+ if (isArrayOfUsernameAndPasswordObjects(credentialsArray)) {
144
+ return credentialsArray.map(encodeBasicAuthCredentials);
145
+ }
145
146
  if (isStringArray(credentialsArray)) {
146
- return credentialsArray.map((encodedCredential) => `Basic ${encodedCredential}`);
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,11 +161,16 @@ 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`, e);
164
+ console.error(`Failed to parse secret '${secretName}' as JSON:`, e);
164
165
  throw new Error();
165
166
  }
166
167
  }
167
- function isSingleUsernameAndPassword(value) {
168
+ function encodeBasicAuthCredentials(credentials) {
169
+ const basicAuthHeader = "Basic " +
170
+ Buffer.from(`${credentials.username}:${credentials.password}`).toString("base64");
171
+ return { basicAuthHeader, username: credentials.username };
172
+ }
173
+ function isUsernameAndPasswordObject(value) {
168
174
  return (typeof value === "object" &&
169
175
  value !== null &&
170
176
  "username" in value &&
@@ -172,6 +178,17 @@ function isSingleUsernameAndPassword(value) {
172
178
  "password" in value &&
173
179
  typeof value.password === "string");
174
180
  }
181
+ function isArrayOfUsernameAndPasswordObjects(value) {
182
+ if (!Array.isArray(value)) {
183
+ return false;
184
+ }
185
+ for (const element of value) {
186
+ if (!isUsernameAndPasswordObject(element)) {
187
+ return false;
188
+ }
189
+ }
190
+ return true;
191
+ }
175
192
  function hasCredentialsKeyWithStringValue(value) {
176
193
  return (typeof value === "object" &&
177
194
  value !== null &&
@@ -189,8 +206,33 @@ function isStringArray(value) {
189
206
  }
190
207
  return true;
191
208
  }
209
+ /**
210
+ * We want to return the requesting username as a context variable in
211
+ * {@link AuthorizerResult.context}, for API Gateway access logs and parameter mapping. So if the
212
+ * basic auth credentials secret is stored as pre-encoded base64 strings, we need to parse them to
213
+ * get the username.
214
+ */
215
+ function parseEncodedBasicAuthCredentials(encodedCredentials) {
216
+ let decodedCredentials;
217
+ try {
218
+ decodedCredentials = Buffer.from(encodedCredentials, "base64").toString();
219
+ }
220
+ catch (e) {
221
+ console.error("Basic auth credentials secret could not be decoded as base64:", e);
222
+ throw new Error();
223
+ }
224
+ const usernameAndPassword = decodedCredentials.split(":", 2);
225
+ if (usernameAndPassword.length !== 2) {
226
+ console.error("Basic auth credentials secret could not be decoded as 'username:password'");
227
+ throw new Error();
228
+ }
229
+ return {
230
+ basicAuthHeader: `Basic ${encodedCredentials}`,
231
+ username: usernameAndPassword[0],
232
+ };
233
+ }
192
234
  export function clearCache() {
193
235
  cachedTokenVerifier = undefined;
194
- cachedBasicAuthHeaders = undefined;
236
+ cachedBasicAuthCredentials = undefined;
195
237
  }
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"]}
238
+ //# 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 *   - Name of secret in AWS Secrets Manager that stores basic auth credentials. See\n *     `BasicAuthAuthorizerProps` on the `ApiGateway` construct for the supported formats.\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 (isUsernameAndPasswordObject(secret)) {\n    return [encodeBasicAuthCredentials(secret)]\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 (isArrayOfUsernameAndPasswordObjects(credentialsArray)) {\n      return credentialsArray.map(encodeBasicAuthCredentials)\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 encodeBasicAuthCredentials(credentials: {\n  username: string\n  password: string\n}): ExpectedBasicAuthCredentials {\n  const basicAuthHeader =\n    \"Basic \" +\n    Buffer.from(`${credentials.username}:${credentials.password}`).toString(\n      \"base64\",\n    )\n\n  return { basicAuthHeader, username: credentials.username }\n}\n\nfunction isUsernameAndPasswordObject(\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 isArrayOfUsernameAndPasswordObjects(\n  value: unknown,\n): value is { username: string; password: string }[] {\n  if (!Array.isArray(value)) {\n    return false\n  }\n\n  for (const element of value) {\n    if (!isUsernameAndPasswordObject(element)) {\n      return false\n    }\n  }\n\n  return true\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 and
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 given
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`.
@@ -349,59 +349,44 @@ export type CognitoUserPoolAuthorizerProps<AuthScopesT extends string = string>
349
349
  };
350
350
  export type BasicAuthAuthorizerProps = {
351
351
  /**
352
- * Name of secret in AWS Secrets Manager that stores basic auth credentials. The secret value
353
- * should follow this format:
354
- * ```json
355
- * { "username": "<username>", "password": "<password>" }
356
- * ```
357
- *
358
- * The following format is also supported:
359
- * ```json
360
- * { "credentials": "[\"<encoded-credential-1>\",\"<encoded-credential-2>\"]" }
361
- * ```
362
- * ...consisting of:
363
- * - A single key, `credentials`
364
- * - With a _string_ value
365
- * - Which is a stringified, escaped JSON array of base64-encoded credentials
366
- * - In which each element is encoded from `<username>:<password>`
367
- *
368
- * If the secret is on this format, the authorizer will match the request's Authorization header
369
- * against any one of these encoded credentials.
370
- *
371
- * The reason that this second format stores stringified JSON _inside_ JSON, is due to a
372
- * limitation in Liflig's `load-secrets` library, which only allows storing strings values.
352
+ * Name of secret in AWS Secrets Manager that stores basic auth credentials.
353
+ *
354
+ * The following formats are supported for the secret value:
355
+ * - Single username and password:
356
+ * ```json
357
+ * { "username": "<username>", "password": "<password>" }
358
+ * ```
359
+ * - Array of username + password objects:
360
+ * ```json
361
+ * { "credentials": "[{\"username\":\"<user-1>\",\"password\":\"password-1\"},{\"username\":\"<user-2>\",\"password\":\"<password-2>\"}]" }
362
+ * ```
363
+ * - The value of the `credentials` field is a string, with a stringified, escaped JSON array of
364
+ * objects with `username` and `password` fields.
365
+ * - The reason that this second format stores stringified JSON _inside_ JSON, is due to a
366
+ * limitation in Liflig's `load-secrets` library, which only allows storing string values.
367
+ * - Array of base64-encoded credentials:
368
+ * ```json
369
+ * { "credentials": "[\"<encoded-credential-1>\",\"<encoded-credential-2>\"]" }
370
+ * ```
371
+ * - Each element is encoded from `<username>:<password>`.
372
+ * - The array is stringified for the same reason as above.
373
+ *
374
+ * If the secret uses one of the array formats, the authorizer will match the request's
375
+ * Authorization header against any one of the credentials.
373
376
  */
374
377
  credentialsSecretName: string;
375
378
  };
376
379
  export type CognitoUserPoolOrBasicAuthAuthorizerProps<AuthScopesT extends string = string> = {
377
380
  userPool: IUserPool;
378
381
  /**
379
- * Name of secret in AWS Secrets Manager that stores basic auth credentials. The secret value
380
- * should follow this format:
381
- * ```json
382
- * { "username": "<username>", "password": "<password>" }
383
- * ```
384
- *
385
- * The following format is also supported:
386
- * ```json
387
- * { "credentials": "[\"<encoded-credential-1>\",\"<encoded-credential-2>\"]" }
388
- * ```
389
- * ...consisting of:
390
- * - A single key, `credentials`
391
- * - With a _string_ value
392
- * - Which is a stringified, escaped JSON array of base64-encoded credentials
393
- * - In which each element is encoded from `<username>:<password>`
394
- *
395
- * If the secret is on this format, the authorizer will match the request's Authorization header
396
- * against any one of these encoded credentials.
382
+ * Name of secret in AWS Secrets Manager that stores basic auth credentials.
397
383
  *
398
- * The reason that this second format stores stringified JSON _inside_ JSON, is due to a
399
- * limitation in Liflig's `load-secrets` library, which only allows storing strings values.
384
+ * See {@link BasicAuthAuthorizerProps.credentialsSecretName} for the supported formats.
400
385
  */
401
386
  basicAuthCredentialsSecretName?: string;
402
387
  /**
403
- * Verifies that token claims contain the given scope. Only applicable for `Bearer` token requests
404
- * checked against the Cognito User Pool (not applicable for basic auth).
388
+ * Verifies that access token claims contain the given scope. Only applicable for requests that
389
+ * use `Authorization: Bearer <access token>` (not applicable for basic auth).
405
390
  *
406
391
  * When defined as part of a resource server, scopes are on the format:
407
392
  * `{resource server identifier}/{scope name}`, e.g. `external/view_users`.