@liflig/cdk-vy 1.0.0 → 2.1.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/README.md CHANGED
@@ -10,11 +10,19 @@ This is based on https://github.com/nsbno/terraform-provider-vy
10
10
  npm install @liflig/cdk-vy
11
11
  ```
12
12
 
13
+ ### Create a VyCognitoProvider
14
+
15
+ ```typescript
16
+ const vyCognitoProvider = new VyCognitoProvider(this, "MyProvider", {
17
+ environment: VyEnvironment.TEST,
18
+ });
19
+ ```
20
+
13
21
  ### Create a resource server
14
22
 
15
23
  ```typescript
16
24
  const resourceServer = new CognitoResourceServer(this, "ApiResourceServer", {
17
- environment: VyEnvironment.PROD,
25
+ resourceServerProvider: vyCognitoProvider.resourceServerProvider,
18
26
  name: "my-api",
19
27
  identifier: "https://my-api.vydev.io",
20
28
  scopes: [
@@ -24,15 +32,15 @@ const resourceServer = new CognitoResourceServer(this, "ApiResourceServer", {
24
32
  });
25
33
  ```
26
34
 
27
- ### Create an app clients
35
+ ### Create app clients
28
36
 
29
37
  ```typescript
30
38
  // Backend app client for M2M authentication
31
- const backendClient = new CognitoAppClient(this, 'BackendClient', {
32
- environment: VyEnvironment.TEST,
33
- name: 'my-backend-service',
39
+ const backendClient = new CognitoAppClient(this, "BackendClient", {
40
+ appClientProvider: vyCognitoProvider.appClientProvider,
41
+ name: "my-backend-service",
34
42
  type: AppClientType.BACKEND,
35
- scopes: ['https://api.vydev.io/read', 'https://api.vydev.io/write']
43
+ scopes: ["https://api.vydev.io/read", "https://api.vydev.io/write"]
36
44
  });
37
45
 
38
46
  // Access credentials
@@ -41,12 +49,12 @@ const clientSecret = backendClient.clientSecret; // Stored in Secrets Manager
41
49
 
42
50
  @example
43
51
  // Frontend app client for user authentication
44
- const frontendClient = new CognitoAppClient(this, 'FrontendClient', {
45
- environment: VyEnvironment.PROD,
46
- name: 'my-web-app',
52
+ const frontendClient = new CognitoAppClient(this, "FrontendClient", {
53
+ appClientProvider: vyCognitoProvider.appClientProvider,
54
+ name: "my-web-app",
47
55
  type: AppClientType.FRONTEND,
48
- scopes: ['email', 'openid', 'profile'],
49
- callbackUrls: ['https://my-app.vydev.io/auth/callback'],
50
- logoutUrls: ['https://my-app.vydev.io/logout']
56
+ scopes: ["email", "openid", "profile"],
57
+ callbackUrls: ["https://my-app.vydev.io/auth/callback"],
58
+ logoutUrls: ["https://my-app.vydev.io/logout"]
51
59
  });
52
60
  ```
@@ -0,0 +1 @@
1
+ import "jest-cdk-snapshot";
@@ -0,0 +1,46 @@
1
+ import { App, Stack } from "aws-cdk-lib";
2
+ import "jest-cdk-snapshot";
3
+ import { VyCognitoProvider } from "../";
4
+ import { VyEnvironment } from "../shared/types";
5
+ test("vyCognitoProvider in TEST", () => {
6
+ const app = new App();
7
+ const stack = new Stack(app, "Stack");
8
+ new VyCognitoProvider(stack, "VyCognitoProvider", {
9
+ environment: VyEnvironment.TEST,
10
+ });
11
+ expect(stack).toMatchCdkSnapshot({
12
+ ignoreAssets: true,
13
+ });
14
+ });
15
+ test("vyCognitoProvider in STAGE", () => {
16
+ const app = new App();
17
+ const stack = new Stack(app, "Stack");
18
+ new VyCognitoProvider(stack, "VyCognitoProvider", {
19
+ environment: VyEnvironment.STAGE,
20
+ });
21
+ expect(stack).toMatchCdkSnapshot({
22
+ ignoreAssets: true,
23
+ });
24
+ });
25
+ test("vyCognitoProvider in PROD", () => {
26
+ const app = new App();
27
+ const stack = new Stack(app, "Stack");
28
+ new VyCognitoProvider(stack, "VyCognitoProvider", {
29
+ environment: VyEnvironment.PROD,
30
+ });
31
+ expect(stack).toMatchCdkSnapshot({
32
+ ignoreAssets: true,
33
+ });
34
+ });
35
+ test("vyCognitoProvider with custom domain", () => {
36
+ const app = new App();
37
+ const stack = new Stack(app, "Stack");
38
+ new VyCognitoProvider(stack, "VyCognitoProvider", {
39
+ environment: VyEnvironment.TEST,
40
+ cognitoBaseDomain: "example.com",
41
+ });
42
+ expect(stack).toMatchCdkSnapshot({
43
+ ignoreAssets: true,
44
+ });
45
+ });
46
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidnktY29nbml0by1wcm92aWRlci50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL19fdGVzdF9fL3Z5LWNvZ25pdG8tcHJvdmlkZXIudGVzdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsR0FBRyxFQUFFLEtBQUssRUFBRSxNQUFNLGFBQWEsQ0FBQTtBQUN4QyxPQUFPLG1CQUFtQixDQUFBO0FBQzFCLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLEtBQUssQ0FBQTtBQUN2QyxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0saUJBQWlCLENBQUE7QUFFL0MsSUFBSSxDQUFDLDJCQUEyQixFQUFFLEdBQUcsRUFBRTtJQUNyQyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxJQUFJLGlCQUFpQixDQUFDLEtBQUssRUFBRSxtQkFBbUIsRUFBRTtRQUNoRCxXQUFXLEVBQUUsYUFBYSxDQUFDLElBQUk7S0FDaEMsQ0FBQyxDQUFBO0lBRUYsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLGtCQUFrQixDQUFDO1FBQy9CLFlBQVksRUFBRSxJQUFJO0tBQ25CLENBQUMsQ0FBQTtBQUNKLENBQUMsQ0FBQyxDQUFBO0FBRUYsSUFBSSxDQUFDLDRCQUE0QixFQUFFLEdBQUcsRUFBRTtJQUN0QyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxJQUFJLGlCQUFpQixDQUFDLEtBQUssRUFBRSxtQkFBbUIsRUFBRTtRQUNoRCxXQUFXLEVBQUUsYUFBYSxDQUFDLEtBQUs7S0FDakMsQ0FBQyxDQUFBO0lBRUYsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLGtCQUFrQixDQUFDO1FBQy9CLFlBQVksRUFBRSxJQUFJO0tBQ25CLENBQUMsQ0FBQTtBQUNKLENBQUMsQ0FBQyxDQUFBO0FBRUYsSUFBSSxDQUFDLDJCQUEyQixFQUFFLEdBQUcsRUFBRTtJQUNyQyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxJQUFJLGlCQUFpQixDQUFDLEtBQUssRUFBRSxtQkFBbUIsRUFBRTtRQUNoRCxXQUFXLEVBQUUsYUFBYSxDQUFDLElBQUk7S0FDaEMsQ0FBQyxDQUFBO0lBRUYsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLGtCQUFrQixDQUFDO1FBQy9CLFlBQVksRUFBRSxJQUFJO0tBQ25CLENBQUMsQ0FBQTtBQUNKLENBQUMsQ0FBQyxDQUFBO0FBRUYsSUFBSSxDQUFDLHNDQUFzQyxFQUFFLEdBQUcsRUFBRTtJQUNoRCxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxJQUFJLGlCQUFpQixDQUFDLEtBQUssRUFBRSxtQkFBbUIsRUFBRTtRQUNoRCxXQUFXLEVBQUUsYUFBYSxDQUFDLElBQUk7UUFDL0IsaUJBQWlCLEVBQUUsYUFBYTtLQUNqQyxDQUFDLENBQUE7SUFFRixNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsa0JBQWtCLENBQUM7UUFDL0IsWUFBWSxFQUFFLElBQUk7S0FDbkIsQ0FBQyxDQUFBO0FBQ0osQ0FBQyxDQUFDLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBcHAsIFN0YWNrIH0gZnJvbSBcImF3cy1jZGstbGliXCJcbmltcG9ydCBcImplc3QtY2RrLXNuYXBzaG90XCJcbmltcG9ydCB7IFZ5Q29nbml0b1Byb3ZpZGVyIH0gZnJvbSBcIi4uL1wiXG5pbXBvcnQgeyBWeUVudmlyb25tZW50IH0gZnJvbSBcIi4uL3NoYXJlZC90eXBlc1wiXG5cbnRlc3QoXCJ2eUNvZ25pdG9Qcm92aWRlciBpbiBURVNUXCIsICgpID0+IHtcbiAgY29uc3QgYXBwID0gbmV3IEFwcCgpXG4gIGNvbnN0IHN0YWNrID0gbmV3IFN0YWNrKGFwcCwgXCJTdGFja1wiKVxuXG4gIG5ldyBWeUNvZ25pdG9Qcm92aWRlcihzdGFjaywgXCJWeUNvZ25pdG9Qcm92aWRlclwiLCB7XG4gICAgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnQuVEVTVCxcbiAgfSlcblxuICBleHBlY3Qoc3RhY2spLnRvTWF0Y2hDZGtTbmFwc2hvdCh7XG4gICAgaWdub3JlQXNzZXRzOiB0cnVlLFxuICB9KVxufSlcblxudGVzdChcInZ5Q29nbml0b1Byb3ZpZGVyIGluIFNUQUdFXCIsICgpID0+IHtcbiAgY29uc3QgYXBwID0gbmV3IEFwcCgpXG4gIGNvbnN0IHN0YWNrID0gbmV3IFN0YWNrKGFwcCwgXCJTdGFja1wiKVxuXG4gIG5ldyBWeUNvZ25pdG9Qcm92aWRlcihzdGFjaywgXCJWeUNvZ25pdG9Qcm92aWRlclwiLCB7XG4gICAgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnQuU1RBR0UsXG4gIH0pXG5cbiAgZXhwZWN0KHN0YWNrKS50b01hdGNoQ2RrU25hcHNob3Qoe1xuICAgIGlnbm9yZUFzc2V0czogdHJ1ZSxcbiAgfSlcbn0pXG5cbnRlc3QoXCJ2eUNvZ25pdG9Qcm92aWRlciBpbiBQUk9EXCIsICgpID0+IHtcbiAgY29uc3QgYXBwID0gbmV3IEFwcCgpXG4gIGNvbnN0IHN0YWNrID0gbmV3IFN0YWNrKGFwcCwgXCJTdGFja1wiKVxuXG4gIG5ldyBWeUNvZ25pdG9Qcm92aWRlcihzdGFjaywgXCJWeUNvZ25pdG9Qcm92aWRlclwiLCB7XG4gICAgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnQuUFJPRCxcbiAgfSlcblxuICBleHBlY3Qoc3RhY2spLnRvTWF0Y2hDZGtTbmFwc2hvdCh7XG4gICAgaWdub3JlQXNzZXRzOiB0cnVlLFxuICB9KVxufSlcblxudGVzdChcInZ5Q29nbml0b1Byb3ZpZGVyIHdpdGggY3VzdG9tIGRvbWFpblwiLCAoKSA9PiB7XG4gIGNvbnN0IGFwcCA9IG5ldyBBcHAoKVxuICBjb25zdCBzdGFjayA9IG5ldyBTdGFjayhhcHAsIFwiU3RhY2tcIilcblxuICBuZXcgVnlDb2duaXRvUHJvdmlkZXIoc3RhY2ssIFwiVnlDb2duaXRvUHJvdmlkZXJcIiwge1xuICAgIGVudmlyb25tZW50OiBWeUVudmlyb25tZW50LlRFU1QsXG4gICAgY29nbml0b0Jhc2VEb21haW46IFwiZXhhbXBsZS5jb21cIixcbiAgfSlcblxuICBleHBlY3Qoc3RhY2spLnRvTWF0Y2hDZGtTbmFwc2hvdCh7XG4gICAgaWdub3JlQXNzZXRzOiB0cnVlLFxuICB9KVxufSlcbiJdfQ==
@@ -5,8 +5,13 @@ import { AppClientType, CognitoAppClient } from "../cognito-app-client";
5
5
  test("cognito frontend app client", () => {
6
6
  const app = new App();
7
7
  const stack = new Stack(app, "Stack");
8
+ const mockAppClientProvider = {
9
+ environment: VyEnvironment.TEST,
10
+ serviceToken: "serviceToken",
11
+ auth_url: "auth_url",
12
+ };
8
13
  new CognitoAppClient(stack, "CognitoAppClient", {
9
- environment: VyEnvironment.PROD,
14
+ appClientProvider: mockAppClientProvider,
10
15
  name: "my-web-app",
11
16
  type: AppClientType.FRONTEND,
12
17
  scopes: ["email", "openid", "profile"],
@@ -20,8 +25,13 @@ test("cognito frontend app client", () => {
20
25
  test("cognito backend app client", () => {
21
26
  const app = new App();
22
27
  const stack = new Stack(app, "Stack");
28
+ const mockAppClientProvider = {
29
+ environment: VyEnvironment.STAGE,
30
+ serviceToken: "serviceToken",
31
+ auth_url: "auth_url",
32
+ };
23
33
  new CognitoAppClient(stack, "CognitoAppClient", {
24
- environment: VyEnvironment.TEST,
34
+ appClientProvider: mockAppClientProvider,
25
35
  name: "my-backend-service",
26
36
  type: AppClientType.BACKEND,
27
37
  scopes: ["https://api.vydev.io/read", "https://api.vydev.io/write"],
@@ -33,8 +43,13 @@ test("cognito backend app client", () => {
33
43
  test("cognito frontend app client with custom secretName", () => {
34
44
  const app = new App();
35
45
  const stack = new Stack(app, "Stack");
46
+ const mockAppClientProvider = {
47
+ environment: VyEnvironment.PROD,
48
+ serviceToken: "serviceToken",
49
+ auth_url: "auth_url",
50
+ };
36
51
  new CognitoAppClient(stack, "CognitoAppClient", {
37
- environment: VyEnvironment.STAGE,
52
+ appClientProvider: mockAppClientProvider,
38
53
  name: "my-backend-service",
39
54
  type: AppClientType.BACKEND,
40
55
  scopes: ["https://api.vydev.io/read", "https://api.vydev.io/write"],
@@ -44,4 +59,4 @@ test("cognito frontend app client with custom secretName", () => {
44
59
  ignoreAssets: true,
45
60
  });
46
61
  });
47
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXgudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb2duaXRvLWFwcC1jbGllbnQvX190ZXN0X18vaW5kZXgudGVzdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsR0FBRyxFQUFFLEtBQUssRUFBRSxNQUFNLGFBQWEsQ0FBQTtBQUN4QyxPQUFPLG1CQUFtQixDQUFBO0FBQzFCLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQTtBQUNsRCxPQUFPLEVBQUUsYUFBYSxFQUFFLGdCQUFnQixFQUFFLE1BQU0sdUJBQXVCLENBQUE7QUFFdkUsSUFBSSxDQUFDLDZCQUE2QixFQUFFLEdBQUcsRUFBRTtJQUN2QyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxJQUFJLGdCQUFnQixDQUFDLEtBQUssRUFBRSxrQkFBa0IsRUFBRTtRQUM5QyxXQUFXLEVBQUUsYUFBYSxDQUFDLElBQUk7UUFDL0IsSUFBSSxFQUFFLFlBQVk7UUFDbEIsSUFBSSxFQUFFLGFBQWEsQ0FBQyxRQUFRO1FBQzVCLE1BQU0sRUFBRSxDQUFDLE9BQU8sRUFBRSxRQUFRLEVBQUUsU0FBUyxDQUFDO1FBQ3RDLFlBQVksRUFBRSxDQUFDLHVDQUF1QyxDQUFDO1FBQ3ZELFVBQVUsRUFBRSxDQUFDLGdDQUFnQyxDQUFDO0tBQy9DLENBQUMsQ0FBQTtJQUVGLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQztRQUMvQixZQUFZLEVBQUUsSUFBSTtLQUNuQixDQUFDLENBQUE7QUFDSixDQUFDLENBQUMsQ0FBQTtBQUVGLElBQUksQ0FBQyw0QkFBNEIsRUFBRSxHQUFHLEVBQUU7SUFDdEMsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLEVBQUUsQ0FBQTtJQUNyQixNQUFNLEtBQUssR0FBRyxJQUFJLEtBQUssQ0FBQyxHQUFHLEVBQUUsT0FBTyxDQUFDLENBQUE7SUFFckMsSUFBSSxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsa0JBQWtCLEVBQUU7UUFDOUMsV0FBVyxFQUFFLGFBQWEsQ0FBQyxJQUFJO1FBQy9CLElBQUksRUFBRSxvQkFBb0I7UUFDMUIsSUFBSSxFQUFFLGFBQWEsQ0FBQyxPQUFPO1FBQzNCLE1BQU0sRUFBRSxDQUFDLDJCQUEyQixFQUFFLDRCQUE0QixDQUFDO0tBQ3BFLENBQUMsQ0FBQTtJQUVGLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQztRQUMvQixZQUFZLEVBQUUsSUFBSTtLQUNuQixDQUFDLENBQUE7QUFDSixDQUFDLENBQUMsQ0FBQTtBQUVGLElBQUksQ0FBQyxvREFBb0QsRUFBRSxHQUFHLEVBQUU7SUFDOUQsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLEVBQUUsQ0FBQTtJQUNyQixNQUFNLEtBQUssR0FBRyxJQUFJLEtBQUssQ0FBQyxHQUFHLEVBQUUsT0FBTyxDQUFDLENBQUE7SUFFckMsSUFBSSxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsa0JBQWtCLEVBQUU7UUFDOUMsV0FBVyxFQUFFLGFBQWEsQ0FBQyxLQUFLO1FBQ2hDLElBQUksRUFBRSxvQkFBb0I7UUFDMUIsSUFBSSxFQUFFLGFBQWEsQ0FBQyxPQUFPO1FBQzNCLE1BQU0sRUFBRSxDQUFDLDJCQUEyQixFQUFFLDRCQUE0QixDQUFDO1FBQ25FLFVBQVUsRUFBRSxXQUFXO0tBQ3hCLENBQUMsQ0FBQTtJQUVGLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQztRQUMvQixZQUFZLEVBQUUsSUFBSTtLQUNuQixDQUFDLENBQUE7QUFDSixDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFwcCwgU3RhY2sgfSBmcm9tIFwiYXdzLWNkay1saWJcIlxuaW1wb3J0IFwiamVzdC1jZGstc25hcHNob3RcIlxuaW1wb3J0IHsgVnlFbnZpcm9ubWVudCB9IGZyb20gXCIuLi8uLi9zaGFyZWQvdHlwZXNcIlxuaW1wb3J0IHsgQXBwQ2xpZW50VHlwZSwgQ29nbml0b0FwcENsaWVudCB9IGZyb20gXCIuLi9jb2duaXRvLWFwcC1jbGllbnRcIlxuXG50ZXN0KFwiY29nbml0byBmcm9udGVuZCBhcHAgY2xpZW50XCIsICgpID0+IHtcbiAgY29uc3QgYXBwID0gbmV3IEFwcCgpXG4gIGNvbnN0IHN0YWNrID0gbmV3IFN0YWNrKGFwcCwgXCJTdGFja1wiKVxuXG4gIG5ldyBDb2duaXRvQXBwQ2xpZW50KHN0YWNrLCBcIkNvZ25pdG9BcHBDbGllbnRcIiwge1xuICAgIGVudmlyb25tZW50OiBWeUVudmlyb25tZW50LlBST0QsXG4gICAgbmFtZTogXCJteS13ZWItYXBwXCIsXG4gICAgdHlwZTogQXBwQ2xpZW50VHlwZS5GUk9OVEVORCxcbiAgICBzY29wZXM6IFtcImVtYWlsXCIsIFwib3BlbmlkXCIsIFwicHJvZmlsZVwiXSxcbiAgICBjYWxsYmFja1VybHM6IFtcImh0dHBzOi8vbXktYXBwLnZ5ZGV2LmlvL2F1dGgvY2FsbGJhY2tcIl0sXG4gICAgbG9nb3V0VXJsczogW1wiaHR0cHM6Ly9teS1hcHAudnlkZXYuaW8vbG9nb3V0XCJdLFxuICB9KVxuXG4gIGV4cGVjdChzdGFjaykudG9NYXRjaENka1NuYXBzaG90KHtcbiAgICBpZ25vcmVBc3NldHM6IHRydWUsXG4gIH0pXG59KVxuXG50ZXN0KFwiY29nbml0byBiYWNrZW5kIGFwcCBjbGllbnRcIiwgKCkgPT4ge1xuICBjb25zdCBhcHAgPSBuZXcgQXBwKClcbiAgY29uc3Qgc3RhY2sgPSBuZXcgU3RhY2soYXBwLCBcIlN0YWNrXCIpXG5cbiAgbmV3IENvZ25pdG9BcHBDbGllbnQoc3RhY2ssIFwiQ29nbml0b0FwcENsaWVudFwiLCB7XG4gICAgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnQuVEVTVCxcbiAgICBuYW1lOiBcIm15LWJhY2tlbmQtc2VydmljZVwiLFxuICAgIHR5cGU6IEFwcENsaWVudFR5cGUuQkFDS0VORCxcbiAgICBzY29wZXM6IFtcImh0dHBzOi8vYXBpLnZ5ZGV2LmlvL3JlYWRcIiwgXCJodHRwczovL2FwaS52eWRldi5pby93cml0ZVwiXSxcbiAgfSlcblxuICBleHBlY3Qoc3RhY2spLnRvTWF0Y2hDZGtTbmFwc2hvdCh7XG4gICAgaWdub3JlQXNzZXRzOiB0cnVlLFxuICB9KVxufSlcblxudGVzdChcImNvZ25pdG8gZnJvbnRlbmQgYXBwIGNsaWVudCB3aXRoIGN1c3RvbSBzZWNyZXROYW1lXCIsICgpID0+IHtcbiAgY29uc3QgYXBwID0gbmV3IEFwcCgpXG4gIGNvbnN0IHN0YWNrID0gbmV3IFN0YWNrKGFwcCwgXCJTdGFja1wiKVxuXG4gIG5ldyBDb2duaXRvQXBwQ2xpZW50KHN0YWNrLCBcIkNvZ25pdG9BcHBDbGllbnRcIiwge1xuICAgIGVudmlyb25tZW50OiBWeUVudmlyb25tZW50LlNUQUdFLFxuICAgIG5hbWU6IFwibXktYmFja2VuZC1zZXJ2aWNlXCIsXG4gICAgdHlwZTogQXBwQ2xpZW50VHlwZS5CQUNLRU5ELFxuICAgIHNjb3BlczogW1wiaHR0cHM6Ly9hcGkudnlkZXYuaW8vcmVhZFwiLCBcImh0dHBzOi8vYXBpLnZ5ZGV2LmlvL3dyaXRlXCJdLFxuICAgIHNlY3JldE5hbWU6IFwibXktc2VjcmV0XCIsXG4gIH0pXG5cbiAgZXhwZWN0KHN0YWNrKS50b01hdGNoQ2RrU25hcHNob3Qoe1xuICAgIGlnbm9yZUFzc2V0czogdHJ1ZSxcbiAgfSlcbn0pXG4iXX0=
62
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXgudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb2duaXRvLWFwcC1jbGllbnQvX190ZXN0X18vaW5kZXgudGVzdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsR0FBRyxFQUFFLEtBQUssRUFBRSxNQUFNLGFBQWEsQ0FBQTtBQUN4QyxPQUFPLG1CQUFtQixDQUFBO0FBQzFCLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQTtBQUVsRCxPQUFPLEVBQUUsYUFBYSxFQUFFLGdCQUFnQixFQUFFLE1BQU0sdUJBQXVCLENBQUE7QUFFdkUsSUFBSSxDQUFDLDZCQUE2QixFQUFFLEdBQUcsRUFBRTtJQUN2QyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxNQUFNLHFCQUFxQixHQUFzQjtRQUMvQyxXQUFXLEVBQUUsYUFBYSxDQUFDLElBQUk7UUFDL0IsWUFBWSxFQUFFLGNBQWM7UUFDNUIsUUFBUSxFQUFFLFVBQVU7S0FDckIsQ0FBQTtJQUVELElBQUksZ0JBQWdCLENBQUMsS0FBSyxFQUFFLGtCQUFrQixFQUFFO1FBQzlDLGlCQUFpQixFQUFFLHFCQUFxQjtRQUN4QyxJQUFJLEVBQUUsWUFBWTtRQUNsQixJQUFJLEVBQUUsYUFBYSxDQUFDLFFBQVE7UUFDNUIsTUFBTSxFQUFFLENBQUMsT0FBTyxFQUFFLFFBQVEsRUFBRSxTQUFTLENBQUM7UUFDdEMsWUFBWSxFQUFFLENBQUMsdUNBQXVDLENBQUM7UUFDdkQsVUFBVSxFQUFFLENBQUMsZ0NBQWdDLENBQUM7S0FDL0MsQ0FBQyxDQUFBO0lBRUYsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLGtCQUFrQixDQUFDO1FBQy9CLFlBQVksRUFBRSxJQUFJO0tBQ25CLENBQUMsQ0FBQTtBQUNKLENBQUMsQ0FBQyxDQUFBO0FBRUYsSUFBSSxDQUFDLDRCQUE0QixFQUFFLEdBQUcsRUFBRTtJQUN0QyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxNQUFNLHFCQUFxQixHQUFzQjtRQUMvQyxXQUFXLEVBQUUsYUFBYSxDQUFDLEtBQUs7UUFDaEMsWUFBWSxFQUFFLGNBQWM7UUFDNUIsUUFBUSxFQUFFLFVBQVU7S0FDckIsQ0FBQTtJQUVELElBQUksZ0JBQWdCLENBQUMsS0FBSyxFQUFFLGtCQUFrQixFQUFFO1FBQzlDLGlCQUFpQixFQUFFLHFCQUFxQjtRQUN4QyxJQUFJLEVBQUUsb0JBQW9CO1FBQzFCLElBQUksRUFBRSxhQUFhLENBQUMsT0FBTztRQUMzQixNQUFNLEVBQUUsQ0FBQywyQkFBMkIsRUFBRSw0QkFBNEIsQ0FBQztLQUNwRSxDQUFDLENBQUE7SUFFRixNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsa0JBQWtCLENBQUM7UUFDL0IsWUFBWSxFQUFFLElBQUk7S0FDbkIsQ0FBQyxDQUFBO0FBQ0osQ0FBQyxDQUFDLENBQUE7QUFFRixJQUFJLENBQUMsb0RBQW9ELEVBQUUsR0FBRyxFQUFFO0lBQzlELE1BQU0sR0FBRyxHQUFHLElBQUksR0FBRyxFQUFFLENBQUE7SUFDckIsTUFBTSxLQUFLLEdBQUcsSUFBSSxLQUFLLENBQUMsR0FBRyxFQUFFLE9BQU8sQ0FBQyxDQUFBO0lBRXJDLE1BQU0scUJBQXFCLEdBQXNCO1FBQy9DLFdBQVcsRUFBRSxhQUFhLENBQUMsSUFBSTtRQUMvQixZQUFZLEVBQUUsY0FBYztRQUM1QixRQUFRLEVBQUUsVUFBVTtLQUNyQixDQUFBO0lBRUQsSUFBSSxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsa0JBQWtCLEVBQUU7UUFDOUMsaUJBQWlCLEVBQUUscUJBQXFCO1FBQ3hDLElBQUksRUFBRSxvQkFBb0I7UUFDMUIsSUFBSSxFQUFFLGFBQWEsQ0FBQyxPQUFPO1FBQzNCLE1BQU0sRUFBRSxDQUFDLDJCQUEyQixFQUFFLDRCQUE0QixDQUFDO1FBQ25FLFVBQVUsRUFBRSxXQUFXO0tBQ3hCLENBQUMsQ0FBQTtJQUVGLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQztRQUMvQixZQUFZLEVBQUUsSUFBSTtLQUNuQixDQUFDLENBQUE7QUFDSixDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFwcCwgU3RhY2sgfSBmcm9tIFwiYXdzLWNkay1saWJcIlxuaW1wb3J0IFwiamVzdC1jZGstc25hcHNob3RcIlxuaW1wb3J0IHsgVnlFbnZpcm9ubWVudCB9IGZyb20gXCIuLi8uLi9zaGFyZWQvdHlwZXNcIlxuaW1wb3J0IHR5cGUgeyBBcHBDbGllbnRQcm92aWRlciB9IGZyb20gXCIuLi8uLi92eS1jb2duaXRvLXByb3ZpZGVyXCJcbmltcG9ydCB7IEFwcENsaWVudFR5cGUsIENvZ25pdG9BcHBDbGllbnQgfSBmcm9tIFwiLi4vY29nbml0by1hcHAtY2xpZW50XCJcblxudGVzdChcImNvZ25pdG8gZnJvbnRlbmQgYXBwIGNsaWVudFwiLCAoKSA9PiB7XG4gIGNvbnN0IGFwcCA9IG5ldyBBcHAoKVxuICBjb25zdCBzdGFjayA9IG5ldyBTdGFjayhhcHAsIFwiU3RhY2tcIilcblxuICBjb25zdCBtb2NrQXBwQ2xpZW50UHJvdmlkZXI6IEFwcENsaWVudFByb3ZpZGVyID0ge1xuICAgIGVudmlyb25tZW50OiBWeUVudmlyb25tZW50LlRFU1QsXG4gICAgc2VydmljZVRva2VuOiBcInNlcnZpY2VUb2tlblwiLFxuICAgIGF1dGhfdXJsOiBcImF1dGhfdXJsXCIsXG4gIH1cblxuICBuZXcgQ29nbml0b0FwcENsaWVudChzdGFjaywgXCJDb2duaXRvQXBwQ2xpZW50XCIsIHtcbiAgICBhcHBDbGllbnRQcm92aWRlcjogbW9ja0FwcENsaWVudFByb3ZpZGVyLFxuICAgIG5hbWU6IFwibXktd2ViLWFwcFwiLFxuICAgIHR5cGU6IEFwcENsaWVudFR5cGUuRlJPTlRFTkQsXG4gICAgc2NvcGVzOiBbXCJlbWFpbFwiLCBcIm9wZW5pZFwiLCBcInByb2ZpbGVcIl0sXG4gICAgY2FsbGJhY2tVcmxzOiBbXCJodHRwczovL215LWFwcC52eWRldi5pby9hdXRoL2NhbGxiYWNrXCJdLFxuICAgIGxvZ291dFVybHM6IFtcImh0dHBzOi8vbXktYXBwLnZ5ZGV2LmlvL2xvZ291dFwiXSxcbiAgfSlcblxuICBleHBlY3Qoc3RhY2spLnRvTWF0Y2hDZGtTbmFwc2hvdCh7XG4gICAgaWdub3JlQXNzZXRzOiB0cnVlLFxuICB9KVxufSlcblxudGVzdChcImNvZ25pdG8gYmFja2VuZCBhcHAgY2xpZW50XCIsICgpID0+IHtcbiAgY29uc3QgYXBwID0gbmV3IEFwcCgpXG4gIGNvbnN0IHN0YWNrID0gbmV3IFN0YWNrKGFwcCwgXCJTdGFja1wiKVxuXG4gIGNvbnN0IG1vY2tBcHBDbGllbnRQcm92aWRlcjogQXBwQ2xpZW50UHJvdmlkZXIgPSB7XG4gICAgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnQuU1RBR0UsXG4gICAgc2VydmljZVRva2VuOiBcInNlcnZpY2VUb2tlblwiLFxuICAgIGF1dGhfdXJsOiBcImF1dGhfdXJsXCIsXG4gIH1cblxuICBuZXcgQ29nbml0b0FwcENsaWVudChzdGFjaywgXCJDb2duaXRvQXBwQ2xpZW50XCIsIHtcbiAgICBhcHBDbGllbnRQcm92aWRlcjogbW9ja0FwcENsaWVudFByb3ZpZGVyLFxuICAgIG5hbWU6IFwibXktYmFja2VuZC1zZXJ2aWNlXCIsXG4gICAgdHlwZTogQXBwQ2xpZW50VHlwZS5CQUNLRU5ELFxuICAgIHNjb3BlczogW1wiaHR0cHM6Ly9hcGkudnlkZXYuaW8vcmVhZFwiLCBcImh0dHBzOi8vYXBpLnZ5ZGV2LmlvL3dyaXRlXCJdLFxuICB9KVxuXG4gIGV4cGVjdChzdGFjaykudG9NYXRjaENka1NuYXBzaG90KHtcbiAgICBpZ25vcmVBc3NldHM6IHRydWUsXG4gIH0pXG59KVxuXG50ZXN0KFwiY29nbml0byBmcm9udGVuZCBhcHAgY2xpZW50IHdpdGggY3VzdG9tIHNlY3JldE5hbWVcIiwgKCkgPT4ge1xuICBjb25zdCBhcHAgPSBuZXcgQXBwKClcbiAgY29uc3Qgc3RhY2sgPSBuZXcgU3RhY2soYXBwLCBcIlN0YWNrXCIpXG5cbiAgY29uc3QgbW9ja0FwcENsaWVudFByb3ZpZGVyOiBBcHBDbGllbnRQcm92aWRlciA9IHtcbiAgICBlbnZpcm9ubWVudDogVnlFbnZpcm9ubWVudC5QUk9ELFxuICAgIHNlcnZpY2VUb2tlbjogXCJzZXJ2aWNlVG9rZW5cIixcbiAgICBhdXRoX3VybDogXCJhdXRoX3VybFwiLFxuICB9XG5cbiAgbmV3IENvZ25pdG9BcHBDbGllbnQoc3RhY2ssIFwiQ29nbml0b0FwcENsaWVudFwiLCB7XG4gICAgYXBwQ2xpZW50UHJvdmlkZXI6IG1vY2tBcHBDbGllbnRQcm92aWRlcixcbiAgICBuYW1lOiBcIm15LWJhY2tlbmQtc2VydmljZVwiLFxuICAgIHR5cGU6IEFwcENsaWVudFR5cGUuQkFDS0VORCxcbiAgICBzY29wZXM6IFtcImh0dHBzOi8vYXBpLnZ5ZGV2LmlvL3JlYWRcIiwgXCJodHRwczovL2FwaS52eWRldi5pby93cml0ZVwiXSxcbiAgICBzZWNyZXROYW1lOiBcIm15LXNlY3JldFwiLFxuICB9KVxuXG4gIGV4cGVjdChzdGFjaykudG9NYXRjaENka1NuYXBzaG90KHtcbiAgICBpZ25vcmVBc3NldHM6IHRydWUsXG4gIH0pXG59KVxuIl19
@@ -2,11 +2,11 @@
2
2
  * CDK Construct for Cognito App Client
3
3
  */
4
4
  import * as cdk from "aws-cdk-lib";
5
- import * as iam from "aws-cdk-lib/aws-iam";
6
- import * as logs from "aws-cdk-lib/aws-logs";
5
+ import type * as iam from "aws-cdk-lib/aws-iam";
6
+ import type * as logs from "aws-cdk-lib/aws-logs";
7
7
  import * as secretsmanager from "aws-cdk-lib/aws-secretsmanager";
8
8
  import { Construct } from "constructs";
9
- import type { VyEnvironment } from "../shared/types";
9
+ import type { AppClientProvider } from "../vy-cognito-provider";
10
10
  /**
11
11
  * Type of app client
12
12
  */
@@ -26,9 +26,9 @@ export declare enum AppClientType {
26
26
  }
27
27
  export interface CognitoAppClientProps {
28
28
  /**
29
- * The Vy environment to provision in (e.g., VyEnvironment.PROD, VyEnvironment.STAGE, VyEnvironment.TEST)
29
+ * An AppClientProvider provided from a VyCognitoProvider
30
30
  */
31
- readonly environment: VyEnvironment;
31
+ readonly appClientProvider: AppClientProvider;
32
32
  /**
33
33
  * The name of the app client
34
34
  * Must be unique within the environment
@@ -70,13 +70,6 @@ export interface CognitoAppClientProps {
70
70
  * @default - true for backend, false for frontend
71
71
  */
72
72
  readonly generateSecret?: boolean;
73
- /**
74
- * Whether to store the client secret in AWS Secrets Manager
75
- * Only applicable if generateSecret is true
76
- *
77
- * @default true
78
- */
79
- readonly storeSecretInSecretsManager?: boolean;
80
73
  /**
81
74
  * Name of the secret in AWS Secrets Manager
82
75
  * Only applicable if generateSecret is true
@@ -103,9 +96,15 @@ export interface CognitoAppClientProps {
103
96
  * - **Frontend**: User authentication using OAuth 2.0 Authorization Code or Implicit Grant
104
97
  *
105
98
  * @example
99
+ * ```typescript
100
+ * // Create a VyCognitoProvider
101
+ * const vyCognitoProvider = new VyCognitoProvider(this, 'MyProvider', {
102
+ * environment: VyEnvironment.TEST,
103
+ * });
104
+ *
106
105
  * // Backend app client for M2M authentication
107
106
  * const backendClient = new CognitoAppClient(this, 'BackendClient', {
108
- * environment: VyEnvironment.TEST,
107
+ * appClientProvider: vyCognitoProvider.appClientProvider,
109
108
  * name: 'my-backend-service',
110
109
  * type: AppClientType.BACKEND,
111
110
  * scopes: ['https://api.vydev.io/read', 'https://api.vydev.io/write']
@@ -114,17 +113,25 @@ export interface CognitoAppClientProps {
114
113
  * // Access credentials
115
114
  * const clientId = backendClient.clientId;
116
115
  * const clientSecret = backendClient.clientSecret; // Stored in Secrets Manager
116
+ * ```
117
117
  *
118
118
  * @example
119
+ * ```typescript
120
+ * // Create a VyCognitoProvider
121
+ * const vyCognitoProvider = new VyCognitoProvider(this, 'MyProvider', {
122
+ * environment: VyEnvironment.TEST,
123
+ * });
124
+ *
119
125
  * // Frontend app client for user authentication
120
126
  * const frontendClient = new CognitoAppClient(this, 'FrontendClient', {
121
- * environment: VyEnvironment.PROD,
127
+ * appClientProvider: vyCognitoProvider.appClientProvider,
122
128
  * name: 'my-web-app',
123
129
  * type: AppClientType.FRONTEND,
124
130
  * scopes: ['email', 'openid', 'profile'],
125
131
  * callbackUrls: ['https://my-app.vydev.io/auth/callback'],
126
132
  * logoutUrls: ['https://my-app.vydev.io/logout']
127
133
  * });
134
+ * ```
128
135
  */
129
136
  export declare class CognitoAppClient extends Construct {
130
137
  /**
@@ -135,26 +142,16 @@ export declare class CognitoAppClient extends Construct {
135
142
  * The underlying custom resource
136
143
  */
137
144
  readonly resource: cdk.CustomResource;
138
- /**
139
- * The logGroup for the event handler lambda
140
- */
141
- readonly lambdaLogGroup: logs.LogGroup;
142
- /**
143
- * The logGroup for the custom resource provider
144
- */
145
- readonly providerLogGroup: logs.LogGroup;
146
145
  /**
147
146
  * The client ID for authentication
148
147
  */
149
148
  readonly clientId: string;
150
149
  /**
151
- * The client secret (only for backend clients)
152
- * If storeSecretInSecretsManager is true, this returns a reference to the secret value
150
+ * A reference to the client secret
153
151
  */
154
- readonly clientSecret?: string;
152
+ readonly clientSecretArn?: string;
155
153
  /**
156
154
  * The Secrets Manager secret containing the client secret
157
- * Only available if storeSecretInSecretsManager is true and type is backend
158
155
  */
159
156
  readonly clientSecretSecret?: secretsmanager.ISecret;
160
157
  constructor(scope: Construct, id: string, props: CognitoAppClientProps);
@@ -5,12 +5,8 @@ import { createRequire } from "node:module";
5
5
  import * as path from "node:path";
6
6
  import { fileURLToPath } from "node:url";
7
7
  import * as cdk from "aws-cdk-lib";
8
- import * as iam from "aws-cdk-lib/aws-iam";
9
- import * as lambda from "aws-cdk-lib/aws-lambda";
10
- import { NodejsFunction } from "aws-cdk-lib/aws-lambda-nodejs";
11
- import * as logs from "aws-cdk-lib/aws-logs";
8
+ import { SecretValue } from "aws-cdk-lib";
12
9
  import * as secretsmanager from "aws-cdk-lib/aws-secretsmanager";
13
- import * as cr from "aws-cdk-lib/custom-resources";
14
10
  import { Construct } from "constructs";
15
11
  const require = createRequire(import.meta.url);
16
12
  const __filename = fileURLToPath(import.meta.url);
@@ -44,9 +40,15 @@ export var AppClientType;
44
40
  * - **Frontend**: User authentication using OAuth 2.0 Authorization Code or Implicit Grant
45
41
  *
46
42
  * @example
43
+ * ```typescript
44
+ * // Create a VyCognitoProvider
45
+ * const vyCognitoProvider = new VyCognitoProvider(this, 'MyProvider', {
46
+ * environment: VyEnvironment.TEST,
47
+ * });
48
+ *
47
49
  * // Backend app client for M2M authentication
48
50
  * const backendClient = new CognitoAppClient(this, 'BackendClient', {
49
- * environment: VyEnvironment.TEST,
51
+ * appClientProvider: vyCognitoProvider.appClientProvider,
50
52
  * name: 'my-backend-service',
51
53
  * type: AppClientType.BACKEND,
52
54
  * scopes: ['https://api.vydev.io/read', 'https://api.vydev.io/write']
@@ -55,17 +57,25 @@ export var AppClientType;
55
57
  * // Access credentials
56
58
  * const clientId = backendClient.clientId;
57
59
  * const clientSecret = backendClient.clientSecret; // Stored in Secrets Manager
60
+ * ```
58
61
  *
59
62
  * @example
63
+ * ```typescript
64
+ * // Create a VyCognitoProvider
65
+ * const vyCognitoProvider = new VyCognitoProvider(this, 'MyProvider', {
66
+ * environment: VyEnvironment.TEST,
67
+ * });
68
+ *
60
69
  * // Frontend app client for user authentication
61
70
  * const frontendClient = new CognitoAppClient(this, 'FrontendClient', {
62
- * environment: VyEnvironment.PROD,
71
+ * appClientProvider: vyCognitoProvider.appClientProvider,
63
72
  * name: 'my-web-app',
64
73
  * type: AppClientType.FRONTEND,
65
74
  * scopes: ['email', 'openid', 'profile'],
66
75
  * callbackUrls: ['https://my-app.vydev.io/auth/callback'],
67
76
  * logoutUrls: ['https://my-app.vydev.io/logout']
68
77
  * });
78
+ * ```
69
79
  */
70
80
  export class CognitoAppClient extends Construct {
71
81
  /**
@@ -76,26 +86,16 @@ export class CognitoAppClient extends Construct {
76
86
  * The underlying custom resource
77
87
  */
78
88
  resource;
79
- /**
80
- * The logGroup for the event handler lambda
81
- */
82
- lambdaLogGroup;
83
- /**
84
- * The logGroup for the custom resource provider
85
- */
86
- providerLogGroup;
87
89
  /**
88
90
  * The client ID for authentication
89
91
  */
90
92
  clientId;
91
93
  /**
92
- * The client secret (only for backend clients)
93
- * If storeSecretInSecretsManager is true, this returns a reference to the secret value
94
+ * A reference to the client secret
94
95
  */
95
- clientSecret;
96
+ clientSecretArn;
96
97
  /**
97
98
  * The Secrets Manager secret containing the client secret
98
- * Only available if storeSecretInSecretsManager is true and type is backend
99
99
  */
100
100
  clientSecretSecret;
101
101
  constructor(scope, id, props) {
@@ -106,68 +106,41 @@ export class CognitoAppClient extends Construct {
106
106
  props.callbackUrls.length === 0) {
107
107
  cdk.Annotations.of(this).addWarning("Frontend app clients typically require callbackUrls for OAuth flows");
108
108
  }
109
- this.lambdaLogGroup = new logs.LogGroup(this, "LambdaLogGroup", {
110
- retention: props.logsRetention ?? logs.RetentionDays.ONE_WEEK,
111
- });
112
- const onEventHandler = new NodejsFunction(this, "OnEventHandler", {
113
- runtime: lambda.Runtime.NODEJS_22_X,
114
- handler: "handler",
115
- entry: require.resolve(`${__dirname}/handler`),
116
- timeout: cdk.Duration.minutes(2),
117
- memorySize: 256,
118
- logGroup: this.lambdaLogGroup,
119
- environment: props.cognitoBaseDomain
120
- ? {
121
- COGNITO_BASE_DOMAIN: props.cognitoBaseDomain,
122
- }
123
- : undefined,
124
- bundling: {
125
- minify: true,
126
- sourceMap: true,
127
- target: "es2020",
128
- externalModules: ["aws-sdk"],
129
- },
130
- });
131
- onEventHandler.addToRolePolicy(new iam.PolicyStatement({
132
- effect: iam.Effect.ALLOW,
133
- actions: ["execute-api:Invoke"],
134
- resources: ["*"], // Can be scoped down if API Gateway ARN is known
135
- }));
136
- this.providerLogGroup = new logs.LogGroup(this, "ProviderLogGroup", {
137
- retention: props.logsRetention ?? logs.RetentionDays.ONE_WEEK,
138
- });
139
- const provider = new cr.Provider(this, "Provider", {
140
- onEventHandler,
141
- logGroup: this.providerLogGroup,
142
- });
109
+ // props.generateSecret overrides type-specific defaults
110
+ // Default for backend clients is true; for frontend clients false
111
+ const generateSecret = props.generateSecret ?? props.type === AppClientType.BACKEND;
143
112
  this.resource = new cdk.CustomResource(this, "Resource", {
144
- serviceToken: provider.serviceToken,
113
+ serviceToken: props.appClientProvider.serviceToken,
145
114
  properties: {
146
- Environment: props.environment,
115
+ Environment: props.appClientProvider.environment,
147
116
  Name: props.name,
148
117
  Type: props.type,
149
118
  Scopes: props.scopes || [],
150
119
  CallbackUrls: props.callbackUrls || [],
151
120
  LogoutUrls: props.logoutUrls || [],
152
- GenerateSecret: props.generateSecret,
121
+ GenerateSecret: generateSecret,
153
122
  },
154
123
  // Force replacement if type changes
155
124
  resourceType: "Custom::VyCognitoAppClient",
156
125
  });
157
- this.clientId = this.resource.getAttString("ClientId");
158
- const shouldStoreSecret = props.storeSecretInSecretsManager !== false;
159
- if (props.type === AppClientType.BACKEND && shouldStoreSecret) {
160
- const secretValue = this.resource.getAttString("ClientSecret");
126
+ const client_id = this.resource.getAttString("ClientId");
127
+ this.clientId = client_id;
128
+ if (generateSecret) {
129
+ const client_secret = this.resource.getAttString("ClientSecret");
130
+ const auth_url = props.appClientProvider.auth_url;
131
+ // For convenience, store auth_url, client_id, and client_secret in a single JSON secret
132
+ const secretString = JSON.stringify({
133
+ auth_url,
134
+ client_id,
135
+ client_secret,
136
+ });
161
137
  this.clientSecretSecret = new secretsmanager.Secret(this, "ClientSecret", {
162
138
  secretName: props.secretName ??
163
- `vy/${props.environment}/cognito/app-client/${props.name}`,
164
- description: `Client secret for Vy Cognito app client ${props.name}`,
165
- secretStringValue: cdk.SecretValue.unsafePlainText(secretValue),
139
+ `/vy/${props.appClientProvider.environment}/cognito/app-client/${props.name}`,
140
+ description: `auth_url, client_id and client_secret for Vy Cognito app client ${props.name}`,
141
+ secretStringValue: SecretValue.unsafePlainText(secretString),
166
142
  });
167
- this.clientSecret = this.clientSecretSecret.secretValue.unsafeUnwrap();
168
- }
169
- else if (props.type === AppClientType.BACKEND) {
170
- this.clientSecret = this.resource.getAttString("ClientSecret");
143
+ this.clientSecretArn = this.clientSecretSecret.secretFullArn;
171
144
  }
172
145
  }
173
146
  /**
@@ -180,4 +153,4 @@ export class CognitoAppClient extends Construct {
180
153
  return this.clientSecretSecret.grantRead(grantee);
181
154
  }
182
155
  }
183
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cognito-app-client.js","sourceRoot":"","sources":["../../src/cognito-app-client/cognito-app-client.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CDK Construct for Cognito App Client\n */\n\nimport { createRequire } from \"node:module\"\nimport * as path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as secretsmanager from \"aws-cdk-lib/aws-secretsmanager\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { Construct } from \"constructs\"\nimport type { VyEnvironment } from \"../shared/types\"\n\nconst require = createRequire(import.meta.url)\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\n/**\n * Type of app client\n */\nexport enum AppClientType {\n  /**\n   * Backend app client for machine-to-machine (M2M) authentication\n   * Uses OAuth 2.0 Client Credentials grant type\n   * Generates a client secret for authentication\n   */\n  BACKEND = \"backend\",\n\n  /**\n   * Frontend app client for user authentication\n   * Uses OAuth 2.0 Authorization Code or Implicit grant types\n   * Supports callback URLs and logout URLs for browser-based flows\n   */\n  FRONTEND = \"frontend\",\n}\n\nexport interface CognitoAppClientProps {\n  /**\n   * The Vy environment to provision in (e.g., VyEnvironment.PROD, VyEnvironment.STAGE, VyEnvironment.TEST)\n   */\n  readonly environment: VyEnvironment\n\n  /**\n   * The name of the app client\n   * Must be unique within the environment\n   */\n  readonly name: string\n\n  /**\n   * The type of app client\n   * - backend: Machine-to-machine authentication with client credentials\n   * - frontend: User authentication with authorization code or implicit grant\n   */\n  readonly type: AppClientType\n\n  /**\n   * OAuth scopes for this client\n   *\n   * For backend clients: Use resource server scopes (e.g., 'https://api.vydev.io/read')\n   * For frontend clients: Use OIDC scopes (e.g., 'email', 'openid', 'profile')\n   *\n   * @default - Empty array\n   */\n  readonly scopes?: string[]\n\n  /**\n   * Callback URLs for OAuth flows\n   * Only used with frontend clients\n   *\n   * @default - No callback URLs\n   */\n  readonly callbackUrls?: string[]\n\n  /**\n   * Logout URLs for OAuth flows\n   * Only used with frontend clients\n   *\n   * @default - No logout URLs\n   */\n  readonly logoutUrls?: string[]\n\n  /**\n   * Whether to generate a client secret\n   * Automatically set based on type, but can be overridden\n   *\n   * @default - true for backend, false for frontend\n   */\n  readonly generateSecret?: boolean\n\n  /**\n   * Whether to store the client secret in AWS Secrets Manager\n   * Only applicable if generateSecret is true\n   *\n   * @default true\n   */\n  readonly storeSecretInSecretsManager?: boolean\n\n  /**\n   * Name of the secret in AWS Secrets Manager\n   * Only applicable if generateSecret is true\n   */\n  readonly secretName?: string\n\n  /**\n   * Base domain for Cognito service\n   * @default 'cognito.vydev.io'\n   */\n  readonly cognitoBaseDomain?: string\n\n  /**\n   * @default logs.RetentionDays.ONE_WEEK\n   */\n  readonly logsRetention?: logs.RetentionDays\n}\n\n/**\n * A Cognito App Client managed through Vy's central Cognito service\n *\n * App clients are the user pool authentication resources attached to your app.\n * Use an app client to configure the permitted authentication actions for an app.\n *\n * There are two types of app clients:\n * - **Backend**: Machine-to-machine authentication using OAuth 2.0 Client Credentials\n * - **Frontend**: User authentication using OAuth 2.0 Authorization Code or Implicit Grant\n *\n * @example\n * // Backend app client for M2M authentication\n * const backendClient = new CognitoAppClient(this, 'BackendClient', {\n *   environment: VyEnvironment.TEST,\n *   name: 'my-backend-service',\n *   type: AppClientType.BACKEND,\n *   scopes: ['https://api.vydev.io/read', 'https://api.vydev.io/write']\n * });\n *\n * // Access credentials\n * const clientId = backendClient.clientId;\n * const clientSecret = backendClient.clientSecret; // Stored in Secrets Manager\n *\n * @example\n * // Frontend app client for user authentication\n * const frontendClient = new CognitoAppClient(this, 'FrontendClient', {\n *   environment: VyEnvironment.PROD,\n *   name: 'my-web-app',\n *   type: AppClientType.FRONTEND,\n *   scopes: ['email', 'openid', 'profile'],\n *   callbackUrls: ['https://my-app.vydev.io/auth/callback'],\n *   logoutUrls: ['https://my-app.vydev.io/logout']\n * });\n */\nexport class CognitoAppClient extends Construct {\n  /**\n   * The name of the app client\n   */\n  public readonly name: string\n\n  /**\n   * The underlying custom resource\n   */\n  public readonly resource: cdk.CustomResource\n\n  /**\n   * The logGroup for the event handler lambda\n   */\n  public readonly lambdaLogGroup: logs.LogGroup\n\n  /**\n   * The logGroup for the custom resource provider\n   */\n  public readonly providerLogGroup: logs.LogGroup\n\n  /**\n   * The client ID for authentication\n   */\n  public readonly clientId: string\n\n  /**\n   * The client secret (only for backend clients)\n   * If storeSecretInSecretsManager is true, this returns a reference to the secret value\n   */\n  public readonly clientSecret?: string\n\n  /**\n   * The Secrets Manager secret containing the client secret\n   * Only available if storeSecretInSecretsManager is true and type is backend\n   */\n  public readonly clientSecretSecret?: secretsmanager.ISecret\n\n  constructor(scope: Construct, id: string, props: CognitoAppClientProps) {\n    super(scope, id)\n\n    this.name = props.name\n\n    if (\n      props.type === AppClientType.FRONTEND &&\n      props.callbackUrls &&\n      props.callbackUrls.length === 0\n    ) {\n      cdk.Annotations.of(this).addWarning(\n        \"Frontend app clients typically require callbackUrls for OAuth flows\",\n      )\n    }\n\n    this.lambdaLogGroup = new logs.LogGroup(this, \"LambdaLogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.ONE_WEEK,\n    })\n\n    const onEventHandler = new NodejsFunction(this, \"OnEventHandler\", {\n      runtime: lambda.Runtime.NODEJS_22_X,\n      handler: \"handler\",\n      entry: require.resolve(`${__dirname}/handler`),\n      timeout: cdk.Duration.minutes(2),\n      memorySize: 256,\n      logGroup: this.lambdaLogGroup,\n      environment: props.cognitoBaseDomain\n        ? {\n            COGNITO_BASE_DOMAIN: props.cognitoBaseDomain,\n          }\n        : undefined,\n      bundling: {\n        minify: true,\n        sourceMap: true,\n        target: \"es2020\",\n        externalModules: [\"aws-sdk\"],\n      },\n    })\n\n    onEventHandler.addToRolePolicy(\n      new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: [\"execute-api:Invoke\"],\n        resources: [\"*\"], // Can be scoped down if API Gateway ARN is known\n      }),\n    )\n\n    this.providerLogGroup = new logs.LogGroup(this, \"ProviderLogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.ONE_WEEK,\n    })\n\n    const provider = new cr.Provider(this, \"Provider\", {\n      onEventHandler,\n      logGroup: this.providerLogGroup,\n    })\n\n    this.resource = new cdk.CustomResource(this, \"Resource\", {\n      serviceToken: provider.serviceToken,\n      properties: {\n        Environment: props.environment,\n        Name: props.name,\n        Type: props.type,\n        Scopes: props.scopes || [],\n        CallbackUrls: props.callbackUrls || [],\n        LogoutUrls: props.logoutUrls || [],\n        GenerateSecret: props.generateSecret,\n      },\n      // Force replacement if type changes\n      resourceType: \"Custom::VyCognitoAppClient\",\n    })\n\n    this.clientId = this.resource.getAttString(\"ClientId\")\n\n    const shouldStoreSecret = props.storeSecretInSecretsManager !== false\n    if (props.type === AppClientType.BACKEND && shouldStoreSecret) {\n      const secretValue = this.resource.getAttString(\"ClientSecret\")\n\n      this.clientSecretSecret = new secretsmanager.Secret(\n        this,\n        \"ClientSecret\",\n        {\n          secretName:\n            props.secretName ??\n            `vy/${props.environment}/cognito/app-client/${props.name}`,\n          description: `Client secret for Vy Cognito app client ${props.name}`,\n          secretStringValue: cdk.SecretValue.unsafePlainText(secretValue),\n        },\n      )\n\n      this.clientSecret = this.clientSecretSecret.secretValue.unsafeUnwrap()\n    } else if (props.type === AppClientType.BACKEND) {\n      this.clientSecret = this.resource.getAttString(\"ClientSecret\")\n    }\n  }\n\n  /**\n   * Grant read access to the client secret (if it exists in Secrets Manager)\n   */\n  public grantReadSecret(grantee: iam.IGrantable): iam.Grant {\n    if (!this.clientSecretSecret) {\n      throw new Error(\"Client secret is not stored in Secrets Manager\")\n    }\n    return this.clientSecretSecret.grantRead(grantee)\n  }\n}\n"]}
156
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cognito-app-client.js","sourceRoot":"","sources":["../../src/cognito-app-client/cognito-app-client.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CDK Construct for Cognito App Client\n */\n\nimport { createRequire } from \"node:module\"\nimport * as path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\nimport * as cdk from \"aws-cdk-lib\"\nimport { SecretValue } from \"aws-cdk-lib\"\nimport type * as iam from \"aws-cdk-lib/aws-iam\"\nimport type * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as secretsmanager from \"aws-cdk-lib/aws-secretsmanager\"\nimport { Construct } from \"constructs\"\nimport type { AppClientProvider } from \"../vy-cognito-provider\"\n\nconst require = createRequire(import.meta.url)\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\n/**\n * Type of app client\n */\nexport enum AppClientType {\n  /**\n   * Backend app client for machine-to-machine (M2M) authentication\n   * Uses OAuth 2.0 Client Credentials grant type\n   * Generates a client secret for authentication\n   */\n  BACKEND = \"backend\",\n\n  /**\n   * Frontend app client for user authentication\n   * Uses OAuth 2.0 Authorization Code or Implicit grant types\n   * Supports callback URLs and logout URLs for browser-based flows\n   */\n  FRONTEND = \"frontend\",\n}\n\nexport interface CognitoAppClientProps {\n  /**\n   * An AppClientProvider provided from a VyCognitoProvider\n   */\n  readonly appClientProvider: AppClientProvider\n\n  /**\n   * The name of the app client\n   * Must be unique within the environment\n   */\n  readonly name: string\n\n  /**\n   * The type of app client\n   * - backend: Machine-to-machine authentication with client credentials\n   * - frontend: User authentication with authorization code or implicit grant\n   */\n  readonly type: AppClientType\n\n  /**\n   * OAuth scopes for this client\n   *\n   * For backend clients: Use resource server scopes (e.g., 'https://api.vydev.io/read')\n   * For frontend clients: Use OIDC scopes (e.g., 'email', 'openid', 'profile')\n   *\n   * @default - Empty array\n   */\n  readonly scopes?: string[]\n\n  /**\n   * Callback URLs for OAuth flows\n   * Only used with frontend clients\n   *\n   * @default - No callback URLs\n   */\n  readonly callbackUrls?: string[]\n\n  /**\n   * Logout URLs for OAuth flows\n   * Only used with frontend clients\n   *\n   * @default - No logout URLs\n   */\n  readonly logoutUrls?: string[]\n\n  /**\n   * Whether to generate a client secret\n   * Automatically set based on type, but can be overridden\n   *\n   * @default - true for backend, false for frontend\n   */\n  readonly generateSecret?: boolean\n\n  /**\n   * Name of the secret in AWS Secrets Manager\n   * Only applicable if generateSecret is true\n   */\n  readonly secretName?: string\n\n  /**\n   * Base domain for Cognito service\n   * @default 'cognito.vydev.io'\n   */\n  readonly cognitoBaseDomain?: string\n\n  /**\n   * @default logs.RetentionDays.ONE_WEEK\n   */\n  readonly logsRetention?: logs.RetentionDays\n}\n\n/**\n * A Cognito App Client managed through Vy's central Cognito service\n *\n * App clients are the user pool authentication resources attached to your app.\n * Use an app client to configure the permitted authentication actions for an app.\n *\n * There are two types of app clients:\n * - **Backend**: Machine-to-machine authentication using OAuth 2.0 Client Credentials\n * - **Frontend**: User authentication using OAuth 2.0 Authorization Code or Implicit Grant\n *\n * @example\n * ```typescript\n *  // Create a VyCognitoProvider\n * const vyCognitoProvider = new VyCognitoProvider(this, 'MyProvider', {\n *   environment: VyEnvironment.TEST,\n * });\n *\n * // Backend app client for M2M authentication\n * const backendClient = new CognitoAppClient(this, 'BackendClient', {\n *   appClientProvider: vyCognitoProvider.appClientProvider,\n *   name: 'my-backend-service',\n *   type: AppClientType.BACKEND,\n *   scopes: ['https://api.vydev.io/read', 'https://api.vydev.io/write']\n * });\n *\n * // Access credentials\n * const clientId = backendClient.clientId;\n * const clientSecret = backendClient.clientSecret; // Stored in Secrets Manager\n * ```\n *\n * @example\n * ```typescript\n * // Create a VyCognitoProvider\n * const vyCognitoProvider = new VyCognitoProvider(this, 'MyProvider', {\n *   environment: VyEnvironment.TEST,\n * });\n *\n * // Frontend app client for user authentication\n * const frontendClient = new CognitoAppClient(this, 'FrontendClient', {\n *   appClientProvider: vyCognitoProvider.appClientProvider,\n *   name: 'my-web-app',\n *   type: AppClientType.FRONTEND,\n *   scopes: ['email', 'openid', 'profile'],\n *   callbackUrls: ['https://my-app.vydev.io/auth/callback'],\n *   logoutUrls: ['https://my-app.vydev.io/logout']\n * });\n * ```\n */\nexport class CognitoAppClient extends Construct {\n  /**\n   * The name of the app client\n   */\n  public readonly name: string\n\n  /**\n   * The underlying custom resource\n   */\n  public readonly resource: cdk.CustomResource\n\n  /**\n   * The client ID for authentication\n   */\n  public readonly clientId: string\n\n  /**\n   * A reference to the client secret\n   */\n  public readonly clientSecretArn?: string\n\n  /**\n   * The Secrets Manager secret containing the client secret\n   */\n  public readonly clientSecretSecret?: secretsmanager.ISecret\n\n  constructor(scope: Construct, id: string, props: CognitoAppClientProps) {\n    super(scope, id)\n\n    this.name = props.name\n\n    if (\n      props.type === AppClientType.FRONTEND &&\n      props.callbackUrls &&\n      props.callbackUrls.length === 0\n    ) {\n      cdk.Annotations.of(this).addWarning(\n        \"Frontend app clients typically require callbackUrls for OAuth flows\",\n      )\n    }\n\n    // props.generateSecret overrides type-specific defaults\n    // Default for backend clients is true; for frontend clients false\n    const generateSecret =\n      props.generateSecret ?? props.type === AppClientType.BACKEND\n\n    this.resource = new cdk.CustomResource(this, \"Resource\", {\n      serviceToken: props.appClientProvider.serviceToken,\n      properties: {\n        Environment: props.appClientProvider.environment,\n        Name: props.name,\n        Type: props.type,\n        Scopes: props.scopes || [],\n        CallbackUrls: props.callbackUrls || [],\n        LogoutUrls: props.logoutUrls || [],\n        GenerateSecret: generateSecret,\n      },\n      // Force replacement if type changes\n      resourceType: \"Custom::VyCognitoAppClient\",\n    })\n\n    const client_id = this.resource.getAttString(\"ClientId\")\n    this.clientId = client_id\n\n    if (generateSecret) {\n      const client_secret = this.resource.getAttString(\"ClientSecret\")\n      const auth_url = props.appClientProvider.auth_url\n\n      // For convenience, store auth_url, client_id, and client_secret in a single JSON secret\n      const secretString = JSON.stringify({\n        auth_url,\n        client_id,\n        client_secret,\n      })\n\n      this.clientSecretSecret = new secretsmanager.Secret(\n        this,\n        \"ClientSecret\",\n        {\n          secretName:\n            props.secretName ??\n            `/vy/${props.appClientProvider.environment}/cognito/app-client/${props.name}`,\n          description: `auth_url, client_id and client_secret for Vy Cognito app client ${props.name}`,\n          secretStringValue: SecretValue.unsafePlainText(secretString),\n        },\n      )\n\n      this.clientSecretArn = this.clientSecretSecret.secretFullArn\n    }\n  }\n\n  /**\n   * Grant read access to the client secret (if it exists in Secrets Manager)\n   */\n  public grantReadSecret(grantee: iam.IGrantable): iam.Grant {\n    if (!this.clientSecretSecret) {\n      throw new Error(\"Client secret is not stored in Secrets Manager\")\n    }\n    return this.clientSecretSecret.grantRead(grantee)\n  }\n}\n"]}
@@ -5,8 +5,12 @@ import { CognitoResourceServer } from "../cognito-resource-server";
5
5
  test("cognito resource server", () => {
6
6
  const app = new App();
7
7
  const stack = new Stack(app, "Stack");
8
- new CognitoResourceServer(stack, "CognitoResourceServer", {
8
+ const mockResourceServerProvider = {
9
9
  environment: VyEnvironment.TEST,
10
+ serviceToken: "serviceToken",
11
+ };
12
+ new CognitoResourceServer(stack, "CognitoResourceServer", {
13
+ resourceServerProvider: mockResourceServerProvider,
10
14
  name: "testName",
11
15
  identifier: "testIdentifier",
12
16
  scopes: [
@@ -20,4 +24,4 @@ test("cognito resource server", () => {
20
24
  ignoreAssets: true,
21
25
  });
22
26
  });
23
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXgudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb2duaXRvLXJlc291cmNlLXNlcnZlci9fX3Rlc3RfXy9pbmRleC50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxHQUFHLEVBQUUsS0FBSyxFQUFFLE1BQU0sYUFBYSxDQUFBO0FBQ3hDLE9BQU8sbUJBQW1CLENBQUE7QUFDMUIsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLG9CQUFvQixDQUFBO0FBQ2xELE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDRCQUE0QixDQUFBO0FBRWxFLElBQUksQ0FBQyx5QkFBeUIsRUFBRSxHQUFHLEVBQUU7SUFDbkMsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLEVBQUUsQ0FBQTtJQUNyQixNQUFNLEtBQUssR0FBRyxJQUFJLEtBQUssQ0FBQyxHQUFHLEVBQUUsT0FBTyxDQUFDLENBQUE7SUFFckMsSUFBSSxxQkFBcUIsQ0FBQyxLQUFLLEVBQUUsdUJBQXVCLEVBQUU7UUFDeEQsV0FBVyxFQUFFLGFBQWEsQ0FBQyxJQUFJO1FBQy9CLElBQUksRUFBRSxVQUFVO1FBQ2hCLFVBQVUsRUFBRSxnQkFBZ0I7UUFDNUIsTUFBTSxFQUFFO1lBQ047Z0JBQ0UsSUFBSSxFQUFFLGVBQWU7Z0JBQ3JCLFdBQVcsRUFBRSxzQkFBc0I7YUFDcEM7U0FDRjtLQUNGLENBQUMsQ0FBQTtJQUVGLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQztRQUMvQixZQUFZLEVBQUUsSUFBSTtLQUNuQixDQUFDLENBQUE7QUFDSixDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFwcCwgU3RhY2sgfSBmcm9tIFwiYXdzLWNkay1saWJcIlxuaW1wb3J0IFwiamVzdC1jZGstc25hcHNob3RcIlxuaW1wb3J0IHsgVnlFbnZpcm9ubWVudCB9IGZyb20gXCIuLi8uLi9zaGFyZWQvdHlwZXNcIlxuaW1wb3J0IHsgQ29nbml0b1Jlc291cmNlU2VydmVyIH0gZnJvbSBcIi4uL2NvZ25pdG8tcmVzb3VyY2Utc2VydmVyXCJcblxudGVzdChcImNvZ25pdG8gcmVzb3VyY2Ugc2VydmVyXCIsICgpID0+IHtcbiAgY29uc3QgYXBwID0gbmV3IEFwcCgpXG4gIGNvbnN0IHN0YWNrID0gbmV3IFN0YWNrKGFwcCwgXCJTdGFja1wiKVxuXG4gIG5ldyBDb2duaXRvUmVzb3VyY2VTZXJ2ZXIoc3RhY2ssIFwiQ29nbml0b1Jlc291cmNlU2VydmVyXCIsIHtcbiAgICBlbnZpcm9ubWVudDogVnlFbnZpcm9ubWVudC5URVNULFxuICAgIG5hbWU6IFwidGVzdE5hbWVcIixcbiAgICBpZGVudGlmaWVyOiBcInRlc3RJZGVudGlmaWVyXCIsXG4gICAgc2NvcGVzOiBbXG4gICAgICB7XG4gICAgICAgIG5hbWU6IFwidGVzdFNjb3BlTmFtZVwiLFxuICAgICAgICBkZXNjcmlwdGlvbjogXCJ0ZXN0U2NvcGVEZXNjcmlwdGlvblwiLFxuICAgICAgfSxcbiAgICBdLFxuICB9KVxuXG4gIGV4cGVjdChzdGFjaykudG9NYXRjaENka1NuYXBzaG90KHtcbiAgICBpZ25vcmVBc3NldHM6IHRydWUsXG4gIH0pXG59KVxuIl19
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXgudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb2duaXRvLXJlc291cmNlLXNlcnZlci9fX3Rlc3RfXy9pbmRleC50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxHQUFHLEVBQUUsS0FBSyxFQUFFLE1BQU0sYUFBYSxDQUFBO0FBQ3hDLE9BQU8sbUJBQW1CLENBQUE7QUFDMUIsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLG9CQUFvQixDQUFBO0FBRWxELE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDRCQUE0QixDQUFBO0FBRWxFLElBQUksQ0FBQyx5QkFBeUIsRUFBRSxHQUFHLEVBQUU7SUFDbkMsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLEVBQUUsQ0FBQTtJQUNyQixNQUFNLEtBQUssR0FBRyxJQUFJLEtBQUssQ0FBQyxHQUFHLEVBQUUsT0FBTyxDQUFDLENBQUE7SUFFckMsTUFBTSwwQkFBMEIsR0FBMkI7UUFDekQsV0FBVyxFQUFFLGFBQWEsQ0FBQyxJQUFJO1FBQy9CLFlBQVksRUFBRSxjQUFjO0tBQzdCLENBQUE7SUFFRCxJQUFJLHFCQUFxQixDQUFDLEtBQUssRUFBRSx1QkFBdUIsRUFBRTtRQUN4RCxzQkFBc0IsRUFBRSwwQkFBMEI7UUFDbEQsSUFBSSxFQUFFLFVBQVU7UUFDaEIsVUFBVSxFQUFFLGdCQUFnQjtRQUM1QixNQUFNLEVBQUU7WUFDTjtnQkFDRSxJQUFJLEVBQUUsZUFBZTtnQkFDckIsV0FBVyxFQUFFLHNCQUFzQjthQUNwQztTQUNGO0tBQ0YsQ0FBQyxDQUFBO0lBRUYsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLGtCQUFrQixDQUFDO1FBQy9CLFlBQVksRUFBRSxJQUFJO0tBQ25CLENBQUMsQ0FBQTtBQUNKLENBQUMsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQXBwLCBTdGFjayB9IGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgXCJqZXN0LWNkay1zbmFwc2hvdFwiXG5pbXBvcnQgeyBWeUVudmlyb25tZW50IH0gZnJvbSBcIi4uLy4uL3NoYXJlZC90eXBlc1wiXG5pbXBvcnQgdHlwZSB7IFJlc291cmNlU2VydmVyUHJvdmlkZXIgfSBmcm9tIFwiLi4vLi4vdnktY29nbml0by1wcm92aWRlclwiXG5pbXBvcnQgeyBDb2duaXRvUmVzb3VyY2VTZXJ2ZXIgfSBmcm9tIFwiLi4vY29nbml0by1yZXNvdXJjZS1zZXJ2ZXJcIlxuXG50ZXN0KFwiY29nbml0byByZXNvdXJjZSBzZXJ2ZXJcIiwgKCkgPT4ge1xuICBjb25zdCBhcHAgPSBuZXcgQXBwKClcbiAgY29uc3Qgc3RhY2sgPSBuZXcgU3RhY2soYXBwLCBcIlN0YWNrXCIpXG5cbiAgY29uc3QgbW9ja1Jlc291cmNlU2VydmVyUHJvdmlkZXI6IFJlc291cmNlU2VydmVyUHJvdmlkZXIgPSB7XG4gICAgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnQuVEVTVCxcbiAgICBzZXJ2aWNlVG9rZW46IFwic2VydmljZVRva2VuXCIsXG4gIH1cblxuICBuZXcgQ29nbml0b1Jlc291cmNlU2VydmVyKHN0YWNrLCBcIkNvZ25pdG9SZXNvdXJjZVNlcnZlclwiLCB7XG4gICAgcmVzb3VyY2VTZXJ2ZXJQcm92aWRlcjogbW9ja1Jlc291cmNlU2VydmVyUHJvdmlkZXIsXG4gICAgbmFtZTogXCJ0ZXN0TmFtZVwiLFxuICAgIGlkZW50aWZpZXI6IFwidGVzdElkZW50aWZpZXJcIixcbiAgICBzY29wZXM6IFtcbiAgICAgIHtcbiAgICAgICAgbmFtZTogXCJ0ZXN0U2NvcGVOYW1lXCIsXG4gICAgICAgIGRlc2NyaXB0aW9uOiBcInRlc3RTY29wZURlc2NyaXB0aW9uXCIsXG4gICAgICB9LFxuICAgIF0sXG4gIH0pXG5cbiAgZXhwZWN0KHN0YWNrKS50b01hdGNoQ2RrU25hcHNob3Qoe1xuICAgIGlnbm9yZUFzc2V0czogdHJ1ZSxcbiAgfSlcbn0pXG4iXX0=
@@ -2,9 +2,8 @@
2
2
  * CDK Construct for Cognito Resource Server
3
3
  */
4
4
  import * as cdk from "aws-cdk-lib";
5
- import * as logs from "aws-cdk-lib/aws-logs";
6
5
  import { Construct } from "constructs";
7
- import type { VyEnvironment } from "../shared/types";
6
+ import type { ResourceServerProvider } from "../vy-cognito-provider";
8
7
  export interface Scope {
9
8
  /**
10
9
  * The name of the scope
@@ -17,9 +16,9 @@ export interface Scope {
17
16
  }
18
17
  export interface CognitoResourceServerProps {
19
18
  /**
20
- * The Vy environment to provision in (e.g., VyEnvironment.PROD, VyEnvironment.STAGE, VyEnvironment.TEST)
19
+ * An ResourceServerProvider provided from a VyCognitoProvider
21
20
  */
22
- readonly environment: VyEnvironment;
21
+ readonly resourceServerProvider: ResourceServerProvider;
23
22
  /**
24
23
  * The name of the resource server
25
24
  */
@@ -39,10 +38,6 @@ export interface CognitoResourceServerProps {
39
38
  * @default 'cognito.vydev.io'
40
39
  */
41
40
  readonly cognitoBaseDomain?: string;
42
- /**
43
- * @default logs.RetentionDays.ONE_WEEK
44
- */
45
- readonly logsRetention?: logs.RetentionDays;
46
41
  }
47
42
  /**
48
43
  * A Cognito Resource Server managed through Vy's central Cognito service
@@ -54,8 +49,13 @@ export interface CognitoResourceServerProps {
54
49
  *
55
50
  * @example
56
51
  * ```typescript
52
+ * // Create a VyCognitoProvider
53
+ * const vyCognitoProvider = new VyCognitoProvider(this, 'MyProvider', {
54
+ * environment: VyEnvironment.TEST,
55
+ * });
56
+ *
57
57
  * const resourceServer = new CognitoResourceServer(this, 'ApiResourceServer', {
58
- * environment: VyEnvironment.PROD,
58
+ * resourceServerProvider: vyCognitoProvider.resourceServerProvider,
59
59
  * name: 'my-api',
60
60
  * identifier: 'https://my-api.vydev.io',
61
61
  * scopes: [
@@ -78,14 +78,6 @@ export declare class CognitoResourceServer extends Construct {
78
78
  * The underlying custom resource
79
79
  */
80
80
  readonly resource: cdk.CustomResource;
81
- /**
82
- * The logGroup for the event handler lambda
83
- */
84
- readonly lambdaLogGroup: logs.LogGroup;
85
- /**
86
- * The logGroup for the custom resource provider
87
- */
88
- readonly providerLogGroup: logs.LogGroup;
89
81
  constructor(scope: Construct, id: string, props: CognitoResourceServerProps);
90
82
  /**
91
83
  * Get a reference to a scope in the format expected by app clients
@@ -5,11 +5,6 @@ import { createRequire } from "node:module";
5
5
  import * as path from "node:path";
6
6
  import { fileURLToPath } from "node:url";
7
7
  import * as cdk from "aws-cdk-lib";
8
- import * as iam from "aws-cdk-lib/aws-iam";
9
- import * as lambda from "aws-cdk-lib/aws-lambda";
10
- import { NodejsFunction } from "aws-cdk-lib/aws-lambda-nodejs";
11
- import * as logs from "aws-cdk-lib/aws-logs";
12
- import * as cr from "aws-cdk-lib/custom-resources";
13
8
  import { Construct } from "constructs";
14
9
  const require = createRequire(import.meta.url);
15
10
  const __filename = fileURLToPath(import.meta.url);
@@ -24,8 +19,13 @@ const __dirname = path.dirname(__filename);
24
19
  *
25
20
  * @example
26
21
  * ```typescript
22
+ * // Create a VyCognitoProvider
23
+ * const vyCognitoProvider = new VyCognitoProvider(this, 'MyProvider', {
24
+ * environment: VyEnvironment.TEST,
25
+ * });
26
+ *
27
27
  * const resourceServer = new CognitoResourceServer(this, 'ApiResourceServer', {
28
- * environment: VyEnvironment.PROD,
28
+ * resourceServerProvider: vyCognitoProvider.resourceServerProvider,
29
29
  * name: 'my-api',
30
30
  * identifier: 'https://my-api.vydev.io',
31
31
  * scopes: [
@@ -48,56 +48,14 @@ export class CognitoResourceServer extends Construct {
48
48
  * The underlying custom resource
49
49
  */
50
50
  resource;
51
- /**
52
- * The logGroup for the event handler lambda
53
- */
54
- lambdaLogGroup;
55
- /**
56
- * The logGroup for the custom resource provider
57
- */
58
- providerLogGroup;
59
51
  constructor(scope, id, props) {
60
52
  super(scope, id);
61
53
  this.identifier = props.identifier;
62
54
  this.name = props.name;
63
- this.lambdaLogGroup = new logs.LogGroup(this, "LambdaLogGroup", {
64
- retention: props.logsRetention ?? logs.RetentionDays.ONE_WEEK,
65
- });
66
- const onEventHandler = new NodejsFunction(this, "OnEventHandler", {
67
- runtime: lambda.Runtime.NODEJS_22_X,
68
- handler: "handler",
69
- entry: require.resolve(`${__dirname}/handler`),
70
- timeout: cdk.Duration.minutes(2),
71
- memorySize: 256,
72
- logGroup: this.lambdaLogGroup,
73
- environment: props.cognitoBaseDomain
74
- ? {
75
- COGNITO_BASE_DOMAIN: props.cognitoBaseDomain,
76
- }
77
- : undefined,
78
- bundling: {
79
- minify: true,
80
- sourceMap: true,
81
- target: "es2020",
82
- externalModules: ["aws-sdk"],
83
- },
84
- });
85
- onEventHandler.addToRolePolicy(new iam.PolicyStatement({
86
- effect: iam.Effect.ALLOW,
87
- actions: ["execute-api:Invoke"],
88
- resources: ["*"], // Can be scoped down if API Gateway ARN is known
89
- }));
90
- this.providerLogGroup = new logs.LogGroup(this, "ProviderLogGroup", {
91
- retention: props.logsRetention ?? logs.RetentionDays.ONE_WEEK,
92
- });
93
- const provider = new cr.Provider(this, "Provider", {
94
- onEventHandler,
95
- logGroup: this.providerLogGroup,
96
- });
97
55
  this.resource = new cdk.CustomResource(this, "Resource", {
98
- serviceToken: provider.serviceToken,
56
+ serviceToken: props.resourceServerProvider.serviceToken,
99
57
  properties: {
100
- Environment: props.environment,
58
+ Environment: props.resourceServerProvider.environment,
101
59
  Name: props.name,
102
60
  Identifier: props.identifier,
103
61
  Scopes: props.scopes?.map((s) => ({
@@ -117,4 +75,4 @@ export class CognitoResourceServer extends Construct {
117
75
  return `${this.identifier}/${scopeName}`;
118
76
  }
119
77
  }
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cognito-resource-server.js","sourceRoot":"","sources":["../../src/cognito-resource-server/cognito-resource-server.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CDK Construct for Cognito Resource Server\n */\n\nimport { createRequire } from \"node:module\"\nimport * as path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { Construct } from \"constructs\"\nimport type { VyEnvironment } from \"../shared/types\"\n\nconst require = createRequire(import.meta.url)\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\nexport interface Scope {\n  /**\n   * The name of the scope\n   */\n  readonly name: string\n\n  /**\n   * A description of what this scope is for\n   */\n  readonly description: string\n}\n\nexport interface CognitoResourceServerProps {\n  /**\n   * The Vy environment to provision in (e.g., VyEnvironment.PROD, VyEnvironment.STAGE, VyEnvironment.TEST)\n   */\n  readonly environment: VyEnvironment\n\n  /**\n   * The name of the resource server\n   */\n  readonly name: string\n\n  /**\n   * The identifier for this resource server (usually a URL)\n   * @example 'https://api.vydev.io'\n   */\n  readonly identifier: string\n\n  /**\n   * Custom scopes for this resource server\n   * @default - No scopes\n   */\n  readonly scopes?: Scope[]\n\n  /**\n   * Base domain for Cognito service\n   * @default 'cognito.vydev.io'\n   */\n  readonly cognitoBaseDomain?: string\n\n  /**\n   * @default logs.RetentionDays.ONE_WEEK\n   */\n  readonly logsRetention?: logs.RetentionDays\n}\n\n/**\n * A Cognito Resource Server managed through Vy's central Cognito service\n *\n * A resource server is an integration between a user pool and an API.\n * Each resource server has custom scopes that you must activate in your app client.\n * When you configure a resource server, your app can generate access tokens with\n * OAuth scopes that authorize read and write operations to an API server.\n *\n * @example\n * ```typescript\n * const resourceServer = new CognitoResourceServer(this, 'ApiResourceServer', {\n *   environment: VyEnvironment.PROD,\n *   name: 'my-api',\n *   identifier: 'https://my-api.vydev.io',\n *   scopes: [\n *     { name: 'read', description: 'Read access to the API' },\n *     { name: 'write', description: 'Write access to the API' }\n *   ]\n * });\n * ```\n */\nexport class CognitoResourceServer extends Construct {\n  /**\n   * The identifier of the resource server\n   */\n  public readonly identifier: string\n\n  /**\n   * The name of the resource server\n   */\n  public readonly name: string\n\n  /**\n   * The underlying custom resource\n   */\n  public readonly resource: cdk.CustomResource\n\n  /**\n   * The logGroup for the event handler lambda\n   */\n  public readonly lambdaLogGroup: logs.LogGroup\n\n  /**\n   * The logGroup for the custom resource provider\n   */\n  public readonly providerLogGroup: logs.LogGroup\n\n  constructor(scope: Construct, id: string, props: CognitoResourceServerProps) {\n    super(scope, id)\n\n    this.identifier = props.identifier\n    this.name = props.name\n\n    this.lambdaLogGroup = new logs.LogGroup(this, \"LambdaLogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.ONE_WEEK,\n    })\n\n    const onEventHandler = new NodejsFunction(this, \"OnEventHandler\", {\n      runtime: lambda.Runtime.NODEJS_22_X,\n      handler: \"handler\",\n      entry: require.resolve(`${__dirname}/handler`),\n      timeout: cdk.Duration.minutes(2),\n      memorySize: 256,\n      logGroup: this.lambdaLogGroup,\n      environment: props.cognitoBaseDomain\n        ? {\n            COGNITO_BASE_DOMAIN: props.cognitoBaseDomain,\n          }\n        : undefined,\n      bundling: {\n        minify: true,\n        sourceMap: true,\n        target: \"es2020\",\n        externalModules: [\"aws-sdk\"],\n      },\n    })\n\n    onEventHandler.addToRolePolicy(\n      new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: [\"execute-api:Invoke\"],\n        resources: [\"*\"], // Can be scoped down if API Gateway ARN is known\n      }),\n    )\n\n    this.providerLogGroup = new logs.LogGroup(this, \"ProviderLogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.ONE_WEEK,\n    })\n\n    const provider = new cr.Provider(this, \"Provider\", {\n      onEventHandler,\n      logGroup: this.providerLogGroup,\n    })\n\n    this.resource = new cdk.CustomResource(this, \"Resource\", {\n      serviceToken: provider.serviceToken,\n      properties: {\n        Environment: props.environment,\n        Name: props.name,\n        Identifier: props.identifier,\n        Scopes: props.scopes?.map((s) => ({\n          Name: s.name,\n          Description: s.description,\n        })),\n      },\n      resourceType: \"Custom::VyCognitoResourceServer\",\n    })\n  }\n\n  /**\n   * Get a reference to a scope in the format expected by app clients\n   * @param scopeName The name of the scope\n   * @returns The full scope identifier (e.g., 'https://api.vydev.io/read')\n   */\n  public scopeIdentifier(scopeName: string): string {\n    return `${this.identifier}/${scopeName}`\n  }\n}\n"]}
78
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29nbml0by1yZXNvdXJjZS1zZXJ2ZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29nbml0by1yZXNvdXJjZS1zZXJ2ZXIvY29nbml0by1yZXNvdXJjZS1zZXJ2ZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sYUFBYSxDQUFBO0FBQzNDLE9BQU8sS0FBSyxJQUFJLE1BQU0sV0FBVyxDQUFBO0FBQ2pDLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxVQUFVLENBQUE7QUFDeEMsT0FBTyxLQUFLLEdBQUcsTUFBTSxhQUFhLENBQUE7QUFDbEMsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLFlBQVksQ0FBQTtBQUd0QyxNQUFNLE9BQU8sR0FBRyxhQUFhLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQTtBQUM5QyxNQUFNLFVBQVUsR0FBRyxhQUFhLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQTtBQUNqRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxDQUFBO0FBNEMxQzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXlCRztBQUNILE1BQU0sT0FBTyxxQkFBc0IsU0FBUSxTQUFTO0lBQ2xEOztPQUVHO0lBQ2EsVUFBVSxDQUFRO0lBRWxDOztPQUVHO0lBQ2EsSUFBSSxDQUFRO0lBRTVCOztPQUVHO0lBQ2EsUUFBUSxDQUFvQjtJQUU1QyxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQWlDO1FBQ3pFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLFVBQVUsR0FBRyxLQUFLLENBQUMsVUFBVSxDQUFBO1FBQ2xDLElBQUksQ0FBQyxJQUFJLEdBQUcsS0FBSyxDQUFDLElBQUksQ0FBQTtRQUV0QixJQUFJLENBQUMsUUFBUSxHQUFHLElBQUksR0FBRyxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQ3ZELFlBQVksRUFBRSxLQUFLLENBQUMsc0JBQXNCLENBQUMsWUFBWTtZQUN2RCxVQUFVLEVBQUU7Z0JBQ1YsV0FBVyxFQUFFLEtBQUssQ0FBQyxzQkFBc0IsQ0FBQyxXQUFXO2dCQUNyRCxJQUFJLEVBQUUsS0FBSyxDQUFDLElBQUk7Z0JBQ2hCLFVBQVUsRUFBRSxLQUFLLENBQUMsVUFBVTtnQkFDNUIsTUFBTSxFQUFFLEtBQUssQ0FBQyxNQUFNLEVBQUUsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDO29CQUNoQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLElBQUk7b0JBQ1osV0FBVyxFQUFFLENBQUMsQ0FBQyxXQUFXO2lCQUMzQixDQUFDLENBQUM7YUFDSjtZQUNELFlBQVksRUFBRSxpQ0FBaUM7U0FDaEQsQ0FBQyxDQUFBO0lBQ0osQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxlQUFlLENBQUMsU0FBaUI7UUFDdEMsT0FBTyxHQUFHLElBQUksQ0FBQyxVQUFVLElBQUksU0FBUyxFQUFFLENBQUE7SUFDMUMsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDREsgQ29uc3RydWN0IGZvciBDb2duaXRvIFJlc291cmNlIFNlcnZlclxuICovXG5cbmltcG9ydCB7IGNyZWF0ZVJlcXVpcmUgfSBmcm9tIFwibm9kZTptb2R1bGVcIlxuaW1wb3J0ICogYXMgcGF0aCBmcm9tIFwibm9kZTpwYXRoXCJcbmltcG9ydCB7IGZpbGVVUkxUb1BhdGggfSBmcm9tIFwibm9kZTp1cmxcIlxuaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tIFwiY29uc3RydWN0c1wiXG5pbXBvcnQgdHlwZSB7IFJlc291cmNlU2VydmVyUHJvdmlkZXIgfSBmcm9tIFwiLi4vdnktY29nbml0by1wcm92aWRlclwiXG5cbmNvbnN0IHJlcXVpcmUgPSBjcmVhdGVSZXF1aXJlKGltcG9ydC5tZXRhLnVybClcbmNvbnN0IF9fZmlsZW5hbWUgPSBmaWxlVVJMVG9QYXRoKGltcG9ydC5tZXRhLnVybClcbmNvbnN0IF9fZGlybmFtZSA9IHBhdGguZGlybmFtZShfX2ZpbGVuYW1lKVxuXG5leHBvcnQgaW50ZXJmYWNlIFNjb3BlIHtcbiAgLyoqXG4gICAqIFRoZSBuYW1lIG9mIHRoZSBzY29wZVxuICAgKi9cbiAgcmVhZG9ubHkgbmFtZTogc3RyaW5nXG5cbiAgLyoqXG4gICAqIEEgZGVzY3JpcHRpb24gb2Ygd2hhdCB0aGlzIHNjb3BlIGlzIGZvclxuICAgKi9cbiAgcmVhZG9ubHkgZGVzY3JpcHRpb246IHN0cmluZ1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENvZ25pdG9SZXNvdXJjZVNlcnZlclByb3BzIHtcbiAgLyoqXG4gICAqIEFuIFJlc291cmNlU2VydmVyUHJvdmlkZXIgcHJvdmlkZWQgZnJvbSBhIFZ5Q29nbml0b1Byb3ZpZGVyXG4gICAqL1xuICByZWFkb25seSByZXNvdXJjZVNlcnZlclByb3ZpZGVyOiBSZXNvdXJjZVNlcnZlclByb3ZpZGVyXG5cbiAgLyoqXG4gICAqIFRoZSBuYW1lIG9mIHRoZSByZXNvdXJjZSBzZXJ2ZXJcbiAgICovXG4gIHJlYWRvbmx5IG5hbWU6IHN0cmluZ1xuXG4gIC8qKlxuICAgKiBUaGUgaWRlbnRpZmllciBmb3IgdGhpcyByZXNvdXJjZSBzZXJ2ZXIgKHVzdWFsbHkgYSBVUkwpXG4gICAqIEBleGFtcGxlICdodHRwczovL2FwaS52eWRldi5pbydcbiAgICovXG4gIHJlYWRvbmx5IGlkZW50aWZpZXI6IHN0cmluZ1xuXG4gIC8qKlxuICAgKiBDdXN0b20gc2NvcGVzIGZvciB0aGlzIHJlc291cmNlIHNlcnZlclxuICAgKiBAZGVmYXVsdCAtIE5vIHNjb3Blc1xuICAgKi9cbiAgcmVhZG9ubHkgc2NvcGVzPzogU2NvcGVbXVxuXG4gIC8qKlxuICAgKiBCYXNlIGRvbWFpbiBmb3IgQ29nbml0byBzZXJ2aWNlXG4gICAqIEBkZWZhdWx0ICdjb2duaXRvLnZ5ZGV2LmlvJ1xuICAgKi9cbiAgcmVhZG9ubHkgY29nbml0b0Jhc2VEb21haW4/OiBzdHJpbmdcbn1cblxuLyoqXG4gKiBBIENvZ25pdG8gUmVzb3VyY2UgU2VydmVyIG1hbmFnZWQgdGhyb3VnaCBWeSdzIGNlbnRyYWwgQ29nbml0byBzZXJ2aWNlXG4gKlxuICogQSByZXNvdXJjZSBzZXJ2ZXIgaXMgYW4gaW50ZWdyYXRpb24gYmV0d2VlbiBhIHVzZXIgcG9vbCBhbmQgYW4gQVBJLlxuICogRWFjaCByZXNvdXJjZSBzZXJ2ZXIgaGFzIGN1c3RvbSBzY29wZXMgdGhhdCB5b3UgbXVzdCBhY3RpdmF0ZSBpbiB5b3VyIGFwcCBjbGllbnQuXG4gKiBXaGVuIHlvdSBjb25maWd1cmUgYSByZXNvdXJjZSBzZXJ2ZXIsIHlvdXIgYXBwIGNhbiBnZW5lcmF0ZSBhY2Nlc3MgdG9rZW5zIHdpdGhcbiAqIE9BdXRoIHNjb3BlcyB0aGF0IGF1dGhvcml6ZSByZWFkIGFuZCB3cml0ZSBvcGVyYXRpb25zIHRvIGFuIEFQSSBzZXJ2ZXIuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqICAvLyBDcmVhdGUgYSBWeUNvZ25pdG9Qcm92aWRlclxuICogY29uc3QgdnlDb2duaXRvUHJvdmlkZXIgPSBuZXcgVnlDb2duaXRvUHJvdmlkZXIodGhpcywgJ015UHJvdmlkZXInLCB7XG4gKiAgIGVudmlyb25tZW50OiBWeUVudmlyb25tZW50LlRFU1QsXG4gKiB9KTtcbiAqXG4gKiBjb25zdCByZXNvdXJjZVNlcnZlciA9IG5ldyBDb2duaXRvUmVzb3VyY2VTZXJ2ZXIodGhpcywgJ0FwaVJlc291cmNlU2VydmVyJywge1xuICogICByZXNvdXJjZVNlcnZlclByb3ZpZGVyOiB2eUNvZ25pdG9Qcm92aWRlci5yZXNvdXJjZVNlcnZlclByb3ZpZGVyLFxuICogICBuYW1lOiAnbXktYXBpJyxcbiAqICAgaWRlbnRpZmllcjogJ2h0dHBzOi8vbXktYXBpLnZ5ZGV2LmlvJyxcbiAqICAgc2NvcGVzOiBbXG4gKiAgICAgeyBuYW1lOiAncmVhZCcsIGRlc2NyaXB0aW9uOiAnUmVhZCBhY2Nlc3MgdG8gdGhlIEFQSScgfSxcbiAqICAgICB7IG5hbWU6ICd3cml0ZScsIGRlc2NyaXB0aW9uOiAnV3JpdGUgYWNjZXNzIHRvIHRoZSBBUEknIH1cbiAqICAgXVxuICogfSk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGNsYXNzIENvZ25pdG9SZXNvdXJjZVNlcnZlciBleHRlbmRzIENvbnN0cnVjdCB7XG4gIC8qKlxuICAgKiBUaGUgaWRlbnRpZmllciBvZiB0aGUgcmVzb3VyY2Ugc2VydmVyXG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgaWRlbnRpZmllcjogc3RyaW5nXG5cbiAgLyoqXG4gICAqIFRoZSBuYW1lIG9mIHRoZSByZXNvdXJjZSBzZXJ2ZXJcbiAgICovXG4gIHB1YmxpYyByZWFkb25seSBuYW1lOiBzdHJpbmdcblxuICAvKipcbiAgICogVGhlIHVuZGVybHlpbmcgY3VzdG9tIHJlc291cmNlXG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgcmVzb3VyY2U6IGNkay5DdXN0b21SZXNvdXJjZVxuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBDb2duaXRvUmVzb3VyY2VTZXJ2ZXJQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMuaWRlbnRpZmllciA9IHByb3BzLmlkZW50aWZpZXJcbiAgICB0aGlzLm5hbWUgPSBwcm9wcy5uYW1lXG5cbiAgICB0aGlzLnJlc291cmNlID0gbmV3IGNkay5DdXN0b21SZXNvdXJjZSh0aGlzLCBcIlJlc291cmNlXCIsIHtcbiAgICAgIHNlcnZpY2VUb2tlbjogcHJvcHMucmVzb3VyY2VTZXJ2ZXJQcm92aWRlci5zZXJ2aWNlVG9rZW4sXG4gICAgICBwcm9wZXJ0aWVzOiB7XG4gICAgICAgIEVudmlyb25tZW50OiBwcm9wcy5yZXNvdXJjZVNlcnZlclByb3ZpZGVyLmVudmlyb25tZW50LFxuICAgICAgICBOYW1lOiBwcm9wcy5uYW1lLFxuICAgICAgICBJZGVudGlmaWVyOiBwcm9wcy5pZGVudGlmaWVyLFxuICAgICAgICBTY29wZXM6IHByb3BzLnNjb3Blcz8ubWFwKChzKSA9PiAoe1xuICAgICAgICAgIE5hbWU6IHMubmFtZSxcbiAgICAgICAgICBEZXNjcmlwdGlvbjogcy5kZXNjcmlwdGlvbixcbiAgICAgICAgfSkpLFxuICAgICAgfSxcbiAgICAgIHJlc291cmNlVHlwZTogXCJDdXN0b206OlZ5Q29nbml0b1Jlc291cmNlU2VydmVyXCIsXG4gICAgfSlcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgYSByZWZlcmVuY2UgdG8gYSBzY29wZSBpbiB0aGUgZm9ybWF0IGV4cGVjdGVkIGJ5IGFwcCBjbGllbnRzXG4gICAqIEBwYXJhbSBzY29wZU5hbWUgVGhlIG5hbWUgb2YgdGhlIHNjb3BlXG4gICAqIEByZXR1cm5zIFRoZSBmdWxsIHNjb3BlIGlkZW50aWZpZXIgKGUuZy4sICdodHRwczovL2FwaS52eWRldi5pby9yZWFkJylcbiAgICovXG4gIHB1YmxpYyBzY29wZUlkZW50aWZpZXIoc2NvcGVOYW1lOiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIHJldHVybiBgJHt0aGlzLmlkZW50aWZpZXJ9LyR7c2NvcGVOYW1lfWBcbiAgfVxufVxuIl19
package/lib/index.d.ts CHANGED
@@ -5,5 +5,5 @@
5
5
  * This library provides CDK equivalents to the Terraform provider `terraform-provider-vy`.
6
6
  */
7
7
  export * from "./cognito-app-client/cognito-app-client";
8
- export * from "./cognito-info";
9
8
  export * from "./cognito-resource-server/cognito-resource-server";
9
+ export * from "./vy-cognito-provider";
package/lib/index.js CHANGED
@@ -4,9 +4,7 @@
4
4
  *
5
5
  * This library provides CDK equivalents to the Terraform provider `terraform-provider-vy`.
6
6
  */
7
- // Resources
8
7
  export * from "./cognito-app-client/cognito-app-client";
9
- // Data Sources
10
- export * from "./cognito-info";
11
8
  export * from "./cognito-resource-server/cognito-resource-server";
12
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7O0dBS0c7QUFFSCxZQUFZO0FBQ1osY0FBYyx5Q0FBeUMsQ0FBQTtBQUN2RCxlQUFlO0FBQ2YsY0FBYyxnQkFBZ0IsQ0FBQTtBQUM5QixjQUFjLG1EQUFtRCxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAdnkvY2RrLXJlc291cmNlc1xuICogQ0RLIGNvbnN0cnVjdHMgZm9yIFZ5IGludGVybmFsIHNlcnZpY2VzXG4gKlxuICogVGhpcyBsaWJyYXJ5IHByb3ZpZGVzIENESyBlcXVpdmFsZW50cyB0byB0aGUgVGVycmFmb3JtIHByb3ZpZGVyIGB0ZXJyYWZvcm0tcHJvdmlkZXItdnlgLlxuICovXG5cbi8vIFJlc291cmNlc1xuZXhwb3J0ICogZnJvbSBcIi4vY29nbml0by1hcHAtY2xpZW50L2NvZ25pdG8tYXBwLWNsaWVudFwiXG4vLyBEYXRhIFNvdXJjZXNcbmV4cG9ydCAqIGZyb20gXCIuL2NvZ25pdG8taW5mb1wiXG5leHBvcnQgKiBmcm9tIFwiLi9jb2duaXRvLXJlc291cmNlLXNlcnZlci9jb2duaXRvLXJlc291cmNlLXNlcnZlclwiXG4iXX0=
9
+ export * from "./vy-cognito-provider";
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7O0dBS0c7QUFFSCxjQUFjLHlDQUF5QyxDQUFBO0FBQ3ZELGNBQWMsbURBQW1ELENBQUE7QUFDakUsY0FBYyx1QkFBdUIsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQHZ5L2Nkay1yZXNvdXJjZXNcbiAqIENESyBjb25zdHJ1Y3RzIGZvciBWeSBpbnRlcm5hbCBzZXJ2aWNlc1xuICpcbiAqIFRoaXMgbGlicmFyeSBwcm92aWRlcyBDREsgZXF1aXZhbGVudHMgdG8gdGhlIFRlcnJhZm9ybSBwcm92aWRlciBgdGVycmFmb3JtLXByb3ZpZGVyLXZ5YC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tIFwiLi9jb2duaXRvLWFwcC1jbGllbnQvY29nbml0by1hcHAtY2xpZW50XCJcbmV4cG9ydCAqIGZyb20gXCIuL2NvZ25pdG8tcmVzb3VyY2Utc2VydmVyL2NvZ25pdG8tcmVzb3VyY2Utc2VydmVyXCJcbmV4cG9ydCAqIGZyb20gXCIuL3Z5LWNvZ25pdG8tcHJvdmlkZXJcIlxuIl19
@@ -63,10 +63,26 @@ export interface ArtifactVersion {
63
63
  version: string;
64
64
  }
65
65
  export interface CognitoDetails {
66
+ /**
67
+ * The URL where users can authenticate
68
+ */
66
69
  authUrl: string;
70
+ /**
71
+ * The URL for the /.well-known/jwks.json endpoint
72
+ */
67
73
  jwksUrl: string;
74
+ /**
75
+ * The URL for the /.well-known/openid-configuration endpoint
76
+ */
68
77
  openIdUrl: string;
78
+ /**
79
+ * The URI for the issuer
80
+ */
69
81
  issuer: string;
82
+ /**
83
+ * The Vy Cognito Userpool Id
84
+ */
85
+ userPoolId: string;
70
86
  }
71
87
  /**
72
88
  * Custom Resource event types
@@ -19,4 +19,4 @@ export var VyEnvironment;
19
19
  */
20
20
  VyEnvironment["PROD"] = "prod";
21
21
  })(VyEnvironment || (VyEnvironment = {}));
22
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2hhcmVkL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUg7O0dBRUc7QUFDSCxNQUFNLENBQU4sSUFBWSxhQWVYO0FBZkQsV0FBWSxhQUFhO0lBQ3ZCOztPQUVHO0lBQ0gsOEJBQWEsQ0FBQTtJQUViOztPQUVHO0lBQ0gsZ0NBQWUsQ0FBQTtJQUVmOztPQUVHO0lBQ0gsOEJBQWEsQ0FBQTtBQUNmLENBQUMsRUFmVyxhQUFhLEtBQWIsYUFBYSxRQWV4QiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogU2hhcmVkIHR5cGUgZGVmaW5pdGlvbnMgZm9yIFZ5IGN1c3RvbSByZXNvdXJjZXNcbiAqL1xuXG4vKipcbiAqIFRoZSBkaWZmZXJlbnQgVnkgZW52aXJvbm1lbnRzXG4gKi9cbmV4cG9ydCBlbnVtIFZ5RW52aXJvbm1lbnQge1xuICAvKipcbiAgICogRGV2ZWxvcG1lbnQgZW52aXJvbm1lbnRcbiAgICovXG4gIFRFU1QgPSBcInRlc3RcIixcblxuICAvKipcbiAgICogUHJvZHVjdGlvbi1saWtlIGVudmlyb25tZW50XG4gICAqL1xuICBTVEFHRSA9IFwic3RhZ2VcIixcblxuICAvKipcbiAgICogUHJvZHVjdGlvbiBlbnZpcm9ubWVudFxuICAgKi9cbiAgUFJPRCA9IFwicHJvZFwiLFxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNjb3BlIHtcbiAgbmFtZTogc3RyaW5nXG4gIGRlc2NyaXB0aW9uOiBzdHJpbmdcbn1cblxuZXhwb3J0IGludGVyZmFjZSBSZXNvdXJjZVNlcnZlciB7XG4gIGlkZW50aWZpZXI6IHN0cmluZ1xuICBuYW1lOiBzdHJpbmdcbiAgc2NvcGVzPzogU2NvcGVbXVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFJlc291cmNlU2VydmVyVXBkYXRlUmVxdWVzdCB7XG4gIGlkZW50aWZpZXI6IHN0cmluZ1xuICBuYW1lOiBzdHJpbmdcbiAgc2NvcGVzPzogU2NvcGVbXVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIEFwcENsaWVudCB7XG4gIG5hbWU6IHN0cmluZ1xuICBzY29wZXM6IHN0cmluZ1tdXG4gIHR5cGU6IFwiZnJvbnRlbmRcIiB8IFwiYmFja2VuZFwiXG4gIGNhbGxiYWNrX3VybHM6IHN0cmluZ1tdXG4gIGxvZ291dF91cmxzOiBzdHJpbmdbXVxuICBnZW5lcmF0ZV9zZWNyZXQ/OiBib29sZWFuXG4gIGNsaWVudF9pZD86IHN0cmluZ1xuICBjbGllbnRfc2VjcmV0Pzogc3RyaW5nXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXBwQ2xpZW50VXBkYXRlUmVxdWVzdCB7XG4gIG5hbWU6IHN0cmluZ1xuICBzY29wZXM6IHN0cmluZ1tdXG4gIGNhbGxiYWNrX3VybHM6IHN0cmluZ1tdXG4gIGxvZ291dF91cmxzOiBzdHJpbmdbXVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIERlcGxveW1lbnRBY2NvdW50IHtcbiAgYWNjb3VudElkOiBzdHJpbmdcbiAgc2xhY2tDaGFubmVsOiBzdHJpbmdcbn1cblxuZXhwb3J0IGludGVyZmFjZSBFbnZpcm9ubWVudEFjY291bnQge1xuICBhY2NvdW50SWQ6IHN0cmluZ1xuICBvd25lckFjY291bnRJZDogc3RyaW5nXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXJ0aWZhY3RWZXJzaW9uIHtcbiAgdXJpOiBzdHJpbmdcbiAgc3RvcmU6IHN0cmluZ1xuICBwYXRoOiBzdHJpbmdcbiAgdmVyc2lvbjogc3RyaW5nXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29nbml0b0RldGFpbHMge1xuICBhdXRoVXJsOiBzdHJpbmdcbiAgandrc1VybDogc3RyaW5nXG4gIG9wZW5JZFVybDogc3RyaW5nXG4gIGlzc3Vlcjogc3RyaW5nXG59XG5cbi8qKlxuICogQ3VzdG9tIFJlc291cmNlIGV2ZW50IHR5cGVzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ3VzdG9tUmVzb3VyY2VSZXF1ZXN0IHtcbiAgUmVxdWVzdFR5cGU6IFwiQ3JlYXRlXCIgfCBcIlVwZGF0ZVwiIHwgXCJEZWxldGVcIlxuICBSZXF1ZXN0SWQ6IHN0cmluZ1xuICBSZXNwb25zZVVSTDogc3RyaW5nXG4gIFJlc291cmNlVHlwZTogc3RyaW5nXG4gIExvZ2ljYWxSZXNvdXJjZUlkOiBzdHJpbmdcbiAgU3RhY2tJZDogc3RyaW5nXG4gIFBoeXNpY2FsUmVzb3VyY2VJZD86IHN0cmluZ1xuICBSZXNvdXJjZVByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIGFueT5cbiAgT2xkUmVzb3VyY2VQcm9wZXJ0aWVzPzogUmVjb3JkPHN0cmluZywgYW55PlxufVxuXG5leHBvcnQgaW50ZXJmYWNlIEN1c3RvbVJlc291cmNlUmVzcG9uc2Uge1xuICBTdGF0dXM6IFwiU1VDQ0VTU1wiIHwgXCJGQUlMRURcIlxuICBSZWFzb24/OiBzdHJpbmdcbiAgUGh5c2ljYWxSZXNvdXJjZUlkOiBzdHJpbmdcbiAgU3RhY2tJZDogc3RyaW5nXG4gIFJlcXVlc3RJZDogc3RyaW5nXG4gIExvZ2ljYWxSZXNvdXJjZUlkOiBzdHJpbmdcbiAgRGF0YT86IFJlY29yZDxzdHJpbmcsIGFueT5cbn1cbiJdfQ==
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2hhcmVkL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUg7O0dBRUc7QUFDSCxNQUFNLENBQU4sSUFBWSxhQWVYO0FBZkQsV0FBWSxhQUFhO0lBQ3ZCOztPQUVHO0lBQ0gsOEJBQWEsQ0FBQTtJQUViOztPQUVHO0lBQ0gsZ0NBQWUsQ0FBQTtJQUVmOztPQUVHO0lBQ0gsOEJBQWEsQ0FBQTtBQUNmLENBQUMsRUFmVyxhQUFhLEtBQWIsYUFBYSxRQWV4QiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogU2hhcmVkIHR5cGUgZGVmaW5pdGlvbnMgZm9yIFZ5IGN1c3RvbSByZXNvdXJjZXNcbiAqL1xuXG4vKipcbiAqIFRoZSBkaWZmZXJlbnQgVnkgZW52aXJvbm1lbnRzXG4gKi9cbmV4cG9ydCBlbnVtIFZ5RW52aXJvbm1lbnQge1xuICAvKipcbiAgICogRGV2ZWxvcG1lbnQgZW52aXJvbm1lbnRcbiAgICovXG4gIFRFU1QgPSBcInRlc3RcIixcblxuICAvKipcbiAgICogUHJvZHVjdGlvbi1saWtlIGVudmlyb25tZW50XG4gICAqL1xuICBTVEFHRSA9IFwic3RhZ2VcIixcblxuICAvKipcbiAgICogUHJvZHVjdGlvbiBlbnZpcm9ubWVudFxuICAgKi9cbiAgUFJPRCA9IFwicHJvZFwiLFxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNjb3BlIHtcbiAgbmFtZTogc3RyaW5nXG4gIGRlc2NyaXB0aW9uOiBzdHJpbmdcbn1cblxuZXhwb3J0IGludGVyZmFjZSBSZXNvdXJjZVNlcnZlciB7XG4gIGlkZW50aWZpZXI6IHN0cmluZ1xuICBuYW1lOiBzdHJpbmdcbiAgc2NvcGVzPzogU2NvcGVbXVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFJlc291cmNlU2VydmVyVXBkYXRlUmVxdWVzdCB7XG4gIGlkZW50aWZpZXI6IHN0cmluZ1xuICBuYW1lOiBzdHJpbmdcbiAgc2NvcGVzPzogU2NvcGVbXVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIEFwcENsaWVudCB7XG4gIG5hbWU6IHN0cmluZ1xuICBzY29wZXM6IHN0cmluZ1tdXG4gIHR5cGU6IFwiZnJvbnRlbmRcIiB8IFwiYmFja2VuZFwiXG4gIGNhbGxiYWNrX3VybHM6IHN0cmluZ1tdXG4gIGxvZ291dF91cmxzOiBzdHJpbmdbXVxuICBnZW5lcmF0ZV9zZWNyZXQ/OiBib29sZWFuXG4gIGNsaWVudF9pZD86IHN0cmluZ1xuICBjbGllbnRfc2VjcmV0Pzogc3RyaW5nXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXBwQ2xpZW50VXBkYXRlUmVxdWVzdCB7XG4gIG5hbWU6IHN0cmluZ1xuICBzY29wZXM6IHN0cmluZ1tdXG4gIGNhbGxiYWNrX3VybHM6IHN0cmluZ1tdXG4gIGxvZ291dF91cmxzOiBzdHJpbmdbXVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIERlcGxveW1lbnRBY2NvdW50IHtcbiAgYWNjb3VudElkOiBzdHJpbmdcbiAgc2xhY2tDaGFubmVsOiBzdHJpbmdcbn1cblxuZXhwb3J0IGludGVyZmFjZSBFbnZpcm9ubWVudEFjY291bnQge1xuICBhY2NvdW50SWQ6IHN0cmluZ1xuICBvd25lckFjY291bnRJZDogc3RyaW5nXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXJ0aWZhY3RWZXJzaW9uIHtcbiAgdXJpOiBzdHJpbmdcbiAgc3RvcmU6IHN0cmluZ1xuICBwYXRoOiBzdHJpbmdcbiAgdmVyc2lvbjogc3RyaW5nXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29nbml0b0RldGFpbHMge1xuICAvKipcbiAgICogVGhlIFVSTCB3aGVyZSB1c2VycyBjYW4gYXV0aGVudGljYXRlXG4gICAqL1xuICBhdXRoVXJsOiBzdHJpbmdcblxuICAvKipcbiAgICogVGhlIFVSTCBmb3IgdGhlIC8ud2VsbC1rbm93bi9qd2tzLmpzb24gZW5kcG9pbnRcbiAgICovXG4gIGp3a3NVcmw6IHN0cmluZ1xuXG4gIC8qKlxuICAgKiBUaGUgVVJMIGZvciB0aGUgLy53ZWxsLWtub3duL29wZW5pZC1jb25maWd1cmF0aW9uIGVuZHBvaW50XG4gICAqL1xuICBvcGVuSWRVcmw6IHN0cmluZ1xuXG4gIC8qKlxuICAgKiBUaGUgVVJJIGZvciB0aGUgaXNzdWVyXG4gICAqL1xuICBpc3N1ZXI6IHN0cmluZ1xuXG4gIC8qKlxuICAgKiBUaGUgVnkgQ29nbml0byBVc2VycG9vbCBJZFxuICAgKi9cbiAgdXNlclBvb2xJZDogc3RyaW5nXG59XG5cbi8qKlxuICogQ3VzdG9tIFJlc291cmNlIGV2ZW50IHR5cGVzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ3VzdG9tUmVzb3VyY2VSZXF1ZXN0IHtcbiAgUmVxdWVzdFR5cGU6IFwiQ3JlYXRlXCIgfCBcIlVwZGF0ZVwiIHwgXCJEZWxldGVcIlxuICBSZXF1ZXN0SWQ6IHN0cmluZ1xuICBSZXNwb25zZVVSTDogc3RyaW5nXG4gIFJlc291cmNlVHlwZTogc3RyaW5nXG4gIExvZ2ljYWxSZXNvdXJjZUlkOiBzdHJpbmdcbiAgU3RhY2tJZDogc3RyaW5nXG4gIFBoeXNpY2FsUmVzb3VyY2VJZD86IHN0cmluZ1xuICBSZXNvdXJjZVByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIGFueT5cbiAgT2xkUmVzb3VyY2VQcm9wZXJ0aWVzPzogUmVjb3JkPHN0cmluZywgYW55PlxufVxuXG5leHBvcnQgaW50ZXJmYWNlIEN1c3RvbVJlc291cmNlUmVzcG9uc2Uge1xuICBTdGF0dXM6IFwiU1VDQ0VTU1wiIHwgXCJGQUlMRURcIlxuICBSZWFzb24/OiBzdHJpbmdcbiAgUGh5c2ljYWxSZXNvdXJjZUlkOiBzdHJpbmdcbiAgU3RhY2tJZDogc3RyaW5nXG4gIFJlcXVlc3RJZDogc3RyaW5nXG4gIExvZ2ljYWxSZXNvdXJjZUlkOiBzdHJpbmdcbiAgRGF0YT86IFJlY29yZDxzdHJpbmcsIGFueT5cbn1cbiJdfQ==
@@ -0,0 +1,36 @@
1
+ import * as logs from "aws-cdk-lib/aws-logs";
2
+ import { Construct } from "constructs";
3
+ import type { CognitoDetails, VyEnvironment } from "./shared/types";
4
+ export interface VyCognitoProviderProps {
5
+ /**
6
+ * The Vy Cognito environment to connect to (e.g., VyEnvironment.PROD, VyEnvironment.STAGE, VyEnvironment.TEST)
7
+ */
8
+ readonly environment: VyEnvironment;
9
+ /**
10
+ * Base domain for Cognito service
11
+ * @default 'cognito.vydev.io'
12
+ */
13
+ readonly cognitoBaseDomain?: string;
14
+ /**
15
+ * @default logs.RetentionDays.ONE_WEEK
16
+ */
17
+ readonly logsRetention?: logs.RetentionDays;
18
+ }
19
+ export interface AppClientProvider {
20
+ environment: VyEnvironment;
21
+ serviceToken: string;
22
+ auth_url: string;
23
+ }
24
+ export interface ResourceServerProvider {
25
+ environment: VyEnvironment;
26
+ serviceToken: string;
27
+ }
28
+ export declare class VyCognitoProvider extends Construct {
29
+ readonly environment: VyEnvironment;
30
+ readonly cognitoBaseDomain: string;
31
+ readonly details: CognitoDetails;
32
+ readonly appClientProvider: AppClientProvider;
33
+ readonly resourceServerProvider: ResourceServerProvider;
34
+ constructor(scope: Construct, id: string, props: VyCognitoProviderProps);
35
+ }
36
+ export declare function getCognitoDetailsForEnvironment(environment: VyEnvironment): CognitoDetails;
@@ -0,0 +1,131 @@
1
+ import { createRequire } from "node:module";
2
+ import path from "node:path";
3
+ import { fileURLToPath } from "node:url";
4
+ import * as cdk from "aws-cdk-lib";
5
+ import * as iam from "aws-cdk-lib/aws-iam";
6
+ import * as lambda from "aws-cdk-lib/aws-lambda";
7
+ import { NodejsFunction } from "aws-cdk-lib/aws-lambda-nodejs";
8
+ import * as logs from "aws-cdk-lib/aws-logs";
9
+ import * as cr from "aws-cdk-lib/custom-resources";
10
+ import { Construct } from "constructs";
11
+ const require = createRequire(import.meta.url);
12
+ const __filename = fileURLToPath(import.meta.url);
13
+ const __dirname = path.dirname(__filename);
14
+ export class VyCognitoProvider extends Construct {
15
+ environment;
16
+ cognitoBaseDomain;
17
+ details;
18
+ appClientProvider;
19
+ resourceServerProvider;
20
+ constructor(scope, id, props) {
21
+ super(scope, id);
22
+ this.environment = props.environment;
23
+ this.cognitoBaseDomain = props.cognitoBaseDomain ?? "cognito.vydev.io";
24
+ this.details = getCognitoDetailsForEnvironment(this.environment);
25
+ const appClientProvider = new LambdaProvider(this, "AppClientProvider", {
26
+ cognitoBaseDomain: this.cognitoBaseDomain,
27
+ runtime: lambda.Runtime.NODEJS_22_X,
28
+ handler: "handler",
29
+ entry: require.resolve(`${__dirname}/cognito-app-client/handler`),
30
+ logRetention: props.logsRetention,
31
+ });
32
+ this.appClientProvider = {
33
+ environment: this.environment,
34
+ serviceToken: appClientProvider.serviceToken,
35
+ auth_url: this.details.authUrl,
36
+ };
37
+ const resourceServerProvider = new LambdaProvider(this, "ResourceServerProvider", {
38
+ cognitoBaseDomain: this.cognitoBaseDomain,
39
+ runtime: lambda.Runtime.NODEJS_22_X,
40
+ handler: "handler",
41
+ entry: require.resolve(`${__dirname}/cognito-resource-server/handler`),
42
+ logRetention: props.logsRetention,
43
+ });
44
+ this.resourceServerProvider = {
45
+ environment: this.environment,
46
+ serviceToken: resourceServerProvider.serviceToken,
47
+ };
48
+ }
49
+ }
50
+ class LambdaProvider extends Construct {
51
+ /**
52
+ * The logGroup for the event handler lambda
53
+ */
54
+ lambdaLogGroup;
55
+ /**
56
+ * The logGroup for the custom resource provider
57
+ */
58
+ providerLogGroup;
59
+ /**
60
+ * The service token for the provider
61
+ */
62
+ serviceToken;
63
+ constructor(scope, id, props) {
64
+ super(scope, id);
65
+ this.lambdaLogGroup = new logs.LogGroup(this, "LambdaLogGroup", {
66
+ retention: props.logRetention ?? logs.RetentionDays.ONE_WEEK,
67
+ });
68
+ this.providerLogGroup = new logs.LogGroup(this, "ProviderLogGroup", {
69
+ retention: props.logRetention ?? logs.RetentionDays.ONE_WEEK,
70
+ });
71
+ const onEventHandler = new NodejsFunction(this, "OnEventHandler", {
72
+ runtime: props.runtime,
73
+ handler: props.handler,
74
+ entry: props.entry,
75
+ timeout: props.timeout ?? cdk.Duration.minutes(2),
76
+ memorySize: props.memorySize ?? 256,
77
+ logGroup: this.lambdaLogGroup,
78
+ environment: {
79
+ COGNITO_BASE_DOMAIN: props.cognitoBaseDomain,
80
+ },
81
+ bundling: {
82
+ minify: true,
83
+ sourceMap: true,
84
+ target: "es2020",
85
+ externalModules: ["aws-sdk"],
86
+ },
87
+ });
88
+ onEventHandler.addToRolePolicy(new iam.PolicyStatement({
89
+ effect: iam.Effect.ALLOW,
90
+ actions: ["execute-api:Invoke"],
91
+ resources: ["*"], // Can be scoped down if API Gateway ARN is known
92
+ }));
93
+ const provider = new cr.Provider(this, "Provider", {
94
+ onEventHandler,
95
+ logGroup: this.providerLogGroup,
96
+ });
97
+ this.serviceToken = provider.serviceToken;
98
+ }
99
+ }
100
+ export function getCognitoDetailsForEnvironment(environment) {
101
+ const config = envConfigs[environment];
102
+ if (!config) {
103
+ throw new Error(`Unknown environment: ${environment}. Valid values are: prod, stage, test`);
104
+ }
105
+ return config;
106
+ }
107
+ // Static config for each environment
108
+ const envConfigs = {
109
+ prod: {
110
+ authUrl: "https://auth.cognito.vydev.io",
111
+ jwksUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE/.well-known/jwks.json",
112
+ openIdUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE/.well-known/openid-configuration",
113
+ issuer: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE",
114
+ userPoolId: "eu-west-1_e6o46c1oE",
115
+ },
116
+ stage: {
117
+ authUrl: "https://auth.stage.cognito.vydev.io",
118
+ jwksUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW/.well-known/jwks.json",
119
+ openIdUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW/.well-known/openid-configuration",
120
+ issuer: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW",
121
+ userPoolId: "eu-west-1_AUYQ679zW",
122
+ },
123
+ test: {
124
+ authUrl: "https://auth.test.cognito.vydev.io",
125
+ jwksUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT/.well-known/jwks.json",
126
+ openIdUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT/.well-known/openid-configuration",
127
+ issuer: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT",
128
+ userPoolId: "eu-west-1_Z53b9AbeT",
129
+ },
130
+ };
131
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vy-cognito-provider.js","sourceRoot":"","sources":["../src/vy-cognito-provider.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAA;AAC3C,OAAO,IAAI,MAAM,WAAW,CAAA;AAC5B,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAA;AACxC,OAAO,KAAK,GAAG,MAAM,aAAa,CAAA;AAClC,OAAO,KAAK,GAAG,MAAM,qBAAqB,CAAA;AAC1C,OAAO,KAAK,MAAM,MAAM,wBAAwB,CAAA;AAChD,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAA;AAC9D,OAAO,KAAK,IAAI,MAAM,sBAAsB,CAAA;AAC5C,OAAO,KAAK,EAAE,MAAM,8BAA8B,CAAA;AAClD,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAA;AAGtC,MAAM,OAAO,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;AAC9C,MAAM,UAAU,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;AACjD,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAA;AA+B1C,MAAM,OAAO,iBAAkB,SAAQ,SAAS;IAC9B,WAAW,CAAe;IAC1B,iBAAiB,CAAQ;IACzB,OAAO,CAAgB;IACvB,iBAAiB,CAAmB;IACpC,sBAAsB,CAAwB;IAE9D,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA6B;QACrE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhB,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAA;QACpC,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC,iBAAiB,IAAI,kBAAkB,CAAA;QACtE,IAAI,CAAC,OAAO,GAAG,+BAA+B,CAAC,IAAI,CAAC,WAAW,CAAC,CAAA;QAEhE,MAAM,iBAAiB,GAAG,IAAI,cAAc,CAAC,IAAI,EAAE,mBAAmB,EAAE;YACtE,iBAAiB,EAAE,IAAI,CAAC,iBAAiB;YACzC,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,OAAO,EAAE,SAAS;YAClB,KAAK,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG,SAAS,6BAA6B,CAAC;YACjE,YAAY,EAAE,KAAK,CAAC,aAAa;SAClC,CAAC,CAAA;QAEF,IAAI,CAAC,iBAAiB,GAAG;YACvB,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,YAAY,EAAE,iBAAiB,CAAC,YAAY;YAC5C,QAAQ,EAAE,IAAI,CAAC,OAAO,CAAC,OAAO;SAC/B,CAAA;QAED,MAAM,sBAAsB,GAAG,IAAI,cAAc,CAC/C,IAAI,EACJ,wBAAwB,EACxB;YACE,iBAAiB,EAAE,IAAI,CAAC,iBAAiB;YACzC,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,OAAO,EAAE,SAAS;YAClB,KAAK,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG,SAAS,kCAAkC,CAAC;YACtE,YAAY,EAAE,KAAK,CAAC,aAAa;SAClC,CACF,CAAA;QAED,IAAI,CAAC,sBAAsB,GAAG;YAC5B,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,YAAY,EAAE,sBAAsB,CAAC,YAAY;SAClD,CAAA;IACH,CAAC;CACF;AAwBD,MAAM,cAAe,SAAQ,SAAS;IACpC;;OAEG;IACa,cAAc,CAAe;IAE7C;;OAEG;IACa,gBAAgB,CAAe;IAE/C;;OAEG;IACa,YAAY,CAAQ;IAEpC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA0B;QAClE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhB,IAAI,CAAC,cAAc,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,gBAAgB,EAAE;YAC9D,SAAS,EAAE,KAAK,CAAC,YAAY,IAAI,IAAI,CAAC,aAAa,CAAC,QAAQ;SAC7D,CAAC,CAAA;QAEF,IAAI,CAAC,gBAAgB,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAClE,SAAS,EAAE,KAAK,CAAC,YAAY,IAAI,IAAI,CAAC,aAAa,CAAC,QAAQ;SAC7D,CAAC,CAAA;QAEF,MAAM,cAAc,GAAG,IAAI,cAAc,CAAC,IAAI,EAAE,gBAAgB,EAAE;YAChE,OAAO,EAAE,KAAK,CAAC,OAAO;YACtB,OAAO,EAAE,KAAK,CAAC,OAAO;YACtB,KAAK,EAAE,KAAK,CAAC,KAAK;YAClB,OAAO,EAAE,KAAK,CAAC,OAAO,IAAI,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YACjD,UAAU,EAAE,KAAK,CAAC,UAAU,IAAI,GAAG;YACnC,QAAQ,EAAE,IAAI,CAAC,cAAc;YAC7B,WAAW,EAAE;gBACX,mBAAmB,EAAE,KAAK,CAAC,iBAAiB;aAC7C;YACD,QAAQ,EAAE;gBACR,MAAM,EAAE,IAAI;gBACZ,SAAS,EAAE,IAAI;gBACf,MAAM,EAAE,QAAQ;gBAChB,eAAe,EAAE,CAAC,SAAS,CAAC;aAC7B;SACF,CAAC,CAAA;QAEF,cAAc,CAAC,eAAe,CAC5B,IAAI,GAAG,CAAC,eAAe,CAAC;YACtB,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC,KAAK;YACxB,OAAO,EAAE,CAAC,oBAAoB,CAAC;YAC/B,SAAS,EAAE,CAAC,GAAG,CAAC,EAAE,iDAAiD;SACpE,CAAC,CACH,CAAA;QAED,MAAM,QAAQ,GAAG,IAAI,EAAE,CAAC,QAAQ,CAAC,IAAI,EAAE,UAAU,EAAE;YACjD,cAAc;YACd,QAAQ,EAAE,IAAI,CAAC,gBAAgB;SAChC,CAAC,CAAA;QAEF,IAAI,CAAC,YAAY,GAAG,QAAQ,CAAC,YAAY,CAAA;IAC3C,CAAC;CACF;AAED,MAAM,UAAU,+BAA+B,CAC7C,WAA0B;IAE1B,MAAM,MAAM,GAAG,UAAU,CAAC,WAAW,CAAC,CAAA;IAEtC,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,MAAM,IAAI,KAAK,CACb,wBAAwB,WAAW,uCAAuC,CAC3E,CAAA;IACH,CAAC;IAED,OAAO,MAAM,CAAA;AACf,CAAC;AAED,qCAAqC;AACrC,MAAM,UAAU,GAAmC;IACjD,IAAI,EAAE;QACJ,OAAO,EAAE,+BAA+B;QACxC,OAAO,EACL,uFAAuF;QACzF,SAAS,EACP,kGAAkG;QACpG,MAAM,EAAE,iEAAiE;QACzE,UAAU,EAAE,qBAAqB;KAClC;IACD,KAAK,EAAE;QACL,OAAO,EAAE,qCAAqC;QAC9C,OAAO,EACL,uFAAuF;QACzF,SAAS,EACP,kGAAkG;QACpG,MAAM,EAAE,iEAAiE;QACzE,UAAU,EAAE,qBAAqB;KAClC;IACD,IAAI,EAAE;QACJ,OAAO,EAAE,oCAAoC;QAC7C,OAAO,EACL,uFAAuF;QACzF,SAAS,EACP,kGAAkG;QACpG,MAAM,EAAE,iEAAiE;QACzE,UAAU,EAAE,qBAAqB;KAClC;CACF,CAAA","sourcesContent":["import { createRequire } from \"node:module\"\nimport path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport { NodejsFunction } from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as cr from \"aws-cdk-lib/custom-resources\"\nimport { Construct } from \"constructs\"\nimport type { CognitoDetails, VyEnvironment } from \"./shared/types\"\n\nconst require = createRequire(import.meta.url)\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\nexport interface VyCognitoProviderProps {\n  /**\n   * The Vy Cognito environment to connect to (e.g., VyEnvironment.PROD, VyEnvironment.STAGE, VyEnvironment.TEST)\n   */\n  readonly environment: VyEnvironment\n\n  /**\n   * Base domain for Cognito service\n   * @default 'cognito.vydev.io'\n   */\n  readonly cognitoBaseDomain?: string\n\n  /**\n   * @default logs.RetentionDays.ONE_WEEK\n   */\n  readonly logsRetention?: logs.RetentionDays\n}\n\nexport interface AppClientProvider {\n  environment: VyEnvironment\n  serviceToken: string\n  auth_url: string\n}\n\nexport interface ResourceServerProvider {\n  environment: VyEnvironment\n  serviceToken: string\n}\n\nexport class VyCognitoProvider extends Construct {\n  public readonly environment: VyEnvironment\n  public readonly cognitoBaseDomain: string\n  public readonly details: CognitoDetails\n  public readonly appClientProvider: AppClientProvider\n  public readonly resourceServerProvider: ResourceServerProvider\n\n  constructor(scope: Construct, id: string, props: VyCognitoProviderProps) {\n    super(scope, id)\n\n    this.environment = props.environment\n    this.cognitoBaseDomain = props.cognitoBaseDomain ?? \"cognito.vydev.io\"\n    this.details = getCognitoDetailsForEnvironment(this.environment)\n\n    const appClientProvider = new LambdaProvider(this, \"AppClientProvider\", {\n      cognitoBaseDomain: this.cognitoBaseDomain,\n      runtime: lambda.Runtime.NODEJS_22_X,\n      handler: \"handler\",\n      entry: require.resolve(`${__dirname}/cognito-app-client/handler`),\n      logRetention: props.logsRetention,\n    })\n\n    this.appClientProvider = {\n      environment: this.environment,\n      serviceToken: appClientProvider.serviceToken,\n      auth_url: this.details.authUrl,\n    }\n\n    const resourceServerProvider = new LambdaProvider(\n      this,\n      \"ResourceServerProvider\",\n      {\n        cognitoBaseDomain: this.cognitoBaseDomain,\n        runtime: lambda.Runtime.NODEJS_22_X,\n        handler: \"handler\",\n        entry: require.resolve(`${__dirname}/cognito-resource-server/handler`),\n        logRetention: props.logsRetention,\n      },\n    )\n\n    this.resourceServerProvider = {\n      environment: this.environment,\n      serviceToken: resourceServerProvider.serviceToken,\n    }\n  }\n}\n\ninterface LambdaProviderProps {\n  readonly cognitoBaseDomain: string\n  readonly runtime: lambda.Runtime\n  readonly handler: string\n  readonly entry: string\n\n  /**\n   * @default cdk.Duration.minutes(2)\n   */\n  readonly timeout?: cdk.Duration\n\n  /**\n   * @default 256\n   */\n  readonly memorySize?: number\n\n  /**\n   * @default logs.RetentionDays.ONE_WEEK\n   */\n  readonly logRetention?: logs.RetentionDays\n}\n\nclass LambdaProvider extends Construct {\n  /**\n   * The logGroup for the event handler lambda\n   */\n  public readonly lambdaLogGroup: logs.LogGroup\n\n  /**\n   * The logGroup for the custom resource provider\n   */\n  public readonly providerLogGroup: logs.LogGroup\n\n  /**\n   * The service token for the provider\n   */\n  public readonly serviceToken: string\n\n  constructor(scope: Construct, id: string, props: LambdaProviderProps) {\n    super(scope, id)\n\n    this.lambdaLogGroup = new logs.LogGroup(this, \"LambdaLogGroup\", {\n      retention: props.logRetention ?? logs.RetentionDays.ONE_WEEK,\n    })\n\n    this.providerLogGroup = new logs.LogGroup(this, \"ProviderLogGroup\", {\n      retention: props.logRetention ?? logs.RetentionDays.ONE_WEEK,\n    })\n\n    const onEventHandler = new NodejsFunction(this, \"OnEventHandler\", {\n      runtime: props.runtime,\n      handler: props.handler,\n      entry: props.entry,\n      timeout: props.timeout ?? cdk.Duration.minutes(2),\n      memorySize: props.memorySize ?? 256,\n      logGroup: this.lambdaLogGroup,\n      environment: {\n        COGNITO_BASE_DOMAIN: props.cognitoBaseDomain,\n      },\n      bundling: {\n        minify: true,\n        sourceMap: true,\n        target: \"es2020\",\n        externalModules: [\"aws-sdk\"],\n      },\n    })\n\n    onEventHandler.addToRolePolicy(\n      new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: [\"execute-api:Invoke\"],\n        resources: [\"*\"], // Can be scoped down if API Gateway ARN is known\n      }),\n    )\n\n    const provider = new cr.Provider(this, \"Provider\", {\n      onEventHandler,\n      logGroup: this.providerLogGroup,\n    })\n\n    this.serviceToken = provider.serviceToken\n  }\n}\n\nexport function getCognitoDetailsForEnvironment(\n  environment: VyEnvironment,\n): CognitoDetails {\n  const config = envConfigs[environment]\n\n  if (!config) {\n    throw new Error(\n      `Unknown environment: ${environment}. Valid values are: prod, stage, test`,\n    )\n  }\n\n  return config\n}\n\n// Static config for each environment\nconst envConfigs: Record<string, CognitoDetails> = {\n  prod: {\n    authUrl: \"https://auth.cognito.vydev.io\",\n    jwksUrl:\n      \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE/.well-known/jwks.json\",\n    openIdUrl:\n      \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE/.well-known/openid-configuration\",\n    issuer: \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE\",\n    userPoolId: \"eu-west-1_e6o46c1oE\",\n  },\n  stage: {\n    authUrl: \"https://auth.stage.cognito.vydev.io\",\n    jwksUrl:\n      \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW/.well-known/jwks.json\",\n    openIdUrl:\n      \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW/.well-known/openid-configuration\",\n    issuer: \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW\",\n    userPoolId: \"eu-west-1_AUYQ679zW\",\n  },\n  test: {\n    authUrl: \"https://auth.test.cognito.vydev.io\",\n    jwksUrl:\n      \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT/.well-known/jwks.json\",\n    openIdUrl:\n      \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT/.well-known/openid-configuration\",\n    issuer: \"https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT\",\n    userPoolId: \"eu-west-1_Z53b9AbeT\",\n  },\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@liflig/cdk-vy",
3
- "version": "1.0.0",
3
+ "version": "2.1.0",
4
4
  "description": "CDK constructs for the Vy internal services, based on nsbno/terraform-provider-vy",
5
5
  "keywords": [
6
6
  "aws-cdk",
@@ -47,7 +47,7 @@
47
47
  "@types/aws-lambda": "8.10.157",
48
48
  "@types/jest": "30.0.0",
49
49
  "@types/node": "24.9.2",
50
- "aws-cdk-lib": "2.221.0",
50
+ "aws-cdk-lib": "2.221.1",
51
51
  "constructs": "10.4.2",
52
52
  "jest": "30.2.0",
53
53
  "jest-cdk-snapshot": "2.3.6",
@@ -60,7 +60,7 @@
60
60
  },
61
61
  "dependencies": {
62
62
  "@aws-crypto/sha256-js": "5.2.0",
63
- "@aws-sdk/client-sts": "3.919.0",
63
+ "@aws-sdk/client-sts": "3.920.0",
64
64
  "@smithy/protocol-http": "5.3.3",
65
65
  "@smithy/signature-v4": "5.3.3"
66
66
  },
@@ -1,49 +0,0 @@
1
- /**
2
- * CDK Construct for Cognito Info data source
3
- */
4
- import { Construct } from "constructs";
5
- import type { VyEnvironment } from "./shared/types";
6
- export interface CognitoInfoProps {
7
- /**
8
- * The environment to get Cognito info for (e.g., VyEnvironment.PROD, VyEnvironment.STAGE, VyEnvironment.TEST)
9
- */
10
- readonly environment: VyEnvironment;
11
- }
12
- /**
13
- * Holds information about the centralized Cognito User Pool
14
- *
15
- * Use this to reference the shared Cognito User Pool in your CDK applications without
16
- * hardcoding values.
17
- *
18
- * @example
19
- * ```typescript
20
- * const cognitoInfo = new CognitoInfo(this, 'CognitoInfo', {
21
- * environment: VyEnvironment.PROD
22
- * });
23
- *
24
- * // Access authentication endpoints
25
- * console.log('Auth URL:', cognitoInfo.authUrl);
26
- * console.log('JWKS URL:', cognitoInfo.jwksUrl);
27
- * console.log('OpenID URL:', cognitoInfo.openIdUrl);
28
- * console.log('Issuer:', cognitoInfo.issuer);
29
- * ```
30
- */
31
- export declare class CognitoInfo extends Construct {
32
- /**
33
- * The URL where users can authenticate
34
- */
35
- readonly authUrl: string;
36
- /**
37
- * The URL for the /.well-known/jwks.json endpoint
38
- */
39
- readonly jwksUrl: string;
40
- /**
41
- * The URL for the /.well-known/openid-configuration endpoint
42
- */
43
- readonly openIdUrl: string;
44
- /**
45
- * The URI for the issuer
46
- */
47
- readonly issuer: string;
48
- constructor(scope: Construct, id: string, props: CognitoInfoProps);
49
- }
@@ -1,76 +0,0 @@
1
- /**
2
- * CDK Construct for Cognito Info data source
3
- */
4
- import { Construct } from "constructs";
5
- /**
6
- * Holds information about the centralized Cognito User Pool
7
- *
8
- * Use this to reference the shared Cognito User Pool in your CDK applications without
9
- * hardcoding values.
10
- *
11
- * @example
12
- * ```typescript
13
- * const cognitoInfo = new CognitoInfo(this, 'CognitoInfo', {
14
- * environment: VyEnvironment.PROD
15
- * });
16
- *
17
- * // Access authentication endpoints
18
- * console.log('Auth URL:', cognitoInfo.authUrl);
19
- * console.log('JWKS URL:', cognitoInfo.jwksUrl);
20
- * console.log('OpenID URL:', cognitoInfo.openIdUrl);
21
- * console.log('Issuer:', cognitoInfo.issuer);
22
- * ```
23
- */
24
- export class CognitoInfo extends Construct {
25
- /**
26
- * The URL where users can authenticate
27
- */
28
- authUrl;
29
- /**
30
- * The URL for the /.well-known/jwks.json endpoint
31
- */
32
- jwksUrl;
33
- /**
34
- * The URL for the /.well-known/openid-configuration endpoint
35
- */
36
- openIdUrl;
37
- /**
38
- * The URI for the issuer
39
- */
40
- issuer;
41
- constructor(scope, id, props) {
42
- super(scope, id);
43
- this.authUrl = getCognitoInfo(props.environment).authUrl;
44
- this.jwksUrl = getCognitoInfo(props.environment).jwksUrl;
45
- this.openIdUrl = getCognitoInfo(props.environment).openIdUrl;
46
- this.issuer = getCognitoInfo(props.environment).issuer;
47
- }
48
- }
49
- function getCognitoInfo(environment) {
50
- const envConfigs = {
51
- prod: {
52
- authUrl: "https://auth.cognito.vydev.io",
53
- jwksUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE/.well-known/jwks.json",
54
- openIdUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE/.well-known/openid-configuration",
55
- issuer: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_e6o46c1oE",
56
- },
57
- stage: {
58
- authUrl: "https://auth.stage.cognito.vydev.io",
59
- jwksUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW/.well-known/jwks.json",
60
- openIdUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW/.well-known/openid-configuration",
61
- issuer: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_AUYQ679zW",
62
- },
63
- test: {
64
- authUrl: "https://auth.test.cognito.vydev.io",
65
- jwksUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT/.well-known/jwks.json",
66
- openIdUrl: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT/.well-known/openid-configuration",
67
- issuer: "https://cognito-idp.eu-west-1.amazonaws.com/eu-west-1_Z53b9AbeT",
68
- },
69
- };
70
- const config = envConfigs[environment];
71
- if (!config) {
72
- throw new Error(`Unknown environment: ${environment}. Valid values are: prod, stage, test, dev`);
73
- }
74
- return config;
75
- }
76
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29nbml0by1pbmZvLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2NvZ25pdG8taW5mby50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxZQUFZLENBQUE7QUFVdEM7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQWtCRztBQUNILE1BQU0sT0FBTyxXQUFZLFNBQVEsU0FBUztJQUN4Qzs7T0FFRztJQUNhLE9BQU8sQ0FBUTtJQUUvQjs7T0FFRztJQUNhLE9BQU8sQ0FBUTtJQUUvQjs7T0FFRztJQUNhLFNBQVMsQ0FBUTtJQUVqQzs7T0FFRztJQUNhLE1BQU0sQ0FBUTtJQUU5QixZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQXVCO1FBQy9ELEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLE9BQU8sR0FBRyxjQUFjLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFDLE9BQU8sQ0FBQTtRQUN4RCxJQUFJLENBQUMsT0FBTyxHQUFHLGNBQWMsQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLENBQUMsT0FBTyxDQUFBO1FBQ3hELElBQUksQ0FBQyxTQUFTLEdBQUcsY0FBYyxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQyxTQUFTLENBQUE7UUFDNUQsSUFBSSxDQUFDLE1BQU0sR0FBRyxjQUFjLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFDLE1BQU0sQ0FBQTtJQUN4RCxDQUFDO0NBQ0Y7QUFFRCxTQUFTLGNBQWMsQ0FBQyxXQUFtQjtJQUN6QyxNQUFNLFVBQVUsR0FBbUM7UUFDakQsSUFBSSxFQUFFO1lBQ0osT0FBTyxFQUFFLCtCQUErQjtZQUN4QyxPQUFPLEVBQ0wsdUZBQXVGO1lBQ3pGLFNBQVMsRUFDUCxrR0FBa0c7WUFDcEcsTUFBTSxFQUFFLGlFQUFpRTtTQUMxRTtRQUNELEtBQUssRUFBRTtZQUNMLE9BQU8sRUFBRSxxQ0FBcUM7WUFDOUMsT0FBTyxFQUNMLHVGQUF1RjtZQUN6RixTQUFTLEVBQ1Asa0dBQWtHO1lBQ3BHLE1BQU0sRUFBRSxpRUFBaUU7U0FDMUU7UUFDRCxJQUFJLEVBQUU7WUFDSixPQUFPLEVBQUUsb0NBQW9DO1lBQzdDLE9BQU8sRUFDTCx1RkFBdUY7WUFDekYsU0FBUyxFQUNQLGtHQUFrRztZQUNwRyxNQUFNLEVBQUUsaUVBQWlFO1NBQzFFO0tBQ0YsQ0FBQTtJQUVELE1BQU0sTUFBTSxHQUFHLFVBQVUsQ0FBQyxXQUFXLENBQUMsQ0FBQTtJQUN0QyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDWixNQUFNLElBQUksS0FBSyxDQUNiLHdCQUF3QixXQUFXLDRDQUE0QyxDQUNoRixDQUFBO0lBQ0gsQ0FBQztJQUVELE9BQU8sTUFBTSxDQUFBO0FBQ2YsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ0RLIENvbnN0cnVjdCBmb3IgQ29nbml0byBJbmZvIGRhdGEgc291cmNlXG4gKi9cblxuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSBcImNvbnN0cnVjdHNcIlxuaW1wb3J0IHR5cGUgeyBDb2duaXRvRGV0YWlscywgVnlFbnZpcm9ubWVudCB9IGZyb20gXCIuL3NoYXJlZC90eXBlc1wiXG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29nbml0b0luZm9Qcm9wcyB7XG4gIC8qKlxuICAgKiBUaGUgZW52aXJvbm1lbnQgdG8gZ2V0IENvZ25pdG8gaW5mbyBmb3IgKGUuZy4sIFZ5RW52aXJvbm1lbnQuUFJPRCwgVnlFbnZpcm9ubWVudC5TVEFHRSwgVnlFbnZpcm9ubWVudC5URVNUKVxuICAgKi9cbiAgcmVhZG9ubHkgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnRcbn1cblxuLyoqXG4gKiBIb2xkcyBpbmZvcm1hdGlvbiBhYm91dCB0aGUgY2VudHJhbGl6ZWQgQ29nbml0byBVc2VyIFBvb2xcbiAqXG4gKiBVc2UgdGhpcyB0byByZWZlcmVuY2UgdGhlIHNoYXJlZCBDb2duaXRvIFVzZXIgUG9vbCBpbiB5b3VyIENESyBhcHBsaWNhdGlvbnMgd2l0aG91dFxuICogaGFyZGNvZGluZyB2YWx1ZXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGNvbnN0IGNvZ25pdG9JbmZvID0gbmV3IENvZ25pdG9JbmZvKHRoaXMsICdDb2duaXRvSW5mbycsIHtcbiAqICAgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnQuUFJPRFxuICogfSk7XG4gKlxuICogLy8gQWNjZXNzIGF1dGhlbnRpY2F0aW9uIGVuZHBvaW50c1xuICogY29uc29sZS5sb2coJ0F1dGggVVJMOicsIGNvZ25pdG9JbmZvLmF1dGhVcmwpO1xuICogY29uc29sZS5sb2coJ0pXS1MgVVJMOicsIGNvZ25pdG9JbmZvLmp3a3NVcmwpO1xuICogY29uc29sZS5sb2coJ09wZW5JRCBVUkw6JywgY29nbml0b0luZm8ub3BlbklkVXJsKTtcbiAqIGNvbnNvbGUubG9nKCdJc3N1ZXI6JywgY29nbml0b0luZm8uaXNzdWVyKTtcbiAqIGBgYFxuICovXG5leHBvcnQgY2xhc3MgQ29nbml0b0luZm8gZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICAvKipcbiAgICogVGhlIFVSTCB3aGVyZSB1c2VycyBjYW4gYXV0aGVudGljYXRlXG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgYXV0aFVybDogc3RyaW5nXG5cbiAgLyoqXG4gICAqIFRoZSBVUkwgZm9yIHRoZSAvLndlbGwta25vd24vandrcy5qc29uIGVuZHBvaW50XG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgandrc1VybDogc3RyaW5nXG5cbiAgLyoqXG4gICAqIFRoZSBVUkwgZm9yIHRoZSAvLndlbGwta25vd24vb3BlbmlkLWNvbmZpZ3VyYXRpb24gZW5kcG9pbnRcbiAgICovXG4gIHB1YmxpYyByZWFkb25seSBvcGVuSWRVcmw6IHN0cmluZ1xuXG4gIC8qKlxuICAgKiBUaGUgVVJJIGZvciB0aGUgaXNzdWVyXG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgaXNzdWVyOiBzdHJpbmdcblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogQ29nbml0b0luZm9Qcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMuYXV0aFVybCA9IGdldENvZ25pdG9JbmZvKHByb3BzLmVudmlyb25tZW50KS5hdXRoVXJsXG4gICAgdGhpcy5qd2tzVXJsID0gZ2V0Q29nbml0b0luZm8ocHJvcHMuZW52aXJvbm1lbnQpLmp3a3NVcmxcbiAgICB0aGlzLm9wZW5JZFVybCA9IGdldENvZ25pdG9JbmZvKHByb3BzLmVudmlyb25tZW50KS5vcGVuSWRVcmxcbiAgICB0aGlzLmlzc3VlciA9IGdldENvZ25pdG9JbmZvKHByb3BzLmVudmlyb25tZW50KS5pc3N1ZXJcbiAgfVxufVxuXG5mdW5jdGlvbiBnZXRDb2duaXRvSW5mbyhlbnZpcm9ubWVudDogc3RyaW5nKTogQ29nbml0b0RldGFpbHMge1xuICBjb25zdCBlbnZDb25maWdzOiBSZWNvcmQ8c3RyaW5nLCBDb2duaXRvRGV0YWlscz4gPSB7XG4gICAgcHJvZDoge1xuICAgICAgYXV0aFVybDogXCJodHRwczovL2F1dGguY29nbml0by52eWRldi5pb1wiLFxuICAgICAgandrc1VybDpcbiAgICAgICAgXCJodHRwczovL2NvZ25pdG8taWRwLmV1LXdlc3QtMS5hbWF6b25hd3MuY29tL2V1LXdlc3QtMV9lNm80NmMxb0UvLndlbGwta25vd24vandrcy5qc29uXCIsXG4gICAgICBvcGVuSWRVcmw6XG4gICAgICAgIFwiaHR0cHM6Ly9jb2duaXRvLWlkcC5ldS13ZXN0LTEuYW1hem9uYXdzLmNvbS9ldS13ZXN0LTFfZTZvNDZjMW9FLy53ZWxsLWtub3duL29wZW5pZC1jb25maWd1cmF0aW9uXCIsXG4gICAgICBpc3N1ZXI6IFwiaHR0cHM6Ly9jb2duaXRvLWlkcC5ldS13ZXN0LTEuYW1hem9uYXdzLmNvbS9ldS13ZXN0LTFfZTZvNDZjMW9FXCIsXG4gICAgfSxcbiAgICBzdGFnZToge1xuICAgICAgYXV0aFVybDogXCJodHRwczovL2F1dGguc3RhZ2UuY29nbml0by52eWRldi5pb1wiLFxuICAgICAgandrc1VybDpcbiAgICAgICAgXCJodHRwczovL2NvZ25pdG8taWRwLmV1LXdlc3QtMS5hbWF6b25hd3MuY29tL2V1LXdlc3QtMV9BVVlRNjc5elcvLndlbGwta25vd24vandrcy5qc29uXCIsXG4gICAgICBvcGVuSWRVcmw6XG4gICAgICAgIFwiaHR0cHM6Ly9jb2duaXRvLWlkcC5ldS13ZXN0LTEuYW1hem9uYXdzLmNvbS9ldS13ZXN0LTFfQVVZUTY3OXpXLy53ZWxsLWtub3duL29wZW5pZC1jb25maWd1cmF0aW9uXCIsXG4gICAgICBpc3N1ZXI6IFwiaHR0cHM6Ly9jb2duaXRvLWlkcC5ldS13ZXN0LTEuYW1hem9uYXdzLmNvbS9ldS13ZXN0LTFfQVVZUTY3OXpXXCIsXG4gICAgfSxcbiAgICB0ZXN0OiB7XG4gICAgICBhdXRoVXJsOiBcImh0dHBzOi8vYXV0aC50ZXN0LmNvZ25pdG8udnlkZXYuaW9cIixcbiAgICAgIGp3a3NVcmw6XG4gICAgICAgIFwiaHR0cHM6Ly9jb2duaXRvLWlkcC5ldS13ZXN0LTEuYW1hem9uYXdzLmNvbS9ldS13ZXN0LTFfWjUzYjlBYmVULy53ZWxsLWtub3duL2p3a3MuanNvblwiLFxuICAgICAgb3BlbklkVXJsOlxuICAgICAgICBcImh0dHBzOi8vY29nbml0by1pZHAuZXUtd2VzdC0xLmFtYXpvbmF3cy5jb20vZXUtd2VzdC0xX1o1M2I5QWJlVC8ud2VsbC1rbm93bi9vcGVuaWQtY29uZmlndXJhdGlvblwiLFxuICAgICAgaXNzdWVyOiBcImh0dHBzOi8vY29nbml0by1pZHAuZXUtd2VzdC0xLmFtYXpvbmF3cy5jb20vZXUtd2VzdC0xX1o1M2I5QWJlVFwiLFxuICAgIH0sXG4gIH1cblxuICBjb25zdCBjb25maWcgPSBlbnZDb25maWdzW2Vudmlyb25tZW50XVxuICBpZiAoIWNvbmZpZykge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgIGBVbmtub3duIGVudmlyb25tZW50OiAke2Vudmlyb25tZW50fS4gVmFsaWQgdmFsdWVzIGFyZTogcHJvZCwgc3RhZ2UsIHRlc3QsIGRldmAsXG4gICAgKVxuICB9XG5cbiAgcmV0dXJuIGNvbmZpZ1xufVxuIl19