From d10786e9f11741c526d8da1ace15804f5d18e0d4 Mon Sep 17 00:00:00 2001 From: Akber Choudhry Date: Tue, 6 Jan 2026 11:02:45 -0500 Subject: [PATCH 1/3] fix: add AWS SDK peer dependencies for @dyanet/config-aws --- examples/gmail-viewer/package-lock.json | 98 +++++++++++++++++++++++++ examples/gmail-viewer/package.json | 1 + 2 files changed, 99 insertions(+) diff --git a/examples/gmail-viewer/package-lock.json b/examples/gmail-viewer/package-lock.json index 82025b0..e8c2d87 100644 --- a/examples/gmail-viewer/package-lock.json +++ b/examples/gmail-viewer/package-lock.json @@ -12,6 +12,7 @@ "@aws-sdk/client-s3": "^3.0.0", "@aws-sdk/client-secrets-manager": "^3.0.0", "@aws-sdk/client-ssm": "^3.0.0", + "@aws-sdk/credential-providers": "^3.0.0", "@dyanet/config-aws": "^1.0.3", "@dyanet/imap": "^0.5.2", "express": "^4.21.0", @@ -231,6 +232,56 @@ "node": ">=14.0.0" } }, + "node_modules/@aws-sdk/client-cognito-identity": { + "version": "3.962.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/client-cognito-identity/-/client-cognito-identity-3.962.0.tgz", + "integrity": "sha512-gQKsnvC4Rlg0uVSDZIo5Ditp/oqea21omsJsftqkAXGyFQsNVLMd1Toz0Akg0ecbLoLhrWPCMylugH6El1buYw==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/sha256-browser": "5.2.0", + "@aws-crypto/sha256-js": "5.2.0", + "@aws-sdk/core": "3.957.0", + "@aws-sdk/credential-provider-node": "3.962.0", + "@aws-sdk/middleware-host-header": "3.957.0", + "@aws-sdk/middleware-logger": "3.957.0", + "@aws-sdk/middleware-recursion-detection": "3.957.0", + "@aws-sdk/middleware-user-agent": "3.957.0", + "@aws-sdk/region-config-resolver": "3.957.0", + "@aws-sdk/types": "3.957.0", + "@aws-sdk/util-endpoints": "3.957.0", + "@aws-sdk/util-user-agent-browser": "3.957.0", + "@aws-sdk/util-user-agent-node": "3.957.0", + "@smithy/config-resolver": "^4.4.5", + "@smithy/core": "^3.20.0", + "@smithy/fetch-http-handler": "^5.3.8", + "@smithy/hash-node": "^4.2.7", + "@smithy/invalid-dependency": "^4.2.7", + "@smithy/middleware-content-length": "^4.2.7", + "@smithy/middleware-endpoint": "^4.4.1", + "@smithy/middleware-retry": "^4.4.17", + "@smithy/middleware-serde": "^4.2.8", + "@smithy/middleware-stack": "^4.2.7", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/node-http-handler": "^4.4.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/smithy-client": "^4.10.2", + "@smithy/types": "^4.11.0", + "@smithy/url-parser": "^4.2.7", + "@smithy/util-base64": "^4.3.0", + "@smithy/util-body-length-browser": "^4.2.0", + "@smithy/util-body-length-node": "^4.2.1", + "@smithy/util-defaults-mode-browser": "^4.3.16", + "@smithy/util-defaults-mode-node": "^4.2.19", + "@smithy/util-endpoints": "^3.2.7", + "@smithy/util-middleware": "^4.2.7", + "@smithy/util-retry": "^4.2.7", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, "node_modules/@aws-sdk/client-s3": { "version": "3.962.0", "resolved": "https://registry.npmjs.org/@aws-sdk/client-s3/-/client-s3-3.962.0.tgz", @@ -484,6 +535,22 @@ "node": ">=18.0.0" } }, + "node_modules/@aws-sdk/credential-provider-cognito-identity": { + "version": "3.962.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-cognito-identity/-/credential-provider-cognito-identity-3.962.0.tgz", + "integrity": "sha512-0B1iwgCNY3cg/s3JwWtBP1PaB4uldd0sgx6WgVVcnQ9rpyuCMOUF61enA1OcmSWgRRndylznGXjok8cva5Pw4w==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/client-cognito-identity": "3.962.0", + "@aws-sdk/types": "3.957.0", + "@smithy/property-provider": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, "node_modules/@aws-sdk/credential-provider-env": { "version": "3.957.0", "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-env/-/credential-provider-env-3.957.0.tgz", @@ -642,6 +709,37 @@ "node": ">=18.0.0" } }, + "node_modules/@aws-sdk/credential-providers": { + "version": "3.962.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-providers/-/credential-providers-3.962.0.tgz", + "integrity": "sha512-KIFu4zkMxJ9TeBm/TWv64W9en5fSP/pPWxm3tkFMleEmyotWmC2xkihZZYcHKgLhezA3fyZxTVz2v++6DpO76A==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/client-cognito-identity": "3.962.0", + "@aws-sdk/core": "3.957.0", + "@aws-sdk/credential-provider-cognito-identity": "3.962.0", + "@aws-sdk/credential-provider-env": "3.957.0", + "@aws-sdk/credential-provider-http": "3.957.0", + "@aws-sdk/credential-provider-ini": "3.962.0", + "@aws-sdk/credential-provider-login": "3.962.0", + "@aws-sdk/credential-provider-node": "3.962.0", + "@aws-sdk/credential-provider-process": "3.957.0", + "@aws-sdk/credential-provider-sso": "3.958.0", + "@aws-sdk/credential-provider-web-identity": "3.958.0", + "@aws-sdk/nested-clients": "3.958.0", + "@aws-sdk/types": "3.957.0", + "@smithy/config-resolver": "^4.4.5", + "@smithy/core": "^3.20.0", + "@smithy/credential-provider-imds": "^4.2.7", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/property-provider": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, "node_modules/@aws-sdk/middleware-bucket-endpoint": { "version": "3.957.0", "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-bucket-endpoint/-/middleware-bucket-endpoint-3.957.0.tgz", diff --git a/examples/gmail-viewer/package.json b/examples/gmail-viewer/package.json index 20e4523..494d45e 100644 --- a/examples/gmail-viewer/package.json +++ b/examples/gmail-viewer/package.json @@ -23,6 +23,7 @@ "@aws-sdk/client-s3": "^3.0.0", "@aws-sdk/client-secrets-manager": "^3.0.0", "@aws-sdk/client-ssm": "^3.0.0", + "@aws-sdk/credential-providers": "^3.0.0", "@dyanet/config-aws": "^1.0.3", "@dyanet/imap": "^0.5.2", "express": "^4.21.0", From 12dc58822ef3ff7aea4c05357f08ba5784e939bd Mon Sep 17 00:00:00 2001 From: Akber Choudhry Date: Tue, 6 Jan 2026 11:11:20 -0500 Subject: [PATCH 2/3] fix: exclude NIL from atom generator in property tests --- tests/property/protocol.property.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/tests/property/protocol.property.ts b/tests/property/protocol.property.ts index 62d7b3d..38d5b41 100644 --- a/tests/property/protocol.property.ts +++ b/tests/property/protocol.property.ts @@ -41,11 +41,12 @@ const responseTextArb = fc.stringOf( /** * Generates valid IMAP atoms (no special characters) + * Excludes NIL (case-insensitive) as it's a reserved keyword */ const atomArb = fc.stringOf( fc.constantFrom(...'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789._-'), { minLength: 1, maxLength: 20 } -); +).filter(s => s.toUpperCase() !== 'NIL'); /** * Generates valid quoted strings (may contain spaces) From 3c17e4e51f0f41ac65605fc893466e3dcd775c1c Mon Sep 17 00:00:00 2001 From: Akber Choudhry Date: Tue, 6 Jan 2026 14:48:02 -0500 Subject: [PATCH 3/3] fix(cdk): minor fixes to SSM and secrets params --- .../gmail-viewer-cdk/bin/gmail-viewer-cdk.js | 6 +++--- .../gmail-viewer-cdk/bin/gmail-viewer-cdk.ts | 4 ++-- .../lib/gmail-viewer-cdk-stack.js | 21 ++++++++----------- .../lib/gmail-viewer-cdk-stack.ts | 19 +++++++---------- 4 files changed, 22 insertions(+), 28 deletions(-) diff --git a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js index 332c32c..4f2feb2 100644 --- a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js +++ b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js @@ -19,7 +19,7 @@ new gmail_viewer_cdk_stack_1.GmailViewerCdkStack(app, 'GmailViewerCdkStack', { gitHubBranch: 'main', codeConnectionArn: 'arn:aws:codeconnections:ca-central-1:239030031457:connection/29fe5c11-23d2-4c10-a81d-4e880b12e2c6', useGitHubWebhooks: false, - apiCustomDomainName: 'mail.dyanet.com', - certificateArn: 'arn:aws:acm:ca-central-1:239030031457:certificate/af9150f1-4635-4154-8b19-e5449d57e971', + apiCustomDomainName: 'demo.dyanet.com', + certificateArn: 'arn:aws:acm:ca-central-1:239030031457:certificate/cb01610f-dddd-44e1-abd9-0672f0dc3355', }); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ21haWwtdmlld2VyLWNkay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImdtYWlsLXZpZXdlci1jZGsudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EsNkNBQWtDO0FBQ2xDLDBFQUFvRTtBQUVwRSxNQUFNLEdBQUcsR0FBRyxJQUFJLGlCQUFHLEVBQUUsQ0FBQztBQUV0QiwwRkFBMEY7QUFDMUYsK0RBQStEO0FBQy9ELE1BQU0sY0FBYyxHQUFHLEdBQUcsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLGdCQUFnQixDQUFDLEVBQUUsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBRTVFLElBQUksNENBQW1CLENBQUMsR0FBRyxFQUFFLHFCQUFxQixFQUFFO0lBQ2xELEdBQUcsRUFBRTtRQUNILE9BQU8sRUFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLG1CQUFtQjtRQUN4QyxNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0I7S0FDdkM7SUFDRCxjQUFjLEVBQUUsQ0FBQywwQkFBMEIsRUFBRSwwQkFBMEIsQ0FBQztJQUN4RSxhQUFhLEVBQUUsQ0FBQywwQkFBMEIsRUFBRSwwQkFBMEIsQ0FBQztJQUN2RSxXQUFXLEVBQUUsUUFBUTtJQUNyQixVQUFVLEVBQUUsTUFBTTtJQUNsQixZQUFZLEVBQUUsTUFBTTtJQUNwQixpQkFBaUIsRUFBRSxtR0FBbUc7SUFDdEgsaUJBQWlCLEVBQUUsS0FBSztJQUN4QixtQkFBbUIsRUFBRSxpQkFBaUI7SUFDdEMsY0FBYyxFQUFFLHdGQUF3RjtDQUN6RyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXHJcbmltcG9ydCB7IEFwcCB9IGZyb20gJ2F3cy1jZGstbGliJztcclxuaW1wb3J0IHsgR21haWxWaWV3ZXJDZGtTdGFjayB9IGZyb20gJy4uL2xpYi9nbWFpbC12aWV3ZXItY2RrLXN0YWNrJztcclxuXHJcbmNvbnN0IGFwcCA9IG5ldyBBcHAoKTtcclxuXHJcbi8vIFRvIHNwZWNpZnkgc3VibmV0cywgdXNlIHRoZSAtLWNvbnRleHQgb3B0aW9uIHdpdGggYSBjb21tYS1zZXBhcmF0ZWQgbGlzdCBvZiBzdWJuZXQgSURzOlxyXG4vLyBjZGsgZGVwbG95IC1jIHByaXZhdGVTdWJuZXRzPXN1Ym5ldC14eHh4eHh4eCxzdWJuZXQteXl5eXl5eXlcclxuY29uc3QgcHJpdmF0ZVN1Ym5ldHMgPSBhcHAubm9kZS50cnlHZXRDb250ZXh0KCdwcml2YXRlU3VibmV0cycpPy5zcGxpdCgnLCcpO1xyXG5cclxubmV3IEdtYWlsVmlld2VyQ2RrU3RhY2soYXBwLCAnR21haWxWaWV3ZXJDZGtTdGFjaycsIHtcclxuICBlbnY6IHtcclxuICAgIGFjY291bnQ6IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX0FDQ09VTlQsXHJcbiAgICByZWdpb246IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX1JFR0lPTixcclxuICB9LFxyXG4gIHByaXZhdGVTdWJuZXRzOiBbJ3N1Ym5ldC0wZDgyOTBkMGVhNTc1MzMxYycsICdzdWJuZXQtMDMyMjkwNTY5YTdlY2YyMTAnXSxcclxuICBwdWJsaWNTdWJuZXRzOiBbJ3N1Ym5ldC0wNzEzYWRlMzBkZDhjMzA4YycsICdzdWJuZXQtMGI0ZGQ0NWI1MWJiODI1MmYnXSxcclxuICBnaXRIdWJPd25lcjogJ2R5YW5ldCcsXHJcbiAgZ2l0SHViUmVwbzogJ2ltYXAnLFxyXG4gIGdpdEh1YkJyYW5jaDogJ21haW4nLFxyXG4gIGNvZGVDb25uZWN0aW9uQXJuOiAnYXJuOmF3czpjb2RlY29ubmVjdGlvbnM6Y2EtY2VudHJhbC0xOjIzOTAzMDAzMTQ1Nzpjb25uZWN0aW9uLzI5ZmU1YzExLTIzZDItNGMxMC1hODFkLTRlODgwYjEyZTJjNicsXHJcbiAgdXNlR2l0SHViV2ViaG9va3M6IGZhbHNlLFxyXG4gIGFwaUN1c3RvbURvbWFpbk5hbWU6ICdtYWlsLmR5YW5ldC5jb20nLFxyXG4gIGNlcnRpZmljYXRlQXJuOiAnYXJuOmF3czphY206Y2EtY2VudHJhbC0xOjIzOTAzMDAzMTQ1NzpjZXJ0aWZpY2F0ZS9hZjkxNTBmMS00NjM1LTQxNTQtOGIxOS1lNTQ0OWQ1N2U5NzEnLFxyXG59KTtcclxuIl19 \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ21haWwtdmlld2VyLWNkay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImdtYWlsLXZpZXdlci1jZGsudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EsNkNBQWtDO0FBQ2xDLDBFQUFvRTtBQUVwRSxNQUFNLEdBQUcsR0FBRyxJQUFJLGlCQUFHLEVBQUUsQ0FBQztBQUV0QiwwRkFBMEY7QUFDMUYsK0RBQStEO0FBQy9ELE1BQU0sY0FBYyxHQUFHLEdBQUcsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLGdCQUFnQixDQUFDLEVBQUUsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBRTVFLElBQUksNENBQW1CLENBQUMsR0FBRyxFQUFFLHFCQUFxQixFQUFFO0lBQ2xELEdBQUcsRUFBRTtRQUNILE9BQU8sRUFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLG1CQUFtQjtRQUN4QyxNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0I7S0FDdkM7SUFDRCxjQUFjLEVBQUUsQ0FBQywwQkFBMEIsRUFBRSwwQkFBMEIsQ0FBQztJQUN4RSxhQUFhLEVBQUUsQ0FBQywwQkFBMEIsRUFBRSwwQkFBMEIsQ0FBQztJQUN2RSxXQUFXLEVBQUUsUUFBUTtJQUNyQixVQUFVLEVBQUUsTUFBTTtJQUNsQixZQUFZLEVBQUUsTUFBTTtJQUNwQixpQkFBaUIsRUFBRSxtR0FBbUc7SUFDdEgsaUJBQWlCLEVBQUUsS0FBSztJQUN4QixtQkFBbUIsRUFBRSxpQkFBaUI7SUFDdEMsY0FBYyxFQUFFLHdGQUF3RjtDQUN6RyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXHJcbmltcG9ydCB7IEFwcCB9IGZyb20gJ2F3cy1jZGstbGliJztcclxuaW1wb3J0IHsgR21haWxWaWV3ZXJDZGtTdGFjayB9IGZyb20gJy4uL2xpYi9nbWFpbC12aWV3ZXItY2RrLXN0YWNrJztcclxuXHJcbmNvbnN0IGFwcCA9IG5ldyBBcHAoKTtcclxuXHJcbi8vIFRvIHNwZWNpZnkgc3VibmV0cywgdXNlIHRoZSAtLWNvbnRleHQgb3B0aW9uIHdpdGggYSBjb21tYS1zZXBhcmF0ZWQgbGlzdCBvZiBzdWJuZXQgSURzOlxyXG4vLyBjZGsgZGVwbG95IC1jIHByaXZhdGVTdWJuZXRzPXN1Ym5ldC14eHh4eHh4eCxzdWJuZXQteXl5eXl5eXlcclxuY29uc3QgcHJpdmF0ZVN1Ym5ldHMgPSBhcHAubm9kZS50cnlHZXRDb250ZXh0KCdwcml2YXRlU3VibmV0cycpPy5zcGxpdCgnLCcpO1xyXG5cclxubmV3IEdtYWlsVmlld2VyQ2RrU3RhY2soYXBwLCAnR21haWxWaWV3ZXJDZGtTdGFjaycsIHtcclxuICBlbnY6IHtcclxuICAgIGFjY291bnQ6IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX0FDQ09VTlQsXHJcbiAgICByZWdpb246IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX1JFR0lPTixcclxuICB9LFxyXG4gIHByaXZhdGVTdWJuZXRzOiBbJ3N1Ym5ldC0wZDgyOTBkMGVhNTc1MzMxYycsICdzdWJuZXQtMDMyMjkwNTY5YTdlY2YyMTAnXSxcclxuICBwdWJsaWNTdWJuZXRzOiBbJ3N1Ym5ldC0wNzEzYWRlMzBkZDhjMzA4YycsICdzdWJuZXQtMGI0ZGQ0NWI1MWJiODI1MmYnXSxcclxuICBnaXRIdWJPd25lcjogJ2R5YW5ldCcsXHJcbiAgZ2l0SHViUmVwbzogJ2ltYXAnLFxyXG4gIGdpdEh1YkJyYW5jaDogJ21haW4nLFxyXG4gIGNvZGVDb25uZWN0aW9uQXJuOiAnYXJuOmF3czpjb2RlY29ubmVjdGlvbnM6Y2EtY2VudHJhbC0xOjIzOTAzMDAzMTQ1Nzpjb25uZWN0aW9uLzI5ZmU1YzExLTIzZDItNGMxMC1hODFkLTRlODgwYjEyZTJjNicsXHJcbiAgdXNlR2l0SHViV2ViaG9va3M6IGZhbHNlLFxyXG4gIGFwaUN1c3RvbURvbWFpbk5hbWU6ICdkZW1vLmR5YW5ldC5jb20nLFxyXG4gIGNlcnRpZmljYXRlQXJuOiAnYXJuOmF3czphY206Y2EtY2VudHJhbC0xOjIzOTAzMDAzMTQ1NzpjZXJ0aWZpY2F0ZS9jYjAxNjEwZi1kZGRkLTQ0ZTEtYWJkOS0wNjcyZjBkYzMzNTUnLFxyXG59KTtcclxuIl19 \ No newline at end of file diff --git a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.ts b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.ts index b984745..e30480b 100644 --- a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.ts +++ b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.ts @@ -20,6 +20,6 @@ new GmailViewerCdkStack(app, 'GmailViewerCdkStack', { gitHubBranch: 'main', codeConnectionArn: 'arn:aws:codeconnections:ca-central-1:239030031457:connection/29fe5c11-23d2-4c10-a81d-4e880b12e2c6', useGitHubWebhooks: false, - apiCustomDomainName: 'mail.dyanet.com', - certificateArn: 'arn:aws:acm:ca-central-1:239030031457:certificate/af9150f1-4635-4154-8b19-e5449d57e971', + apiCustomDomainName: 'demo.dyanet.com', + certificateArn: 'arn:aws:acm:ca-central-1:239030031457:certificate/cb01610f-dddd-44e1-abd9-0672f0dc3355', }); diff --git a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.js b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.js index d3db55e..3b677c0 100644 --- a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.js +++ b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.js @@ -46,7 +46,6 @@ const logs = __importStar(require("aws-cdk-lib/aws-logs")); const route53 = __importStar(require("aws-cdk-lib/aws-route53")); const servicediscovery = __importStar(require("aws-cdk-lib/aws-servicediscovery")); const secretsmanager = __importStar(require("aws-cdk-lib/aws-secretsmanager")); -const ssm = __importStar(require("aws-cdk-lib/aws-ssm")); class GmailViewerCdkStack extends cdk.Stack { clusterName; repositoryUri; @@ -80,7 +79,7 @@ class GmailViewerCdkStack extends cdk.Stack { const googleClientId = props.googleClientId || 'PLACEHOLDER_GOOGLE_CLIENT_ID'; const googleClientSecret = props.googleClientSecret || 'PLACEHOLDER_GOOGLE_CLIENT_SECRET'; const sessionSecretValue = props.sessionSecretValue || 'PLACEHOLDER_SESSION_SECRET'; - const apiDomain = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'mail.dyanet.com'; + const apiDomain = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'demo.dyanet.com'; const certArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN; const zoneId = props.hostedZoneId ?? ''; const hasCustomDomain = Boolean(certArn && apiDomain); @@ -162,13 +161,16 @@ class GmailViewerCdkStack extends cdk.Stack { const cloudMapService = new servicediscovery.Service(this, 'AppCloudMapService', { name: 'mail-example', namespace, dnsRecordType: servicediscovery.DnsRecordType.SRV, dnsTtl: nsTtl, routingPolicy: servicediscovery.RoutingPolicy.WEIGHTED, customHealthCheck: { failureThreshold: 1 }, }); - // Secrets Manager - single secret with GOOGLE_CLIENT_SECRET and SESSION_SECRET + // Secrets Manager - all config values for the application const appSecrets = new secretsmanager.Secret(this, 'AppSecrets', { - secretName: `${ssmPrefix}/secrets`, - description: 'Application secrets for mail-example', + secretName: `${ssmPrefix}/config`, + description: 'Application configuration for mail-example', secretObjectValue: { + GOOGLE_CLIENT_ID: cdk.SecretValue.unsafePlainText(googleClientId), GOOGLE_CLIENT_SECRET: cdk.SecretValue.unsafePlainText(googleClientSecret), SESSION_SECRET: cdk.SecretValue.unsafePlainText(sessionSecretValue), + BASE_URL: cdk.SecretValue.unsafePlainText(baseUrl.toString()), + PORT: cdk.SecretValue.unsafePlainText(containerPort.toString()), }, }); // IAM Roles (L2) @@ -178,11 +180,6 @@ class GmailViewerCdkStack extends cdk.Stack { const taskRole = new iam.Role(this, 'TaskRole', { assumedBy: new iam.ServicePrincipal('ecs-tasks.amazonaws.com') }); taskRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], resources: [`arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, `arn:aws:kms:${this.region}:${this.account}:key/*`] })); taskRole.addToPolicy(new iam.PolicyStatement({ actions: ['secretsmanager:GetSecretValue'], resources: [appSecrets.secretArn] })); - // SSM Parameters (L2) - create before task definition so they can be referenced - const ssmBaseUrl = new ssm.StringParameter(this, 'SsmBaseUrl', { parameterName: `${ssmPrefix}/env/BASE_URL`, stringValue: baseUrl.toString() }); - const ssmGoogleClientId = new ssm.StringParameter(this, 'SsmGoogleClientId', { parameterName: `${ssmPrefix}/env/GOOGLE_CLIENT_ID`, stringValue: googleClientId }); - new ssm.StringParameter(this, 'SsmPort', { parameterName: `${ssmPrefix}/env/PORT`, stringValue: containerPort.toString() }); - new ssm.StringParameter(this, 'SsmPublicEnv', { parameterName: `${ssmPrefix}/env/NODE_ENV`, stringValue: publicEnvValue }); // ECS Task Definition (L2) const taskDef = new ecs.FargateTaskDefinition(this, 'AppTaskDefinition', { family: 'mail-example', cpu: containerCpu, memoryLimitMiB: containerMemory, executionRole: taskExecRole, taskRole }); taskDef.addContainer('mail-example', { @@ -190,7 +187,7 @@ class GmailViewerCdkStack extends cdk.Stack { logging: ecs.LogDrivers.awsLogs({ logGroup: appLogGroup, streamPrefix: 'mail-example' }), environment: { CONFIG_SSM_PREFIX: ssmPrefix, PORT: containerPort.toString(), BASE_URL: baseUrl.toString(), NODE_ENV: publicEnvValue }, secrets: { - GOOGLE_CLIENT_ID: ecs.Secret.fromSsmParameter(ssmGoogleClientId), + GOOGLE_CLIENT_ID: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_ID'), GOOGLE_CLIENT_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_SECRET'), SESSION_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'SESSION_SECRET'), }, @@ -243,4 +240,4 @@ class GmailViewerCdkStack extends cdk.Stack { } } exports.GmailViewerCdkStack = GmailViewerCdkStack; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gmail-viewer-cdk-stack.js","sourceRoot":"","sources":["gmail-viewer-cdk-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\r\nimport * as apigatewayv2 from 'aws-cdk-lib/aws-apigatewayv2';\r\nimport * as codebuild from 'aws-cdk-lib/aws-codebuild';\r\nimport * as cr from 'aws-cdk-lib/custom-resources';\r\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\r\nimport * as ecr from 'aws-cdk-lib/aws-ecr';\r\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\r\nimport * as iam from 'aws-cdk-lib/aws-iam';\r\nimport * as logs from 'aws-cdk-lib/aws-logs';\r\nimport * as route53 from 'aws-cdk-lib/aws-route53';\r\nimport * as servicediscovery from 'aws-cdk-lib/aws-servicediscovery';\r\nimport * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';\r\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\r\n\r\nexport interface GmailViewerCdkStackProps extends cdk.StackProps {\r\n  readonly vpcId?: string;\r\n  readonly privateSubnets?: string[];\r\n  readonly publicSubnets?: string[];\r\n  readonly containerPort?: number | string;\r\n  readonly containerCpu?: number | string;\r\n  readonly containerMemory?: number | string;\r\n  readonly desiredCount?: number | string;\r\n  readonly certificateArn?: string;\r\n  readonly apiCustomDomainName?: string;\r\n  readonly hostedZoneId?: string;\r\n  readonly serviceDiscoveryNamespaceName?: string;\r\n  readonly serviceDiscoveryTtl?: number | string;\r\n  readonly ssmPrefix?: string;\r\n  readonly gitHubOwner?: string;\r\n  readonly gitHubRepo?: string;\r\n  readonly gitHubBranch?: string;\r\n  readonly publicEnvValue?: string;\r\n  readonly googleClientId?: string;\r\n  readonly googleClientSecret?: string;\r\n  readonly sessionSecretValue?: string;\r\n  readonly useGitHubWebhooks?: boolean;\r\n  readonly codeConnectionArn?: string;\r\n}\r\n\r\nexport class GmailViewerCdkStack extends cdk.Stack {\r\n  public readonly clusterName: string;\r\n  public readonly repositoryUri: string;\r\n  public readonly apiInvokeUrl: string;\r\n  public readonly cloudMapServiceArn: string;\r\n\r\n  public constructor(scope: cdk.App, id: string, props: GmailViewerCdkStackProps) {\r\n    super(scope, id, props);\r\n\r\n    const toNumber = (value: string | number | undefined, fallback: number): number => {\r\n      const candidate = value ?? fallback;\r\n      const parsed = typeof candidate === 'number' ? candidate : Number(candidate);\r\n      return Number.isFinite(parsed) ? parsed : fallback;\r\n    };\r\n    const toCpuUnits = (value: string | number | undefined): number => {\r\n      const parsed = toNumber(value, 0.5);\r\n      const vcpu = parsed <= 4 ? parsed * 1024 : parsed;\r\n      return Math.max(256, Math.round(vcpu / 256) * 256);\r\n    };\r\n\r\n    const repoParts = (process.env.GITHUB_REPOSITORY ?? '').split('/');\r\n    const containerPort = toNumber(props.containerPort, 3000);\r\n    const containerCpu = toCpuUnits(props.containerCpu);\r\n    const containerMemory = toNumber(props.containerMemory, 1024);\r\n    const desiredCount = toNumber(props.desiredCount, 1);\r\n    const nsName = props.serviceDiscoveryNamespaceName ?? 'mail.local';\r\n    const nsTtl = cdk.Duration.seconds(toNumber(props.serviceDiscoveryTtl, 60));\r\n    const ssmPrefixRaw = props.ssmPrefix ?? '/mail-example';\r\n    const ssmPrefix = '/' + ssmPrefixRaw.split('/').filter(p => p).join('/');\r\n    const gitHubOwner = props.gitHubOwner ?? (repoParts[0] || 'dyanet');\r\n    const gitHubRepo = props.gitHubRepo ?? (repoParts[1] || 'imap');\r\n    const gitHubBranch = props.gitHubBranch ?? process.env.GITHUB_REF_NAME ?? 'main';\r\n    const publicEnvValue = props.publicEnvValue ?? 'production';\r\n    const googleClientId = props.googleClientId || 'PLACEHOLDER_GOOGLE_CLIENT_ID';\r\n    const googleClientSecret = props.googleClientSecret || 'PLACEHOLDER_GOOGLE_CLIENT_SECRET';\r\n    const sessionSecretValue = props.sessionSecretValue || 'PLACEHOLDER_SESSION_SECRET';\r\n    const apiDomain = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'mail.dyanet.com';\r\n    const certArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN;\r\n    const zoneId = props.hostedZoneId ?? '';\r\n    const hasCustomDomain = Boolean(certArn && apiDomain);\r\n    const hasHostedZone = Boolean(zoneId);\r\n    const usePublicSubnets = props.publicSubnets && props.publicSubnets.length > 0;\r\n\r\n    // VPC setup\r\n    let vpc: ec2.IVpc;\r\n    let serviceSubnetIds: string[];\r\n    const useVpcParameter = this.node.tryGetContext('useVpcParameter');\r\n\r\n    if (!useVpcParameter && this.account && this.region) {\r\n      vpc = ec2.Vpc.fromLookup(this, 'MailVpc', { tags: { Name: 'dya-vpc' } });\r\n      if (usePublicSubnets) {\r\n        serviceSubnetIds = props.publicSubnets!;\r\n      } else if (props.privateSubnets && props.privateSubnets.length > 0) {\r\n        serviceSubnetIds = props.privateSubnets;\r\n      } else {\r\n        const badAz = 'cac1-az4';\r\n        let subnets = vpc.privateSubnets.filter(s => s.availabilityZone !== badAz);\r\n        if (subnets.length === 0) subnets = vpc.publicSubnets.filter(s => s.availabilityZone !== badAz);\r\n        serviceSubnetIds = subnets.slice(0, 2).map(s => s.subnetId);\r\n        if (serviceSubnetIds.length < 1) throw new Error(`VPC must have at least one subnet in an allowed AZ.`);\r\n      }\r\n    } else {\r\n      const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', { type: 'AWS::EC2::VPC::Id', default: props.vpcId ?? '' });\r\n      const subnetParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', { type: 'CommaDelimitedList', default: props.privateSubnets?.join(',') ?? '' });\r\n      vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', { vpcId: vpcIdParam.valueAsString, availabilityZones: cdk.Fn.getAzs(), privateSubnetIds: subnetParam.valueAsList });\r\n      serviceSubnetIds = props.privateSubnets ?? subnetParam.valueAsList;\r\n    }\r\n\r\n    // Security Groups (L2)\r\n    const apiVpcLinkSg = new ec2.SecurityGroup(this, 'ApiVpcLinkSecurityGroup', { vpc, description: 'Egress from API Gateway VPC Link', allowAllOutbound: false });\r\n    apiVpcLinkSg.addEgressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(containerPort));\r\n\r\n    const serviceSg = new ec2.SecurityGroup(this, 'ServiceSecurityGroup', { vpc, description: 'Allow traffic to Fargate tasks', allowAllOutbound: true });\r\n    serviceSg.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(containerPort));\r\n\r\n    // Log Group (L2)\r\n    const appLogGroup = new logs.LogGroup(this, 'AppLogGroup', { logGroupName: '/ecs/mail-example', retention: logs.RetentionDays.ONE_MONTH, removalPolicy: cdk.RemovalPolicy.DESTROY });\r\n\r\n    // ECS Cluster (L2)\r\n    const cluster = new ecs.Cluster(this, 'MailCluster', { vpc, clusterName: 'public-cluster', enableFargateCapacityProviders: true });\r\n\r\n\r\n    // ECR Repository - create if not exists using custom resource\r\n    const ecrRepoName = 'mail-example';\r\n    const checkEcrRepo = new cr.AwsCustomResource(this, 'CheckEcrRepo', {\r\n      onCreate: { service: 'ECR', action: 'describeRepositories', parameters: { repositoryNames: [ecrRepoName] }, physicalResourceId: cr.PhysicalResourceId.of(ecrRepoName) },\r\n      policy: cr.AwsCustomResourcePolicy.fromSdkCalls({ resources: cr.AwsCustomResourcePolicy.ANY_RESOURCE }),\r\n    });\r\n    const createEcrRepo = new cr.AwsCustomResource(this, 'CreateEcrRepo', {\r\n      onCreate: { service: 'ECR', action: 'createRepository', parameters: { repositoryName: ecrRepoName, imageTagMutability: 'MUTABLE', encryptionConfiguration: { encryptionType: 'AES256' } }, physicalResourceId: cr.PhysicalResourceId.of(ecrRepoName), ignoreErrorCodesMatching: 'RepositoryAlreadyExistsException' },\r\n      policy: cr.AwsCustomResourcePolicy.fromSdkCalls({ resources: cr.AwsCustomResourcePolicy.ANY_RESOURCE }),\r\n    });\r\n    createEcrRepo.node.addDependency(checkEcrRepo);\r\n    const ecrRepo = ecr.Repository.fromRepositoryName(this, 'MailExampleRepositoryRef', ecrRepoName);\r\n\r\n    // API Gateway HTTP API (L1 - no L2 for HTTP API with VPC Link)\r\n    const httpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', { name: 'mail-example', protocolType: 'HTTP' });\r\n    const apiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', { stageName: 'prod', apiId: httpApi.ref, autoDeploy: true });\r\n\r\n    const apiDomainName = hasCustomDomain ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', {\r\n      domainName: apiDomain, domainNameConfigurations: [{ certificateArn: certArn!, endpointType: 'REGIONAL', securityPolicy: 'TLS_1_2' }],\r\n    }) : undefined;\r\n\r\n    const apiMapping = hasCustomDomain && apiDomainName ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', {\r\n      apiId: httpApi.ref, domainName: apiDomainName.ref, stage: apiStage.ref, apiMappingKey: 'examples',\r\n    }) : undefined;\r\n    if (apiMapping && apiDomainName) { apiMapping.addDependency(apiDomainName); apiMapping.addDependency(apiStage); }\r\n\r\n    const apiRecord = hasCustomDomain && hasHostedZone && apiDomainName ? new route53.CfnRecordSet(this, 'ApiDomainRecord', {\r\n      hostedZoneId: zoneId, name: apiDomain, type: 'A', aliasTarget: { dnsName: apiDomainName.attrRegionalDomainName, hostedZoneId: apiDomainName.attrRegionalHostedZoneId },\r\n    }) : undefined;\r\n    if (apiRecord && apiDomainName) apiRecord.addDependency(apiDomainName);\r\n\r\n    const baseUrl = hasCustomDomain ? `https://${apiDomain}/examples/gmail-viewer` : cdk.Fn.join('', [httpApi.attrApiEndpoint, '/', apiStage.ref, '/gmail-viewer']);\r\n\r\n    // Cloud Map Namespace (L2)\r\n    const namespace = new servicediscovery.PrivateDnsNamespace(this, 'ServiceNamespace', { name: nsName, vpc, description: 'Namespace for mail services' });\r\n\r\n    // VPC Link (L1)\r\n    const vpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', { name: 'mail-example-vpclink', subnetIds: serviceSubnetIds, securityGroupIds: [apiVpcLinkSg.securityGroupId] });\r\n\r\n    // Cloud Map Service (L2)\r\n    const cloudMapService = new servicediscovery.Service(this, 'AppCloudMapService', {\r\n      name: 'mail-example', namespace, dnsRecordType: servicediscovery.DnsRecordType.SRV, dnsTtl: nsTtl, routingPolicy: servicediscovery.RoutingPolicy.WEIGHTED, customHealthCheck: { failureThreshold: 1 },\r\n    });\r\n\r\n    // Secrets Manager - single secret with GOOGLE_CLIENT_SECRET and SESSION_SECRET\r\n    const appSecrets = new secretsmanager.Secret(this, 'AppSecrets', {\r\n      secretName: `${ssmPrefix}/secrets`,\r\n      description: 'Application secrets for mail-example',\r\n      secretObjectValue: {\r\n        GOOGLE_CLIENT_SECRET: cdk.SecretValue.unsafePlainText(googleClientSecret),\r\n        SESSION_SECRET: cdk.SecretValue.unsafePlainText(sessionSecretValue),\r\n      },\r\n    });\r\n\r\n    // IAM Roles (L2)\r\n    const taskExecRole = new iam.Role(this, 'TaskExecutionRole', { assumedBy: new iam.ServicePrincipal('ecs-tasks.amazonaws.com'), managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AmazonECSTaskExecutionRolePolicy')] });\r\n    taskExecRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], resources: [`arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, `arn:aws:kms:${this.region}:${this.account}:key/*`] }));\r\n    taskExecRole.addToPolicy(new iam.PolicyStatement({ actions: ['secretsmanager:GetSecretValue'], resources: [appSecrets.secretArn] }));\r\n\r\n    const taskRole = new iam.Role(this, 'TaskRole', { assumedBy: new iam.ServicePrincipal('ecs-tasks.amazonaws.com') });\r\n    taskRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], resources: [`arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, `arn:aws:kms:${this.region}:${this.account}:key/*`] }));\r\n    taskRole.addToPolicy(new iam.PolicyStatement({ actions: ['secretsmanager:GetSecretValue'], resources: [appSecrets.secretArn] }));\r\n\r\n    // SSM Parameters (L2) - create before task definition so they can be referenced\r\n    const ssmBaseUrl = new ssm.StringParameter(this, 'SsmBaseUrl', { parameterName: `${ssmPrefix}/env/BASE_URL`, stringValue: baseUrl.toString() });\r\n    const ssmGoogleClientId = new ssm.StringParameter(this, 'SsmGoogleClientId', { parameterName: `${ssmPrefix}/env/GOOGLE_CLIENT_ID`, stringValue: googleClientId });\r\n    new ssm.StringParameter(this, 'SsmPort', { parameterName: `${ssmPrefix}/env/PORT`, stringValue: containerPort.toString() });\r\n    new ssm.StringParameter(this, 'SsmPublicEnv', { parameterName: `${ssmPrefix}/env/NODE_ENV`, stringValue: publicEnvValue });\r\n\r\n    // ECS Task Definition (L2)\r\n    const taskDef = new ecs.FargateTaskDefinition(this, 'AppTaskDefinition', { family: 'mail-example', cpu: containerCpu, memoryLimitMiB: containerMemory, executionRole: taskExecRole, taskRole });\r\n    taskDef.addContainer('mail-example', {\r\n      image: ecs.ContainerImage.fromEcrRepository(ecrRepo, 'latest'), portMappings: [{ containerPort }],\r\n      logging: ecs.LogDrivers.awsLogs({ logGroup: appLogGroup, streamPrefix: 'mail-example' }),\r\n      environment: { CONFIG_SSM_PREFIX: ssmPrefix, PORT: containerPort.toString(), BASE_URL: baseUrl.toString(), NODE_ENV: publicEnvValue },\r\n      secrets: {\r\n        GOOGLE_CLIENT_ID: ecs.Secret.fromSsmParameter(ssmGoogleClientId),\r\n        GOOGLE_CLIENT_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_SECRET'),\r\n        SESSION_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'SESSION_SECRET'),\r\n      },\r\n    });\r\n\r\n    // ECS Service (L2) - associate with standalone Cloud Map service for API Gateway integration\r\n    const service = new ecs.FargateService(this, 'AppService', {\r\n      serviceName: 'mail-example', cluster, taskDefinition: taskDef, desiredCount,\r\n      capacityProviderStrategies: [{ capacityProvider: 'FARGATE_SPOT', weight: 4 }, { capacityProvider: 'FARGATE', weight: 1 }],\r\n      vpcSubnets: { subnets: serviceSubnetIds.map((id, i) => ec2.Subnet.fromSubnetId(this, `ServiceSubnet${i}`, id)) },\r\n      securityGroups: [serviceSg], assignPublicIp: usePublicSubnets ?? false, minHealthyPercent: 50, maxHealthyPercent: 200,\r\n    });\r\n    service.associateCloudMapService({ service: cloudMapService, containerPort, container: taskDef.defaultContainer! });\r\n\r\n    // API Gateway Integration (L1)\r\n    const integration = new apigatewayv2.CfnIntegration(this, 'MailIntegration', { apiId: httpApi.ref, integrationType: 'HTTP_PROXY', integrationMethod: 'ANY', integrationUri: cloudMapService.serviceArn, connectionType: 'VPC_LINK', connectionId: vpcLink.ref, payloadFormatVersion: '1.0' });\r\n    new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', { apiId: httpApi.ref, routeKey: 'ANY /gmail-viewer', target: `integrations/${integration.ref}` });\r\n    new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', { apiId: httpApi.ref, routeKey: 'ANY /gmail-viewer/{proxy+}', target: `integrations/${integration.ref}` });\r\n\r\n    // CodeBuild Role (L2)\r\n    const buildRole = new iam.Role(this, 'CodeBuildRole', { assumedBy: new iam.ServicePrincipal('codebuild.amazonaws.com') });\r\n    buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'], resources: ['*'] }));\r\n    buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ecr:GetAuthorizationToken', 'ecr:BatchCheckLayerAvailability', 'ecr:CompleteLayerUpload', 'ecr:BatchGetImage', 'ecr:DescribeRepositories', 'ecr:InitiateLayerUpload', 'ecr:PutImage', 'ecr:UploadLayerPart'], resources: ['*'] }));\r\n    buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'], resources: ['*'] }));\r\n    buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'], resources: ['*'] }));\r\n    if (props.codeConnectionArn) buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['codeconnections:GetConnection', 'codeconnections:GetConnectionToken', 'codeconnections:UseConnection'], resources: [props.codeConnectionArn] }));\r\n\r\n    // CodeBuild Project (L2)\r\n    new codebuild.Project(this, 'CodeBuildProject', {\r\n      projectName: 'mail-example', role: buildRole,\r\n      environment: { buildImage: codebuild.LinuxBuildImage.STANDARD_7_0, computeType: codebuild.ComputeType.SMALL, privileged: true, environmentVariables: { ECR_URI: { value: ecrRepo.repositoryUri }, CLUSTER_NAME: { value: cluster.clusterName }, SERVICE_NAME: { value: 'mail-example' } } },\r\n      source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch }),\r\n      timeout: cdk.Duration.minutes(30), queuedTimeout: cdk.Duration.minutes(30), badge: true, description: 'Build & deploy mail-example to ECR then force ECS deploy',\r\n      buildSpec: codebuild.BuildSpec.fromObject({\r\n        version: '0.2',\r\n        phases: {\r\n          pre_build: { commands: ['echo \"Logging in to ECR\"', 'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ECR_URI', 'IMAGE_TAG=${CODEBUILD_RESOLVED_SOURCE_VERSION:-latest}'] },\r\n          build: { commands: ['cd examples/gmail-viewer', 'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .'] },\r\n          post_build: { commands: ['docker push $ECR_URI:latest', 'docker push $ECR_URI:$IMAGE_TAG', 'aws ecs update-service --cluster $CLUSTER_NAME --service $SERVICE_NAME --force-new-deployment'] },\r\n        },\r\n        artifacts: { files: [] }, env: { shell: 'bash' },\r\n      }),\r\n    });\r\n\r\n    // Outputs\r\n    this.clusterName = cluster.clusterName;\r\n    new cdk.CfnOutput(this, 'ClusterName', { value: this.clusterName });\r\n    this.repositoryUri = ecrRepo.repositoryUri;\r\n    new cdk.CfnOutput(this, 'RepositoryUri', { value: this.repositoryUri });\r\n    this.apiInvokeUrl = baseUrl.toString();\r\n    new cdk.CfnOutput(this, 'ApiInvokeUrl', { value: this.apiInvokeUrl });\r\n    this.cloudMapServiceArn = cloudMapService.serviceArn;\r\n    new cdk.CfnOutput(this, 'CloudMapServiceArn', { value: this.cloudMapServiceArn });\r\n  }\r\n}\r\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gmail-viewer-cdk-stack.js","sourceRoot":"","sources":["gmail-viewer-cdk-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\r\nimport * as apigatewayv2 from 'aws-cdk-lib/aws-apigatewayv2';\r\nimport * as codebuild from 'aws-cdk-lib/aws-codebuild';\r\nimport * as cr from 'aws-cdk-lib/custom-resources';\r\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\r\nimport * as ecr from 'aws-cdk-lib/aws-ecr';\r\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\r\nimport * as iam from 'aws-cdk-lib/aws-iam';\r\nimport * as logs from 'aws-cdk-lib/aws-logs';\r\nimport * as route53 from 'aws-cdk-lib/aws-route53';\r\nimport * as servicediscovery from 'aws-cdk-lib/aws-servicediscovery';\r\nimport * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';\r\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\r\n\r\nexport interface GmailViewerCdkStackProps extends cdk.StackProps {\r\n  readonly vpcId?: string;\r\n  readonly privateSubnets?: string[];\r\n  readonly publicSubnets?: string[];\r\n  readonly containerPort?: number | string;\r\n  readonly containerCpu?: number | string;\r\n  readonly containerMemory?: number | string;\r\n  readonly desiredCount?: number | string;\r\n  readonly certificateArn?: string;\r\n  readonly apiCustomDomainName?: string;\r\n  readonly hostedZoneId?: string;\r\n  readonly serviceDiscoveryNamespaceName?: string;\r\n  readonly serviceDiscoveryTtl?: number | string;\r\n  readonly ssmPrefix?: string;\r\n  readonly gitHubOwner?: string;\r\n  readonly gitHubRepo?: string;\r\n  readonly gitHubBranch?: string;\r\n  readonly publicEnvValue?: string;\r\n  readonly googleClientId?: string;\r\n  readonly googleClientSecret?: string;\r\n  readonly sessionSecretValue?: string;\r\n  readonly useGitHubWebhooks?: boolean;\r\n  readonly codeConnectionArn?: string;\r\n}\r\n\r\nexport class GmailViewerCdkStack extends cdk.Stack {\r\n  public readonly clusterName: string;\r\n  public readonly repositoryUri: string;\r\n  public readonly apiInvokeUrl: string;\r\n  public readonly cloudMapServiceArn: string;\r\n\r\n  public constructor(scope: cdk.App, id: string, props: GmailViewerCdkStackProps) {\r\n    super(scope, id, props);\r\n\r\n    const toNumber = (value: string | number | undefined, fallback: number): number => {\r\n      const candidate = value ?? fallback;\r\n      const parsed = typeof candidate === 'number' ? candidate : Number(candidate);\r\n      return Number.isFinite(parsed) ? parsed : fallback;\r\n    };\r\n    const toCpuUnits = (value: string | number | undefined): number => {\r\n      const parsed = toNumber(value, 0.5);\r\n      const vcpu = parsed <= 4 ? parsed * 1024 : parsed;\r\n      return Math.max(256, Math.round(vcpu / 256) * 256);\r\n    };\r\n\r\n    const repoParts = (process.env.GITHUB_REPOSITORY ?? '').split('/');\r\n    const containerPort = toNumber(props.containerPort, 3000);\r\n    const containerCpu = toCpuUnits(props.containerCpu);\r\n    const containerMemory = toNumber(props.containerMemory, 1024);\r\n    const desiredCount = toNumber(props.desiredCount, 1);\r\n    const nsName = props.serviceDiscoveryNamespaceName ?? 'mail.local';\r\n    const nsTtl = cdk.Duration.seconds(toNumber(props.serviceDiscoveryTtl, 60));\r\n    const ssmPrefixRaw = props.ssmPrefix ?? '/mail-example';\r\n    const ssmPrefix = '/' + ssmPrefixRaw.split('/').filter(p => p).join('/');\r\n    const gitHubOwner = props.gitHubOwner ?? (repoParts[0] || 'dyanet');\r\n    const gitHubRepo = props.gitHubRepo ?? (repoParts[1] || 'imap');\r\n    const gitHubBranch = props.gitHubBranch ?? process.env.GITHUB_REF_NAME ?? 'main';\r\n    const publicEnvValue = props.publicEnvValue ?? 'production';\r\n    const googleClientId = props.googleClientId || 'PLACEHOLDER_GOOGLE_CLIENT_ID';\r\n    const googleClientSecret = props.googleClientSecret || 'PLACEHOLDER_GOOGLE_CLIENT_SECRET';\r\n    const sessionSecretValue = props.sessionSecretValue || 'PLACEHOLDER_SESSION_SECRET';\r\n    const apiDomain = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'demo.dyanet.com';\r\n    const certArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN;\r\n    const zoneId = props.hostedZoneId ?? '';\r\n    const hasCustomDomain = Boolean(certArn && apiDomain);\r\n    const hasHostedZone = Boolean(zoneId);\r\n    const usePublicSubnets = props.publicSubnets && props.publicSubnets.length > 0;\r\n\r\n    // VPC setup\r\n    let vpc: ec2.IVpc;\r\n    let serviceSubnetIds: string[];\r\n    const useVpcParameter = this.node.tryGetContext('useVpcParameter');\r\n\r\n    if (!useVpcParameter && this.account && this.region) {\r\n      vpc = ec2.Vpc.fromLookup(this, 'MailVpc', { tags: { Name: 'dya-vpc' } });\r\n      if (usePublicSubnets) {\r\n        serviceSubnetIds = props.publicSubnets!;\r\n      } else if (props.privateSubnets && props.privateSubnets.length > 0) {\r\n        serviceSubnetIds = props.privateSubnets;\r\n      } else {\r\n        const badAz = 'cac1-az4';\r\n        let subnets = vpc.privateSubnets.filter(s => s.availabilityZone !== badAz);\r\n        if (subnets.length === 0) subnets = vpc.publicSubnets.filter(s => s.availabilityZone !== badAz);\r\n        serviceSubnetIds = subnets.slice(0, 2).map(s => s.subnetId);\r\n        if (serviceSubnetIds.length < 1) throw new Error(`VPC must have at least one subnet in an allowed AZ.`);\r\n      }\r\n    } else {\r\n      const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', { type: 'AWS::EC2::VPC::Id', default: props.vpcId ?? '' });\r\n      const subnetParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', { type: 'CommaDelimitedList', default: props.privateSubnets?.join(',') ?? '' });\r\n      vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', { vpcId: vpcIdParam.valueAsString, availabilityZones: cdk.Fn.getAzs(), privateSubnetIds: subnetParam.valueAsList });\r\n      serviceSubnetIds = props.privateSubnets ?? subnetParam.valueAsList;\r\n    }\r\n\r\n    // Security Groups (L2)\r\n    const apiVpcLinkSg = new ec2.SecurityGroup(this, 'ApiVpcLinkSecurityGroup', { vpc, description: 'Egress from API Gateway VPC Link', allowAllOutbound: false });\r\n    apiVpcLinkSg.addEgressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(containerPort));\r\n\r\n    const serviceSg = new ec2.SecurityGroup(this, 'ServiceSecurityGroup', { vpc, description: 'Allow traffic to Fargate tasks', allowAllOutbound: true });\r\n    serviceSg.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(containerPort));\r\n\r\n    // Log Group (L2)\r\n    const appLogGroup = new logs.LogGroup(this, 'AppLogGroup', { logGroupName: '/ecs/mail-example', retention: logs.RetentionDays.ONE_MONTH, removalPolicy: cdk.RemovalPolicy.DESTROY });\r\n\r\n    // ECS Cluster (L2)\r\n    const cluster = new ecs.Cluster(this, 'MailCluster', { vpc, clusterName: 'public-cluster', enableFargateCapacityProviders: true });\r\n\r\n\r\n    // ECR Repository - create if not exists using custom resource\r\n    const ecrRepoName = 'mail-example';\r\n    const checkEcrRepo = new cr.AwsCustomResource(this, 'CheckEcrRepo', {\r\n      onCreate: { service: 'ECR', action: 'describeRepositories', parameters: { repositoryNames: [ecrRepoName] }, physicalResourceId: cr.PhysicalResourceId.of(ecrRepoName) },\r\n      policy: cr.AwsCustomResourcePolicy.fromSdkCalls({ resources: cr.AwsCustomResourcePolicy.ANY_RESOURCE }),\r\n    });\r\n    const createEcrRepo = new cr.AwsCustomResource(this, 'CreateEcrRepo', {\r\n      onCreate: { service: 'ECR', action: 'createRepository', parameters: { repositoryName: ecrRepoName, imageTagMutability: 'MUTABLE', encryptionConfiguration: { encryptionType: 'AES256' } }, physicalResourceId: cr.PhysicalResourceId.of(ecrRepoName), ignoreErrorCodesMatching: 'RepositoryAlreadyExistsException' },\r\n      policy: cr.AwsCustomResourcePolicy.fromSdkCalls({ resources: cr.AwsCustomResourcePolicy.ANY_RESOURCE }),\r\n    });\r\n    createEcrRepo.node.addDependency(checkEcrRepo);\r\n    const ecrRepo = ecr.Repository.fromRepositoryName(this, 'MailExampleRepositoryRef', ecrRepoName);\r\n\r\n    // API Gateway HTTP API (L1 - no L2 for HTTP API with VPC Link)\r\n    const httpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', { name: 'mail-example', protocolType: 'HTTP' });\r\n    const apiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', { stageName: 'prod', apiId: httpApi.ref, autoDeploy: true });\r\n\r\n    const apiDomainName = hasCustomDomain ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', {\r\n      domainName: apiDomain, domainNameConfigurations: [{ certificateArn: certArn!, endpointType: 'REGIONAL', securityPolicy: 'TLS_1_2' }],\r\n    }) : undefined;\r\n\r\n    const apiMapping = hasCustomDomain && apiDomainName ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', {\r\n      apiId: httpApi.ref, domainName: apiDomainName.ref, stage: apiStage.ref, apiMappingKey: 'examples',\r\n    }) : undefined;\r\n    if (apiMapping && apiDomainName) { apiMapping.addDependency(apiDomainName); apiMapping.addDependency(apiStage); }\r\n\r\n    const apiRecord = hasCustomDomain && hasHostedZone && apiDomainName ? new route53.CfnRecordSet(this, 'ApiDomainRecord', {\r\n      hostedZoneId: zoneId, name: apiDomain, type: 'A', aliasTarget: { dnsName: apiDomainName.attrRegionalDomainName, hostedZoneId: apiDomainName.attrRegionalHostedZoneId },\r\n    }) : undefined;\r\n    if (apiRecord && apiDomainName) apiRecord.addDependency(apiDomainName);\r\n\r\n    const baseUrl = hasCustomDomain ? `https://${apiDomain}/examples/gmail-viewer` : cdk.Fn.join('', [httpApi.attrApiEndpoint, '/', apiStage.ref, '/gmail-viewer']);\r\n\r\n    // Cloud Map Namespace (L2)\r\n    const namespace = new servicediscovery.PrivateDnsNamespace(this, 'ServiceNamespace', { name: nsName, vpc, description: 'Namespace for mail services' });\r\n\r\n    // VPC Link (L1)\r\n    const vpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', { name: 'mail-example-vpclink', subnetIds: serviceSubnetIds, securityGroupIds: [apiVpcLinkSg.securityGroupId] });\r\n\r\n    // Cloud Map Service (L2)\r\n    const cloudMapService = new servicediscovery.Service(this, 'AppCloudMapService', {\r\n      name: 'mail-example', namespace, dnsRecordType: servicediscovery.DnsRecordType.SRV, dnsTtl: nsTtl, routingPolicy: servicediscovery.RoutingPolicy.WEIGHTED, customHealthCheck: { failureThreshold: 1 },\r\n    });\r\n\r\n    // Secrets Manager - all config values for the application\r\n    const appSecrets = new secretsmanager.Secret(this, 'AppSecrets', {\r\n      secretName: `${ssmPrefix}/config`,\r\n      description: 'Application configuration for mail-example',\r\n      secretObjectValue: {\r\n        GOOGLE_CLIENT_ID: cdk.SecretValue.unsafePlainText(googleClientId),\r\n        GOOGLE_CLIENT_SECRET: cdk.SecretValue.unsafePlainText(googleClientSecret),\r\n        SESSION_SECRET: cdk.SecretValue.unsafePlainText(sessionSecretValue),\r\n        BASE_URL: cdk.SecretValue.unsafePlainText(baseUrl.toString()),\r\n        PORT: cdk.SecretValue.unsafePlainText(containerPort.toString()),\r\n      },\r\n    });\r\n\r\n    // IAM Roles (L2)\r\n    const taskExecRole = new iam.Role(this, 'TaskExecutionRole', { assumedBy: new iam.ServicePrincipal('ecs-tasks.amazonaws.com'), managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AmazonECSTaskExecutionRolePolicy')] });\r\n    taskExecRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], resources: [`arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, `arn:aws:kms:${this.region}:${this.account}:key/*`] }));\r\n    taskExecRole.addToPolicy(new iam.PolicyStatement({ actions: ['secretsmanager:GetSecretValue'], resources: [appSecrets.secretArn] }));\r\n\r\n    const taskRole = new iam.Role(this, 'TaskRole', { assumedBy: new iam.ServicePrincipal('ecs-tasks.amazonaws.com') });\r\n    taskRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], resources: [`arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, `arn:aws:kms:${this.region}:${this.account}:key/*`] }));\r\n    taskRole.addToPolicy(new iam.PolicyStatement({ actions: ['secretsmanager:GetSecretValue'], resources: [appSecrets.secretArn] }));\r\n\r\n    // ECS Task Definition (L2)\r\n    const taskDef = new ecs.FargateTaskDefinition(this, 'AppTaskDefinition', { family: 'mail-example', cpu: containerCpu, memoryLimitMiB: containerMemory, executionRole: taskExecRole, taskRole });\r\n    taskDef.addContainer('mail-example', {\r\n      image: ecs.ContainerImage.fromEcrRepository(ecrRepo, 'latest'), portMappings: [{ containerPort }],\r\n      logging: ecs.LogDrivers.awsLogs({ logGroup: appLogGroup, streamPrefix: 'mail-example' }),\r\n      environment: { CONFIG_SSM_PREFIX: ssmPrefix, PORT: containerPort.toString(), BASE_URL: baseUrl.toString(), NODE_ENV: publicEnvValue },\r\n      secrets: {\r\n        GOOGLE_CLIENT_ID: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_ID'),\r\n        GOOGLE_CLIENT_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_SECRET'),\r\n        SESSION_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'SESSION_SECRET'),\r\n      },\r\n    });\r\n\r\n    // ECS Service (L2) - associate with standalone Cloud Map service for API Gateway integration\r\n    const service = new ecs.FargateService(this, 'AppService', {\r\n      serviceName: 'mail-example', cluster, taskDefinition: taskDef, desiredCount,\r\n      capacityProviderStrategies: [{ capacityProvider: 'FARGATE_SPOT', weight: 4 }, { capacityProvider: 'FARGATE', weight: 1 }],\r\n      vpcSubnets: { subnets: serviceSubnetIds.map((id, i) => ec2.Subnet.fromSubnetId(this, `ServiceSubnet${i}`, id)) },\r\n      securityGroups: [serviceSg], assignPublicIp: usePublicSubnets ?? false, minHealthyPercent: 50, maxHealthyPercent: 200,\r\n    });\r\n    service.associateCloudMapService({ service: cloudMapService, containerPort, container: taskDef.defaultContainer! });\r\n\r\n    // API Gateway Integration (L1)\r\n    const integration = new apigatewayv2.CfnIntegration(this, 'MailIntegration', { apiId: httpApi.ref, integrationType: 'HTTP_PROXY', integrationMethod: 'ANY', integrationUri: cloudMapService.serviceArn, connectionType: 'VPC_LINK', connectionId: vpcLink.ref, payloadFormatVersion: '1.0' });\r\n    new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', { apiId: httpApi.ref, routeKey: 'ANY /gmail-viewer', target: `integrations/${integration.ref}` });\r\n    new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', { apiId: httpApi.ref, routeKey: 'ANY /gmail-viewer/{proxy+}', target: `integrations/${integration.ref}` });\r\n\r\n    // CodeBuild Role (L2)\r\n    const buildRole = new iam.Role(this, 'CodeBuildRole', { assumedBy: new iam.ServicePrincipal('codebuild.amazonaws.com') });\r\n    buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'], resources: ['*'] }));\r\n    buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ecr:GetAuthorizationToken', 'ecr:BatchCheckLayerAvailability', 'ecr:CompleteLayerUpload', 'ecr:BatchGetImage', 'ecr:DescribeRepositories', 'ecr:InitiateLayerUpload', 'ecr:PutImage', 'ecr:UploadLayerPart'], resources: ['*'] }));\r\n    buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'], resources: ['*'] }));\r\n    buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'], resources: ['*'] }));\r\n    if (props.codeConnectionArn) buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['codeconnections:GetConnection', 'codeconnections:GetConnectionToken', 'codeconnections:UseConnection'], resources: [props.codeConnectionArn] }));\r\n\r\n    // CodeBuild Project (L2)\r\n    new codebuild.Project(this, 'CodeBuildProject', {\r\n      projectName: 'mail-example', role: buildRole,\r\n      environment: { buildImage: codebuild.LinuxBuildImage.STANDARD_7_0, computeType: codebuild.ComputeType.SMALL, privileged: true, environmentVariables: { ECR_URI: { value: ecrRepo.repositoryUri }, CLUSTER_NAME: { value: cluster.clusterName }, SERVICE_NAME: { value: 'mail-example' } } },\r\n      source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch }),\r\n      timeout: cdk.Duration.minutes(30), queuedTimeout: cdk.Duration.minutes(30), badge: true, description: 'Build & deploy mail-example to ECR then force ECS deploy',\r\n      buildSpec: codebuild.BuildSpec.fromObject({\r\n        version: '0.2',\r\n        phases: {\r\n          pre_build: { commands: ['echo \"Logging in to ECR\"', 'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ECR_URI', 'IMAGE_TAG=${CODEBUILD_RESOLVED_SOURCE_VERSION:-latest}'] },\r\n          build: { commands: ['cd examples/gmail-viewer', 'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .'] },\r\n          post_build: { commands: ['docker push $ECR_URI:latest', 'docker push $ECR_URI:$IMAGE_TAG', 'aws ecs update-service --cluster $CLUSTER_NAME --service $SERVICE_NAME --force-new-deployment'] },\r\n        },\r\n        artifacts: { files: [] }, env: { shell: 'bash' },\r\n      }),\r\n    });\r\n\r\n    // Outputs\r\n    this.clusterName = cluster.clusterName;\r\n    new cdk.CfnOutput(this, 'ClusterName', { value: this.clusterName });\r\n    this.repositoryUri = ecrRepo.repositoryUri;\r\n    new cdk.CfnOutput(this, 'RepositoryUri', { value: this.repositoryUri });\r\n    this.apiInvokeUrl = baseUrl.toString();\r\n    new cdk.CfnOutput(this, 'ApiInvokeUrl', { value: this.apiInvokeUrl });\r\n    this.cloudMapServiceArn = cloudMapService.serviceArn;\r\n    new cdk.CfnOutput(this, 'CloudMapServiceArn', { value: this.cloudMapServiceArn });\r\n  }\r\n}\r\n"]} \ No newline at end of file diff --git a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.ts b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.ts index caea0e5..2eecb7e 100644 --- a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.ts +++ b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.ts @@ -73,7 +73,7 @@ export class GmailViewerCdkStack extends cdk.Stack { const googleClientId = props.googleClientId || 'PLACEHOLDER_GOOGLE_CLIENT_ID'; const googleClientSecret = props.googleClientSecret || 'PLACEHOLDER_GOOGLE_CLIENT_SECRET'; const sessionSecretValue = props.sessionSecretValue || 'PLACEHOLDER_SESSION_SECRET'; - const apiDomain = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'mail.dyanet.com'; + const apiDomain = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'demo.dyanet.com'; const certArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN; const zoneId = props.hostedZoneId ?? ''; const hasCustomDomain = Boolean(certArn && apiDomain); @@ -163,13 +163,16 @@ export class GmailViewerCdkStack extends cdk.Stack { name: 'mail-example', namespace, dnsRecordType: servicediscovery.DnsRecordType.SRV, dnsTtl: nsTtl, routingPolicy: servicediscovery.RoutingPolicy.WEIGHTED, customHealthCheck: { failureThreshold: 1 }, }); - // Secrets Manager - single secret with GOOGLE_CLIENT_SECRET and SESSION_SECRET + // Secrets Manager - all config values for the application const appSecrets = new secretsmanager.Secret(this, 'AppSecrets', { - secretName: `${ssmPrefix}/secrets`, - description: 'Application secrets for mail-example', + secretName: `${ssmPrefix}/config`, + description: 'Application configuration for mail-example', secretObjectValue: { + GOOGLE_CLIENT_ID: cdk.SecretValue.unsafePlainText(googleClientId), GOOGLE_CLIENT_SECRET: cdk.SecretValue.unsafePlainText(googleClientSecret), SESSION_SECRET: cdk.SecretValue.unsafePlainText(sessionSecretValue), + BASE_URL: cdk.SecretValue.unsafePlainText(baseUrl.toString()), + PORT: cdk.SecretValue.unsafePlainText(containerPort.toString()), }, }); @@ -182,12 +185,6 @@ export class GmailViewerCdkStack extends cdk.Stack { taskRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], resources: [`arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, `arn:aws:kms:${this.region}:${this.account}:key/*`] })); taskRole.addToPolicy(new iam.PolicyStatement({ actions: ['secretsmanager:GetSecretValue'], resources: [appSecrets.secretArn] })); - // SSM Parameters (L2) - create before task definition so they can be referenced - const ssmBaseUrl = new ssm.StringParameter(this, 'SsmBaseUrl', { parameterName: `${ssmPrefix}/env/BASE_URL`, stringValue: baseUrl.toString() }); - const ssmGoogleClientId = new ssm.StringParameter(this, 'SsmGoogleClientId', { parameterName: `${ssmPrefix}/env/GOOGLE_CLIENT_ID`, stringValue: googleClientId }); - new ssm.StringParameter(this, 'SsmPort', { parameterName: `${ssmPrefix}/env/PORT`, stringValue: containerPort.toString() }); - new ssm.StringParameter(this, 'SsmPublicEnv', { parameterName: `${ssmPrefix}/env/NODE_ENV`, stringValue: publicEnvValue }); - // ECS Task Definition (L2) const taskDef = new ecs.FargateTaskDefinition(this, 'AppTaskDefinition', { family: 'mail-example', cpu: containerCpu, memoryLimitMiB: containerMemory, executionRole: taskExecRole, taskRole }); taskDef.addContainer('mail-example', { @@ -195,7 +192,7 @@ export class GmailViewerCdkStack extends cdk.Stack { logging: ecs.LogDrivers.awsLogs({ logGroup: appLogGroup, streamPrefix: 'mail-example' }), environment: { CONFIG_SSM_PREFIX: ssmPrefix, PORT: containerPort.toString(), BASE_URL: baseUrl.toString(), NODE_ENV: publicEnvValue }, secrets: { - GOOGLE_CLIENT_ID: ecs.Secret.fromSsmParameter(ssmGoogleClientId), + GOOGLE_CLIENT_ID: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_ID'), GOOGLE_CLIENT_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_SECRET'), SESSION_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'SESSION_SECRET'), },