@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.
- package/lib/api-gateway/authorizer-lambdas/basic-auth-authorizer.d.ts +22 -5
- package/lib/api-gateway/authorizer-lambdas/basic-auth-authorizer.js +71 -25
- 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 +20 -12
- package/lib/api-gateway/authorizer-lambdas/cognito-user-pool-or-basic-auth-authorizer.js +71 -29
- package/lib/api-gateway/http-api-gateway.d.ts +33 -48
- package/lib/api-gateway/http-api-gateway.js +5 -6
- package/package.json +1 -1
|
@@ -1,18 +1,35 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* This lambda verifies authorization header against static basic auth credentials saved in
|
|
2
|
+
* This lambda verifies authorization header against static basic auth credentials saved in Secrets
|
|
3
3
|
* Manager.
|
|
4
4
|
*
|
|
5
5
|
* Expects the following environment variables:
|
|
6
6
|
* - CREDENTIALS_SECRET_NAME
|
|
7
|
-
* -
|
|
8
|
-
*
|
|
9
|
-
* the `BasicAuthAuthorizerProps` on the `ApiGateway` construct.
|
|
7
|
+
* - Name of secret in AWS Secrets Manager that stores basic auth credentials. See
|
|
8
|
+
* `BasicAuthAuthorizerProps` on the `ApiGateway` construct for the supported formats.
|
|
10
9
|
*/
|
|
11
10
|
import type { APIGatewayRequestAuthorizerEventV2, APIGatewaySimpleAuthorizerResult } from "aws-lambda";
|
|
12
11
|
import { SecretsManager } from "@aws-sdk/client-secrets-manager";
|
|
13
|
-
|
|
12
|
+
type AuthorizerResult = APIGatewaySimpleAuthorizerResult & {
|
|
13
|
+
/**
|
|
14
|
+
* Returning a context object from our authorizer allows our API Gateway to access these variables
|
|
15
|
+
* via `${context.authorizer.<property>}`.
|
|
16
|
+
* https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html
|
|
17
|
+
*/
|
|
18
|
+
context?: {
|
|
19
|
+
/**
|
|
20
|
+
* If the request's credentials are verified, we return the username that was used in this
|
|
21
|
+
* context variable (named `authorizer.username`). We use this to include the requesting user in
|
|
22
|
+
* the API Gateway access logs (see `defaultAccessLogFormat` in our `ApiGateway` construct). You
|
|
23
|
+
* can also use this when mapping parameters to the backend integration (see
|
|
24
|
+
* `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).
|
|
25
|
+
*/
|
|
26
|
+
username: string;
|
|
27
|
+
};
|
|
28
|
+
};
|
|
29
|
+
export declare const handler: (event: APIGatewayRequestAuthorizerEventV2) => Promise<AuthorizerResult>;
|
|
14
30
|
/** For overriding dependency creation in tests. */
|
|
15
31
|
export declare const dependencies: {
|
|
16
32
|
createSecretsManager: () => SecretsManager;
|
|
17
33
|
};
|
|
18
34
|
export declare function clearCache(): void;
|
|
35
|
+
export {};
|
|
@@ -1,12 +1,11 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* This lambda verifies authorization header against static basic auth credentials saved in
|
|
2
|
+
* This lambda verifies authorization header against static basic auth credentials saved in Secrets
|
|
3
3
|
* Manager.
|
|
4
4
|
*
|
|
5
5
|
* Expects the following environment variables:
|
|
6
6
|
* - CREDENTIALS_SECRET_NAME
|
|
7
|
-
* -
|
|
8
|
-
*
|
|
9
|
-
* the `BasicAuthAuthorizerProps` on the `ApiGateway` construct.
|
|
7
|
+
* - Name of secret in AWS Secrets Manager that stores basic auth credentials. See
|
|
8
|
+
* `BasicAuthAuthorizerProps` on the `ApiGateway` construct for the supported formats.
|
|
10
9
|
*/
|
|
11
10
|
import { SecretsManager } from "@aws-sdk/client-secrets-manager";
|
|
12
11
|
export const handler = async (event) => {
|
|
@@ -14,37 +13,40 @@ export const handler = async (event) => {
|
|
|
14
13
|
if (!authHeader || !authHeader.startsWith("Basic ")) {
|
|
15
14
|
return { isAuthorized: false };
|
|
16
15
|
}
|
|
17
|
-
const
|
|
18
|
-
for (const
|
|
19
|
-
if (authHeader ===
|
|
20
|
-
return {
|
|
16
|
+
const expectedCredentials = await getExpectedBasicAuthCredentials();
|
|
17
|
+
for (const expected of expectedCredentials) {
|
|
18
|
+
if (authHeader === expected.basicAuthHeader) {
|
|
19
|
+
return {
|
|
20
|
+
isAuthorized: true,
|
|
21
|
+
context: {
|
|
22
|
+
username: expected.username,
|
|
23
|
+
},
|
|
24
|
+
};
|
|
21
25
|
}
|
|
22
26
|
}
|
|
23
27
|
return { isAuthorized: false };
|
|
24
28
|
};
|
|
25
29
|
/** Cache this value, so that subsequent lambda invocations don't have to refetch. */
|
|
26
|
-
let
|
|
30
|
+
let cachedBasicAuthCredentials = undefined;
|
|
27
31
|
/**
|
|
28
|
-
* Returns an array
|
|
29
|
-
*
|
|
32
|
+
* Returns an array, to support credential secrets with multiple values (see
|
|
33
|
+
* `BasicAuthAuthorizerProps` on the `ApiGateway` construct for more on this).
|
|
30
34
|
*/
|
|
31
|
-
async function
|
|
32
|
-
if (
|
|
35
|
+
async function getExpectedBasicAuthCredentials() {
|
|
36
|
+
if (cachedBasicAuthCredentials === undefined) {
|
|
33
37
|
const secretName = process.env["CREDENTIALS_SECRET_NAME"];
|
|
34
38
|
if (!secretName) {
|
|
35
39
|
console.error("CREDENTIALS_SECRET_NAME env variable is not defined");
|
|
36
40
|
throw new Error();
|
|
37
41
|
}
|
|
38
|
-
|
|
42
|
+
cachedBasicAuthCredentials = await getBasicAuthCredentialsSecret(secretName);
|
|
39
43
|
}
|
|
40
|
-
return
|
|
44
|
+
return cachedBasicAuthCredentials;
|
|
41
45
|
}
|
|
42
|
-
async function
|
|
46
|
+
async function getBasicAuthCredentialsSecret(secretName) {
|
|
43
47
|
const secret = await getSecretValue(secretName);
|
|
44
|
-
if (
|
|
45
|
-
|
|
46
|
-
Buffer.from(`${secret.username}:${secret.password}`).toString("base64");
|
|
47
|
-
return [header];
|
|
48
|
+
if (isUsernameAndPasswordObject(secret)) {
|
|
49
|
+
return [encodeBasicAuthCredentials(secret)];
|
|
48
50
|
}
|
|
49
51
|
// See `BasicAuthAuthorizerProps` on the `ApiGateway` construct for an explanation of the formats
|
|
50
52
|
// we parse here
|
|
@@ -57,8 +59,11 @@ async function getSecretAsBasicAuthHeaders(secretName) {
|
|
|
57
59
|
console.error(`Failed to parse credentials array in secret '${secretName}' as JSON`, e);
|
|
58
60
|
throw new Error();
|
|
59
61
|
}
|
|
62
|
+
if (isArrayOfUsernameAndPasswordObjects(credentialsArray)) {
|
|
63
|
+
return credentialsArray.map(encodeBasicAuthCredentials);
|
|
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,11 +84,16 @@ 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
|
}
|
|
86
|
-
function
|
|
91
|
+
function encodeBasicAuthCredentials(credentials) {
|
|
92
|
+
const basicAuthHeader = "Basic " +
|
|
93
|
+
Buffer.from(`${credentials.username}:${credentials.password}`).toString("base64");
|
|
94
|
+
return { basicAuthHeader, username: credentials.username };
|
|
95
|
+
}
|
|
96
|
+
function isUsernameAndPasswordObject(value) {
|
|
87
97
|
return (typeof value === "object" &&
|
|
88
98
|
value !== null &&
|
|
89
99
|
"username" in value &&
|
|
@@ -91,6 +101,17 @@ function isSingleUsernameAndPassword(value) {
|
|
|
91
101
|
"password" in value &&
|
|
92
102
|
typeof value.password === "string");
|
|
93
103
|
}
|
|
104
|
+
function isArrayOfUsernameAndPasswordObjects(value) {
|
|
105
|
+
if (!Array.isArray(value)) {
|
|
106
|
+
return false;
|
|
107
|
+
}
|
|
108
|
+
for (const element of value) {
|
|
109
|
+
if (!isUsernameAndPasswordObject(element)) {
|
|
110
|
+
return false;
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
return true;
|
|
114
|
+
}
|
|
94
115
|
function hasCredentialsKeyWithStringValue(value) {
|
|
95
116
|
return (typeof value === "object" &&
|
|
96
117
|
value !== null &&
|
|
@@ -108,7 +129,32 @@ function isStringArray(value) {
|
|
|
108
129
|
}
|
|
109
130
|
return true;
|
|
110
131
|
}
|
|
132
|
+
/**
|
|
133
|
+
* We want to return the requesting username as a context variable in
|
|
134
|
+
* {@link AuthorizerResult.context}, for API Gateway access logs and parameter mapping. So if the
|
|
135
|
+
* basic auth credentials secret is stored as pre-encoded base64 strings, we need to parse them to
|
|
136
|
+
* get the username.
|
|
137
|
+
*/
|
|
138
|
+
function parseEncodedBasicAuthCredentials(encodedCredentials) {
|
|
139
|
+
let decodedCredentials;
|
|
140
|
+
try {
|
|
141
|
+
decodedCredentials = Buffer.from(encodedCredentials, "base64").toString();
|
|
142
|
+
}
|
|
143
|
+
catch (e) {
|
|
144
|
+
console.error("Basic auth credentials secret could not be decoded as base64:", e);
|
|
145
|
+
throw new Error();
|
|
146
|
+
}
|
|
147
|
+
const usernameAndPassword = decodedCredentials.split(":", 2);
|
|
148
|
+
if (usernameAndPassword.length !== 2) {
|
|
149
|
+
console.error("Basic auth credentials secret could not be decoded as 'username:password'");
|
|
150
|
+
throw new Error();
|
|
151
|
+
}
|
|
152
|
+
return {
|
|
153
|
+
basicAuthHeader: `Basic ${encodedCredentials}`,
|
|
154
|
+
username: usernameAndPassword[0],
|
|
155
|
+
};
|
|
156
|
+
}
|
|
111
157
|
export function clearCache() {
|
|
112
|
-
|
|
158
|
+
cachedBasicAuthCredentials = undefined;
|
|
113
159
|
}
|
|
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"]}
|
|
160
|
+
//# 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":"AAAA;;;;;;;;GAQG;AAMH,OAAO,EAAE,cAAc,EAAE,MAAM,iCAAiC,CAAA;AAoBhE,MAAM,CAAC,MAAM,OAAO,GAAG,KAAK,EAC1B,KAAyC,EACd,EAAE;IAC7B,MAAM,UAAU,GAAG,KAAK,CAAC,OAAO,EAAE,aAAa,CAAA;IAC/C,IAAI,CAAC,UAAU,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QACpD,OAAO,EAAE,YAAY,EAAE,KAAK,EAAE,CAAA;IAChC,CAAC;IAED,MAAM,mBAAmB,GAAG,MAAM,+BAA+B,EAAE,CAAA;IAEnE,KAAK,MAAM,QAAQ,IAAI,mBAAmB,EAAE,CAAC;QAC3C,IAAI,UAAU,KAAK,QAAQ,CAAC,eAAe,EAAE,CAAC;YAC5C,OAAO;gBACL,YAAY,EAAE,IAAI;gBAClB,OAAO,EAAE;oBACP,QAAQ,EAAE,QAAQ,CAAC,QAAQ;iBAC5B;aACF,CAAA;QACH,CAAC;IACH,CAAC;IAED,OAAO,EAAE,YAAY,EAAE,KAAK,EAAE,CAAA;AAChC,CAAC,CAAA;AAOD,qFAAqF;AACrF,IAAI,0BAA0B,GAC5B,SAAS,CAAA;AAEX;;;GAGG;AACH,KAAK,UAAU,+BAA+B;IAG5C,IAAI,0BAA0B,KAAK,SAAS,EAAE,CAAC;QAC7C,MAAM,UAAU,GACd,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAA;QACxC,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,OAAO,CAAC,KAAK,CAAC,qDAAqD,CAAC,CAAA;YACpE,MAAM,IAAI,KAAK,EAAE,CAAA;QACnB,CAAC;QAED,0BAA0B,GAAG,MAAM,6BAA6B,CAAC,UAAU,CAAC,CAAA;IAC9E,CAAC;IAED,OAAO,0BAA0B,CAAA;AACnC,CAAC;AAED,KAAK,UAAU,6BAA6B,CAC1C,UAAkB;IAElB,MAAM,MAAM,GAAG,MAAM,cAAc,CAAC,UAAU,CAAC,CAAA;IAE/C,IAAI,2BAA2B,CAAC,MAAM,CAAC,EAAE,CAAC;QACxC,OAAO,CAAC,0BAA0B,CAAC,MAAM,CAAC,CAAC,CAAA;IAC7C,CAAC;IAED,iGAAiG;IACjG,gBAAgB;IAChB,IAAI,gCAAgC,CAAC,MAAM,CAAC,EAAE,CAAC;QAC7C,IAAI,gBAAyB,CAAA;QAC7B,IAAI,CAAC;YACH,gBAAgB,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,WAAW,CAAC,CAAA;QACnD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,CAAC,KAAK,CACX,gDAAgD,UAAU,WAAW,EACrE,CAAC,CACF,CAAA;YACD,MAAM,IAAI,KAAK,EAAE,CAAA;QACnB,CAAC;QAED,IAAI,mCAAmC,CAAC,gBAAgB,CAAC,EAAE,CAAC;YAC1D,OAAO,gBAAgB,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAA;QACzD,CAAC;QAED,IAAI,aAAa,CAAC,gBAAgB,CAAC,EAAE,CAAC;YACpC,OAAO,gBAAgB,CAAC,GAAG,CAAC,gCAAgC,CAAC,CAAA;QAC/D,CAAC;IACH,CAAC;IAED,OAAO,CAAC,KAAK,CACX,mFAAmF,UAAU,IAAI,CAClG,CAAA;IACD,MAAM,IAAI,KAAK,EAAE,CAAA;AACnB,CAAC;AAED,mDAAmD;AACnD,MAAM,CAAC,MAAM,YAAY,GAAG;IAC1B,oBAAoB,EAAE,GAAG,EAAE,CAAC,IAAI,cAAc,EAAE;CACjD,CAAA;AAED,KAAK,UAAU,cAAc,CAAC,UAAkB;IAC9C,MAAM,MAAM,GAAG,YAAY,CAAC,oBAAoB,EAAE,CAAA;IAClD,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,cAAc,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,CAAC,CAAA;IAEpE,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC;QACzB,OAAO,CAAC,KAAK,CAAC,+BAA+B,UAAU,GAAG,CAAC,CAAA;QAC3D,MAAM,IAAI,KAAK,EAAE,CAAA;IACnB,CAAC;IAED,IAAI,CAAC;QACH,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,YAAY,CAAC,CAAA;IACxC,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,OAAO,CAAC,KAAK,CAAC,2BAA2B,UAAU,YAAY,EAAE,CAAC,CAAC,CAAA;QACnE,MAAM,IAAI,KAAK,EAAE,CAAA;IACnB,CAAC;AACH,CAAC;AAED,SAAS,0BAA0B,CAAC,WAGnC;IACC,MAAM,eAAe,GACnB,QAAQ;QACR,MAAM,CAAC,IAAI,CAAC,GAAG,WAAW,CAAC,QAAQ,IAAI,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC,QAAQ,CACrE,QAAQ,CACT,CAAA;IAEH,OAAO,EAAE,eAAe,EAAE,QAAQ,EAAE,WAAW,CAAC,QAAQ,EAAE,CAAA;AAC5D,CAAC;AAED,SAAS,2BAA2B,CAClC,KAAc;IAEd,OAAO,CACL,OAAO,KAAK,KAAK,QAAQ;QACzB,KAAK,KAAK,IAAI;QACd,UAAU,IAAI,KAAK;QACnB,OAAO,KAAK,CAAC,QAAQ,KAAK,QAAQ;QAClC,UAAU,IAAI,KAAK;QACnB,OAAO,KAAK,CAAC,QAAQ,KAAK,QAAQ,CACnC,CAAA;AACH,CAAC;AAED,SAAS,mCAAmC,CAC1C,KAAc;IAEd,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QAC1B,OAAO,KAAK,CAAA;IACd,CAAC;IAED,KAAK,MAAM,OAAO,IAAI,KAAK,EAAE,CAAC;QAC5B,IAAI,CAAC,2BAA2B,CAAC,OAAO,CAAC,EAAE,CAAC;YAC1C,OAAO,KAAK,CAAA;QACd,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAA;AACb,CAAC;AAED,SAAS,gCAAgC,CACvC,KAAc;IAEd,OAAO,CACL,OAAO,KAAK,KAAK,QAAQ;QACzB,KAAK,KAAK,IAAI;QACd,aAAa,IAAI,KAAK;QACtB,OAAO,KAAK,CAAC,WAAW,KAAK,QAAQ,CACtC,CAAA;AACH,CAAC;AAED,SAAS,aAAa,CAAC,KAAc;IACnC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QAC1B,OAAO,KAAK,CAAA;IACd,CAAC;IAED,KAAK,MAAM,OAAO,IAAI,KAAK,EAAE,CAAC;QAC5B,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE,CAAC;YAChC,OAAO,KAAK,CAAA;QACd,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAA;AACb,CAAC;AAED;;;;;GAKG;AACH,SAAS,gCAAgC,CACvC,kBAA0B;IAE1B,IAAI,kBAA0B,CAAA;IAC9B,IAAI,CAAC;QACH,kBAAkB,GAAG,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,QAAQ,CAAC,CAAC,QAAQ,EAAE,CAAA;IAC3E,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,OAAO,CAAC,KAAK,CACX,+DAA+D,EAC/D,CAAC,CACF,CAAA;QACD,MAAM,IAAI,KAAK,EAAE,CAAA;IACnB,CAAC;IAED,MAAM,mBAAmB,GAAG,kBAAkB,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,CAAA;IAC5D,IAAI,mBAAmB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACrC,OAAO,CAAC,KAAK,CACX,2EAA2E,CAC5E,CAAA;QACD,MAAM,IAAI,KAAK,EAAE,CAAA;IACnB,CAAC;IAED,OAAO;QACL,eAAe,EAAE,SAAS,kBAAkB,EAAE;QAC9C,QAAQ,EAAE,mBAAmB,CAAC,CAAC,CAAC;KACjC,CAAA;AACH,CAAC;AAED,MAAM,UAAU,UAAU;IACxB,0BAA0B,GAAG,SAAS,CAAA;AACxC,CAAC","sourcesContent":["/**\n * This lambda verifies authorization header against static basic auth credentials saved in Secrets\n * Manager.\n *\n * Expects the following environment variables:\n * - CREDENTIALS_SECRET_NAME\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 */\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 (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\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 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  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":"AAAA;;;;;;;;;;;GAWG;AAMH,OAAO,EAAE,cAAc,EAAE,MAAM,iCAAiC,CAAA;AAChE,OAAO,EAAE,kBAAkB,EAAE,MAAM,gBAAgB,CAAA;AA4BnD,MAAM,CAAC,MAAM,OAAO,GAAG,KAAK,EAC1B,KAAyC,EACd,EAAE;IAC7B,MAAM,UAAU,GAAG,KAAK,CAAC,OAAO,EAAE,aAAa,CAAA;IAC/C,IAAI,CAAC,UAAU,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;QACrD,OAAO,EAAE,YAAY,EAAE,KAAK,EAAE,CAAA;IAChC,CAAC;IAED,MAAM,MAAM,GAAG,MAAM,iBAAiB,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAA,CAAC,kCAAkC;IAClG,QAAQ,MAAM,EAAE,CAAC;QACf,KAAK,SAAS;YACZ,OAAO,EAAE,YAAY,EAAE,KAAK,EAAE,CAAA;QAChC,KAAK,SAAS;YACZ,wFAAwF;YACxF,qFAAqF;YACrF,yFAAyF;YACzF,qEAAqE;YACrE,MAAM,IAAI,KAAK,CAAC,cAAc,CAAC,CAAA;QACjC,OAAO,CAAC,CAAC,CAAC;YACR,OAAO;gBACL,YAAY,EAAE,IAAI;gBAClB,OAAO,EAAE;oBACP,QAAQ,EAAE,MAAM,CAAC,SAAS;oBAC1B,2BAA2B,EAAE,MAAM,8BAA8B,EAAE;iBACpE;aACF,CAAA;QACH,CAAC;IACH,CAAC;AACH,CAAC,CAAA;AAED,4DAA4D;AAC5D,KAAK,UAAU,iBAAiB,CAC9B,KAAa;IAEb,IAAI,CAAC;QACH,MAAM,aAAa,GAAG,gBAAgB,EAAE,CAAA;QACxC,6FAA6F;QAC7F,gBAAgB;QAChB,OAAO,MAAM,aAAa,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;IAC1C,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,4DAA4D;QAC5D,0GAA0G;QAC1G,8FAA8F;QAC9F,cAAc;QACd,IAAI,CAAC,YAAY,KAAK,IAAI,CAAC,CAAC,OAAO,EAAE,QAAQ,CAAC,eAAe,CAAC,EAAE,CAAC;YAC/D,OAAO,SAAS,CAAA;QAClB,CAAC;aAAM,CAAC;YACN,OAAO,SAAS,CAAA;QAClB,CAAC;IACH,CAAC;AACH,CAAC;AAMD;;;GAGG;AACH,IAAI,mBAAmB,GAA8B,SAAS,CAAA;AAE9D,SAAS,gBAAgB;IACvB,IAAI,mBAAmB,KAAK,SAAS,EAAE,CAAC;QACtC,mBAAmB,GAAG,YAAY,CAAC,mBAAmB,EAAE,CAAA;IAC1D,CAAC;IACD,OAAO,mBAAmB,CAAA;AAC5B,CAAC;AAED,mDAAmD;AACnD,MAAM,CAAC,MAAM,YAAY,GAAG;IAC1B,mBAAmB,EAAE,GAAkB,EAAE;QACvC,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,CAAA;QAC9C,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,OAAO,CAAC,KAAK,CAAC,0CAA0C,CAAC,CAAA;YACzD,MAAM,IAAI,KAAK,EAAE,CAAA;QACnB,CAAC;QAED,OAAO,kBAAkB,CAAC,MAAM,CAAC;YAC/B,UAAU;YACV,QAAQ,EAAE,QAAQ;YAClB,QAAQ,EAAE,IAAI;YACd,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,SAAS,EAAE,yCAAyC;SAC1F,CAAC,CAAA;IACJ,CAAC;IACD,oBAAoB,EAAE,GAAG,EAAE,CAAC,IAAI,cAAc,EAAE;CACjD,CAAA;AAED,qFAAqF;AACrF,IAAI,iCAAiC,GAAuB,SAAS,CAAA;AAErE,KAAK,UAAU,8BAA8B;IAC3C,IAAI,iCAAiC,KAAK,SAAS,EAAE,CAAC;QACpD,MAAM,UAAU,GACd,OAAO,CAAC,GAAG,CAAC,wCAAwC,CAAC,CAAA;QACvD,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,OAAO,SAAS,CAAA;QAClB,CAAC;QAED,iCAAiC;YAC/B,MAAM,0BAA0B,CAAC,UAAU,CAAC,CAAA;IAChD,CAAC;IAED,OAAO,iCAAiC,CAAA;AAC1C,CAAC;AAED,KAAK,UAAU,0BAA0B,CAAC,UAAkB;IAC1D,MAAM,WAAW,GAAG,MAAM,cAAc,CAAC,UAAU,CAAC,CAAA;IACpD,IAAI,CAAC,uBAAuB,CAAC,WAAW,CAAC,EAAE,CAAC;QAC1C,OAAO,CAAC,KAAK,CACX,+EAA+E,UAAU,IAAI,CAC9F,CAAA;QACD,MAAM,IAAI,KAAK,EAAE,CAAA;IACnB,CAAC;IAED,OAAO,CACL,QAAQ;QACR,MAAM,CAAC,IAAI,CAAC,GAAG,WAAW,CAAC,QAAQ,IAAI,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC,QAAQ,CACrE,QAAQ,CACT,CACF,CAAA;AACH,CAAC;AAED,KAAK,UAAU,cAAc,CAAC,UAAkB;IAC9C,MAAM,MAAM,GAAG,YAAY,CAAC,oBAAoB,EAAE,CAAA;IAClD,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,cAAc,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,CAAC,CAAA;IAEpE,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC;QACzB,OAAO,CAAC,KAAK,CAAC,+BAA+B,UAAU,GAAG,CAAC,CAAA;QAC3D,MAAM,IAAI,KAAK,EAAE,CAAA;IACnB,CAAC;IAED,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,YAAY,CAAC,CAAA;AACxC,CAAC;AAED,SAAS,uBAAuB,CAC9B,KAAc;IAEd,OAAO,CACL,OAAO,KAAK,KAAK,QAAQ;QACzB,KAAK,KAAK,IAAI;QACd,UAAU,IAAI,KAAK;QACnB,OAAO,KAAK,CAAC,QAAQ,KAAK,QAAQ;QAClC,UAAU,IAAI,KAAK;QACnB,OAAO,KAAK,CAAC,QAAQ,KAAK,QAAQ,CACnC,CAAA;AACH,CAAC;AAED,MAAM,UAAU,UAAU;IACxB,mBAAmB,GAAG,SAAS,CAAA;IAC/B,iCAAiC,GAAG,SAAS,CAAA;AAC/C,CAAC","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,16 +1,15 @@
|
|
|
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
|
-
* 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
|
-
* -
|
|
10
|
-
*
|
|
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
|
|
12
|
+
* - Set this to require that the access token payload contains the given scope
|
|
14
13
|
*/
|
|
15
14
|
import type { APIGatewayRequestAuthorizerEventV2, APIGatewaySimpleAuthorizerResult } from "aws-lambda";
|
|
16
15
|
import { SecretsManager } from "@aws-sdk/client-secrets-manager";
|
|
@@ -19,18 +18,27 @@ type AuthorizerResult = APIGatewaySimpleAuthorizerResult & {
|
|
|
19
18
|
/**
|
|
20
19
|
* Returning a context object from our authorizer allows our API Gateway to access these variables
|
|
21
20
|
* via `${context.authorizer.<property>}`.
|
|
22
|
-
* https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-
|
|
21
|
+
* https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-parameter-mapping.html
|
|
23
22
|
*/
|
|
24
23
|
context?: {
|
|
25
24
|
/**
|
|
26
|
-
* If the token
|
|
27
|
-
* variable (named `authorizer.clientId`).
|
|
28
|
-
*
|
|
29
|
-
*
|
|
25
|
+
* If the request used an access token, and the token was verified, we return the auth client ID
|
|
26
|
+
* from the token's claims in this context variable (named `authorizer.clientId`). We use this
|
|
27
|
+
* to include the requesting client in the API Gateway access logs (see `defaultAccessLogFormat`
|
|
28
|
+
* in our `ApiGateway` construct). You can also use this when mapping parameters to the backend
|
|
29
|
+
* integration (see `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).
|
|
30
30
|
*/
|
|
31
31
|
clientId?: string;
|
|
32
32
|
/**
|
|
33
|
-
*
|
|
33
|
+
* If the request used Basic Auth, and the credentials were verified, we return the username
|
|
34
|
+
* that was used in this context variable (named `authorizer.username`). We use this to include
|
|
35
|
+
* the requesting user in the API Gateway access logs (see `defaultAccessLogFormat` in our
|
|
36
|
+
* `ApiGateway` construct). You can also use this when mapping parameters to the backend
|
|
37
|
+
* integration (see `AlbIntegrationProps.mapParameters` on the `ApiGateway` construct).
|
|
38
|
+
*/
|
|
39
|
+
username?: string;
|
|
40
|
+
/**
|
|
41
|
+
* See `CognitoUserPoolAuthorizerProps.basicAuthForInternalAuthorization` on the `ApiGateway`
|
|
34
42
|
* construct (we provide the same context variable here as in the Cognito User Pool authorizer,
|
|
35
43
|
* using the credentials from BASIC_AUTH_CREDENTIALS_SECRET_NAME).
|
|
36
44
|
*/
|