@liflig/cdk-vy 2.1.0 → 2.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +2 -1
- package/lib/cognito-app-client/__test__/index.test.js +22 -1
- package/lib/cognito-app-client/cognito-app-client.d.ts +6 -0
- package/lib/cognito-app-client/cognito-app-client.js +5 -2
- package/lib/cognito-app-client/handler.js +9 -2
- package/lib/cognito-resource-server/handler.js +9 -2
- package/package.json +2 -2
package/README.md
CHANGED
|
@@ -55,6 +55,7 @@ const frontendClient = new CognitoAppClient(this, "FrontendClient", {
|
|
|
55
55
|
type: AppClientType.FRONTEND,
|
|
56
56
|
scopes: ["email", "openid", "profile"],
|
|
57
57
|
callbackUrls: ["https://my-app.vydev.io/auth/callback"],
|
|
58
|
-
logoutUrls: ["https://my-app.vydev.io/logout"]
|
|
58
|
+
logoutUrls: ["https://my-app.vydev.io/logout"],
|
|
59
|
+
authUrlPath: "/oauth2/token"
|
|
59
60
|
});
|
|
60
61
|
```
|
|
@@ -59,4 +59,25 @@ test("cognito frontend app client with custom secretName", () => {
|
|
|
59
59
|
ignoreAssets: true,
|
|
60
60
|
});
|
|
61
61
|
});
|
|
62
|
-
|
|
62
|
+
test("cognito frontend app client with custom authUrlPath", () => {
|
|
63
|
+
const app = new App();
|
|
64
|
+
const stack = new Stack(app, "Stack");
|
|
65
|
+
const mockAppClientProvider = {
|
|
66
|
+
environment: VyEnvironment.TEST,
|
|
67
|
+
serviceToken: "serviceToken",
|
|
68
|
+
auth_url: "auth_url",
|
|
69
|
+
};
|
|
70
|
+
new CognitoAppClient(stack, "CognitoAppClient", {
|
|
71
|
+
appClientProvider: mockAppClientProvider,
|
|
72
|
+
name: "my-web-app",
|
|
73
|
+
type: AppClientType.FRONTEND,
|
|
74
|
+
scopes: ["email", "openid", "profile"],
|
|
75
|
+
callbackUrls: ["https://my-app.vydev.io/auth/callback"],
|
|
76
|
+
logoutUrls: ["https://my-app.vydev.io/logout"],
|
|
77
|
+
authUrlPath: "/auth/custom-path",
|
|
78
|
+
});
|
|
79
|
+
expect(stack).toMatchCdkSnapshot({
|
|
80
|
+
ignoreAssets: true,
|
|
81
|
+
});
|
|
82
|
+
});
|
|
83
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXgudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb2duaXRvLWFwcC1jbGllbnQvX190ZXN0X18vaW5kZXgudGVzdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsR0FBRyxFQUFFLEtBQUssRUFBRSxNQUFNLGFBQWEsQ0FBQTtBQUN4QyxPQUFPLG1CQUFtQixDQUFBO0FBQzFCLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQTtBQUVsRCxPQUFPLEVBQUUsYUFBYSxFQUFFLGdCQUFnQixFQUFFLE1BQU0sdUJBQXVCLENBQUE7QUFFdkUsSUFBSSxDQUFDLDZCQUE2QixFQUFFLEdBQUcsRUFBRTtJQUN2QyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxNQUFNLHFCQUFxQixHQUFzQjtRQUMvQyxXQUFXLEVBQUUsYUFBYSxDQUFDLElBQUk7UUFDL0IsWUFBWSxFQUFFLGNBQWM7UUFDNUIsUUFBUSxFQUFFLFVBQVU7S0FDckIsQ0FBQTtJQUVELElBQUksZ0JBQWdCLENBQUMsS0FBSyxFQUFFLGtCQUFrQixFQUFFO1FBQzlDLGlCQUFpQixFQUFFLHFCQUFxQjtRQUN4QyxJQUFJLEVBQUUsWUFBWTtRQUNsQixJQUFJLEVBQUUsYUFBYSxDQUFDLFFBQVE7UUFDNUIsTUFBTSxFQUFFLENBQUMsT0FBTyxFQUFFLFFBQVEsRUFBRSxTQUFTLENBQUM7UUFDdEMsWUFBWSxFQUFFLENBQUMsdUNBQXVDLENBQUM7UUFDdkQsVUFBVSxFQUFFLENBQUMsZ0NBQWdDLENBQUM7S0FDL0MsQ0FBQyxDQUFBO0lBRUYsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLGtCQUFrQixDQUFDO1FBQy9CLFlBQVksRUFBRSxJQUFJO0tBQ25CLENBQUMsQ0FBQTtBQUNKLENBQUMsQ0FBQyxDQUFBO0FBRUYsSUFBSSxDQUFDLDRCQUE0QixFQUFFLEdBQUcsRUFBRTtJQUN0QyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFBO0lBQ3JCLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLEdBQUcsRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUVyQyxNQUFNLHFCQUFxQixHQUFzQjtRQUMvQyxXQUFXLEVBQUUsYUFBYSxDQUFDLEtBQUs7UUFDaEMsWUFBWSxFQUFFLGNBQWM7UUFDNUIsUUFBUSxFQUFFLFVBQVU7S0FDckIsQ0FBQTtJQUVELElBQUksZ0JBQWdCLENBQUMsS0FBSyxFQUFFLGtCQUFrQixFQUFFO1FBQzlDLGlCQUFpQixFQUFFLHFCQUFxQjtRQUN4QyxJQUFJLEVBQUUsb0JBQW9CO1FBQzFCLElBQUksRUFBRSxhQUFhLENBQUMsT0FBTztRQUMzQixNQUFNLEVBQUUsQ0FBQywyQkFBMkIsRUFBRSw0QkFBNEIsQ0FBQztLQUNwRSxDQUFDLENBQUE7SUFFRixNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsa0JBQWtCLENBQUM7UUFDL0IsWUFBWSxFQUFFLElBQUk7S0FDbkIsQ0FBQyxDQUFBO0FBQ0osQ0FBQyxDQUFDLENBQUE7QUFFRixJQUFJLENBQUMsb0RBQW9ELEVBQUUsR0FBRyxFQUFFO0lBQzlELE1BQU0sR0FBRyxHQUFHLElBQUksR0FBRyxFQUFFLENBQUE7SUFDckIsTUFBTSxLQUFLLEdBQUcsSUFBSSxLQUFLLENBQUMsR0FBRyxFQUFFLE9BQU8sQ0FBQyxDQUFBO0lBRXJDLE1BQU0scUJBQXFCLEdBQXNCO1FBQy9DLFdBQVcsRUFBRSxhQUFhLENBQUMsSUFBSTtRQUMvQixZQUFZLEVBQUUsY0FBYztRQUM1QixRQUFRLEVBQUUsVUFBVTtLQUNyQixDQUFBO0lBRUQsSUFBSSxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsa0JBQWtCLEVBQUU7UUFDOUMsaUJBQWlCLEVBQUUscUJBQXFCO1FBQ3hDLElBQUksRUFBRSxvQkFBb0I7UUFDMUIsSUFBSSxFQUFFLGFBQWEsQ0FBQyxPQUFPO1FBQzNCLE1BQU0sRUFBRSxDQUFDLDJCQUEyQixFQUFFLDRCQUE0QixDQUFDO1FBQ25FLFVBQVUsRUFBRSxXQUFXO0tBQ3hCLENBQUMsQ0FBQTtJQUVGLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQztRQUMvQixZQUFZLEVBQUUsSUFBSTtLQUNuQixDQUFDLENBQUE7QUFDSixDQUFDLENBQUMsQ0FBQTtBQUVGLElBQUksQ0FBQyxxREFBcUQsRUFBRSxHQUFHLEVBQUU7SUFDL0QsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLEVBQUUsQ0FBQTtJQUNyQixNQUFNLEtBQUssR0FBRyxJQUFJLEtBQUssQ0FBQyxHQUFHLEVBQUUsT0FBTyxDQUFDLENBQUE7SUFFckMsTUFBTSxxQkFBcUIsR0FBc0I7UUFDL0MsV0FBVyxFQUFFLGFBQWEsQ0FBQyxJQUFJO1FBQy9CLFlBQVksRUFBRSxjQUFjO1FBQzVCLFFBQVEsRUFBRSxVQUFVO0tBQ3JCLENBQUE7SUFFRCxJQUFJLGdCQUFnQixDQUFDLEtBQUssRUFBRSxrQkFBa0IsRUFBRTtRQUM5QyxpQkFBaUIsRUFBRSxxQkFBcUI7UUFDeEMsSUFBSSxFQUFFLFlBQVk7UUFDbEIsSUFBSSxFQUFFLGFBQWEsQ0FBQyxRQUFRO1FBQzVCLE1BQU0sRUFBRSxDQUFDLE9BQU8sRUFBRSxRQUFRLEVBQUUsU0FBUyxDQUFDO1FBQ3RDLFlBQVksRUFBRSxDQUFDLHVDQUF1QyxDQUFDO1FBQ3ZELFVBQVUsRUFBRSxDQUFDLGdDQUFnQyxDQUFDO1FBQzlDLFdBQVcsRUFBRSxtQkFBbUI7S0FDakMsQ0FBQyxDQUFBO0lBRUYsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLGtCQUFrQixDQUFDO1FBQy9CLFlBQVksRUFBRSxJQUFJO0tBQ25CLENBQUMsQ0FBQTtBQUNKLENBQUMsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQXBwLCBTdGFjayB9IGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgXCJqZXN0LWNkay1zbmFwc2hvdFwiXG5pbXBvcnQgeyBWeUVudmlyb25tZW50IH0gZnJvbSBcIi4uLy4uL3NoYXJlZC90eXBlc1wiXG5pbXBvcnQgdHlwZSB7IEFwcENsaWVudFByb3ZpZGVyIH0gZnJvbSBcIi4uLy4uL3Z5LWNvZ25pdG8tcHJvdmlkZXJcIlxuaW1wb3J0IHsgQXBwQ2xpZW50VHlwZSwgQ29nbml0b0FwcENsaWVudCB9IGZyb20gXCIuLi9jb2duaXRvLWFwcC1jbGllbnRcIlxuXG50ZXN0KFwiY29nbml0byBmcm9udGVuZCBhcHAgY2xpZW50XCIsICgpID0+IHtcbiAgY29uc3QgYXBwID0gbmV3IEFwcCgpXG4gIGNvbnN0IHN0YWNrID0gbmV3IFN0YWNrKGFwcCwgXCJTdGFja1wiKVxuXG4gIGNvbnN0IG1vY2tBcHBDbGllbnRQcm92aWRlcjogQXBwQ2xpZW50UHJvdmlkZXIgPSB7XG4gICAgZW52aXJvbm1lbnQ6IFZ5RW52aXJvbm1lbnQuVEVTVCxcbiAgICBzZXJ2aWNlVG9rZW46IFwic2VydmljZVRva2VuXCIsXG4gICAgYXV0aF91cmw6IFwiYXV0aF91cmxcIixcbiAgfVxuXG4gIG5ldyBDb2duaXRvQXBwQ2xpZW50KHN0YWNrLCBcIkNvZ25pdG9BcHBDbGllbnRcIiwge1xuICAgIGFwcENsaWVudFByb3ZpZGVyOiBtb2NrQXBwQ2xpZW50UHJvdmlkZXIsXG4gICAgbmFtZTogXCJteS13ZWItYXBwXCIsXG4gICAgdHlwZTogQXBwQ2xpZW50VHlwZS5GUk9OVEVORCxcbiAgICBzY29wZXM6IFtcImVtYWlsXCIsIFwib3BlbmlkXCIsIFwicHJvZmlsZVwiXSxcbiAgICBjYWxsYmFja1VybHM6IFtcImh0dHBzOi8vbXktYXBwLnZ5ZGV2LmlvL2F1dGgvY2FsbGJhY2tcIl0sXG4gICAgbG9nb3V0VXJsczogW1wiaHR0cHM6Ly9teS1hcHAudnlkZXYuaW8vbG9nb3V0XCJdLFxuICB9KVxuXG4gIGV4cGVjdChzdGFjaykudG9NYXRjaENka1NuYXBzaG90KHtcbiAgICBpZ25vcmVBc3NldHM6IHRydWUsXG4gIH0pXG59KVxuXG50ZXN0KFwiY29nbml0byBiYWNrZW5kIGFwcCBjbGllbnRcIiwgKCkgPT4ge1xuICBjb25zdCBhcHAgPSBuZXcgQXBwKClcbiAgY29uc3Qgc3RhY2sgPSBuZXcgU3RhY2soYXBwLCBcIlN0YWNrXCIpXG5cbiAgY29uc3QgbW9ja0FwcENsaWVudFByb3ZpZGVyOiBBcHBDbGllbnRQcm92aWRlciA9IHtcbiAgICBlbnZpcm9ubWVudDogVnlFbnZpcm9ubWVudC5TVEFHRSxcbiAgICBzZXJ2aWNlVG9rZW46IFwic2VydmljZVRva2VuXCIsXG4gICAgYXV0aF91cmw6IFwiYXV0aF91cmxcIixcbiAgfVxuXG4gIG5ldyBDb2duaXRvQXBwQ2xpZW50KHN0YWNrLCBcIkNvZ25pdG9BcHBDbGllbnRcIiwge1xuICAgIGFwcENsaWVudFByb3ZpZGVyOiBtb2NrQXBwQ2xpZW50UHJvdmlkZXIsXG4gICAgbmFtZTogXCJteS1iYWNrZW5kLXNlcnZpY2VcIixcbiAgICB0eXBlOiBBcHBDbGllbnRUeXBlLkJBQ0tFTkQsXG4gICAgc2NvcGVzOiBbXCJodHRwczovL2FwaS52eWRldi5pby9yZWFkXCIsIFwiaHR0cHM6Ly9hcGkudnlkZXYuaW8vd3JpdGVcIl0sXG4gIH0pXG5cbiAgZXhwZWN0KHN0YWNrKS50b01hdGNoQ2RrU25hcHNob3Qoe1xuICAgIGlnbm9yZUFzc2V0czogdHJ1ZSxcbiAgfSlcbn0pXG5cbnRlc3QoXCJjb2duaXRvIGZyb250ZW5kIGFwcCBjbGllbnQgd2l0aCBjdXN0b20gc2VjcmV0TmFtZVwiLCAoKSA9PiB7XG4gIGNvbnN0IGFwcCA9IG5ldyBBcHAoKVxuICBjb25zdCBzdGFjayA9IG5ldyBTdGFjayhhcHAsIFwiU3RhY2tcIilcblxuICBjb25zdCBtb2NrQXBwQ2xpZW50UHJvdmlkZXI6IEFwcENsaWVudFByb3ZpZGVyID0ge1xuICAgIGVudmlyb25tZW50OiBWeUVudmlyb25tZW50LlBST0QsXG4gICAgc2VydmljZVRva2VuOiBcInNlcnZpY2VUb2tlblwiLFxuICAgIGF1dGhfdXJsOiBcImF1dGhfdXJsXCIsXG4gIH1cblxuICBuZXcgQ29nbml0b0FwcENsaWVudChzdGFjaywgXCJDb2duaXRvQXBwQ2xpZW50XCIsIHtcbiAgICBhcHBDbGllbnRQcm92aWRlcjogbW9ja0FwcENsaWVudFByb3ZpZGVyLFxuICAgIG5hbWU6IFwibXktYmFja2VuZC1zZXJ2aWNlXCIsXG4gICAgdHlwZTogQXBwQ2xpZW50VHlwZS5CQUNLRU5ELFxuICAgIHNjb3BlczogW1wiaHR0cHM6Ly9hcGkudnlkZXYuaW8vcmVhZFwiLCBcImh0dHBzOi8vYXBpLnZ5ZGV2LmlvL3dyaXRlXCJdLFxuICAgIHNlY3JldE5hbWU6IFwibXktc2VjcmV0XCIsXG4gIH0pXG5cbiAgZXhwZWN0KHN0YWNrKS50b01hdGNoQ2RrU25hcHNob3Qoe1xuICAgIGlnbm9yZUFzc2V0czogdHJ1ZSxcbiAgfSlcbn0pXG5cbnRlc3QoXCJjb2duaXRvIGZyb250ZW5kIGFwcCBjbGllbnQgd2l0aCBjdXN0b20gYXV0aFVybFBhdGhcIiwgKCkgPT4ge1xuICBjb25zdCBhcHAgPSBuZXcgQXBwKClcbiAgY29uc3Qgc3RhY2sgPSBuZXcgU3RhY2soYXBwLCBcIlN0YWNrXCIpXG5cbiAgY29uc3QgbW9ja0FwcENsaWVudFByb3ZpZGVyOiBBcHBDbGllbnRQcm92aWRlciA9IHtcbiAgICBlbnZpcm9ubWVudDogVnlFbnZpcm9ubWVudC5URVNULFxuICAgIHNlcnZpY2VUb2tlbjogXCJzZXJ2aWNlVG9rZW5cIixcbiAgICBhdXRoX3VybDogXCJhdXRoX3VybFwiLFxuICB9XG5cbiAgbmV3IENvZ25pdG9BcHBDbGllbnQoc3RhY2ssIFwiQ29nbml0b0FwcENsaWVudFwiLCB7XG4gICAgYXBwQ2xpZW50UHJvdmlkZXI6IG1vY2tBcHBDbGllbnRQcm92aWRlcixcbiAgICBuYW1lOiBcIm15LXdlYi1hcHBcIixcbiAgICB0eXBlOiBBcHBDbGllbnRUeXBlLkZST05URU5ELFxuICAgIHNjb3BlczogW1wiZW1haWxcIiwgXCJvcGVuaWRcIiwgXCJwcm9maWxlXCJdLFxuICAgIGNhbGxiYWNrVXJsczogW1wiaHR0cHM6Ly9teS1hcHAudnlkZXYuaW8vYXV0aC9jYWxsYmFja1wiXSxcbiAgICBsb2dvdXRVcmxzOiBbXCJodHRwczovL215LWFwcC52eWRldi5pby9sb2dvdXRcIl0sXG4gICAgYXV0aFVybFBhdGg6IFwiL2F1dGgvY3VzdG9tLXBhdGhcIixcbiAgfSlcblxuICBleHBlY3Qoc3RhY2spLnRvTWF0Y2hDZGtTbmFwc2hvdCh7XG4gICAgaWdub3JlQXNzZXRzOiB0cnVlLFxuICB9KVxufSlcbiJdfQ==
|
|
@@ -84,6 +84,11 @@ export interface CognitoAppClientProps {
|
|
|
84
84
|
* @default logs.RetentionDays.ONE_WEEK
|
|
85
85
|
*/
|
|
86
86
|
readonly logsRetention?: logs.RetentionDays;
|
|
87
|
+
/**
|
|
88
|
+
* @default - No path
|
|
89
|
+
* Append a path to the auth_url for frontend clients.
|
|
90
|
+
*/
|
|
91
|
+
readonly authUrlPath?: string;
|
|
87
92
|
}
|
|
88
93
|
/**
|
|
89
94
|
* A Cognito App Client managed through Vy's central Cognito service
|
|
@@ -130,6 +135,7 @@ export interface CognitoAppClientProps {
|
|
|
130
135
|
* scopes: ['email', 'openid', 'profile'],
|
|
131
136
|
* callbackUrls: ['https://my-app.vydev.io/auth/callback'],
|
|
132
137
|
* logoutUrls: ['https://my-app.vydev.io/logout']
|
|
138
|
+
* authUrlPath: '/oauth2/token'
|
|
133
139
|
* });
|
|
134
140
|
* ```
|
|
135
141
|
*/
|
|
@@ -74,6 +74,7 @@ export var AppClientType;
|
|
|
74
74
|
* scopes: ['email', 'openid', 'profile'],
|
|
75
75
|
* callbackUrls: ['https://my-app.vydev.io/auth/callback'],
|
|
76
76
|
* logoutUrls: ['https://my-app.vydev.io/logout']
|
|
77
|
+
* authUrlPath: '/oauth2/token'
|
|
77
78
|
* });
|
|
78
79
|
* ```
|
|
79
80
|
*/
|
|
@@ -127,7 +128,9 @@ export class CognitoAppClient extends Construct {
|
|
|
127
128
|
this.clientId = client_id;
|
|
128
129
|
if (generateSecret) {
|
|
129
130
|
const client_secret = this.resource.getAttString("ClientSecret");
|
|
130
|
-
const auth_url = props.
|
|
131
|
+
const auth_url = (props.authUrlPath
|
|
132
|
+
? props.appClientProvider.auth_url.concat(props.authUrlPath)
|
|
133
|
+
: props.appClientProvider.auth_url).replaceAll("//", "/");
|
|
131
134
|
// For convenience, store auth_url, client_id, and client_secret in a single JSON secret
|
|
132
135
|
const secretString = JSON.stringify({
|
|
133
136
|
auth_url,
|
|
@@ -153,4 +156,4 @@ export class CognitoAppClient extends Construct {
|
|
|
153
156
|
return this.clientSecretSecret.grantRead(grantee);
|
|
154
157
|
}
|
|
155
158
|
}
|
|
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"]}
|
|
159
|
+
//# 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   * @default - No path\n   * Append a path to the auth_url for frontend clients.\n   */\n  readonly authUrlPath?: string\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 *   authUrlPath: '/oauth2/token'\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\n      const auth_url = (\n        props.authUrlPath\n          ? props.appClientProvider.auth_url.concat(props.authUrlPath)\n          : props.appClientProvider.auth_url\n      ).replaceAll(\"//\", \"/\")\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"]}
|
|
@@ -48,7 +48,14 @@ async function deleteAppClient(baseUrl, name) {
|
|
|
48
48
|
path: `/app-clients/${encodedName}`,
|
|
49
49
|
});
|
|
50
50
|
if (response.statusCode !== 200) {
|
|
51
|
-
|
|
51
|
+
const message = `Could not delete resource: ${response.statusCode} - ${response.body}`;
|
|
52
|
+
if (response.statusCode === 404) {
|
|
53
|
+
// Allow soft fail to avoid ROLLBACK_FAILED status
|
|
54
|
+
console.warn(message);
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
throw new Error(message);
|
|
58
|
+
}
|
|
52
59
|
}
|
|
53
60
|
}
|
|
54
61
|
export async function handler(event) {
|
|
@@ -111,4 +118,4 @@ export async function handler(event) {
|
|
|
111
118
|
return createFailureResponse(event.PhysicalResourceId || props.Name || "unknown", handleError(error));
|
|
112
119
|
}
|
|
113
120
|
}
|
|
114
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../src/cognito-app-client/handler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Lambda handler for CognitoAppClient custom resource\n */\n\nimport {\n  createFailureResponse,\n  createSuccessResponse,\n  handleError,\n} from \"../shared/custom-resource-handler\"\nimport { createUrlFromEnvironment, signedRequest } from \"../shared/sigv4-client\"\nimport type {\n  AppClient,\n  AppClientUpdateRequest,\n  CustomResourceRequest,\n  CustomResourceResponse,\n} from \"../shared/types\"\n\nconst COGNITO_BASE_DOMAIN =\n  process.env.COGNITO_BASE_DOMAIN || \"cognito.vydev.io\"\n\ninterface AppClientProperties {\n  Environment: string\n  Name: string\n  Type: \"frontend\" | \"backend\"\n  Scopes?: string[]\n  CallbackUrls?: string[]\n  LogoutUrls?: string[]\n  GenerateSecret?: boolean\n}\n\nasync function createAppClient(\n  baseUrl: string,\n  client: AppClient,\n): Promise<AppClient> {\n  const response = await signedRequest({\n    method: \"POST\",\n    hostname: baseUrl,\n    path: \"/app-clients\",\n    body: JSON.stringify(client),\n  })\n\n  if (response.statusCode !== 201) {\n    throw new Error(\n      `Could not create resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n\n  return JSON.parse(response.body)\n}\n\nasync function readAppClient(\n  baseUrl: string,\n  name: string,\n): Promise<AppClient> {\n  const encodedName = encodeURIComponent(name)\n  const response = await signedRequest({\n    method: \"GET\",\n    hostname: baseUrl,\n    path: `/app-clients/${encodedName}`,\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not read resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n\n  return JSON.parse(response.body)\n}\n\nasync function updateAppClient(\n  baseUrl: string,\n  update: AppClientUpdateRequest,\n): Promise<void> {\n  const encodedName = encodeURIComponent(update.name)\n  const response = await signedRequest({\n    method: \"PUT\",\n    hostname: baseUrl,\n    path: `/app-clients/${encodedName}`,\n    body: JSON.stringify(update),\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not update resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n}\n\nasync function deleteAppClient(baseUrl: string, name: string): Promise<void> {\n  const encodedName = encodeURIComponent(name)\n  const response = await signedRequest({\n    method: \"DELETE\",\n    hostname: baseUrl,\n    path: `/app-clients/${encodedName}`,\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not delete resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n}\n\nexport async function handler(\n  event: CustomResourceRequest,\n): Promise<CustomResourceResponse> {\n  const props = event.ResourceProperties as AppClientProperties\n  const baseUrl = createUrlFromEnvironment(\n    COGNITO_BASE_DOMAIN,\n    \"delegated\",\n    props.Environment,\n  )\n\n  try {\n    switch (event.RequestType) {\n      case \"Create\": {\n        // We receive a string value for GenerateSecret, but we need a boolean\n        // See https://github.com/aws-cloudformation/cloudformation-coverage-roadmap/issues/1037\n        const generate_secret: boolean =\n          typeof props.GenerateSecret === \"boolean\"\n            ? props.GenerateSecret\n            : props.GenerateSecret === \"true\"\n\n        const client: AppClient = {\n          name: props.Name,\n          type: props.Type,\n          scopes: props.Scopes || [],\n          callback_urls: props.CallbackUrls || [],\n          logout_urls: props.LogoutUrls || [],\n          generate_secret,\n        }\n\n        const created = await createAppClient(baseUrl, client)\n\n        return createSuccessResponse(event.PhysicalResourceId ?? created.name, {\n          Name: created.name,\n          ClientId: created.client_id || \"\",\n          ClientSecret: created.client_secret || \"\",\n          Type: created.type,\n        }) as CustomResourceResponse\n      }\n\n      case \"Update\": {\n        // Check if Type changed (requires replacement)\n        const oldProps = event.OldResourceProperties as AppClientProperties\n        if (oldProps && oldProps.Type !== props.Type) {\n          throw new Error(\n            \"Cannot change app client type. This requires resource replacement.\",\n          )\n        }\n\n        const update: AppClientUpdateRequest = {\n          name: props.Name,\n          scopes: props.Scopes || [],\n          callback_urls: props.CallbackUrls || [],\n          logout_urls: props.LogoutUrls || [],\n        }\n\n        await updateAppClient(baseUrl, update)\n        const updated = await readAppClient(baseUrl, props.Name)\n\n        return createSuccessResponse(event.PhysicalResourceId ?? updated.name, {\n          Name: updated.name,\n          ClientId: updated.client_id || \"\",\n          ClientSecret: updated.client_secret || \"\",\n          Type: updated.type,\n        }) as CustomResourceResponse\n      }\n\n      case \"Delete\": {\n        const name = event.PhysicalResourceId || props.Name\n        await deleteAppClient(baseUrl, name)\n\n        return createSuccessResponse(name, {}) as CustomResourceResponse\n      }\n    }\n  } catch (error) {\n    console.error(\"Error:\", error)\n    return createFailureResponse(\n      event.PhysicalResourceId || props.Name || \"unknown\",\n      handleError(error),\n    ) as CustomResourceResponse\n  }\n}\n"]}
|
|
121
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../src/cognito-app-client/handler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Lambda handler for CognitoAppClient custom resource\n */\n\nimport {\n  createFailureResponse,\n  createSuccessResponse,\n  handleError,\n} from \"../shared/custom-resource-handler\"\nimport { createUrlFromEnvironment, signedRequest } from \"../shared/sigv4-client\"\nimport type {\n  AppClient,\n  AppClientUpdateRequest,\n  CustomResourceRequest,\n  CustomResourceResponse,\n} from \"../shared/types\"\n\nconst COGNITO_BASE_DOMAIN =\n  process.env.COGNITO_BASE_DOMAIN || \"cognito.vydev.io\"\n\ninterface AppClientProperties {\n  Environment: string\n  Name: string\n  Type: \"frontend\" | \"backend\"\n  Scopes?: string[]\n  CallbackUrls?: string[]\n  LogoutUrls?: string[]\n  GenerateSecret?: boolean\n}\n\nasync function createAppClient(\n  baseUrl: string,\n  client: AppClient,\n): Promise<AppClient> {\n  const response = await signedRequest({\n    method: \"POST\",\n    hostname: baseUrl,\n    path: \"/app-clients\",\n    body: JSON.stringify(client),\n  })\n\n  if (response.statusCode !== 201) {\n    throw new Error(\n      `Could not create resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n\n  return JSON.parse(response.body)\n}\n\nasync function readAppClient(\n  baseUrl: string,\n  name: string,\n): Promise<AppClient> {\n  const encodedName = encodeURIComponent(name)\n  const response = await signedRequest({\n    method: \"GET\",\n    hostname: baseUrl,\n    path: `/app-clients/${encodedName}`,\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not read resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n\n  return JSON.parse(response.body)\n}\n\nasync function updateAppClient(\n  baseUrl: string,\n  update: AppClientUpdateRequest,\n): Promise<void> {\n  const encodedName = encodeURIComponent(update.name)\n  const response = await signedRequest({\n    method: \"PUT\",\n    hostname: baseUrl,\n    path: `/app-clients/${encodedName}`,\n    body: JSON.stringify(update),\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not update resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n}\n\nasync function deleteAppClient(baseUrl: string, name: string): Promise<void> {\n  const encodedName = encodeURIComponent(name)\n  const response = await signedRequest({\n    method: \"DELETE\",\n    hostname: baseUrl,\n    path: `/app-clients/${encodedName}`,\n  })\n\n  if (response.statusCode !== 200) {\n    const message = `Could not delete resource: ${response.statusCode} - ${response.body}`\n\n    if (response.statusCode === 404) {\n      // Allow soft fail to avoid ROLLBACK_FAILED status\n      console.warn(message)\n    } else {\n      throw new Error(message)\n    }\n  }\n}\n\nexport async function handler(\n  event: CustomResourceRequest,\n): Promise<CustomResourceResponse> {\n  const props = event.ResourceProperties as AppClientProperties\n  const baseUrl = createUrlFromEnvironment(\n    COGNITO_BASE_DOMAIN,\n    \"delegated\",\n    props.Environment,\n  )\n\n  try {\n    switch (event.RequestType) {\n      case \"Create\": {\n        // We receive a string value for GenerateSecret, but we need a boolean\n        // See https://github.com/aws-cloudformation/cloudformation-coverage-roadmap/issues/1037\n        const generate_secret: boolean =\n          typeof props.GenerateSecret === \"boolean\"\n            ? props.GenerateSecret\n            : props.GenerateSecret === \"true\"\n\n        const client: AppClient = {\n          name: props.Name,\n          type: props.Type,\n          scopes: props.Scopes || [],\n          callback_urls: props.CallbackUrls || [],\n          logout_urls: props.LogoutUrls || [],\n          generate_secret,\n        }\n\n        const created = await createAppClient(baseUrl, client)\n\n        return createSuccessResponse(event.PhysicalResourceId ?? created.name, {\n          Name: created.name,\n          ClientId: created.client_id || \"\",\n          ClientSecret: created.client_secret || \"\",\n          Type: created.type,\n        }) as CustomResourceResponse\n      }\n\n      case \"Update\": {\n        // Check if Type changed (requires replacement)\n        const oldProps = event.OldResourceProperties as AppClientProperties\n        if (oldProps && oldProps.Type !== props.Type) {\n          throw new Error(\n            \"Cannot change app client type. This requires resource replacement.\",\n          )\n        }\n\n        const update: AppClientUpdateRequest = {\n          name: props.Name,\n          scopes: props.Scopes || [],\n          callback_urls: props.CallbackUrls || [],\n          logout_urls: props.LogoutUrls || [],\n        }\n\n        await updateAppClient(baseUrl, update)\n        const updated = await readAppClient(baseUrl, props.Name)\n\n        return createSuccessResponse(event.PhysicalResourceId ?? updated.name, {\n          Name: updated.name,\n          ClientId: updated.client_id || \"\",\n          ClientSecret: updated.client_secret || \"\",\n          Type: updated.type,\n        }) as CustomResourceResponse\n      }\n\n      case \"Delete\": {\n        const name = event.PhysicalResourceId || props.Name\n        await deleteAppClient(baseUrl, name)\n\n        return createSuccessResponse(name, {}) as CustomResourceResponse\n      }\n    }\n  } catch (error) {\n    console.error(\"Error:\", error)\n    return createFailureResponse(\n      event.PhysicalResourceId || props.Name || \"unknown\",\n      handleError(error),\n    ) as CustomResourceResponse\n  }\n}\n"]}
|
|
@@ -48,7 +48,14 @@ async function deleteResourceServer(baseUrl, identifier) {
|
|
|
48
48
|
path: `/resource-servers/${encodedIdentifier}`,
|
|
49
49
|
});
|
|
50
50
|
if (response.statusCode !== 200) {
|
|
51
|
-
|
|
51
|
+
const message = `Could not delete resource: ${response.statusCode} - ${response.body}`;
|
|
52
|
+
if (response.statusCode === 404) {
|
|
53
|
+
// Allow soft fail to avoid ROLLBACK_FAILED status
|
|
54
|
+
console.warn(message);
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
throw new Error(message);
|
|
58
|
+
}
|
|
52
59
|
}
|
|
53
60
|
}
|
|
54
61
|
export async function handler(event) {
|
|
@@ -101,4 +108,4 @@ export async function handler(event) {
|
|
|
101
108
|
return createFailureResponse(event.PhysicalResourceId || props.Identifier || "unknown", handleError(error));
|
|
102
109
|
}
|
|
103
110
|
}
|
|
104
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../src/cognito-resource-server/handler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Lambda handler for CognitoResourceServer custom resource\n */\n\nimport {\n  createFailureResponse,\n  createSuccessResponse,\n  handleError,\n} from \"../shared/custom-resource-handler\"\nimport { createUrlFromEnvironment, signedRequest } from \"../shared/sigv4-client\"\nimport type {\n  CustomResourceRequest,\n  CustomResourceResponse,\n  ResourceServer,\n  ResourceServerUpdateRequest,\n} from \"../shared/types\"\n\nconst COGNITO_BASE_DOMAIN =\n  process.env.COGNITO_BASE_DOMAIN || \"cognito.vydev.io\"\n\ninterface ResourceServerProperties {\n  Environment: string\n  Name: string\n  Identifier: string\n  Scopes?: Array<{ Name: string; Description: string }>\n}\n\nasync function createResourceServer(\n  baseUrl: string,\n  server: ResourceServer,\n): Promise<ResourceServer> {\n  const response = await signedRequest({\n    method: \"POST\",\n    hostname: baseUrl,\n    path: \"/resource-servers\",\n    body: JSON.stringify(server),\n  })\n\n  if (response.statusCode !== 201) {\n    throw new Error(\n      `Could not create resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n\n  return JSON.parse(response.body)\n}\n\nasync function readResourceServer(\n  baseUrl: string,\n  identifier: string,\n): Promise<ResourceServer> {\n  const encodedIdentifier = encodeURIComponent(identifier)\n  const response = await signedRequest({\n    method: \"GET\",\n    hostname: baseUrl,\n    path: `/resource-servers/${encodedIdentifier}`,\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not read resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n\n  return JSON.parse(response.body)\n}\n\nasync function updateResourceServer(\n  baseUrl: string,\n  update: ResourceServerUpdateRequest,\n): Promise<void> {\n  const encodedIdentifier = encodeURIComponent(update.identifier)\n  const response = await signedRequest({\n    method: \"PUT\",\n    hostname: baseUrl,\n    path: `/resource-servers/${encodedIdentifier}`,\n    body: JSON.stringify(update),\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not update resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n}\n\nasync function deleteResourceServer(\n  baseUrl: string,\n  identifier: string,\n): Promise<void> {\n  const encodedIdentifier = encodeURIComponent(identifier)\n  const response = await signedRequest({\n    method: \"DELETE\",\n    hostname: baseUrl,\n    path: `/resource-servers/${encodedIdentifier}`,\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not delete resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n}\n\nexport async function handler(\n  event: CustomResourceRequest,\n): Promise<CustomResourceResponse> {\n  const props = event.ResourceProperties as ResourceServerProperties\n  const baseUrl = createUrlFromEnvironment(\n    COGNITO_BASE_DOMAIN,\n    \"delegated\",\n    props.Environment,\n  )\n\n  try {\n    switch (event.RequestType) {\n      case \"Create\": {\n        const server: ResourceServer = {\n          identifier: props.Identifier,\n          name: props.Name,\n          scopes: props.Scopes?.map((s) => ({\n            name: s.Name,\n            description: s.Description,\n          })),\n        }\n\n        const created = await createResourceServer(baseUrl, server)\n\n        return createSuccessResponse(\n          event.PhysicalResourceId ?? created.identifier,\n          {\n            Identifier: created.identifier,\n            Name: created.name,\n            Scopes: created.scopes,\n          },\n        ) as CustomResourceResponse\n      }\n\n      case \"Update\": {\n        const update: ResourceServerUpdateRequest = {\n          identifier: props.Identifier,\n          name: props.Name,\n          scopes: props.Scopes?.map((s) => ({\n            name: s.Name,\n            description: s.Description,\n          })),\n        }\n\n        await updateResourceServer(baseUrl, update)\n        const updated = await readResourceServer(baseUrl, props.Identifier)\n\n        return createSuccessResponse(\n          event.PhysicalResourceId ?? updated.identifier,\n          {\n            Identifier: updated.identifier,\n            Name: updated.name,\n            Scopes: updated.scopes,\n          },\n        ) as CustomResourceResponse\n      }\n\n      case \"Delete\": {\n        const identifier = event.PhysicalResourceId || props.Identifier\n        await deleteResourceServer(baseUrl, identifier)\n\n        return createSuccessResponse(identifier, {}) as CustomResourceResponse\n      }\n    }\n  } catch (error) {\n    console.error(\"Error:\", error)\n    return createFailureResponse(\n      event.PhysicalResourceId || props.Identifier || \"unknown\",\n      handleError(error),\n    ) as CustomResourceResponse\n  }\n}\n"]}
|
|
111
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../src/cognito-resource-server/handler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Lambda handler for CognitoResourceServer custom resource\n */\n\nimport {\n  createFailureResponse,\n  createSuccessResponse,\n  handleError,\n} from \"../shared/custom-resource-handler\"\nimport { createUrlFromEnvironment, signedRequest } from \"../shared/sigv4-client\"\nimport type {\n  CustomResourceRequest,\n  CustomResourceResponse,\n  ResourceServer,\n  ResourceServerUpdateRequest,\n} from \"../shared/types\"\n\nconst COGNITO_BASE_DOMAIN =\n  process.env.COGNITO_BASE_DOMAIN || \"cognito.vydev.io\"\n\ninterface ResourceServerProperties {\n  Environment: string\n  Name: string\n  Identifier: string\n  Scopes?: Array<{ Name: string; Description: string }>\n}\n\nasync function createResourceServer(\n  baseUrl: string,\n  server: ResourceServer,\n): Promise<ResourceServer> {\n  const response = await signedRequest({\n    method: \"POST\",\n    hostname: baseUrl,\n    path: \"/resource-servers\",\n    body: JSON.stringify(server),\n  })\n\n  if (response.statusCode !== 201) {\n    throw new Error(\n      `Could not create resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n\n  return JSON.parse(response.body)\n}\n\nasync function readResourceServer(\n  baseUrl: string,\n  identifier: string,\n): Promise<ResourceServer> {\n  const encodedIdentifier = encodeURIComponent(identifier)\n  const response = await signedRequest({\n    method: \"GET\",\n    hostname: baseUrl,\n    path: `/resource-servers/${encodedIdentifier}`,\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not read resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n\n  return JSON.parse(response.body)\n}\n\nasync function updateResourceServer(\n  baseUrl: string,\n  update: ResourceServerUpdateRequest,\n): Promise<void> {\n  const encodedIdentifier = encodeURIComponent(update.identifier)\n  const response = await signedRequest({\n    method: \"PUT\",\n    hostname: baseUrl,\n    path: `/resource-servers/${encodedIdentifier}`,\n    body: JSON.stringify(update),\n  })\n\n  if (response.statusCode !== 200) {\n    throw new Error(\n      `Could not update resource: ${response.statusCode} - ${response.body}`,\n    )\n  }\n}\n\nasync function deleteResourceServer(\n  baseUrl: string,\n  identifier: string,\n): Promise<void> {\n  const encodedIdentifier = encodeURIComponent(identifier)\n  const response = await signedRequest({\n    method: \"DELETE\",\n    hostname: baseUrl,\n    path: `/resource-servers/${encodedIdentifier}`,\n  })\n\n  if (response.statusCode !== 200) {\n    const message = `Could not delete resource: ${response.statusCode} - ${response.body}`\n\n    if (response.statusCode === 404) {\n      // Allow soft fail to avoid ROLLBACK_FAILED status\n      console.warn(message)\n    } else {\n      throw new Error(message)\n    }\n  }\n}\n\nexport async function handler(\n  event: CustomResourceRequest,\n): Promise<CustomResourceResponse> {\n  const props = event.ResourceProperties as ResourceServerProperties\n  const baseUrl = createUrlFromEnvironment(\n    COGNITO_BASE_DOMAIN,\n    \"delegated\",\n    props.Environment,\n  )\n\n  try {\n    switch (event.RequestType) {\n      case \"Create\": {\n        const server: ResourceServer = {\n          identifier: props.Identifier,\n          name: props.Name,\n          scopes: props.Scopes?.map((s) => ({\n            name: s.Name,\n            description: s.Description,\n          })),\n        }\n\n        const created = await createResourceServer(baseUrl, server)\n\n        return createSuccessResponse(\n          event.PhysicalResourceId ?? created.identifier,\n          {\n            Identifier: created.identifier,\n            Name: created.name,\n            Scopes: created.scopes,\n          },\n        ) as CustomResourceResponse\n      }\n\n      case \"Update\": {\n        const update: ResourceServerUpdateRequest = {\n          identifier: props.Identifier,\n          name: props.Name,\n          scopes: props.Scopes?.map((s) => ({\n            name: s.Name,\n            description: s.Description,\n          })),\n        }\n\n        await updateResourceServer(baseUrl, update)\n        const updated = await readResourceServer(baseUrl, props.Identifier)\n\n        return createSuccessResponse(\n          event.PhysicalResourceId ?? updated.identifier,\n          {\n            Identifier: updated.identifier,\n            Name: updated.name,\n            Scopes: updated.scopes,\n          },\n        ) as CustomResourceResponse\n      }\n\n      case \"Delete\": {\n        const identifier = event.PhysicalResourceId || props.Identifier\n        await deleteResourceServer(baseUrl, identifier)\n\n        return createSuccessResponse(identifier, {}) as CustomResourceResponse\n      }\n    }\n  } catch (error) {\n    console.error(\"Error:\", error)\n    return createFailureResponse(\n      event.PhysicalResourceId || props.Identifier || \"unknown\",\n      handleError(error),\n    ) as CustomResourceResponse\n  }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@liflig/cdk-vy",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.3.0",
|
|
4
4
|
"description": "CDK constructs for the Vy internal services, based on nsbno/terraform-provider-vy",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"aws-cdk",
|
|
@@ -41,7 +41,7 @@
|
|
|
41
41
|
"provenance": true
|
|
42
42
|
},
|
|
43
43
|
"devDependencies": {
|
|
44
|
-
"@biomejs/biome": "2.3.
|
|
44
|
+
"@biomejs/biome": "2.3.5",
|
|
45
45
|
"@commitlint/cli": "20.1.0",
|
|
46
46
|
"@commitlint/config-conventional": "20.0.0",
|
|
47
47
|
"@types/aws-lambda": "8.10.157",
|