@liflig/cdk-cloudfront-auth 1.0.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.
Files changed (48) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +63 -0
  3. package/dist/check-auth/index.js +2 -0
  4. package/dist/check-auth/index.js.LICENSE.txt +17 -0
  5. package/dist/generate-secret/index.js +1 -0
  6. package/dist/http-headers/index.js +2 -0
  7. package/dist/http-headers/index.js.LICENSE.txt +6 -0
  8. package/dist/parse-auth/index.js +2 -0
  9. package/dist/parse-auth/index.js.LICENSE.txt +31 -0
  10. package/dist/refresh-auth/index.js +2 -0
  11. package/dist/refresh-auth/index.js.LICENSE.txt +31 -0
  12. package/dist/sign-out/index.js +2 -0
  13. package/dist/sign-out/index.js.LICENSE.txt +17 -0
  14. package/lib/client-secret.d.ts +10 -0
  15. package/lib/client-secret.js +54 -0
  16. package/lib/client-update.d.ts +14 -0
  17. package/lib/client-update.js +59 -0
  18. package/lib/cloudfront-auth.d.ts +132 -0
  19. package/lib/cloudfront-auth.js +267 -0
  20. package/lib/generate-secret.d.ts +15 -0
  21. package/lib/generate-secret.js +71 -0
  22. package/lib/handlers/check-auth.d.ts +7 -0
  23. package/lib/handlers/generate-secret.d.ts +9 -0
  24. package/lib/handlers/http-headers.d.ts +1 -0
  25. package/lib/handlers/parse-auth.d.ts +1 -0
  26. package/lib/handlers/refresh-auth.d.ts +1 -0
  27. package/lib/handlers/sign-out.d.ts +1 -0
  28. package/lib/handlers/util/axios.d.ts +4 -0
  29. package/lib/handlers/util/axios.js +42 -0
  30. package/lib/handlers/util/base64.d.ts +8 -0
  31. package/lib/handlers/util/base64.js +26 -0
  32. package/lib/handlers/util/cloudfront.d.ts +17 -0
  33. package/lib/handlers/util/cloudfront.js +102 -0
  34. package/lib/handlers/util/config.d.ts +26 -0
  35. package/lib/handlers/util/config.js +48 -0
  36. package/lib/handlers/util/cookies.d.ts +29 -0
  37. package/lib/handlers/util/cookies.js +115 -0
  38. package/lib/handlers/util/jwt.d.ts +17 -0
  39. package/lib/handlers/util/jwt.js +59 -0
  40. package/lib/handlers/util/logger.d.ts +16 -0
  41. package/lib/handlers/util/logger.js +55 -0
  42. package/lib/handlers/util/nonce.d.ts +9 -0
  43. package/lib/handlers/util/nonce.js +47 -0
  44. package/lib/index.d.ts +2 -0
  45. package/lib/index.js +19 -0
  46. package/lib/lambdas.d.ts +33 -0
  47. package/lib/lambdas.js +88 -0
  48. package/package.json +75 -0
@@ -0,0 +1,17 @@
1
+ /*!
2
+ * cookie
3
+ * Copyright(c) 2012-2014 Roman Shtylman
4
+ * Copyright(c) 2015 Douglas Christopher Wilson
5
+ * MIT Licensed
6
+ */
7
+
8
+ /*! safe-buffer. MIT License. Feross Aboukhadijeh <https://feross.org/opensource> */
9
+
10
+ /**
11
+ * @license
12
+ * Lodash <https://lodash.com/>
13
+ * Copyright OpenJS Foundation and other contributors <https://openjsf.org/>
14
+ * Released under MIT license <https://lodash.com/license>
15
+ * Based on Underscore.js 1.8.3 <http://underscorejs.org/LICENSE>
16
+ * Copyright Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
17
+ */
@@ -0,0 +1,10 @@
1
+ import * as cognito from "aws-cdk-lib/aws-cognito";
2
+ import { Construct } from "constructs";
3
+ export interface RetrieveClientSecretProps {
4
+ client: cognito.IUserPoolClient;
5
+ userPool: cognito.IUserPool;
6
+ }
7
+ export declare class RetrieveClientSecret extends Construct {
8
+ clientSecretValue: string;
9
+ constructor(scope: Construct, id: string, props: RetrieveClientSecretProps);
10
+ }
@@ -0,0 +1,54 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || function (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
25
+ Object.defineProperty(exports, "__esModule", { value: true });
26
+ exports.RetrieveClientSecret = void 0;
27
+ const iam = __importStar(require("aws-cdk-lib/aws-iam"));
28
+ const cr = __importStar(require("aws-cdk-lib/custom-resources"));
29
+ const constructs_1 = require("constructs");
30
+ class RetrieveClientSecret extends constructs_1.Construct {
31
+ constructor(scope, id, props) {
32
+ super(scope, id);
33
+ const clientSecret = new cr.AwsCustomResource(this, "Resource", {
34
+ onUpdate: {
35
+ service: "CognitoIdentityServiceProvider",
36
+ action: "describeUserPoolClient",
37
+ parameters: {
38
+ UserPoolId: props.userPool.userPoolId,
39
+ ClientId: props.client.userPoolClientId,
40
+ },
41
+ physicalResourceId: cr.PhysicalResourceId.of(`${props.userPool.userPoolId}-${props.client.userPoolClientId}`),
42
+ },
43
+ policy: cr.AwsCustomResourcePolicy.fromStatements([
44
+ new iam.PolicyStatement({
45
+ actions: ["cognito-idp:DescribeUserPoolClient"],
46
+ resources: [props.userPool.userPoolArn],
47
+ }),
48
+ ]),
49
+ });
50
+ this.clientSecretValue = clientSecret.getResponseField("UserPoolClient.ClientSecret");
51
+ }
52
+ }
53
+ exports.RetrieveClientSecret = RetrieveClientSecret;
54
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xpZW50LXNlY3JldC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9jbGllbnQtc2VjcmV0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQ0EseURBQTBDO0FBQzFDLGlFQUFrRDtBQUNsRCwyQ0FBc0M7QUFPdEMsTUFBYSxvQkFBcUIsU0FBUSxzQkFBUztJQUdqRCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQWdDO1FBQ3hFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsTUFBTSxZQUFZLEdBQUcsSUFBSSxFQUFFLENBQUMsaUJBQWlCLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUM5RCxRQUFRLEVBQUU7Z0JBQ1IsT0FBTyxFQUFFLGdDQUFnQztnQkFDekMsTUFBTSxFQUFFLHdCQUF3QjtnQkFDaEMsVUFBVSxFQUFFO29CQUNWLFVBQVUsRUFBRSxLQUFLLENBQUMsUUFBUSxDQUFDLFVBQVU7b0JBQ3JDLFFBQVEsRUFBRSxLQUFLLENBQUMsTUFBTSxDQUFDLGdCQUFnQjtpQkFDeEM7Z0JBQ0Qsa0JBQWtCLEVBQUUsRUFBRSxDQUFDLGtCQUFrQixDQUFDLEVBQUUsQ0FDMUMsR0FBRyxLQUFLLENBQUMsUUFBUSxDQUFDLFVBQVUsSUFBSSxLQUFLLENBQUMsTUFBTSxDQUFDLGdCQUFnQixFQUFFLENBQ2hFO2FBQ0Y7WUFDRCxNQUFNLEVBQUUsRUFBRSxDQUFDLHVCQUF1QixDQUFDLGNBQWMsQ0FBQztnQkFDaEQsSUFBSSxHQUFHLENBQUMsZUFBZSxDQUFDO29CQUN0QixPQUFPLEVBQUUsQ0FBQyxvQ0FBb0MsQ0FBQztvQkFDL0MsU0FBUyxFQUFFLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUM7aUJBQ3hDLENBQUM7YUFDSCxDQUFDO1NBQ0gsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLGlCQUFpQixHQUFHLFlBQVksQ0FBQyxnQkFBZ0IsQ0FDcEQsNkJBQTZCLENBQzlCLENBQUE7SUFDSCxDQUFDO0NBQ0Y7QUE5QkQsb0RBOEJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29nbml0byBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNvZ25pdG9cIlxuaW1wb3J0ICogYXMgaWFtIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtaWFtXCJcbmltcG9ydCAqIGFzIGNyIGZyb20gXCJhd3MtY2RrLWxpYi9jdXN0b20tcmVzb3VyY2VzXCJcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCJcblxuZXhwb3J0IGludGVyZmFjZSBSZXRyaWV2ZUNsaWVudFNlY3JldFByb3BzIHtcbiAgY2xpZW50OiBjb2duaXRvLklVc2VyUG9vbENsaWVudFxuICB1c2VyUG9vbDogY29nbml0by5JVXNlclBvb2xcbn1cblxuZXhwb3J0IGNsYXNzIFJldHJpZXZlQ2xpZW50U2VjcmV0IGV4dGVuZHMgQ29uc3RydWN0IHtcbiAgcHVibGljIGNsaWVudFNlY3JldFZhbHVlOiBzdHJpbmdcblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogUmV0cmlldmVDbGllbnRTZWNyZXRQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIGNvbnN0IGNsaWVudFNlY3JldCA9IG5ldyBjci5Bd3NDdXN0b21SZXNvdXJjZSh0aGlzLCBcIlJlc291cmNlXCIsIHtcbiAgICAgIG9uVXBkYXRlOiB7XG4gICAgICAgIHNlcnZpY2U6IFwiQ29nbml0b0lkZW50aXR5U2VydmljZVByb3ZpZGVyXCIsXG4gICAgICAgIGFjdGlvbjogXCJkZXNjcmliZVVzZXJQb29sQ2xpZW50XCIsXG4gICAgICAgIHBhcmFtZXRlcnM6IHtcbiAgICAgICAgICBVc2VyUG9vbElkOiBwcm9wcy51c2VyUG9vbC51c2VyUG9vbElkLFxuICAgICAgICAgIENsaWVudElkOiBwcm9wcy5jbGllbnQudXNlclBvb2xDbGllbnRJZCxcbiAgICAgICAgfSxcbiAgICAgICAgcGh5c2ljYWxSZXNvdXJjZUlkOiBjci5QaHlzaWNhbFJlc291cmNlSWQub2YoXG4gICAgICAgICAgYCR7cHJvcHMudXNlclBvb2wudXNlclBvb2xJZH0tJHtwcm9wcy5jbGllbnQudXNlclBvb2xDbGllbnRJZH1gLFxuICAgICAgICApLFxuICAgICAgfSxcbiAgICAgIHBvbGljeTogY3IuQXdzQ3VzdG9tUmVzb3VyY2VQb2xpY3kuZnJvbVN0YXRlbWVudHMoW1xuICAgICAgICBuZXcgaWFtLlBvbGljeVN0YXRlbWVudCh7XG4gICAgICAgICAgYWN0aW9uczogW1wiY29nbml0by1pZHA6RGVzY3JpYmVVc2VyUG9vbENsaWVudFwiXSxcbiAgICAgICAgICByZXNvdXJjZXM6IFtwcm9wcy51c2VyUG9vbC51c2VyUG9vbEFybl0sXG4gICAgICAgIH0pLFxuICAgICAgXSksXG4gICAgfSlcblxuICAgIHRoaXMuY2xpZW50U2VjcmV0VmFsdWUgPSBjbGllbnRTZWNyZXQuZ2V0UmVzcG9uc2VGaWVsZChcbiAgICAgIFwiVXNlclBvb2xDbGllbnQuQ2xpZW50U2VjcmV0XCIsXG4gICAgKVxuICB9XG59XG4iXX0=
@@ -0,0 +1,14 @@
1
+ import * as cognito from "aws-cdk-lib/aws-cognito";
2
+ import { Construct } from "constructs";
3
+ interface ClientUpdateProps {
4
+ oauthScopes: string[];
5
+ client: cognito.IUserPoolClient;
6
+ userPool: cognito.IUserPool;
7
+ callbackUrl: string;
8
+ signOutUrl: string;
9
+ identityProviders: string[];
10
+ }
11
+ export declare class ClientUpdate extends Construct {
12
+ constructor(scope: Construct, id: string, props: ClientUpdateProps);
13
+ }
14
+ export {};
@@ -0,0 +1,59 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || function (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
25
+ Object.defineProperty(exports, "__esModule", { value: true });
26
+ exports.ClientUpdate = void 0;
27
+ const iam = __importStar(require("aws-cdk-lib/aws-iam"));
28
+ const cr = __importStar(require("aws-cdk-lib/custom-resources"));
29
+ const constructs_1 = require("constructs");
30
+ class ClientUpdate extends constructs_1.Construct {
31
+ constructor(scope, id, props) {
32
+ super(scope, id);
33
+ new cr.AwsCustomResource(this, "Resource", {
34
+ onUpdate: {
35
+ service: "CognitoIdentityServiceProvider",
36
+ action: "updateUserPoolClient",
37
+ parameters: {
38
+ AllowedOAuthFlows: ["code"],
39
+ AllowedOAuthFlowsUserPoolClient: true,
40
+ SupportedIdentityProviders: props.identityProviders,
41
+ AllowedOAuthScopes: props.oauthScopes,
42
+ ClientId: props.client.userPoolClientId,
43
+ CallbackURLs: [props.callbackUrl],
44
+ LogoutURLs: [props.signOutUrl],
45
+ UserPoolId: props.userPool.userPoolId,
46
+ },
47
+ physicalResourceId: cr.PhysicalResourceId.of(`${props.userPool.userPoolId}-${props.client.userPoolClientId}`),
48
+ },
49
+ policy: cr.AwsCustomResourcePolicy.fromStatements([
50
+ new iam.PolicyStatement({
51
+ actions: ["cognito-idp:UpdateUserPoolClient"],
52
+ resources: [props.userPool.userPoolArn],
53
+ }),
54
+ ]),
55
+ });
56
+ }
57
+ }
58
+ exports.ClientUpdate = ClientUpdate;
59
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xpZW50LXVwZGF0ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9jbGllbnQtdXBkYXRlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQ0EseURBQTBDO0FBQzFDLGlFQUFrRDtBQUNsRCwyQ0FBc0M7QUFXdEMsTUFBYSxZQUFhLFNBQVEsc0JBQVM7SUFDekMsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUF3QjtRQUNoRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLElBQUksRUFBRSxDQUFDLGlCQUFpQixDQUFDLElBQUksRUFBRSxVQUFVLEVBQUU7WUFDekMsUUFBUSxFQUFFO2dCQUNSLE9BQU8sRUFBRSxnQ0FBZ0M7Z0JBQ3pDLE1BQU0sRUFBRSxzQkFBc0I7Z0JBQzlCLFVBQVUsRUFBRTtvQkFDVixpQkFBaUIsRUFBRSxDQUFDLE1BQU0sQ0FBQztvQkFDM0IsK0JBQStCLEVBQUUsSUFBSTtvQkFDckMsMEJBQTBCLEVBQUUsS0FBSyxDQUFDLGlCQUFpQjtvQkFDbkQsa0JBQWtCLEVBQUUsS0FBSyxDQUFDLFdBQVc7b0JBQ3JDLFFBQVEsRUFBRSxLQUFLLENBQUMsTUFBTSxDQUFDLGdCQUFnQjtvQkFDdkMsWUFBWSxFQUFFLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQztvQkFDakMsVUFBVSxFQUFFLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQztvQkFDOUIsVUFBVSxFQUFFLEtBQUssQ0FBQyxRQUFRLENBQUMsVUFBVTtpQkFDdEM7Z0JBQ0Qsa0JBQWtCLEVBQUUsRUFBRSxDQUFDLGtCQUFrQixDQUFDLEVBQUUsQ0FDMUMsR0FBRyxLQUFLLENBQUMsUUFBUSxDQUFDLFVBQVUsSUFBSSxLQUFLLENBQUMsTUFBTSxDQUFDLGdCQUFnQixFQUFFLENBQ2hFO2FBQ0Y7WUFDRCxNQUFNLEVBQUUsRUFBRSxDQUFDLHVCQUF1QixDQUFDLGNBQWMsQ0FBQztnQkFDaEQsSUFBSSxHQUFHLENBQUMsZUFBZSxDQUFDO29CQUN0QixPQUFPLEVBQUUsQ0FBQyxrQ0FBa0MsQ0FBQztvQkFDN0MsU0FBUyxFQUFFLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUM7aUJBQ3hDLENBQUM7YUFDSCxDQUFDO1NBQ0gsQ0FBQyxDQUFBO0lBQ0osQ0FBQztDQUNGO0FBOUJELG9DQThCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGNvZ25pdG8gZnJvbSBcImF3cy1jZGstbGliL2F3cy1jb2duaXRvXCJcbmltcG9ydCAqIGFzIGlhbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBjciBmcm9tIFwiYXdzLWNkay1saWIvY3VzdG9tLXJlc291cmNlc1wiXG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tIFwiY29uc3RydWN0c1wiXG5cbmludGVyZmFjZSBDbGllbnRVcGRhdGVQcm9wcyB7XG4gIG9hdXRoU2NvcGVzOiBzdHJpbmdbXVxuICBjbGllbnQ6IGNvZ25pdG8uSVVzZXJQb29sQ2xpZW50XG4gIHVzZXJQb29sOiBjb2duaXRvLklVc2VyUG9vbFxuICBjYWxsYmFja1VybDogc3RyaW5nXG4gIHNpZ25PdXRVcmw6IHN0cmluZ1xuICBpZGVudGl0eVByb3ZpZGVyczogc3RyaW5nW11cbn1cblxuZXhwb3J0IGNsYXNzIENsaWVudFVwZGF0ZSBleHRlbmRzIENvbnN0cnVjdCB7XG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBDbGllbnRVcGRhdGVQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIG5ldyBjci5Bd3NDdXN0b21SZXNvdXJjZSh0aGlzLCBcIlJlc291cmNlXCIsIHtcbiAgICAgIG9uVXBkYXRlOiB7XG4gICAgICAgIHNlcnZpY2U6IFwiQ29nbml0b0lkZW50aXR5U2VydmljZVByb3ZpZGVyXCIsXG4gICAgICAgIGFjdGlvbjogXCJ1cGRhdGVVc2VyUG9vbENsaWVudFwiLFxuICAgICAgICBwYXJhbWV0ZXJzOiB7XG4gICAgICAgICAgQWxsb3dlZE9BdXRoRmxvd3M6IFtcImNvZGVcIl0sXG4gICAgICAgICAgQWxsb3dlZE9BdXRoRmxvd3NVc2VyUG9vbENsaWVudDogdHJ1ZSxcbiAgICAgICAgICBTdXBwb3J0ZWRJZGVudGl0eVByb3ZpZGVyczogcHJvcHMuaWRlbnRpdHlQcm92aWRlcnMsXG4gICAgICAgICAgQWxsb3dlZE9BdXRoU2NvcGVzOiBwcm9wcy5vYXV0aFNjb3BlcyxcbiAgICAgICAgICBDbGllbnRJZDogcHJvcHMuY2xpZW50LnVzZXJQb29sQ2xpZW50SWQsXG4gICAgICAgICAgQ2FsbGJhY2tVUkxzOiBbcHJvcHMuY2FsbGJhY2tVcmxdLFxuICAgICAgICAgIExvZ291dFVSTHM6IFtwcm9wcy5zaWduT3V0VXJsXSxcbiAgICAgICAgICBVc2VyUG9vbElkOiBwcm9wcy51c2VyUG9vbC51c2VyUG9vbElkLFxuICAgICAgICB9LFxuICAgICAgICBwaHlzaWNhbFJlc291cmNlSWQ6IGNyLlBoeXNpY2FsUmVzb3VyY2VJZC5vZihcbiAgICAgICAgICBgJHtwcm9wcy51c2VyUG9vbC51c2VyUG9vbElkfS0ke3Byb3BzLmNsaWVudC51c2VyUG9vbENsaWVudElkfWAsXG4gICAgICAgICksXG4gICAgICB9LFxuICAgICAgcG9saWN5OiBjci5Bd3NDdXN0b21SZXNvdXJjZVBvbGljeS5mcm9tU3RhdGVtZW50cyhbXG4gICAgICAgIG5ldyBpYW0uUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgICBhY3Rpb25zOiBbXCJjb2duaXRvLWlkcDpVcGRhdGVVc2VyUG9vbENsaWVudFwiXSxcbiAgICAgICAgICByZXNvdXJjZXM6IFtwcm9wcy51c2VyUG9vbC51c2VyUG9vbEFybl0sXG4gICAgICAgIH0pLFxuICAgICAgXSksXG4gICAgfSlcbiAgfVxufVxuIl19
@@ -0,0 +1,132 @@
1
+ import * as cloudfront from "aws-cdk-lib/aws-cloudfront";
2
+ import { AddBehaviorOptions, BehaviorOptions, IOrigin } from "aws-cdk-lib/aws-cloudfront";
3
+ import * as cognito from "aws-cdk-lib/aws-cognito";
4
+ import { ClientUpdate } from "./client-update";
5
+ import { AuthLambdas } from "./lambdas";
6
+ import { Construct } from "constructs";
7
+ export interface CloudFrontAuthProps {
8
+ /**
9
+ * Cognito Client that will be used to authenticate the user.
10
+ *
11
+ * If a custom client is provided, the updateClient method cannot
12
+ * be used since we cannot know which parameters was set.
13
+ *
14
+ * @default - a new client will be generated
15
+ */
16
+ client?: cognito.UserPoolClient;
17
+ userPool: cognito.IUserPool;
18
+ /**
19
+ * The domain that is used for Cognito Auth.
20
+ *
21
+ * If not using custom domains this will be a name under amazoncognito.com.
22
+ *
23
+ * @example `${domain.domainName}.auth.${region}.amazoncognito.com`
24
+ */
25
+ cognitoAuthDomain: string;
26
+ authLambdas: AuthLambdas;
27
+ /**
28
+ * @default /auth/callback
29
+ */
30
+ callbackPath?: string;
31
+ /**
32
+ * @default /
33
+ */
34
+ signOutRedirectTo?: string;
35
+ /**
36
+ * @default /auth/sign-out
37
+ */
38
+ signOutPath?: string;
39
+ /**
40
+ * @default /auth/refresh
41
+ */
42
+ refreshAuthPath?: string;
43
+ /**
44
+ * Log level.
45
+ *
46
+ * A log level of debug will log secrets and should only be used in
47
+ * a development environment.
48
+ *
49
+ * @default warn
50
+ */
51
+ logLevel?: "none" | "error" | "warn" | "info" | "debug";
52
+ /**
53
+ * Require the user to be part of a specific Cognito group to
54
+ * access any resource.
55
+ */
56
+ requireGroupAnyOf?: string[];
57
+ }
58
+ export interface UpdateClientProps {
59
+ signOutUrl: string;
60
+ callbackUrl: string;
61
+ /**
62
+ * List of identity providers used for the client.
63
+ *
64
+ * @default - COGNITO and identity providers registered in the UserPool construct
65
+ */
66
+ identityProviders?: string[];
67
+ }
68
+ /**
69
+ * Configure previously deployed lambda functions, Cognito client
70
+ * and CloudFront distribution.
71
+ */
72
+ export declare class CloudFrontAuth extends Construct {
73
+ readonly callbackPath: string;
74
+ readonly signOutRedirectTo: string;
75
+ readonly signOutPath: string;
76
+ readonly refreshAuthPath: string;
77
+ private readonly userPool;
78
+ private readonly clientCreated;
79
+ readonly client: cognito.UserPoolClient;
80
+ private readonly checkAuthFn;
81
+ private readonly httpHeadersFn;
82
+ private readonly parseAuthFn;
83
+ private readonly refreshAuthFn;
84
+ private readonly signOutFn;
85
+ private readonly oauthScopes;
86
+ constructor(scope: Construct, id: string, props: CloudFrontAuthProps);
87
+ private createPathLambda;
88
+ /**
89
+ * Create behaviors for authentication pages:
90
+ *
91
+ * - callback page
92
+ * - refresh page
93
+ * - sign out page
94
+ *
95
+ * This is to be used with CloudFrontWebDistribution. See
96
+ * createAuthPagesBehaviors if using Distribution.
97
+ */
98
+ get authPages(): cloudfront.Behavior[];
99
+ /**
100
+ * Create behaviors for authentication pages.
101
+ *
102
+ * - callback page
103
+ * - refresh page
104
+ * - sign out page
105
+ *
106
+ * This is to be used with Distribution.
107
+ */
108
+ createAuthPagesBehaviors(origin: IOrigin, options?: AddBehaviorOptions): Record<string, BehaviorOptions>;
109
+ /**
110
+ * Create lambda function association for viewer request to check
111
+ * authentication and original response to add headers.
112
+ *
113
+ * This is to be used with CloudFrontWebDistribution. See
114
+ * createProtectedBehavior if using Distribution.
115
+ */
116
+ get authFilters(): cloudfront.LambdaFunctionAssociation[];
117
+ /**
118
+ * Create behavior that includes authorization check.
119
+ *
120
+ * This is to be used with Distribution.
121
+ */
122
+ createProtectedBehavior(origin: IOrigin, options?: AddBehaviorOptions): BehaviorOptions;
123
+ /**
124
+ * Update Cognito client to use the proper URLs and OAuth scopes.
125
+ *
126
+ * TODO: In case the client configuration changes and is updated
127
+ * by CloudFormation, this will not be reapplied causing the client
128
+ * to not be correctly configured.
129
+ * How can we avoid this scenario?
130
+ */
131
+ updateClient(id: string, props: UpdateClientProps): ClientUpdate;
132
+ }
@@ -0,0 +1,267 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || function (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
25
+ Object.defineProperty(exports, "__esModule", { value: true });
26
+ exports.CloudFrontAuth = void 0;
27
+ const cloudfront = __importStar(require("aws-cdk-lib/aws-cloudfront"));
28
+ const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
29
+ const cdk_lambda_config_1 = require("@liflig/cdk-lambda-config");
30
+ const client_secret_1 = require("./client-secret");
31
+ const client_update_1 = require("./client-update");
32
+ const generate_secret_1 = require("./generate-secret");
33
+ const constructs_1 = require("constructs");
34
+ /**
35
+ * Configure previously deployed lambda functions, Cognito client
36
+ * and CloudFront distribution.
37
+ */
38
+ class CloudFrontAuth extends constructs_1.Construct {
39
+ constructor(scope, id, props) {
40
+ var _a, _b, _c, _d, _e, _f;
41
+ super(scope, id);
42
+ this.callbackPath = (_a = props.callbackPath) !== null && _a !== void 0 ? _a : "/auth/callback";
43
+ this.signOutRedirectTo = (_b = props.signOutRedirectTo) !== null && _b !== void 0 ? _b : "/";
44
+ this.signOutPath = (_c = props.signOutPath) !== null && _c !== void 0 ? _c : "/auth/sign-out";
45
+ this.refreshAuthPath = (_d = props.refreshAuthPath) !== null && _d !== void 0 ? _d : "/auth/refresh";
46
+ this.oauthScopes = [
47
+ "phone",
48
+ "email",
49
+ "profile",
50
+ "openid",
51
+ "aws.cognito.signin.user.admin",
52
+ ];
53
+ this.userPool = props.userPool;
54
+ this.clientCreated = !props.client;
55
+ this.client =
56
+ (_e = props.client) !== null && _e !== void 0 ? _e : props.userPool.addClient("UserPoolClient", {
57
+ // Note: The following must be kept in sync with the API
58
+ // call performed in ClientUpdate.
59
+ authFlows: {
60
+ userPassword: true,
61
+ userSrp: true,
62
+ },
63
+ oAuth: {
64
+ flows: {
65
+ authorizationCodeGrant: true,
66
+ },
67
+ },
68
+ preventUserExistenceErrors: true,
69
+ generateSecret: true,
70
+ });
71
+ const nonceSigningSecret = new generate_secret_1.GenerateSecret(this, "NonceSigningSecret")
72
+ .value;
73
+ const { clientSecretValue } = new client_secret_1.RetrieveClientSecret(this, "ClientSecret", {
74
+ client: this.client,
75
+ userPool: this.userPool,
76
+ });
77
+ const config = {
78
+ httpHeaders: {
79
+ "Content-Security-Policy": "default-src 'none'; img-src 'self'; script-src 'self'; style-src 'self' 'unsafe-inline'; object-src 'none'; connect-src 'self'",
80
+ "Strict-Transport-Security": "max-age=31536000; includeSubdomains; preload",
81
+ "Referrer-Policy": "same-origin",
82
+ "X-XSS-Protection": "1; mode=block",
83
+ "X-Frame-Options": "DENY",
84
+ "X-Content-Type-Options": "nosniff",
85
+ "Cache-Control": "no-cache",
86
+ },
87
+ logLevel: (_f = props.logLevel) !== null && _f !== void 0 ? _f : "warn",
88
+ userPoolId: this.userPool.userPoolId,
89
+ clientId: this.client.userPoolClientId,
90
+ clientSecret: clientSecretValue,
91
+ oauthScopes: this.oauthScopes,
92
+ cognitoAuthDomain: props.cognitoAuthDomain,
93
+ callbackPath: this.callbackPath,
94
+ signOutRedirectTo: this.signOutRedirectTo,
95
+ signOutPath: this.signOutPath,
96
+ refreshAuthPath: this.refreshAuthPath,
97
+ requireGroupAnyOf: props.requireGroupAnyOf,
98
+ cookieSettings: {
99
+ /*
100
+ spaMode - consider if this should be supported
101
+ idToken: "Path=/; Secure; SameSite=Lax",
102
+ accessToken: "Path=/; Secure; SameSite=Lax",
103
+ refreshToken: "Path=/; Secure; SameSite=Lax",
104
+ nonce: "Path=/; Secure; HttpOnly; SameSite=Lax",
105
+ */
106
+ idToken: "Path=/; Secure; HttpOnly; SameSite=Lax",
107
+ accessToken: "Path=/; Secure; HttpOnly; SameSite=Lax",
108
+ refreshToken: "Path=/; Secure; HttpOnly; SameSite=Lax",
109
+ nonce: "Path=/; Secure; HttpOnly; SameSite=Lax",
110
+ },
111
+ nonceSigningSecret,
112
+ };
113
+ this.checkAuthFn = new cdk_lambda_config_1.LambdaConfig(this, "CheckAuthFn", {
114
+ function: props.authLambdas.checkAuthFn.get(this, "CheckAuthFnImport"),
115
+ config,
116
+ }).version;
117
+ this.httpHeadersFn = new cdk_lambda_config_1.LambdaConfig(this, "HttpHeadersFn", {
118
+ function: props.authLambdas.httpHeadersFn.get(this, "HttpHeadersFnImport"),
119
+ config,
120
+ }).version;
121
+ this.parseAuthFn = new cdk_lambda_config_1.LambdaConfig(this, "ParseAuthFn", {
122
+ function: props.authLambdas.parseAuthFn.get(this, "ParseAuthFnImport"),
123
+ config,
124
+ }).version;
125
+ this.refreshAuthFn = new cdk_lambda_config_1.LambdaConfig(this, "RefreshAuthFn", {
126
+ function: props.authLambdas.refreshAuthFn.get(this, "RefreshAuthFnImport"),
127
+ config,
128
+ }).version;
129
+ this.signOutFn = new cdk_lambda_config_1.LambdaConfig(this, "SignOutFn", {
130
+ function: props.authLambdas.signOutFn.get(this, "SignOutFnImport"),
131
+ config,
132
+ }).version;
133
+ }
134
+ createPathLambda(path, fn) {
135
+ return {
136
+ pathPattern: path,
137
+ forwardedValues: {
138
+ queryString: true,
139
+ },
140
+ lambdaFunctionAssociations: [
141
+ {
142
+ eventType: cloudfront.LambdaEdgeEventType.VIEWER_REQUEST,
143
+ lambdaFunction: fn,
144
+ },
145
+ ],
146
+ };
147
+ }
148
+ /**
149
+ * Create behaviors for authentication pages:
150
+ *
151
+ * - callback page
152
+ * - refresh page
153
+ * - sign out page
154
+ *
155
+ * This is to be used with CloudFrontWebDistribution. See
156
+ * createAuthPagesBehaviors if using Distribution.
157
+ */
158
+ get authPages() {
159
+ return [
160
+ this.createPathLambda(this.callbackPath, this.parseAuthFn),
161
+ this.createPathLambda(this.refreshAuthPath, this.refreshAuthFn),
162
+ this.createPathLambda(this.signOutPath, this.signOutFn),
163
+ ];
164
+ }
165
+ /**
166
+ * Create behaviors for authentication pages.
167
+ *
168
+ * - callback page
169
+ * - refresh page
170
+ * - sign out page
171
+ *
172
+ * This is to be used with Distribution.
173
+ */
174
+ createAuthPagesBehaviors(origin, options) {
175
+ function path(path, fn) {
176
+ return {
177
+ [path]: {
178
+ origin,
179
+ compress: true,
180
+ viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
181
+ edgeLambdas: [
182
+ {
183
+ eventType: cloudfront.LambdaEdgeEventType.VIEWER_REQUEST,
184
+ functionVersion: fn,
185
+ },
186
+ ],
187
+ ...options,
188
+ },
189
+ };
190
+ }
191
+ return {
192
+ ...path(this.callbackPath, this.parseAuthFn),
193
+ ...path(this.refreshAuthPath, this.refreshAuthFn),
194
+ ...path(this.signOutPath, this.signOutFn),
195
+ };
196
+ }
197
+ /**
198
+ * Create lambda function association for viewer request to check
199
+ * authentication and original response to add headers.
200
+ *
201
+ * This is to be used with CloudFrontWebDistribution. See
202
+ * createProtectedBehavior if using Distribution.
203
+ */
204
+ get authFilters() {
205
+ return [
206
+ {
207
+ eventType: cloudfront.LambdaEdgeEventType.VIEWER_REQUEST,
208
+ lambdaFunction: this.checkAuthFn,
209
+ },
210
+ {
211
+ eventType: cloudfront.LambdaEdgeEventType.ORIGIN_RESPONSE,
212
+ lambdaFunction: this.httpHeadersFn,
213
+ },
214
+ ];
215
+ }
216
+ /**
217
+ * Create behavior that includes authorization check.
218
+ *
219
+ * This is to be used with Distribution.
220
+ */
221
+ createProtectedBehavior(origin, options) {
222
+ if ((options === null || options === void 0 ? void 0 : options.edgeLambdas) != null) {
223
+ throw Error("User-defined edgeLambdas is currently not supported");
224
+ }
225
+ return {
226
+ origin,
227
+ compress: true,
228
+ viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
229
+ edgeLambdas: [
230
+ {
231
+ eventType: cloudfront.LambdaEdgeEventType.VIEWER_REQUEST,
232
+ functionVersion: this.checkAuthFn,
233
+ },
234
+ {
235
+ eventType: cloudfront.LambdaEdgeEventType.ORIGIN_RESPONSE,
236
+ functionVersion: this.httpHeadersFn,
237
+ },
238
+ ],
239
+ ...options,
240
+ };
241
+ }
242
+ /**
243
+ * Update Cognito client to use the proper URLs and OAuth scopes.
244
+ *
245
+ * TODO: In case the client configuration changes and is updated
246
+ * by CloudFormation, this will not be reapplied causing the client
247
+ * to not be correctly configured.
248
+ * How can we avoid this scenario?
249
+ */
250
+ updateClient(id, props) {
251
+ var _a;
252
+ if (!this.clientCreated) {
253
+ throw new Error("You cannot use updateClient with a user-provided Cognito Client " +
254
+ "since it would override the user-provided settings");
255
+ }
256
+ return new client_update_1.ClientUpdate(this, id, {
257
+ client: this.client,
258
+ userPool: this.userPool,
259
+ signOutUrl: props.signOutUrl,
260
+ callbackUrl: props.callbackUrl,
261
+ oauthScopes: this.oauthScopes,
262
+ identityProviders: (_a = props.identityProviders) !== null && _a !== void 0 ? _a : ["COGNITO"].concat(this.userPool.identityProviders.map((it) => it.providerName)),
263
+ });
264
+ }
265
+ }
266
+ exports.CloudFrontAuth = CloudFrontAuth;
267
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudfront-auth.js","sourceRoot":"","sources":["../src/cloudfront-auth.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cloudfront from \"aws-cdk-lib/aws-cloudfront\"\nimport {\n  AddBehaviorOptions,\n  BehaviorOptions,\n  IOrigin,\n  ViewerProtocolPolicy,\n} from \"aws-cdk-lib/aws-cloudfront\"\nimport * as cognito from \"aws-cdk-lib/aws-cognito\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport { IVersion } from \"aws-cdk-lib/aws-lambda\"\nimport { LambdaConfig } from \"@liflig/cdk-lambda-config\"\nimport { RetrieveClientSecret } from \"./client-secret\"\nimport { ClientUpdate } from \"./client-update\"\nimport { GenerateSecret } from \"./generate-secret\"\nimport { StoredConfig } from \"./handlers/util/config\"\nimport { AuthLambdas } from \"./lambdas\"\nimport { Construct } from \"constructs\"\n\nexport interface CloudFrontAuthProps {\n  /**\n   * Cognito Client that will be used to authenticate the user.\n   *\n   * If a custom client is provided, the updateClient method cannot\n   * be used since we cannot know which parameters was set.\n   *\n   * @default - a new client will be generated\n   */\n  client?: cognito.UserPoolClient\n  userPool: cognito.IUserPool\n  /**\n   * The domain that is used for Cognito Auth.\n   *\n   * If not using custom domains this will be a name under amazoncognito.com.\n   *\n   * @example `${domain.domainName}.auth.${region}.amazoncognito.com`\n   */\n  cognitoAuthDomain: string\n  authLambdas: AuthLambdas\n  /**\n   * @default /auth/callback\n   */\n  callbackPath?: string\n  /**\n   * @default /\n   */\n  signOutRedirectTo?: string\n  /**\n   * @default /auth/sign-out\n   */\n  signOutPath?: string\n  /**\n   * @default /auth/refresh\n   */\n  refreshAuthPath?: string\n  /**\n   * Log level.\n   *\n   * A log level of debug will log secrets and should only be used in\n   * a development environment.\n   *\n   * @default warn\n   */\n  logLevel?: \"none\" | \"error\" | \"warn\" | \"info\" | \"debug\"\n  /**\n   * Require the user to be part of a specific Cognito group to\n   * access any resource.\n   */\n  requireGroupAnyOf?: string[]\n}\n\nexport interface UpdateClientProps {\n  signOutUrl: string\n  callbackUrl: string\n  /**\n   * List of identity providers used for the client.\n   *\n   * @default - COGNITO and identity providers registered in the UserPool construct\n   */\n  identityProviders?: string[]\n}\n\n/**\n * Configure previously deployed lambda functions, Cognito client\n * and CloudFront distribution.\n */\nexport class CloudFrontAuth extends Construct {\n  public readonly callbackPath: string\n  public readonly signOutRedirectTo: string\n  public readonly signOutPath: string\n  public readonly refreshAuthPath: string\n\n  private readonly userPool: cognito.IUserPool\n  private readonly clientCreated: boolean\n  public readonly client: cognito.UserPoolClient\n\n  private readonly checkAuthFn: lambda.IVersion\n  private readonly httpHeadersFn: lambda.IVersion\n  private readonly parseAuthFn: lambda.IVersion\n  private readonly refreshAuthFn: lambda.IVersion\n  private readonly signOutFn: lambda.IVersion\n\n  private readonly oauthScopes: string[]\n\n  constructor(scope: Construct, id: string, props: CloudFrontAuthProps) {\n    super(scope, id)\n\n    this.callbackPath = props.callbackPath ?? \"/auth/callback\"\n    this.signOutRedirectTo = props.signOutRedirectTo ?? \"/\"\n    this.signOutPath = props.signOutPath ?? \"/auth/sign-out\"\n    this.refreshAuthPath = props.refreshAuthPath ?? \"/auth/refresh\"\n\n    this.oauthScopes = [\n      \"phone\",\n      \"email\",\n      \"profile\",\n      \"openid\",\n      \"aws.cognito.signin.user.admin\",\n    ]\n\n    this.userPool = props.userPool\n\n    this.clientCreated = !props.client\n    this.client =\n      props.client ??\n      props.userPool.addClient(\"UserPoolClient\", {\n        // Note: The following must be kept in sync with the API\n        // call performed in ClientUpdate.\n        authFlows: {\n          userPassword: true,\n          userSrp: true,\n        },\n        oAuth: {\n          flows: {\n            authorizationCodeGrant: true,\n          },\n        },\n        preventUserExistenceErrors: true,\n        generateSecret: true,\n      })\n\n    const nonceSigningSecret = new GenerateSecret(this, \"NonceSigningSecret\")\n      .value\n\n    const { clientSecretValue } = new RetrieveClientSecret(\n      this,\n      \"ClientSecret\",\n      {\n        client: this.client,\n        userPool: this.userPool,\n      },\n    )\n\n    const config: StoredConfig = {\n      httpHeaders: {\n        \"Content-Security-Policy\":\n          \"default-src 'none'; img-src 'self'; script-src 'self'; style-src 'self' 'unsafe-inline'; object-src 'none'; connect-src 'self'\",\n        \"Strict-Transport-Security\":\n          \"max-age=31536000; includeSubdomains; preload\",\n        \"Referrer-Policy\": \"same-origin\",\n        \"X-XSS-Protection\": \"1; mode=block\",\n        \"X-Frame-Options\": \"DENY\",\n        \"X-Content-Type-Options\": \"nosniff\",\n        \"Cache-Control\": \"no-cache\",\n      },\n      logLevel: props.logLevel ?? \"warn\",\n      userPoolId: this.userPool.userPoolId,\n      clientId: this.client.userPoolClientId,\n      clientSecret: clientSecretValue,\n      oauthScopes: this.oauthScopes,\n      cognitoAuthDomain: props.cognitoAuthDomain,\n      callbackPath: this.callbackPath,\n      signOutRedirectTo: this.signOutRedirectTo,\n      signOutPath: this.signOutPath,\n      refreshAuthPath: this.refreshAuthPath,\n      requireGroupAnyOf: props.requireGroupAnyOf,\n      cookieSettings: {\n        /*\n        spaMode - consider if this should be supported\n        idToken: \"Path=/; Secure; SameSite=Lax\",\n        accessToken: \"Path=/; Secure; SameSite=Lax\",\n        refreshToken: \"Path=/; Secure; SameSite=Lax\",\n        nonce: \"Path=/; Secure; HttpOnly; SameSite=Lax\",\n        */\n        idToken: \"Path=/; Secure; HttpOnly; SameSite=Lax\",\n        accessToken: \"Path=/; Secure; HttpOnly; SameSite=Lax\",\n        refreshToken: \"Path=/; Secure; HttpOnly; SameSite=Lax\",\n        nonce: \"Path=/; Secure; HttpOnly; SameSite=Lax\",\n      },\n      nonceSigningSecret,\n    }\n\n    this.checkAuthFn = new LambdaConfig(this, \"CheckAuthFn\", {\n      function: props.authLambdas.checkAuthFn.get(this, \"CheckAuthFnImport\"),\n      config,\n    }).version\n\n    this.httpHeadersFn = new LambdaConfig(this, \"HttpHeadersFn\", {\n      function: props.authLambdas.httpHeadersFn.get(\n        this,\n        \"HttpHeadersFnImport\",\n      ),\n      config,\n    }).version\n\n    this.parseAuthFn = new LambdaConfig(this, \"ParseAuthFn\", {\n      function: props.authLambdas.parseAuthFn.get(this, \"ParseAuthFnImport\"),\n      config,\n    }).version\n\n    this.refreshAuthFn = new LambdaConfig(this, \"RefreshAuthFn\", {\n      function: props.authLambdas.refreshAuthFn.get(\n        this,\n        \"RefreshAuthFnImport\",\n      ),\n      config,\n    }).version\n\n    this.signOutFn = new LambdaConfig(this, \"SignOutFn\", {\n      function: props.authLambdas.signOutFn.get(this, \"SignOutFnImport\"),\n      config,\n    }).version\n  }\n\n  private createPathLambda(\n    path: string,\n    fn: lambda.IVersion,\n  ): cloudfront.Behavior {\n    return {\n      pathPattern: path,\n      forwardedValues: {\n        queryString: true,\n      },\n      lambdaFunctionAssociations: [\n        {\n          eventType: cloudfront.LambdaEdgeEventType.VIEWER_REQUEST,\n          lambdaFunction: fn,\n        },\n      ],\n    }\n  }\n\n  /**\n   * Create behaviors for authentication pages:\n   *\n   * - callback page\n   * - refresh page\n   * - sign out page\n   *\n   * This is to be used with CloudFrontWebDistribution. See\n   * createAuthPagesBehaviors if using Distribution.\n   */\n  public get authPages(): cloudfront.Behavior[] {\n    return [\n      this.createPathLambda(this.callbackPath, this.parseAuthFn),\n      this.createPathLambda(this.refreshAuthPath, this.refreshAuthFn),\n      this.createPathLambda(this.signOutPath, this.signOutFn),\n    ]\n  }\n\n  /**\n   * Create behaviors for authentication pages.\n   *\n   * - callback page\n   * - refresh page\n   * - sign out page\n   *\n   * This is to be used with Distribution.\n   */\n  public createAuthPagesBehaviors(\n    origin: IOrigin,\n    options?: AddBehaviorOptions,\n  ): Record<string, BehaviorOptions> {\n    function path(path: string, fn: IVersion): Record<string, BehaviorOptions> {\n      return {\n        [path]: {\n          origin,\n          compress: true,\n          viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          edgeLambdas: [\n            {\n              eventType: cloudfront.LambdaEdgeEventType.VIEWER_REQUEST,\n              functionVersion: fn,\n            },\n          ],\n          ...options,\n        },\n      }\n    }\n\n    return {\n      ...path(this.callbackPath, this.parseAuthFn),\n      ...path(this.refreshAuthPath, this.refreshAuthFn),\n      ...path(this.signOutPath, this.signOutFn),\n    }\n  }\n\n  /**\n   * Create lambda function association for viewer request to check\n   * authentication and original response to add headers.\n   *\n   * This is to be used with CloudFrontWebDistribution. See\n   * createProtectedBehavior if using Distribution.\n   */\n  public get authFilters(): cloudfront.LambdaFunctionAssociation[] {\n    return [\n      {\n        eventType: cloudfront.LambdaEdgeEventType.VIEWER_REQUEST,\n        lambdaFunction: this.checkAuthFn,\n      },\n      {\n        eventType: cloudfront.LambdaEdgeEventType.ORIGIN_RESPONSE,\n        lambdaFunction: this.httpHeadersFn,\n      },\n    ]\n  }\n\n  /**\n   * Create behavior that includes authorization check.\n   *\n   * This is to be used with Distribution.\n   */\n  public createProtectedBehavior(\n    origin: IOrigin,\n    options?: AddBehaviorOptions,\n  ): BehaviorOptions {\n    if (options?.edgeLambdas != null) {\n      throw Error(\"User-defined edgeLambdas is currently not supported\")\n    }\n\n    return {\n      origin,\n      compress: true,\n      viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: [\n        {\n          eventType: cloudfront.LambdaEdgeEventType.VIEWER_REQUEST,\n          functionVersion: this.checkAuthFn,\n        },\n        {\n          eventType: cloudfront.LambdaEdgeEventType.ORIGIN_RESPONSE,\n          functionVersion: this.httpHeadersFn,\n        },\n      ],\n      ...options,\n    }\n  }\n\n  /**\n   * Update Cognito client to use the proper URLs and OAuth scopes.\n   *\n   * TODO: In case the client configuration changes and is updated\n   *  by CloudFormation, this will not be reapplied causing the client\n   *  to not be correctly configured.\n   *  How can we avoid this scenario?\n   */\n  public updateClient(id: string, props: UpdateClientProps): ClientUpdate {\n    if (!this.clientCreated) {\n      throw new Error(\n        \"You cannot use updateClient with a user-provided Cognito Client \" +\n          \"since it would override the user-provided settings\",\n      )\n    }\n\n    return new ClientUpdate(this, id, {\n      client: this.client,\n      userPool: this.userPool,\n      signOutUrl: props.signOutUrl,\n      callbackUrl: props.callbackUrl,\n      oauthScopes: this.oauthScopes,\n      identityProviders:\n        props.identityProviders ??\n        [\"COGNITO\"].concat(\n          this.userPool.identityProviders.map((it) => it.providerName),\n        ),\n    })\n  }\n}\n"]}
@@ -0,0 +1,15 @@
1
+ import { Construct } from "constructs";
2
+ interface GenerateSecretProps {
3
+ /**
4
+ * Nonce to force secret update.
5
+ */
6
+ nonce?: string;
7
+ }
8
+ /**
9
+ * Generate a secret to be used in other parts of the deployment.
10
+ */
11
+ export declare class GenerateSecret extends Construct {
12
+ readonly value: string;
13
+ constructor(scope: Construct, id: string, props?: GenerateSecretProps);
14
+ }
15
+ export {};