From 336ec592a51a710aabff6225a17806ffbd5ba7b0 Mon Sep 17 00:00:00 2001 From: Akber Choudhry Date: Sun, 4 Jan 2026 14:34:04 -0500 Subject: [PATCH 1/3] chore: update CI actions to v6, publish to GitHub Packages only --- .github/workflows/ci.yml | 33 +++++++++++---------------------- 1 file changed, 11 insertions(+), 22 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 91c9d58..ee32b4f 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -14,10 +14,10 @@ jobs: node-version: [20, 22, 24] steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 - name: Use Node.js ${{ matrix.node-version }} - uses: actions/setup-node@v4 + uses: actions/setup-node@v6 with: node-version: ${{ matrix.node-version }} cache: 'npm' @@ -38,15 +38,18 @@ jobs: needs: test runs-on: ubuntu-latest if: github.ref == 'refs/heads/main' && github.event_name == 'push' + permissions: + contents: read + packages: write steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 - name: Use Node.js 20 - uses: actions/setup-node@v4 + uses: actions/setup-node@v6 with: node-version: 20 - registry-url: 'https://registry.npmjs.org' + registry-url: 'https://npm.pkg.github.com' cache: 'npm' - name: Install dependencies @@ -55,21 +58,7 @@ jobs: - name: Build run: npm run build - - name: Check if version changed - id: version-check - run: | - PACKAGE_VERSION=$(node -p "require('./package.json').version") - NPM_VERSION=$(npm view @dyanet/imap version 2>/dev/null || echo "0.0.0") - if [ "$PACKAGE_VERSION" != "$NPM_VERSION" ]; then - echo "should_publish=true" >> $GITHUB_OUTPUT - echo "Package version $PACKAGE_VERSION differs from npm version $NPM_VERSION" - else - echo "should_publish=false" >> $GITHUB_OUTPUT - echo "Package version $PACKAGE_VERSION matches npm version" - fi - - - name: Publish to npm - if: steps.version-check.outputs.should_publish == 'true' - run: npm publish --access public + - name: Publish to GitHub Packages + run: npm publish env: - NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }} + NODE_AUTH_TOKEN: ${{ secrets.GITHUB_TOKEN }} From 417c0ef3ca4955478699d6962810f2ac2d837794 Mon Sep 17 00:00:00 2001 From: Akber Choudhry Date: Sun, 4 Jan 2026 15:57:51 -0500 Subject: [PATCH 2/3] chore: lock dependencies for gmail-viewer example --- .../gmail-viewer-cdk/bin/gmail-viewer-cdk.js | 2 +- .../lib/gmail-viewer-cdk-stack.js | 2 +- .../test/gmail-viewer-cdk.test.js | 2 +- .../test/mail_example_stack.test.js | 2 +- examples/gmail-viewer/package-lock.json | 1074 +++++++++++++++++ examples/gmail-viewer/package.json | 2 +- 6 files changed, 1079 insertions(+), 5 deletions(-) create mode 100644 examples/gmail-viewer/package-lock.json diff --git a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js index f41e92b..aad315a 100644 --- a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js +++ b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js @@ -21,4 +21,4 @@ new gmail_viewer_cdk_stack_1.GmailViewerCdkStack(app, 'GmailViewerCdkStack', { apiCustomDomainName: 'mail.dyanet.com', certificateArn: 'arn:aws:acm:ca-central-1:239030031457:certificate/af9150f1-4635-4154-8b19-e5449d57e971', }); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ21haWwtdmlld2VyLWNkay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImdtYWlsLXZpZXdlci1jZGsudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EsNkNBQWtDO0FBQ2xDLDBFQUFvRTtBQUVwRSxNQUFNLEdBQUcsR0FBRyxJQUFJLGlCQUFHLEVBQUUsQ0FBQztBQUV0QiwwRkFBMEY7QUFDMUYsK0RBQStEO0FBQy9ELE1BQU0sY0FBYyxHQUFHLEdBQUcsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLGdCQUFnQixDQUFDLEVBQUUsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBRTVFLElBQUksNENBQW1CLENBQUMsR0FBRyxFQUFFLHFCQUFxQixFQUFFO0lBQ2xELEdBQUcsRUFBRTtRQUNILE9BQU8sRUFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLG1CQUFtQjtRQUN4QyxNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0I7S0FDdkM7SUFDRCxjQUFjLEVBQUUsQ0FBQywwQkFBMEIsRUFBRywwQkFBMEIsQ0FBQztJQUN6RSxXQUFXLEVBQUUsUUFBUTtJQUNyQixVQUFVLEVBQUUsTUFBTTtJQUNsQixZQUFZLEVBQUUsTUFBTTtJQUNwQixpQkFBaUIsRUFBRSxtR0FBbUc7SUFDdEgsaUJBQWlCLEVBQUUsS0FBSztJQUN4QixtQkFBbUIsRUFBRSxpQkFBaUI7SUFDdEMsY0FBYyxFQUFFLHdGQUF3RjtDQUN6RyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXG5pbXBvcnQgeyBBcHAgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBHbWFpbFZpZXdlckNka1N0YWNrIH0gZnJvbSAnLi4vbGliL2dtYWlsLXZpZXdlci1jZGstc3RhY2snO1xuXG5jb25zdCBhcHAgPSBuZXcgQXBwKCk7XG5cbi8vIFRvIHNwZWNpZnkgc3VibmV0cywgdXNlIHRoZSAtLWNvbnRleHQgb3B0aW9uIHdpdGggYSBjb21tYS1zZXBhcmF0ZWQgbGlzdCBvZiBzdWJuZXQgSURzOlxuLy8gY2RrIGRlcGxveSAtYyBwcml2YXRlU3VibmV0cz1zdWJuZXQteHh4eHh4eHgsc3VibmV0LXl5eXl5eXl5XG5jb25zdCBwcml2YXRlU3VibmV0cyA9IGFwcC5ub2RlLnRyeUdldENvbnRleHQoJ3ByaXZhdGVTdWJuZXRzJyk/LnNwbGl0KCcsJyk7XG5cbm5ldyBHbWFpbFZpZXdlckNka1N0YWNrKGFwcCwgJ0dtYWlsVmlld2VyQ2RrU3RhY2snLCB7XG4gIGVudjoge1xuICAgIGFjY291bnQ6IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX0FDQ09VTlQsXG4gICAgcmVnaW9uOiBwcm9jZXNzLmVudi5DREtfREVGQVVMVF9SRUdJT04sXG4gIH0sXG4gIHByaXZhdGVTdWJuZXRzOiBbJ3N1Ym5ldC0wZDgyOTBkMGVhNTc1MzMxYycsIFx0J3N1Ym5ldC0wMzIyOTA1NjlhN2VjZjIxMCddLFxuICBnaXRIdWJPd25lcjogJ2R5YW5ldCcsXG4gIGdpdEh1YlJlcG86ICdpbWFwJyxcbiAgZ2l0SHViQnJhbmNoOiAnbWFpbicsXG4gIGNvZGVDb25uZWN0aW9uQXJuOiAnYXJuOmF3czpjb2RlY29ubmVjdGlvbnM6Y2EtY2VudHJhbC0xOjIzOTAzMDAzMTQ1Nzpjb25uZWN0aW9uLzI5ZmU1YzExLTIzZDItNGMxMC1hODFkLTRlODgwYjEyZTJjNicsXG4gIHVzZUdpdEh1YldlYmhvb2tzOiBmYWxzZSxcbiAgYXBpQ3VzdG9tRG9tYWluTmFtZTogJ21haWwuZHlhbmV0LmNvbScsXG4gIGNlcnRpZmljYXRlQXJuOiAnYXJuOmF3czphY206Y2EtY2VudHJhbC0xOjIzOTAzMDAzMTQ1NzpjZXJ0aWZpY2F0ZS9hZjkxNTBmMS00NjM1LTQxNTQtOGIxOS1lNTQ0OWQ1N2U5NzEnLFxufSk7XG4iXX0= \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ21haWwtdmlld2VyLWNkay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImdtYWlsLXZpZXdlci1jZGsudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EsNkNBQWtDO0FBQ2xDLDBFQUFvRTtBQUVwRSxNQUFNLEdBQUcsR0FBRyxJQUFJLGlCQUFHLEVBQUUsQ0FBQztBQUV0QiwwRkFBMEY7QUFDMUYsK0RBQStEO0FBQy9ELE1BQU0sY0FBYyxHQUFHLEdBQUcsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLGdCQUFnQixDQUFDLEVBQUUsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBRTVFLElBQUksNENBQW1CLENBQUMsR0FBRyxFQUFFLHFCQUFxQixFQUFFO0lBQ2xELEdBQUcsRUFBRTtRQUNILE9BQU8sRUFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLG1CQUFtQjtRQUN4QyxNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0I7S0FDdkM7SUFDRCxjQUFjLEVBQUUsQ0FBQywwQkFBMEIsRUFBRywwQkFBMEIsQ0FBQztJQUN6RSxXQUFXLEVBQUUsUUFBUTtJQUNyQixVQUFVLEVBQUUsTUFBTTtJQUNsQixZQUFZLEVBQUUsTUFBTTtJQUNwQixpQkFBaUIsRUFBRSxtR0FBbUc7SUFDdEgsaUJBQWlCLEVBQUUsS0FBSztJQUN4QixtQkFBbUIsRUFBRSxpQkFBaUI7SUFDdEMsY0FBYyxFQUFFLHdGQUF3RjtDQUN6RyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXHJcbmltcG9ydCB7IEFwcCB9IGZyb20gJ2F3cy1jZGstbGliJztcclxuaW1wb3J0IHsgR21haWxWaWV3ZXJDZGtTdGFjayB9IGZyb20gJy4uL2xpYi9nbWFpbC12aWV3ZXItY2RrLXN0YWNrJztcclxuXHJcbmNvbnN0IGFwcCA9IG5ldyBBcHAoKTtcclxuXHJcbi8vIFRvIHNwZWNpZnkgc3VibmV0cywgdXNlIHRoZSAtLWNvbnRleHQgb3B0aW9uIHdpdGggYSBjb21tYS1zZXBhcmF0ZWQgbGlzdCBvZiBzdWJuZXQgSURzOlxyXG4vLyBjZGsgZGVwbG95IC1jIHByaXZhdGVTdWJuZXRzPXN1Ym5ldC14eHh4eHh4eCxzdWJuZXQteXl5eXl5eXlcclxuY29uc3QgcHJpdmF0ZVN1Ym5ldHMgPSBhcHAubm9kZS50cnlHZXRDb250ZXh0KCdwcml2YXRlU3VibmV0cycpPy5zcGxpdCgnLCcpO1xyXG5cclxubmV3IEdtYWlsVmlld2VyQ2RrU3RhY2soYXBwLCAnR21haWxWaWV3ZXJDZGtTdGFjaycsIHtcclxuICBlbnY6IHtcclxuICAgIGFjY291bnQ6IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX0FDQ09VTlQsXHJcbiAgICByZWdpb246IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX1JFR0lPTixcclxuICB9LFxyXG4gIHByaXZhdGVTdWJuZXRzOiBbJ3N1Ym5ldC0wZDgyOTBkMGVhNTc1MzMxYycsIFx0J3N1Ym5ldC0wMzIyOTA1NjlhN2VjZjIxMCddLFxyXG4gIGdpdEh1Yk93bmVyOiAnZHlhbmV0JyxcclxuICBnaXRIdWJSZXBvOiAnaW1hcCcsXHJcbiAgZ2l0SHViQnJhbmNoOiAnbWFpbicsXHJcbiAgY29kZUNvbm5lY3Rpb25Bcm46ICdhcm46YXdzOmNvZGVjb25uZWN0aW9uczpjYS1jZW50cmFsLTE6MjM5MDMwMDMxNDU3OmNvbm5lY3Rpb24vMjlmZTVjMTEtMjNkMi00YzEwLWE4MWQtNGU4ODBiMTJlMmM2JyxcclxuICB1c2VHaXRIdWJXZWJob29rczogZmFsc2UsXHJcbiAgYXBpQ3VzdG9tRG9tYWluTmFtZTogJ21haWwuZHlhbmV0LmNvbScsXHJcbiAgY2VydGlmaWNhdGVBcm46ICdhcm46YXdzOmFjbTpjYS1jZW50cmFsLTE6MjM5MDMwMDMxNDU3OmNlcnRpZmljYXRlL2FmOTE1MGYxLTQ2MzUtNDE1NC04YjE5LWU1NDQ5ZDU3ZTk3MScsXHJcbn0pO1xyXG4iXX0= \ No newline at end of file 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 8732e6a..ae921de 100644 --- a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.js +++ b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.js @@ -667,4 +667,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';\nimport * as apigatewayv2 from 'aws-cdk-lib/aws-apigatewayv2';\nimport * as codebuild from 'aws-cdk-lib/aws-codebuild';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as ecr from 'aws-cdk-lib/aws-ecr';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport * as servicediscovery from 'aws-cdk-lib/aws-servicediscovery';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\n\nexport interface GmailViewerCdkStackProps extends cdk.StackProps {\n  readonly vpcId?: string;\n  readonly privateSubnets?: string[];\n  readonly containerPort?: number | string;\n  readonly containerCpu?: number | string;\n  readonly containerMemory?: number | string;\n  readonly desiredCount?: number | string;\n  readonly certificateArn?: string;\n  readonly apiCustomDomainName?: string;\n  readonly hostedZoneId?: string;\n  readonly serviceDiscoveryNamespaceName?: string;\n  readonly serviceDiscoveryTtl?: number | string;\n  readonly ssmPrefix?: string;\n  readonly gitHubOwner?: string;\n  readonly gitHubRepo?: string;\n  readonly gitHubBranch?: string;\n  readonly publicEnvValue?: string;\n  readonly googleClientId?: string;\n  readonly googleClientSecret?: string;\n  readonly sessionSecretValue?: string;\n  readonly useGitHubWebhooks?: boolean;\n  readonly codeConnectionArn?: string;\n}\n\nexport class GmailViewerCdkStack extends cdk.Stack {\n  public readonly clusterName: string;\n  public readonly repositoryUri: string;\n  public readonly apiInvokeUrl: string;\n  public readonly cloudMapServiceArn: string;\n\n  public constructor(scope: cdk.App, id: string, props: GmailViewerCdkStackProps) {\n    super(scope, id, props);\n\n    const toNumber = (value: string | number | undefined, fallback: number): number => {\n      const candidate = value ?? fallback;\n      const parsed = typeof candidate === 'number' ? candidate : Number(candidate);\n      return Number.isFinite(parsed) ? parsed : fallback;\n    };\n\n    const toCpuUnits = (value: string | number | undefined): number => {\n      const parsed = toNumber(value, 0.5);\n      const vcpu = parsed <= 4 ? parsed * 1024 : parsed;\n      const rounded = Math.round(vcpu / 256) * 256;\n      return Math.max(256, rounded);\n    };\n\n    const repoParts = (process.env.GITHUB_REPOSITORY ?? '').split('/');\n    const guessedGitHubOwner = repoParts[0] || 'dyanet';\n    const guessedGitHubRepo = repoParts[1] || 'imap';\n\n    const containerPort = toNumber(props.containerPort, 3000);\n    const containerCpu = toCpuUnits(props.containerCpu);\n    const containerMemory = toNumber(props.containerMemory, 1024);\n    const desiredCount = toNumber(props.desiredCount, 1);\n    const serviceDiscoveryNamespaceName = props.serviceDiscoveryNamespaceName ?? 'mail.local';\n    const serviceDiscoveryTtl = toNumber(props.serviceDiscoveryTtl, 60);\n    const ssmPrefixRaw = props.ssmPrefix ?? '/mail-example';\n    const ssmPrefix = '/' + ssmPrefixRaw.split('/').filter(p => p).join('/');\n    const gitHubOwner = props.gitHubOwner ?? guessedGitHubOwner;\n    const gitHubRepo = props.gitHubRepo ?? guessedGitHubRepo;\n    const gitHubBranch = props.gitHubBranch ?? process.env.GITHUB_REF_NAME ?? 'main';\n    const publicEnvValue = props.publicEnvValue ?? 'production';\n    const googleClientId = props.googleClientId || 'not-set';\n    const googleClientSecret = props.googleClientSecret ?? 'not-set';\n    const sessionSecretValue = props.sessionSecretValue ?? 'not-set';\n    const apiCustomDomainName = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'mail.dyanet.com';\n    const certificateArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN;\n    const hostedZoneId = props.hostedZoneId ?? '';\n    const hasCustomDomain = Boolean(certificateArn && apiCustomDomainName);\n    const hasHostedZone = Boolean(hostedZoneId);\n\n    let vpc: ec2.IVpc;\n    let privateSubnetIds: string[];\n    let serviceSubnetIds: string[];\n\n    // If account/region are available, use Vpc.fromLookup to automatically discover subnets.\n    // Otherwise fall back to CloudFormation parameters which let the deployer pick values in the console.\n    const useVpcParameter = this.node.tryGetContext('useVpcParameter');\n\n    if (!useVpcParameter && this.account && this.region) {\n      vpc = ec2.Vpc.fromLookup(this, 'MailVpc', { tags: { Name: 'dya-vpc' } });\n      \n      if (props.privateSubnets && props.privateSubnets.length > 0) {\n          privateSubnetIds = props.privateSubnets;\n      } else {\n          // The service linked to the VPC Link may not be available in all Availability Zones.\n          const badAz = 'cac1-az4';\n          \n          let subnets = vpc.privateSubnets.filter(s => s.availabilityZone !== badAz);\n          if (subnets.length === 0) {\n            // No private subnets, try public subnets\n            subnets = vpc.publicSubnets.filter(s => s.availabilityZone !== badAz);\n          }\n\n          privateSubnetIds = subnets.slice(0, 2).map(s => s.subnetId);\n          \n          if (privateSubnetIds.length < 1) {\n            throw new Error(`VPC '${vpc.vpcId}' must have at least one private or public subnet in an allowed availability zone (not in ${badAz}).`);\n          }\n      }\n      serviceSubnetIds = privateSubnetIds;\n    } else {\n      // CloudFormation parameter for VPC selection (dropdown of VPC IDs in the console)\n      const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', {\n        type: 'AWS::EC2::VPC::Id',\n        default: props.vpcId ?? '',\n      });\n\n      // Optional parameter for private subnet IDs (CommaDelimitedList) when synthesizing without account/region\n      const privateSubnetIdsParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', {\n        type: 'CommaDelimitedList',\n        default: props.privateSubnets && props.privateSubnets.length > 0 ? props.privateSubnets.join(',') : '',\n      });\n      // Use the provided parameter values (deployer must supply private subnet ids when synthesizing without lookup)\n      vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', {\n        vpcId: vpcIdParam.valueAsString,\n        availabilityZones: cdk.Fn.getAzs(),\n        privateSubnetIds: privateSubnetIdsParam.valueAsList,\n      });\n      privateSubnetIds = props.privateSubnets ?? privateSubnetIdsParam.valueAsList;\n      serviceSubnetIds = privateSubnetIds;\n    }\n\n    const apiVpcLinkSecurityGroup = new ec2.CfnSecurityGroup(this, 'ApiVpcLinkSecurityGroup', {\n      groupDescription: 'Egress from API Gateway VPC Link to service',\n      vpcId: vpc.vpcId,\n      securityGroupEgress: [\n        {\n          ipProtocol: 'tcp',\n          fromPort: containerPort,\n          toPort: containerPort,\n          cidrIp: '0.0.0.0/0',\n        },\n      ],\n    });\n    apiVpcLinkSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const serviceSecurityGroup = new ec2.CfnSecurityGroup(this, 'ServiceSecurityGroup', {\n      groupDescription: 'Allow API Gateway VPC Link to reach Fargate tasks',\n      vpcId: vpc.vpcId,\n      securityGroupIngress: [\n        {\n          ipProtocol: 'tcp',\n          fromPort: containerPort,\n          toPort: containerPort,\n          cidrIp: '0.0.0.0/0',\n        },\n      ],\n      securityGroupEgress: [\n        {\n          ipProtocol: '-1',\n          fromPort: 0,\n          toPort: 65535,\n          cidrIp: '0.0.0.0/0',\n        },\n      ],\n    });\n    serviceSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const appLogGroup = new logs.CfnLogGroup(this, 'AppLogGroup', {\n      logGroupName: '/ecs/mail-example',\n      retentionInDays: 30,\n    });\n    appLogGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const mailCluster = new ecs.CfnCluster(this, 'MailCluster', {\n      clusterName: 'mail-cluster',\n      capacityProviders: ['FARGATE', 'FARGATE_SPOT'],\n      defaultCapacityProviderStrategy: [\n        {\n          capacityProvider: 'FARGATE_SPOT',\n          weight: 4,\n        },\n        {\n          capacityProvider: 'FARGATE',\n          weight: 1,\n        },\n      ],\n    });\n    mailCluster.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const mailExampleRepository = new ecr.CfnRepository(this, 'MailExampleRepository', {\n      repositoryName: 'mail-example',\n      imageTagMutability: 'MUTABLE',\n      encryptionConfiguration: {\n        encryptionType: 'AES256',\n      },\n      lifecyclePolicy: {\n        lifecyclePolicyText:\n          '{\\n  \"rules\": [\\n    {\\n      \"rulePriority\": 1,\\n      \"description\": \"Expire images older than 30 days\",\\n      \"selection\": {\\n        \"tagStatus\": \"any\",\\n        \"countType\": \"sinceImagePushed\",\\n        \"countUnit\": \"days\",\\n        \"countNumber\": 30\\n      },\\n      \"action\": { \"type\": \"expire\" }\\n    }\\n  ]\\n}\\n',\n      },\n    });\n    mailExampleRepository.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const mailHttpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', {\n      name: 'mail-example',\n      protocolType: 'HTTP',\n    });\n    mailHttpApi.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const mailApiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', {\n      stageName: 'prod',\n      apiId: mailHttpApi.ref,\n      autoDeploy: true,\n    });\n    mailApiStage.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const apiDomainName = hasCustomDomain\n      ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', {\n          domainName: apiCustomDomainName,\n          domainNameConfigurations: [\n            {\n              certificateArn: certificateArn!,\n              endpointType: 'REGIONAL',\n              securityPolicy: 'TLS_1_2',\n            },\n          ],\n        })\n      : undefined;\n    if (apiDomainName) {\n      apiDomainName.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n    }\n\n    const apiMapping =\n      hasCustomDomain && apiDomainName\n        ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', {\n            apiId: mailHttpApi.ref,\n            domainName: apiDomainName.ref,\n            stage: mailApiStage.ref,\n            apiMappingKey: 'examples',\n          })\n        : undefined;\n    if (apiMapping) {\n      apiMapping.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n    }\n\n    const apiDomainRecord =\n      hasCustomDomain && hasHostedZone && apiDomainName\n        ? new route53.CfnRecordSet(this, 'ApiDomainRecord', {\n            hostedZoneId,\n            name: apiCustomDomainName,\n            type: 'A',\n            aliasTarget: {\n              dnsName: apiDomainName.attrRegionalDomainName,\n              hostedZoneId: apiDomainName.attrRegionalHostedZoneId,\n            },\n          })\n        : undefined;\n    if (apiDomainRecord) {\n      apiDomainRecord.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n    }\n    if (apiMapping && apiDomainName) {\n      apiMapping.addDependency(apiDomainName);\n      apiMapping.addDependency(mailApiStage);\n    }\n    if (apiDomainRecord && apiDomainName) {\n      apiDomainRecord.addDependency(apiDomainName);\n    }\n\n    const baseUrl = hasCustomDomain\n      ? `https://${apiCustomDomainName}/examples/gmail-viewer`\n      : cdk.Fn.join('', [mailHttpApi.attrApiEndpoint, '/', mailApiStage.ref, '/gmail-viewer']);\n\n    const serviceNamespace = new servicediscovery.CfnPrivateDnsNamespace(this, 'ServiceNamespace', {\n      name: serviceDiscoveryNamespaceName,\n      vpc: vpc.vpcId,\n      description: 'Namespace for mail services',\n    });\n    serviceNamespace.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const apiVpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', {\n      name: 'mail-example-vpclink',\n      subnetIds: serviceSubnetIds,\n      securityGroupIds: [apiVpcLinkSecurityGroup.ref],\n    });\n    apiVpcLink.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const appCloudMapService = new servicediscovery.CfnService(this, 'AppCloudMapService', {\n      name: 'mail-example',\n      namespaceId: serviceNamespace.ref,\n      dnsConfig: {\n        routingPolicy: 'WEIGHTED',\n        dnsRecords: [\n          {\n            ttl: serviceDiscoveryTtl,\n            type: 'SRV',\n          },\n        ],\n      },\n      healthCheckCustomConfig: {\n        failureThreshold: 1,\n      },\n    });\n    appCloudMapService.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const taskExecutionRole = new iam.CfnRole(this, 'TaskExecutionRole', {\n      assumeRolePolicyDocument: {\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Effect: 'Allow',\n            Principal: {\n              Service: 'ecs-tasks.amazonaws.com',\n            },\n            Action: 'sts:AssumeRole',\n          },\n        ],\n      },\n      managedPolicyArns: ['arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy'],\n      policies: [\n        {\n          policyName: 'AllowParameterReadForSecrets',\n          policyDocument: {\n            Version: '2012-10-17',\n            Statement: [\n              {\n                Effect: 'Allow',\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'],\n                Resource: [\n                  `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`,\n                  `arn:aws:kms:${this.region}:${this.account}:key/*`,\n                ],\n              },\n            ],\n          },\n        },\n      ],\n    });\n    taskExecutionRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const taskRole = new iam.CfnRole(this, 'TaskRole', {\n      assumeRolePolicyDocument: {\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Effect: 'Allow',\n            Principal: {\n              Service: 'ecs-tasks.amazonaws.com',\n            },\n            Action: 'sts:AssumeRole',\n          },\n        ],\n      },\n      policies: [\n        {\n          policyName: 'AppRuntimeConfigRead',\n          policyDocument: {\n            Version: '2012-10-17',\n            Statement: [\n              {\n                Effect: 'Allow',\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'],\n                Resource: [\n                  `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`,\n                  `arn:aws:kms:${this.region}:${this.account}:key/*`,\n                ],\n              },\n            ],\n          },\n        },\n      ],\n    });\n    taskRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const appTaskDefinition = new ecs.CfnTaskDefinition(this, 'AppTaskDefinition', {\n      family: 'mail-example',\n      cpu: containerCpu.toString(),\n      memory: containerMemory.toString(),\n      networkMode: 'awsvpc',\n      requiresCompatibilities: ['FARGATE'],\n      executionRoleArn: taskExecutionRole.attrArn,\n      taskRoleArn: taskRole.attrArn,\n      containerDefinitions: [\n        {\n          name: 'mail-example',\n          image: `${mailExampleRepository.attrRepositoryUri}:latest`,\n          portMappings: [\n            {\n              containerPort,\n            },\n          ],\n          logConfiguration: {\n            logDriver: 'awslogs',\n            options: {\n              'awslogs-group': appLogGroup.ref,\n              'awslogs-region': this.region,\n              'awslogs-stream-prefix': 'mail-example',\n            },\n          },\n          environment: [\n            {\n              name: 'CONFIG_SSM_PREFIX',\n              value: ssmPrefix,\n            },\n            {\n              name: 'PORT',\n              value: containerPort.toString(),\n            },\n            {\n              name: 'BASE_URL',\n              value: baseUrl.toString(),\n            },\n            {\n              name: 'NODE_ENV',\n              value: publicEnvValue,\n            },\n          ],\n          secrets: [\n            {\n              name: 'GOOGLE_CLIENT_ID',\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/env/GOOGLE_CLIENT_ID`,\n            },\n            {\n              name: 'GOOGLE_CLIENT_SECRET',\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`,\n            },\n            {\n              name: 'SESSION_SECRET',\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/SESSION_SECRET`,\n            },\n          ],\n        },\n      ],\n    });\n    appTaskDefinition.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    new ecs.CfnService(this, 'AppService', {\n      serviceName: 'mail-example',\n      cluster: mailCluster.ref,\n      taskDefinition: appTaskDefinition.ref,\n      desiredCount,\n      capacityProviderStrategy: [\n        {\n          capacityProvider: 'FARGATE_SPOT',\n          weight: 4,\n        },\n        {\n          capacityProvider: 'FARGATE',\n          weight: 1,\n        },\n      ],\n      deploymentConfiguration: {\n        maximumPercent: 200,\n        minimumHealthyPercent: 50,\n      },\n      networkConfiguration: {\n        awsvpcConfiguration: {\n          assignPublicIp: 'DISABLED',\n          subnets: serviceSubnetIds,\n          securityGroups: [serviceSecurityGroup.ref],\n        },\n      },\n      serviceRegistries: [\n        {\n          registryArn: appCloudMapService.attrArn,\n          containerName: 'mail-example',\n          containerPort: containerPort,\n        },\n      ],\n      platformVersion: 'LATEST',\n    });\n\n    const mailIntegration = new apigatewayv2.CfnIntegration(this, 'MailIntegration', {\n      apiId: mailHttpApi.ref,\n      integrationType: 'HTTP_PROXY',\n      integrationMethod: 'ANY',\n      integrationUri: appCloudMapService.attrArn,\n      connectionType: 'VPC_LINK',\n      connectionId: apiVpcLink.ref,\n      payloadFormatVersion: '1.0',\n    });\n\n    new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', {\n      apiId: mailHttpApi.ref,\n      routeKey: 'ANY /gmail-viewer',\n      target: `integrations/${mailIntegration.ref}`,\n    });\n\n    new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', {\n      apiId: mailHttpApi.ref,\n      routeKey: 'ANY /gmail-viewer/{proxy+}',\n      target: `integrations/${mailIntegration.ref}`,\n    });\n\n    const ssmBaseUrl = new ssm.StringParameter(this, 'SsmBaseUrl', {\n      parameterName: `${ssmPrefix}/env/BASE_URL`,\n      stringValue: baseUrl,\n      type: ssm.ParameterType.STRING,\n    });\n    (ssmBaseUrl.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const ssmGoogleClientId = new ssm.StringParameter(this, 'SsmGoogleClientId', {\n      parameterName: `${ssmPrefix}/env/GOOGLE_CLIENT_ID`,\n      stringValue: googleClientId,\n      type: ssm.ParameterType.STRING,\n    });\n    (ssmGoogleClientId.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const ssmGoogleClientSecret = new ssm.StringParameter(this, 'SsmGoogleClientSecret', {\n      parameterName: `${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`,\n      stringValue: googleClientSecret,\n      type: ssm.ParameterType.STRING,\n    });\n    (ssmGoogleClientSecret.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const ssmPort = new ssm.StringParameter(this, 'SsmPort', {\n      parameterName: `${ssmPrefix}/env/PORT`,\n      stringValue: containerPort.toString(),\n      type: ssm.ParameterType.STRING,\n    });\n    (ssmPort.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const ssmPublicEnv = new ssm.StringParameter(this, 'SsmPublicEnv', {\n      parameterName: `${ssmPrefix}/env/NODE_ENV`,\n      stringValue: publicEnvValue,\n      type: ssm.ParameterType.STRING,\n    });\n    (ssmPublicEnv.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const ssmSessionSecret = new ssm.StringParameter(this, 'SsmSessionSecret', {\n      parameterName: `${ssmPrefix}/secrets/SESSION_SECRET`,\n      stringValue: sessionSecretValue,\n      type: ssm.ParameterType.STRING,\n    });\n    (ssmSessionSecret.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    const codeBuildRole = new iam.CfnRole(this, 'CodeBuildRole', {\n      assumeRolePolicyDocument: {\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Effect: 'Allow',\n            Principal: {\n              Service: 'codebuild.amazonaws.com',\n            },\n            Action: 'sts:AssumeRole',\n          },\n        ],\n      },\n      policies: [\n        {\n          policyName: 'BuildLogs',\n          policyDocument: {\n            Version: '2012-10-17',\n            Statement: [\n              {\n                Effect: 'Allow',\n                Action: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'],\n                Resource: '*',\n              },\n            ],\n          },\n        },\n        {\n          policyName: 'ECRPush',\n          policyDocument: {\n            Version: '2012-10-17',\n            Statement: [\n              {\n                Effect: 'Allow',\n                Action: [\n                  'ecr:GetAuthorizationToken',\n                  'ecr:BatchCheckLayerAvailability',\n                  'ecr:CompleteLayerUpload',\n                  'ecr:BatchGetImage',\n                  'ecr:DescribeRepositories',\n                  'ecr:InitiateLayerUpload',\n                  'ecr:PutImage',\n                  'ecr:UploadLayerPart',\n                ],\n                Resource: '*',\n              },\n            ],\n          },\n        },\n        {\n          policyName: 'ECSDeploy',\n          policyDocument: {\n            Version: '2012-10-17',\n            Statement: [\n              {\n                Effect: 'Allow',\n                Action: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'],\n                Resource: '*',\n              },\n            ],\n          },\n        },\n        {\n          policyName: 'SSMReadForBuild',\n          policyDocument: {\n            Version: '2012-10-17',\n            Statement: [\n              {\n                Effect: 'Allow',\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'],\n                Resource: '*',\n              },\n            ],\n          },\n        },\n        {\n          policyName: 'CodeConnectionAccess',\n          policyDocument: {\n            Version: '2012-10-17',\n            Statement: [\n              {\n                Effect: 'Allow',\n                Action: [\n                \"codeconnections:GetConnection\",\n                \"codeconnections:GetConnectionToken\",\n                \"codeconnections:UseConnection\"\n                ],\n                Resource: props.codeConnectionArn,\n              },\n            ],\n          },\n        }\n      ],\n    });\n    codeBuildRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    // Use the higher-level Project construct so we can provide a typed BuildSpec\n    const codebuildRoleRef = iam.Role.fromRoleArn(this, 'CodeBuildRoleRef', codeBuildRole.attrArn);\n\n    const codeBuildProject = new codebuild.Project(this, 'CodeBuildProject', {\n      projectName: 'mail-example',\n      role: codebuildRoleRef,\n      environment: {\n        buildImage: codebuild.LinuxBuildImage.STANDARD_7_0,\n        computeType: codebuild.ComputeType.SMALL,\n        privileged: true,\n        environmentVariables: {\n          ECR_URI: { value: mailExampleRepository.attrRepositoryUri },\n          CLUSTER_NAME: { value: mailCluster.ref },\n          SERVICE_NAME: { value: 'mail-example' },\n        },\n      },\n      source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch, }),\n      timeout: cdk.Duration.minutes(30),\n      queuedTimeout: cdk.Duration.minutes(30),\n      badge: true,\n      description: 'Build & deploy mail-example to ECR then force ECS deploy',\n      cache: codebuild.Cache.none(),\n      buildSpec: codebuild.BuildSpec.fromObject({\n        version: '0.2',\n        phases: {\n          pre_build: { commands: [\n            'echo \"Logging in to ECR\"',\n            'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ECR_URI',\n            'IMAGE_TAG=${CODEBUILD_RESOLVED_SOURCE_VERSION:-latest}',\n          ]},\n          build: { commands: [\n            'cd examples/gmail-viewer',\n            'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .',\n          ] },\n          post_build: { commands: [\n            'docker push $ECR_URI:latest',\n            'docker push $ECR_URI:$IMAGE_TAG',\n            'aws ecs update-service --cluster $CLUSTER_NAME --service $SERVICE_NAME --force-new-deployment',\n          ]},\n        },\n        artifacts: { files: [] },\n        env: { shell: 'bash' },\n      }),\n    });\n    (codeBuildProject.node.defaultChild as codebuild.CfnProject).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\n\n    this.clusterName = mailCluster.ref;\n    new cdk.CfnOutput(this, 'CfnOutputClusterName', {\n      key: 'ClusterName',\n      value: this.clusterName.toString(),\n    });\n    this.repositoryUri = mailExampleRepository.attrRepositoryUri;\n    new cdk.CfnOutput(this, 'CfnOutputRepositoryUri', {\n      key: 'RepositoryUri',\n      value: this.repositoryUri.toString(),\n    });\n    this.apiInvokeUrl = baseUrl.toString();\n    new cdk.CfnOutput(this, 'CfnOutputApiInvokeUrl', {\n      key: 'ApiInvokeUrl',\n      value: this.apiInvokeUrl,\n    });\n    this.cloudMapServiceArn = appCloudMapService.attrArn;\n    new cdk.CfnOutput(this, 'CfnOutputCloudMapServiceArn', {\n      key: 'CloudMapServiceArn',\n      value: this.cloudMapServiceArn.toString(),\n    });\n  }\n}\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 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 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 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\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      const rounded = Math.round(vcpu / 256) * 256;\r\n      return Math.max(256, rounded);\r\n    };\r\n\r\n    const repoParts = (process.env.GITHUB_REPOSITORY ?? '').split('/');\r\n    const guessedGitHubOwner = repoParts[0] || 'dyanet';\r\n    const guessedGitHubRepo = repoParts[1] || 'imap';\r\n\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 serviceDiscoveryNamespaceName = props.serviceDiscoveryNamespaceName ?? 'mail.local';\r\n    const serviceDiscoveryTtl = 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 ?? guessedGitHubOwner;\r\n    const gitHubRepo = props.gitHubRepo ?? guessedGitHubRepo;\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 || 'not-set';\r\n    const googleClientSecret = props.googleClientSecret ?? 'not-set';\r\n    const sessionSecretValue = props.sessionSecretValue ?? 'not-set';\r\n    const apiCustomDomainName = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'mail.dyanet.com';\r\n    const certificateArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN;\r\n    const hostedZoneId = props.hostedZoneId ?? '';\r\n    const hasCustomDomain = Boolean(certificateArn && apiCustomDomainName);\r\n    const hasHostedZone = Boolean(hostedZoneId);\r\n\r\n    let vpc: ec2.IVpc;\r\n    let privateSubnetIds: string[];\r\n    let serviceSubnetIds: string[];\r\n\r\n    // If account/region are available, use Vpc.fromLookup to automatically discover subnets.\r\n    // Otherwise fall back to CloudFormation parameters which let the deployer pick values in the console.\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      \r\n      if (props.privateSubnets && props.privateSubnets.length > 0) {\r\n          privateSubnetIds = props.privateSubnets;\r\n      } else {\r\n          // The service linked to the VPC Link may not be available in all Availability Zones.\r\n          const badAz = 'cac1-az4';\r\n          \r\n          let subnets = vpc.privateSubnets.filter(s => s.availabilityZone !== badAz);\r\n          if (subnets.length === 0) {\r\n            // No private subnets, try public subnets\r\n            subnets = vpc.publicSubnets.filter(s => s.availabilityZone !== badAz);\r\n          }\r\n\r\n          privateSubnetIds = subnets.slice(0, 2).map(s => s.subnetId);\r\n          \r\n          if (privateSubnetIds.length < 1) {\r\n            throw new Error(`VPC '${vpc.vpcId}' must have at least one private or public subnet in an allowed availability zone (not in ${badAz}).`);\r\n          }\r\n      }\r\n      serviceSubnetIds = privateSubnetIds;\r\n    } else {\r\n      // CloudFormation parameter for VPC selection (dropdown of VPC IDs in the console)\r\n      const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', {\r\n        type: 'AWS::EC2::VPC::Id',\r\n        default: props.vpcId ?? '',\r\n      });\r\n\r\n      // Optional parameter for private subnet IDs (CommaDelimitedList) when synthesizing without account/region\r\n      const privateSubnetIdsParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', {\r\n        type: 'CommaDelimitedList',\r\n        default: props.privateSubnets && props.privateSubnets.length > 0 ? props.privateSubnets.join(',') : '',\r\n      });\r\n      // Use the provided parameter values (deployer must supply private subnet ids when synthesizing without lookup)\r\n      vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', {\r\n        vpcId: vpcIdParam.valueAsString,\r\n        availabilityZones: cdk.Fn.getAzs(),\r\n        privateSubnetIds: privateSubnetIdsParam.valueAsList,\r\n      });\r\n      privateSubnetIds = props.privateSubnets ?? privateSubnetIdsParam.valueAsList;\r\n      serviceSubnetIds = privateSubnetIds;\r\n    }\r\n\r\n    const apiVpcLinkSecurityGroup = new ec2.CfnSecurityGroup(this, 'ApiVpcLinkSecurityGroup', {\r\n      groupDescription: 'Egress from API Gateway VPC Link to service',\r\n      vpcId: vpc.vpcId,\r\n      securityGroupEgress: [\r\n        {\r\n          ipProtocol: 'tcp',\r\n          fromPort: containerPort,\r\n          toPort: containerPort,\r\n          cidrIp: '0.0.0.0/0',\r\n        },\r\n      ],\r\n    });\r\n    apiVpcLinkSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const serviceSecurityGroup = new ec2.CfnSecurityGroup(this, 'ServiceSecurityGroup', {\r\n      groupDescription: 'Allow API Gateway VPC Link to reach Fargate tasks',\r\n      vpcId: vpc.vpcId,\r\n      securityGroupIngress: [\r\n        {\r\n          ipProtocol: 'tcp',\r\n          fromPort: containerPort,\r\n          toPort: containerPort,\r\n          cidrIp: '0.0.0.0/0',\r\n        },\r\n      ],\r\n      securityGroupEgress: [\r\n        {\r\n          ipProtocol: '-1',\r\n          fromPort: 0,\r\n          toPort: 65535,\r\n          cidrIp: '0.0.0.0/0',\r\n        },\r\n      ],\r\n    });\r\n    serviceSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const appLogGroup = new logs.CfnLogGroup(this, 'AppLogGroup', {\r\n      logGroupName: '/ecs/mail-example',\r\n      retentionInDays: 30,\r\n    });\r\n    appLogGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const mailCluster = new ecs.CfnCluster(this, 'MailCluster', {\r\n      clusterName: 'mail-cluster',\r\n      capacityProviders: ['FARGATE', 'FARGATE_SPOT'],\r\n      defaultCapacityProviderStrategy: [\r\n        {\r\n          capacityProvider: 'FARGATE_SPOT',\r\n          weight: 4,\r\n        },\r\n        {\r\n          capacityProvider: 'FARGATE',\r\n          weight: 1,\r\n        },\r\n      ],\r\n    });\r\n    mailCluster.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const mailExampleRepository = new ecr.CfnRepository(this, 'MailExampleRepository', {\r\n      repositoryName: 'mail-example',\r\n      imageTagMutability: 'MUTABLE',\r\n      encryptionConfiguration: {\r\n        encryptionType: 'AES256',\r\n      },\r\n      lifecyclePolicy: {\r\n        lifecyclePolicyText:\r\n          '{\\n  \"rules\": [\\n    {\\n      \"rulePriority\": 1,\\n      \"description\": \"Expire images older than 30 days\",\\n      \"selection\": {\\n        \"tagStatus\": \"any\",\\n        \"countType\": \"sinceImagePushed\",\\n        \"countUnit\": \"days\",\\n        \"countNumber\": 30\\n      },\\n      \"action\": { \"type\": \"expire\" }\\n    }\\n  ]\\n}\\n',\r\n      },\r\n    });\r\n    mailExampleRepository.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const mailHttpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', {\r\n      name: 'mail-example',\r\n      protocolType: 'HTTP',\r\n    });\r\n    mailHttpApi.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const mailApiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', {\r\n      stageName: 'prod',\r\n      apiId: mailHttpApi.ref,\r\n      autoDeploy: true,\r\n    });\r\n    mailApiStage.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const apiDomainName = hasCustomDomain\r\n      ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', {\r\n          domainName: apiCustomDomainName,\r\n          domainNameConfigurations: [\r\n            {\r\n              certificateArn: certificateArn!,\r\n              endpointType: 'REGIONAL',\r\n              securityPolicy: 'TLS_1_2',\r\n            },\r\n          ],\r\n        })\r\n      : undefined;\r\n    if (apiDomainName) {\r\n      apiDomainName.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n    }\r\n\r\n    const apiMapping =\r\n      hasCustomDomain && apiDomainName\r\n        ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', {\r\n            apiId: mailHttpApi.ref,\r\n            domainName: apiDomainName.ref,\r\n            stage: mailApiStage.ref,\r\n            apiMappingKey: 'examples',\r\n          })\r\n        : undefined;\r\n    if (apiMapping) {\r\n      apiMapping.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n    }\r\n\r\n    const apiDomainRecord =\r\n      hasCustomDomain && hasHostedZone && apiDomainName\r\n        ? new route53.CfnRecordSet(this, 'ApiDomainRecord', {\r\n            hostedZoneId,\r\n            name: apiCustomDomainName,\r\n            type: 'A',\r\n            aliasTarget: {\r\n              dnsName: apiDomainName.attrRegionalDomainName,\r\n              hostedZoneId: apiDomainName.attrRegionalHostedZoneId,\r\n            },\r\n          })\r\n        : undefined;\r\n    if (apiDomainRecord) {\r\n      apiDomainRecord.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n    }\r\n    if (apiMapping && apiDomainName) {\r\n      apiMapping.addDependency(apiDomainName);\r\n      apiMapping.addDependency(mailApiStage);\r\n    }\r\n    if (apiDomainRecord && apiDomainName) {\r\n      apiDomainRecord.addDependency(apiDomainName);\r\n    }\r\n\r\n    const baseUrl = hasCustomDomain\r\n      ? `https://${apiCustomDomainName}/examples/gmail-viewer`\r\n      : cdk.Fn.join('', [mailHttpApi.attrApiEndpoint, '/', mailApiStage.ref, '/gmail-viewer']);\r\n\r\n    const serviceNamespace = new servicediscovery.CfnPrivateDnsNamespace(this, 'ServiceNamespace', {\r\n      name: serviceDiscoveryNamespaceName,\r\n      vpc: vpc.vpcId,\r\n      description: 'Namespace for mail services',\r\n    });\r\n    serviceNamespace.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const apiVpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', {\r\n      name: 'mail-example-vpclink',\r\n      subnetIds: serviceSubnetIds,\r\n      securityGroupIds: [apiVpcLinkSecurityGroup.ref],\r\n    });\r\n    apiVpcLink.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const appCloudMapService = new servicediscovery.CfnService(this, 'AppCloudMapService', {\r\n      name: 'mail-example',\r\n      namespaceId: serviceNamespace.ref,\r\n      dnsConfig: {\r\n        routingPolicy: 'WEIGHTED',\r\n        dnsRecords: [\r\n          {\r\n            ttl: serviceDiscoveryTtl,\r\n            type: 'SRV',\r\n          },\r\n        ],\r\n      },\r\n      healthCheckCustomConfig: {\r\n        failureThreshold: 1,\r\n      },\r\n    });\r\n    appCloudMapService.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const taskExecutionRole = new iam.CfnRole(this, 'TaskExecutionRole', {\r\n      assumeRolePolicyDocument: {\r\n        Version: '2012-10-17',\r\n        Statement: [\r\n          {\r\n            Effect: 'Allow',\r\n            Principal: {\r\n              Service: 'ecs-tasks.amazonaws.com',\r\n            },\r\n            Action: 'sts:AssumeRole',\r\n          },\r\n        ],\r\n      },\r\n      managedPolicyArns: ['arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy'],\r\n      policies: [\r\n        {\r\n          policyName: 'AllowParameterReadForSecrets',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'],\r\n                Resource: [\r\n                  `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`,\r\n                  `arn:aws:kms:${this.region}:${this.account}:key/*`,\r\n                ],\r\n              },\r\n            ],\r\n          },\r\n        },\r\n      ],\r\n    });\r\n    taskExecutionRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const taskRole = new iam.CfnRole(this, 'TaskRole', {\r\n      assumeRolePolicyDocument: {\r\n        Version: '2012-10-17',\r\n        Statement: [\r\n          {\r\n            Effect: 'Allow',\r\n            Principal: {\r\n              Service: 'ecs-tasks.amazonaws.com',\r\n            },\r\n            Action: 'sts:AssumeRole',\r\n          },\r\n        ],\r\n      },\r\n      policies: [\r\n        {\r\n          policyName: 'AppRuntimeConfigRead',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'],\r\n                Resource: [\r\n                  `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`,\r\n                  `arn:aws:kms:${this.region}:${this.account}:key/*`,\r\n                ],\r\n              },\r\n            ],\r\n          },\r\n        },\r\n      ],\r\n    });\r\n    taskRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const appTaskDefinition = new ecs.CfnTaskDefinition(this, 'AppTaskDefinition', {\r\n      family: 'mail-example',\r\n      cpu: containerCpu.toString(),\r\n      memory: containerMemory.toString(),\r\n      networkMode: 'awsvpc',\r\n      requiresCompatibilities: ['FARGATE'],\r\n      executionRoleArn: taskExecutionRole.attrArn,\r\n      taskRoleArn: taskRole.attrArn,\r\n      containerDefinitions: [\r\n        {\r\n          name: 'mail-example',\r\n          image: `${mailExampleRepository.attrRepositoryUri}:latest`,\r\n          portMappings: [\r\n            {\r\n              containerPort,\r\n            },\r\n          ],\r\n          logConfiguration: {\r\n            logDriver: 'awslogs',\r\n            options: {\r\n              'awslogs-group': appLogGroup.ref,\r\n              'awslogs-region': this.region,\r\n              'awslogs-stream-prefix': 'mail-example',\r\n            },\r\n          },\r\n          environment: [\r\n            {\r\n              name: 'CONFIG_SSM_PREFIX',\r\n              value: ssmPrefix,\r\n            },\r\n            {\r\n              name: 'PORT',\r\n              value: containerPort.toString(),\r\n            },\r\n            {\r\n              name: 'BASE_URL',\r\n              value: baseUrl.toString(),\r\n            },\r\n            {\r\n              name: 'NODE_ENV',\r\n              value: publicEnvValue,\r\n            },\r\n          ],\r\n          secrets: [\r\n            {\r\n              name: 'GOOGLE_CLIENT_ID',\r\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/env/GOOGLE_CLIENT_ID`,\r\n            },\r\n            {\r\n              name: 'GOOGLE_CLIENT_SECRET',\r\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`,\r\n            },\r\n            {\r\n              name: 'SESSION_SECRET',\r\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/SESSION_SECRET`,\r\n            },\r\n          ],\r\n        },\r\n      ],\r\n    });\r\n    appTaskDefinition.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    new ecs.CfnService(this, 'AppService', {\r\n      serviceName: 'mail-example',\r\n      cluster: mailCluster.ref,\r\n      taskDefinition: appTaskDefinition.ref,\r\n      desiredCount,\r\n      capacityProviderStrategy: [\r\n        {\r\n          capacityProvider: 'FARGATE_SPOT',\r\n          weight: 4,\r\n        },\r\n        {\r\n          capacityProvider: 'FARGATE',\r\n          weight: 1,\r\n        },\r\n      ],\r\n      deploymentConfiguration: {\r\n        maximumPercent: 200,\r\n        minimumHealthyPercent: 50,\r\n      },\r\n      networkConfiguration: {\r\n        awsvpcConfiguration: {\r\n          assignPublicIp: 'DISABLED',\r\n          subnets: serviceSubnetIds,\r\n          securityGroups: [serviceSecurityGroup.ref],\r\n        },\r\n      },\r\n      serviceRegistries: [\r\n        {\r\n          registryArn: appCloudMapService.attrArn,\r\n          containerName: 'mail-example',\r\n          containerPort: containerPort,\r\n        },\r\n      ],\r\n      platformVersion: 'LATEST',\r\n    });\r\n\r\n    const mailIntegration = new apigatewayv2.CfnIntegration(this, 'MailIntegration', {\r\n      apiId: mailHttpApi.ref,\r\n      integrationType: 'HTTP_PROXY',\r\n      integrationMethod: 'ANY',\r\n      integrationUri: appCloudMapService.attrArn,\r\n      connectionType: 'VPC_LINK',\r\n      connectionId: apiVpcLink.ref,\r\n      payloadFormatVersion: '1.0',\r\n    });\r\n\r\n    new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', {\r\n      apiId: mailHttpApi.ref,\r\n      routeKey: 'ANY /gmail-viewer',\r\n      target: `integrations/${mailIntegration.ref}`,\r\n    });\r\n\r\n    new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', {\r\n      apiId: mailHttpApi.ref,\r\n      routeKey: 'ANY /gmail-viewer/{proxy+}',\r\n      target: `integrations/${mailIntegration.ref}`,\r\n    });\r\n\r\n    const ssmBaseUrl = new ssm.StringParameter(this, 'SsmBaseUrl', {\r\n      parameterName: `${ssmPrefix}/env/BASE_URL`,\r\n      stringValue: baseUrl,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmBaseUrl.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmGoogleClientId = new ssm.StringParameter(this, 'SsmGoogleClientId', {\r\n      parameterName: `${ssmPrefix}/env/GOOGLE_CLIENT_ID`,\r\n      stringValue: googleClientId,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmGoogleClientId.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmGoogleClientSecret = new ssm.StringParameter(this, 'SsmGoogleClientSecret', {\r\n      parameterName: `${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`,\r\n      stringValue: googleClientSecret,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmGoogleClientSecret.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmPort = new ssm.StringParameter(this, 'SsmPort', {\r\n      parameterName: `${ssmPrefix}/env/PORT`,\r\n      stringValue: containerPort.toString(),\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmPort.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmPublicEnv = new ssm.StringParameter(this, 'SsmPublicEnv', {\r\n      parameterName: `${ssmPrefix}/env/NODE_ENV`,\r\n      stringValue: publicEnvValue,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmPublicEnv.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmSessionSecret = new ssm.StringParameter(this, 'SsmSessionSecret', {\r\n      parameterName: `${ssmPrefix}/secrets/SESSION_SECRET`,\r\n      stringValue: sessionSecretValue,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmSessionSecret.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const codeBuildRole = new iam.CfnRole(this, 'CodeBuildRole', {\r\n      assumeRolePolicyDocument: {\r\n        Version: '2012-10-17',\r\n        Statement: [\r\n          {\r\n            Effect: 'Allow',\r\n            Principal: {\r\n              Service: 'codebuild.amazonaws.com',\r\n            },\r\n            Action: 'sts:AssumeRole',\r\n          },\r\n        ],\r\n      },\r\n      policies: [\r\n        {\r\n          policyName: 'BuildLogs',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'],\r\n                Resource: '*',\r\n              },\r\n            ],\r\n          },\r\n        },\r\n        {\r\n          policyName: 'ECRPush',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: [\r\n                  'ecr:GetAuthorizationToken',\r\n                  'ecr:BatchCheckLayerAvailability',\r\n                  'ecr:CompleteLayerUpload',\r\n                  'ecr:BatchGetImage',\r\n                  'ecr:DescribeRepositories',\r\n                  'ecr:InitiateLayerUpload',\r\n                  'ecr:PutImage',\r\n                  'ecr:UploadLayerPart',\r\n                ],\r\n                Resource: '*',\r\n              },\r\n            ],\r\n          },\r\n        },\r\n        {\r\n          policyName: 'ECSDeploy',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'],\r\n                Resource: '*',\r\n              },\r\n            ],\r\n          },\r\n        },\r\n        {\r\n          policyName: 'SSMReadForBuild',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'],\r\n                Resource: '*',\r\n              },\r\n            ],\r\n          },\r\n        },\r\n        {\r\n          policyName: 'CodeConnectionAccess',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: [\r\n                \"codeconnections:GetConnection\",\r\n                \"codeconnections:GetConnectionToken\",\r\n                \"codeconnections:UseConnection\"\r\n                ],\r\n                Resource: props.codeConnectionArn,\r\n              },\r\n            ],\r\n          },\r\n        }\r\n      ],\r\n    });\r\n    codeBuildRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    // Use the higher-level Project construct so we can provide a typed BuildSpec\r\n    const codebuildRoleRef = iam.Role.fromRoleArn(this, 'CodeBuildRoleRef', codeBuildRole.attrArn);\r\n\r\n    const codeBuildProject = new codebuild.Project(this, 'CodeBuildProject', {\r\n      projectName: 'mail-example',\r\n      role: codebuildRoleRef,\r\n      environment: {\r\n        buildImage: codebuild.LinuxBuildImage.STANDARD_7_0,\r\n        computeType: codebuild.ComputeType.SMALL,\r\n        privileged: true,\r\n        environmentVariables: {\r\n          ECR_URI: { value: mailExampleRepository.attrRepositoryUri },\r\n          CLUSTER_NAME: { value: mailCluster.ref },\r\n          SERVICE_NAME: { value: 'mail-example' },\r\n        },\r\n      },\r\n      source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch, }),\r\n      timeout: cdk.Duration.minutes(30),\r\n      queuedTimeout: cdk.Duration.minutes(30),\r\n      badge: true,\r\n      description: 'Build & deploy mail-example to ECR then force ECS deploy',\r\n      cache: codebuild.Cache.none(),\r\n      buildSpec: codebuild.BuildSpec.fromObject({\r\n        version: '0.2',\r\n        phases: {\r\n          pre_build: { commands: [\r\n            'echo \"Logging in to ECR\"',\r\n            'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ECR_URI',\r\n            'IMAGE_TAG=${CODEBUILD_RESOLVED_SOURCE_VERSION:-latest}',\r\n          ]},\r\n          build: { commands: [\r\n            'cd examples/gmail-viewer',\r\n            'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .',\r\n          ] },\r\n          post_build: { commands: [\r\n            'docker push $ECR_URI:latest',\r\n            'docker push $ECR_URI:$IMAGE_TAG',\r\n            'aws ecs update-service --cluster $CLUSTER_NAME --service $SERVICE_NAME --force-new-deployment',\r\n          ]},\r\n        },\r\n        artifacts: { files: [] },\r\n        env: { shell: 'bash' },\r\n      }),\r\n    });\r\n    (codeBuildProject.node.defaultChild as codebuild.CfnProject).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    this.clusterName = mailCluster.ref;\r\n    new cdk.CfnOutput(this, 'CfnOutputClusterName', {\r\n      key: 'ClusterName',\r\n      value: this.clusterName.toString(),\r\n    });\r\n    this.repositoryUri = mailExampleRepository.attrRepositoryUri;\r\n    new cdk.CfnOutput(this, 'CfnOutputRepositoryUri', {\r\n      key: 'RepositoryUri',\r\n      value: this.repositoryUri.toString(),\r\n    });\r\n    this.apiInvokeUrl = baseUrl.toString();\r\n    new cdk.CfnOutput(this, 'CfnOutputApiInvokeUrl', {\r\n      key: 'ApiInvokeUrl',\r\n      value: this.apiInvokeUrl,\r\n    });\r\n    this.cloudMapServiceArn = appCloudMapService.attrArn;\r\n    new cdk.CfnOutput(this, 'CfnOutputCloudMapServiceArn', {\r\n      key: 'CloudMapServiceArn',\r\n      value: this.cloudMapServiceArn.toString(),\r\n    });\r\n  }\r\n}\r\n"]} \ No newline at end of file diff --git a/examples/gmail-viewer-cdk/test/gmail-viewer-cdk.test.js b/examples/gmail-viewer-cdk/test/gmail-viewer-cdk.test.js index a0206f1..e9f9dbe 100644 --- a/examples/gmail-viewer-cdk/test/gmail-viewer-cdk.test.js +++ b/examples/gmail-viewer-cdk/test/gmail-viewer-cdk.test.js @@ -15,4 +15,4 @@ test('SQS Queue Created', () => { // VisibilityTimeout: 300 // }); }); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ21haWwtdmlld2VyLWNkay50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZ21haWwtdmlld2VyLWNkay50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwyQ0FBMkM7QUFDM0MscURBQXFEO0FBQ3JELG1FQUFtRTs7QUFFbkUsdUVBQXVFO0FBQ3ZFLG9EQUFvRDtBQUNwRCxJQUFJLENBQUMsbUJBQW1CLEVBQUUsR0FBRyxFQUFFO0lBQy9CLCtCQUErQjtJQUMvQixjQUFjO0lBQ2QsOEVBQThFO0lBQzlFLGNBQWM7SUFDZCxnREFBZ0Q7SUFFaEQsd0RBQXdEO0lBQ3hELDZCQUE2QjtJQUM3QixRQUFRO0FBQ1IsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBpbXBvcnQgKiBhcyBjZGsgZnJvbSAnYXdzLWNkay1saWIvY29yZSc7XG4vLyBpbXBvcnQgeyBUZW1wbGF0ZSB9IGZyb20gJ2F3cy1jZGstbGliL2Fzc2VydGlvbnMnO1xuLy8gaW1wb3J0ICogYXMgR21haWxWaWV3ZXJDZGsgZnJvbSAnLi4vbGliL2dtYWlsLXZpZXdlci1jZGstc3RhY2snO1xuXG4vLyBleGFtcGxlIHRlc3QuIFRvIHJ1biB0aGVzZSB0ZXN0cywgdW5jb21tZW50IHRoaXMgZmlsZSBhbG9uZyB3aXRoIHRoZVxuLy8gZXhhbXBsZSByZXNvdXJjZSBpbiBsaWIvZ21haWwtdmlld2VyLWNkay1zdGFjay50c1xudGVzdCgnU1FTIFF1ZXVlIENyZWF0ZWQnLCAoKSA9PiB7XG4vLyAgIGNvbnN0IGFwcCA9IG5ldyBjZGsuQXBwKCk7XG4vLyAgICAgLy8gV0hFTlxuLy8gICBjb25zdCBzdGFjayA9IG5ldyBHbWFpbFZpZXdlckNkay5HbWFpbFZpZXdlckNka1N0YWNrKGFwcCwgJ015VGVzdFN0YWNrJyk7XG4vLyAgICAgLy8gVEhFTlxuLy8gICBjb25zdCB0ZW1wbGF0ZSA9IFRlbXBsYXRlLmZyb21TdGFjayhzdGFjayk7XG5cbi8vICAgdGVtcGxhdGUuaGFzUmVzb3VyY2VQcm9wZXJ0aWVzKCdBV1M6OlNRUzo6UXVldWUnLCB7XG4vLyAgICAgVmlzaWJpbGl0eVRpbWVvdXQ6IDMwMFxuLy8gICB9KTtcbn0pO1xuIl19 \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ21haWwtdmlld2VyLWNkay50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZ21haWwtdmlld2VyLWNkay50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwyQ0FBMkM7QUFDM0MscURBQXFEO0FBQ3JELG1FQUFtRTs7QUFFbkUsdUVBQXVFO0FBQ3ZFLG9EQUFvRDtBQUNwRCxJQUFJLENBQUMsbUJBQW1CLEVBQUUsR0FBRyxFQUFFO0lBQy9CLCtCQUErQjtJQUMvQixjQUFjO0lBQ2QsOEVBQThFO0lBQzlFLGNBQWM7SUFDZCxnREFBZ0Q7SUFFaEQsd0RBQXdEO0lBQ3hELDZCQUE2QjtJQUM3QixRQUFRO0FBQ1IsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBpbXBvcnQgKiBhcyBjZGsgZnJvbSAnYXdzLWNkay1saWIvY29yZSc7XHJcbi8vIGltcG9ydCB7IFRlbXBsYXRlIH0gZnJvbSAnYXdzLWNkay1saWIvYXNzZXJ0aW9ucyc7XHJcbi8vIGltcG9ydCAqIGFzIEdtYWlsVmlld2VyQ2RrIGZyb20gJy4uL2xpYi9nbWFpbC12aWV3ZXItY2RrLXN0YWNrJztcclxuXHJcbi8vIGV4YW1wbGUgdGVzdC4gVG8gcnVuIHRoZXNlIHRlc3RzLCB1bmNvbW1lbnQgdGhpcyBmaWxlIGFsb25nIHdpdGggdGhlXHJcbi8vIGV4YW1wbGUgcmVzb3VyY2UgaW4gbGliL2dtYWlsLXZpZXdlci1jZGstc3RhY2sudHNcclxudGVzdCgnU1FTIFF1ZXVlIENyZWF0ZWQnLCAoKSA9PiB7XHJcbi8vICAgY29uc3QgYXBwID0gbmV3IGNkay5BcHAoKTtcclxuLy8gICAgIC8vIFdIRU5cclxuLy8gICBjb25zdCBzdGFjayA9IG5ldyBHbWFpbFZpZXdlckNkay5HbWFpbFZpZXdlckNka1N0YWNrKGFwcCwgJ015VGVzdFN0YWNrJyk7XHJcbi8vICAgICAvLyBUSEVOXHJcbi8vICAgY29uc3QgdGVtcGxhdGUgPSBUZW1wbGF0ZS5mcm9tU3RhY2soc3RhY2spO1xyXG5cclxuLy8gICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoJ0FXUzo6U1FTOjpRdWV1ZScsIHtcclxuLy8gICAgIFZpc2liaWxpdHlUaW1lb3V0OiAzMDBcclxuLy8gICB9KTtcclxufSk7XHJcbiJdfQ== \ No newline at end of file diff --git a/examples/gmail-viewer-cdk/test/mail_example_stack.test.js b/examples/gmail-viewer-cdk/test/mail_example_stack.test.js index d04103a..7ea5ea2 100644 --- a/examples/gmail-viewer-cdk/test/mail_example_stack.test.js +++ b/examples/gmail-viewer-cdk/test/mail_example_stack.test.js @@ -15,4 +15,4 @@ test('SQS Queue Created', () => { // VisibilityTimeout: 300 // }); }); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFpbF9leGFtcGxlX3N0YWNrLnRlc3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJtYWlsX2V4YW1wbGVfc3RhY2sudGVzdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsMkNBQTJDO0FBQzNDLHFEQUFxRDtBQUNyRCx1RUFBdUU7O0FBRXZFLHVFQUF1RTtBQUN2RSxzREFBc0Q7QUFDdEQsSUFBSSxDQUFDLG1CQUFtQixFQUFFLEdBQUcsRUFBRTtJQUMvQiwrQkFBK0I7SUFDL0IsY0FBYztJQUNkLGtGQUFrRjtJQUNsRixjQUFjO0lBQ2QsZ0RBQWdEO0lBRWhELHdEQUF3RDtJQUN4RCw2QkFBNkI7SUFDN0IsUUFBUTtBQUNSLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gaW1wb3J0ICogYXMgY2RrIGZyb20gJ2F3cy1jZGstbGliL2NvcmUnO1xuLy8gaW1wb3J0IHsgVGVtcGxhdGUgfSBmcm9tICdhd3MtY2RrLWxpYi9hc3NlcnRpb25zJztcbi8vIGltcG9ydCAqIGFzIE1haWxFeGFtcGxlU3RhY2sgZnJvbSAnLi4vbGliL21haWxfZXhhbXBsZV9zdGFjay1zdGFjayc7XG5cbi8vIGV4YW1wbGUgdGVzdC4gVG8gcnVuIHRoZXNlIHRlc3RzLCB1bmNvbW1lbnQgdGhpcyBmaWxlIGFsb25nIHdpdGggdGhlXG4vLyBleGFtcGxlIHJlc291cmNlIGluIGxpYi9tYWlsX2V4YW1wbGVfc3RhY2stc3RhY2sudHNcbnRlc3QoJ1NRUyBRdWV1ZSBDcmVhdGVkJywgKCkgPT4ge1xuLy8gICBjb25zdCBhcHAgPSBuZXcgY2RrLkFwcCgpO1xuLy8gICAgIC8vIFdIRU5cbi8vICAgY29uc3Qgc3RhY2sgPSBuZXcgTWFpbEV4YW1wbGVTdGFjay5NYWlsRXhhbXBsZVN0YWNrU3RhY2soYXBwLCAnTXlUZXN0U3RhY2snKTtcbi8vICAgICAvLyBUSEVOXG4vLyAgIGNvbnN0IHRlbXBsYXRlID0gVGVtcGxhdGUuZnJvbVN0YWNrKHN0YWNrKTtcblxuLy8gICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoJ0FXUzo6U1FTOjpRdWV1ZScsIHtcbi8vICAgICBWaXNpYmlsaXR5VGltZW91dDogMzAwXG4vLyAgIH0pO1xufSk7XG4iXX0= \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFpbF9leGFtcGxlX3N0YWNrLnRlc3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJtYWlsX2V4YW1wbGVfc3RhY2sudGVzdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsMkNBQTJDO0FBQzNDLHFEQUFxRDtBQUNyRCx1RUFBdUU7O0FBRXZFLHVFQUF1RTtBQUN2RSxzREFBc0Q7QUFDdEQsSUFBSSxDQUFDLG1CQUFtQixFQUFFLEdBQUcsRUFBRTtJQUMvQiwrQkFBK0I7SUFDL0IsY0FBYztJQUNkLGtGQUFrRjtJQUNsRixjQUFjO0lBQ2QsZ0RBQWdEO0lBRWhELHdEQUF3RDtJQUN4RCw2QkFBNkI7SUFDN0IsUUFBUTtBQUNSLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gaW1wb3J0ICogYXMgY2RrIGZyb20gJ2F3cy1jZGstbGliL2NvcmUnO1xyXG4vLyBpbXBvcnQgeyBUZW1wbGF0ZSB9IGZyb20gJ2F3cy1jZGstbGliL2Fzc2VydGlvbnMnO1xyXG4vLyBpbXBvcnQgKiBhcyBNYWlsRXhhbXBsZVN0YWNrIGZyb20gJy4uL2xpYi9tYWlsX2V4YW1wbGVfc3RhY2stc3RhY2snO1xyXG5cclxuLy8gZXhhbXBsZSB0ZXN0LiBUbyBydW4gdGhlc2UgdGVzdHMsIHVuY29tbWVudCB0aGlzIGZpbGUgYWxvbmcgd2l0aCB0aGVcclxuLy8gZXhhbXBsZSByZXNvdXJjZSBpbiBsaWIvbWFpbF9leGFtcGxlX3N0YWNrLXN0YWNrLnRzXHJcbnRlc3QoJ1NRUyBRdWV1ZSBDcmVhdGVkJywgKCkgPT4ge1xyXG4vLyAgIGNvbnN0IGFwcCA9IG5ldyBjZGsuQXBwKCk7XHJcbi8vICAgICAvLyBXSEVOXHJcbi8vICAgY29uc3Qgc3RhY2sgPSBuZXcgTWFpbEV4YW1wbGVTdGFjay5NYWlsRXhhbXBsZVN0YWNrU3RhY2soYXBwLCAnTXlUZXN0U3RhY2snKTtcclxuLy8gICAgIC8vIFRIRU5cclxuLy8gICBjb25zdCB0ZW1wbGF0ZSA9IFRlbXBsYXRlLmZyb21TdGFjayhzdGFjayk7XHJcblxyXG4vLyAgIHRlbXBsYXRlLmhhc1Jlc291cmNlUHJvcGVydGllcygnQVdTOjpTUVM6OlF1ZXVlJywge1xyXG4vLyAgICAgVmlzaWJpbGl0eVRpbWVvdXQ6IDMwMFxyXG4vLyAgIH0pO1xyXG59KTtcclxuIl19 \ No newline at end of file diff --git a/examples/gmail-viewer/package-lock.json b/examples/gmail-viewer/package-lock.json new file mode 100644 index 0000000..bfb97ff --- /dev/null +++ b/examples/gmail-viewer/package-lock.json @@ -0,0 +1,1074 @@ +{ + "name": "gmail-viewer-example", + "version": "0.5.2", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "gmail-viewer-example", + "version": "0.5.2", + "license": "MIT", + "dependencies": { + "@dyanet/config-aws": "^1.0.3", + "@dyanet/imap": "^0.5.2", + "express": "^4.21.0", + "express-session": "^1.18.0", + "helmet": "^8.0.0", + "zod": "^3.22.0" + }, + "devDependencies": { + "@types/express": "^5.0.0", + "@types/express-session": "^1.18.0", + "@types/node": "^20.10.0", + "typescript": "^5.3.0" + }, + "engines": { + "node": ">=20.0.0" + } + }, + "node_modules/@dyanet/config-aws": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@dyanet/config-aws/-/config-aws-1.0.3.tgz", + "integrity": "sha512-Y7Wr9MWU+5rRrwg8Wr/5bqufdZWZvo8l3OZd18oCetRT9935+aPVdNP0IuGwLocitQoz+pP91x4G/3bIe/PIOg==", + "license": "MIT", + "engines": { + "node": ">=20.0.0", + "npm": ">=10.0.0" + }, + "peerDependencies": { + "@aws-sdk/client-s3": "^3.0.0", + "@aws-sdk/client-secrets-manager": "^3.0.0", + "@aws-sdk/client-ssm": "^3.0.0", + "zod": "^3.22.0" + }, + "peerDependenciesMeta": { + "@aws-sdk/client-s3": { + "optional": true + }, + "@aws-sdk/client-secrets-manager": { + "optional": true + }, + "@aws-sdk/client-ssm": { + "optional": true + } + } + }, + "node_modules/@dyanet/imap": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/@dyanet/imap/-/imap-0.5.2.tgz", + "integrity": "sha512-VwxFsqCY4DKct4A+bHzLpBex7UrwWE4p0DsLrA6Bjuwftxmtm7HkkENuRwEZOlw020XDIAfbWMKXWd/n8M/N0A==", + "license": "MIT", + "engines": { + "node": ">=20.0.0" + } + }, + "node_modules/@types/body-parser": { + "version": "1.19.6", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.6.tgz", + "integrity": "sha512-HLFeCYgz89uk22N5Qg3dvGvsv46B8GLvKKo1zKG4NybA8U2DiEO3w9lqGg29t/tfLRJpJ6iQxnVw4OnB7MoM9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "node_modules/@types/connect": { + "version": "3.4.38", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", + "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/express": { + "version": "5.0.6", + "resolved": "https://registry.npmjs.org/@types/express/-/express-5.0.6.tgz", + "integrity": "sha512-sKYVuV7Sv9fbPIt/442koC7+IIwK5olP1KWeD88e/idgoJqDm3JV/YUiPwkoKK92ylff2MGxSz1CSjsXelx0YA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "^5.0.0", + "@types/serve-static": "^2" + } + }, + "node_modules/@types/express-serve-static-core": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-5.1.0.tgz", + "integrity": "sha512-jnHMsrd0Mwa9Cf4IdOzbz543y4XJepXrbia2T4b6+spXC2We3t1y6K44D3mR8XMFSXMCf3/l7rCgddfx7UNVBA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*", + "@types/send": "*" + } + }, + "node_modules/@types/express-session": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/@types/express-session/-/express-session-1.18.2.tgz", + "integrity": "sha512-k+I0BxwVXsnEU2hV77cCobC08kIsn4y44C3gC0b46uxZVMaXA04lSPgRLR/bSL2w0t0ShJiG8o4jPzRG/nscFg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/express": "*" + } + }, + "node_modules/@types/http-errors": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.5.tgz", + "integrity": "sha512-r8Tayk8HJnX0FztbZN7oVqGccWgw98T/0neJphO91KkmOzug1KkofZURD4UaD5uH8AqcFLfdPErnBod0u71/qg==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "20.19.27", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.27.tgz", + "integrity": "sha512-N2clP5pJhB2YnZJ3PIHFk5RkygRX5WO/5f0WC08tp0wd+sv0rsJk3MqWn3CbNmT2J505a5336jaQj4ph1AdMug==", + "dev": true, + "license": "MIT", + "dependencies": { + "undici-types": "~6.21.0" + } + }, + "node_modules/@types/qs": { + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.14.0.tgz", + "integrity": "sha512-eOunJqu0K1923aExK6y8p6fsihYEn/BYuQ4g0CxAAgFc4b/ZLN4CrsRZ55srTdqoiLzU2B2evC+apEIxprEzkQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/range-parser": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.7.tgz", + "integrity": "sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/send": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@types/send/-/send-1.2.1.tgz", + "integrity": "sha512-arsCikDvlU99zl1g69TcAB3mzZPpxgw0UQnaHeC1Nwb015xp8bknZv5rIfri9xTOcMuaVgvabfIRA7PSZVuZIQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/serve-static": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-2.2.0.tgz", + "integrity": "sha512-8mam4H1NHLtu7nmtalF7eyBH14QyOASmcxHhSfEoRyr0nP/YdoesEtU+uSRvMe96TW/HPTtkoKqQLl53N7UXMQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/http-errors": "*", + "@types/node": "*" + } + }, + "node_modules/accepts": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", + "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", + "license": "MIT", + "dependencies": { + "mime-types": "~2.1.34", + "negotiator": "0.6.3" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==", + "license": "MIT" + }, + "node_modules/body-parser": { + "version": "1.20.4", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.4.tgz", + "integrity": "sha512-ZTgYYLMOXY9qKU/57FAo8F+HA2dGX7bqGc71txDRC1rS4frdFI5R7NhluHxH6M0YItAP0sHB4uqAOcYKxO6uGA==", + "license": "MIT", + "dependencies": { + "bytes": "~3.1.2", + "content-type": "~1.0.5", + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "~1.2.0", + "http-errors": "~2.0.1", + "iconv-lite": "~0.4.24", + "on-finished": "~2.4.1", + "qs": "~6.14.0", + "raw-body": "~2.5.3", + "type-is": "~1.6.18", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/call-bind-apply-helpers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", + "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/call-bound": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", + "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "get-intrinsic": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/content-disposition": { + "version": "0.5.4", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", + "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", + "license": "MIT", + "dependencies": { + "safe-buffer": "5.2.1" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-type": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", + "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz", + "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.7.tgz", + "integrity": "sha512-NXdYc3dLr47pBkpUCHtKSwIOQXLVn8dZEuywboCOJY/osA0wFSLlSawr3KN8qXJEyX66FcONTH8EIlVuK0yyFA==", + "license": "MIT" + }, + "node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "license": "MIT", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/destroy": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", + "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", + "license": "MIT", + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-errors": "^1.3.0", + "gopd": "^1.2.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", + "license": "MIT" + }, + "node_modules/encodeurl": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", + "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/es-define-property": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", + "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-object-atoms": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", + "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", + "license": "MIT" + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/express": { + "version": "4.22.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.22.1.tgz", + "integrity": "sha512-F2X8g9P1X7uCPZMA3MVf9wcTqlyNp7IhH5qPCI0izhaOIYXaW9L535tGA3qmjRzpH+bZczqq7hVKxTR4NWnu+g==", + "license": "MIT", + "dependencies": { + "accepts": "~1.3.8", + "array-flatten": "1.1.1", + "body-parser": "~1.20.3", + "content-disposition": "~0.5.4", + "content-type": "~1.0.4", + "cookie": "~0.7.1", + "cookie-signature": "~1.0.6", + "debug": "2.6.9", + "depd": "2.0.0", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.3.1", + "fresh": "~0.5.2", + "http-errors": "~2.0.0", + "merge-descriptors": "1.0.3", + "methods": "~1.1.2", + "on-finished": "~2.4.1", + "parseurl": "~1.3.3", + "path-to-regexp": "~0.1.12", + "proxy-addr": "~2.0.7", + "qs": "~6.14.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.2.1", + "send": "~0.19.0", + "serve-static": "~1.16.2", + "setprototypeof": "1.2.0", + "statuses": "~2.0.1", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/express-session": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/express-session/-/express-session-1.18.2.tgz", + "integrity": "sha512-SZjssGQC7TzTs9rpPDuUrR23GNZ9+2+IkA/+IJWmvQilTr5OSliEHGF+D9scbIpdC6yGtTI0/VhaHoVes2AN/A==", + "license": "MIT", + "dependencies": { + "cookie": "0.7.2", + "cookie-signature": "1.0.7", + "debug": "2.6.9", + "depd": "~2.0.0", + "on-headers": "~1.1.0", + "parseurl": "~1.3.3", + "safe-buffer": "5.2.1", + "uid-safe": "~2.1.5" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/finalhandler": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.2.tgz", + "integrity": "sha512-aA4RyPcd3badbdABGDuTXCMTtOneUCAYH/gxoYRTZlIJdF0YPWuGqiAsIrhNnnqdXGswYk6dGujem4w80UJFhg==", + "license": "MIT", + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "on-finished": "~2.4.1", + "parseurl": "~1.3.3", + "statuses": "~2.0.2", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-intrinsic": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.1.1", + "function-bind": "^1.1.2", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/gopd": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-symbols": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", + "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "license": "MIT", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/helmet": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/helmet/-/helmet-8.1.0.tgz", + "integrity": "sha512-jOiHyAZsmnr8LqoPGmCjYAaiuWwjAPLgY8ZX2XrmHawt99/u1y6RgrZMTeoPfpUbV96HOalYgz1qzkRbw54Pmg==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/http-errors": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz", + "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==", + "license": "MIT", + "dependencies": { + "depd": "~2.0.0", + "inherits": "~2.0.4", + "setprototypeof": "~1.2.0", + "statuses": "~2.0.2", + "toidentifier": "~1.0.1" + }, + "engines": { + "node": ">= 0.8" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "license": "ISC" + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "license": "MIT", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", + "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "license": "MIT", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", + "license": "MIT" + }, + "node_modules/negotiator": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/object-inspect": { + "version": "1.13.4", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", + "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "license": "MIT", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/on-headers": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.1.0.tgz", + "integrity": "sha512-737ZY3yNnXy37FHkQxPzt4UZ2UWPWiCZWLvFZ4fu5cueciegX0zGPnrlY6bwRg4FdQOe9YU8MkmJwGhoMybl8A==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/path-to-regexp": { + "version": "0.1.12", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz", + "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==", + "license": "MIT" + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "license": "MIT", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/qs": { + "version": "6.14.1", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.1.tgz", + "integrity": "sha512-4EK3+xJl8Ts67nLYNwqw/dsFVnCf+qR7RgXSK9jEEm9unao3njwMDdmsdvoKBKHzxd7tCYz5e5M+SnMjdtXGQQ==", + "license": "BSD-3-Clause", + "dependencies": { + "side-channel": "^1.1.0" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/random-bytes": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/random-bytes/-/random-bytes-1.0.0.tgz", + "integrity": "sha512-iv7LhNVO047HzYR3InF6pUcUsPQiHTM1Qal51DcGSuZFBil1aBBWG5eHPNek7bvILMaYJ/8RU1e8w1AMdHmLQQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "2.5.3", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.3.tgz", + "integrity": "sha512-s4VSOf6yN0rvbRZGxs8Om5CWj6seneMwK3oDb4lWDH0UPhWcxwOWw5+qk24bxq87szX1ydrwylIOp2uG1ojUpA==", + "license": "MIT", + "dependencies": { + "bytes": "~3.1.2", + "http-errors": "~2.0.1", + "iconv-lite": "~0.4.24", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "license": "MIT" + }, + "node_modules/send": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/send/-/send-0.19.2.tgz", + "integrity": "sha512-VMbMxbDeehAxpOtWJXlcUS5E8iXh6QmN+BkRX1GARS3wRaXEEgzCcB10gTQazO42tpNIya8xIyNx8fll1OFPrg==", + "license": "MIT", + "dependencies": { + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "~0.5.2", + "http-errors": "~2.0.1", + "mime": "1.6.0", + "ms": "2.1.3", + "on-finished": "~2.4.1", + "range-parser": "~1.2.1", + "statuses": "~2.0.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "license": "MIT" + }, + "node_modules/serve-static": { + "version": "1.16.3", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.3.tgz", + "integrity": "sha512-x0RTqQel6g5SY7Lg6ZreMmsOzncHFU7nhnRWkKgWuMTu5NN0DR5oruckMqRvacAN9d5w6ARnRBXl9xhDCgfMeA==", + "license": "MIT", + "dependencies": { + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "~0.19.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", + "license": "ISC" + }, + "node_modules/side-channel": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", + "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3", + "side-channel-list": "^1.0.0", + "side-channel-map": "^1.0.1", + "side-channel-weakmap": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-list": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", + "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", + "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-weakmap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", + "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3", + "side-channel-map": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/statuses": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz", + "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "license": "MIT", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "license": "MIT", + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typescript": { + "version": "5.9.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", + "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/uid-safe": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/uid-safe/-/uid-safe-2.1.5.tgz", + "integrity": "sha512-KPHm4VL5dDXKz01UuEd88Df+KzynaohSL9fBh096KWAxSKZQDI2uBrVqtvRM4rwrIrRRKsdLNML/lnaaVSRioA==", + "license": "MIT", + "dependencies": { + "random-bytes": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/undici-types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz", + "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", + "license": "MIT", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/zod": { + "version": "3.25.76", + "resolved": "https://registry.npmjs.org/zod/-/zod-3.25.76.tgz", + "integrity": "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/colinhacks" + } + } + } +} diff --git a/examples/gmail-viewer/package.json b/examples/gmail-viewer/package.json index f4ec4c3..0ce9b40 100644 --- a/examples/gmail-viewer/package.json +++ b/examples/gmail-viewer/package.json @@ -25,7 +25,7 @@ "express": "^4.21.0", "express-session": "^1.18.0", "helmet": "^8.0.0", - "zod": "^4.2.1" + "zod": "^3.22.0" }, "devDependencies": { "@types/express": "^5.0.0", From 1f6c68c1cf9caae3313df813d34bc52670d6cbc4 Mon Sep 17 00:00:00 2001 From: Akber Choudhry Date: Tue, 6 Jan 2026 10:52:55 -0500 Subject: [PATCH 3/3] fix(buid) : changes to cdk and to example gmail viewer --- .../gmail-viewer-cdk/bin/gmail-viewer-cdk.js | 3 +- .../gmail-viewer-cdk/bin/gmail-viewer-cdk.ts | 3 +- .../lib/gmail-viewer-cdk-stack.d.ts | 1 + .../lib/gmail-viewer-cdk-stack.js | 710 ++----- .../lib/gmail-viewer-cdk-stack.ts | 739 ++----- examples/gmail-viewer/package-lock.json | 1844 ++++++++++++++++- examples/gmail-viewer/package.json | 3 + 7 files changed, 2106 insertions(+), 1197 deletions(-) diff --git a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js index aad315a..332c32c 100644 --- a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js +++ b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.js @@ -13,6 +13,7 @@ new gmail_viewer_cdk_stack_1.GmailViewerCdkStack(app, 'GmailViewerCdkStack', { region: process.env.CDK_DEFAULT_REGION, }, privateSubnets: ['subnet-0d8290d0ea575331c', 'subnet-032290569a7ecf210'], + publicSubnets: ['subnet-0713ade30dd8c308c', 'subnet-0b4dd45b51bb8252f'], gitHubOwner: 'dyanet', gitHubRepo: 'imap', gitHubBranch: 'main', @@ -21,4 +22,4 @@ new gmail_viewer_cdk_stack_1.GmailViewerCdkStack(app, 'GmailViewerCdkStack', { apiCustomDomainName: 'mail.dyanet.com', certificateArn: 'arn:aws:acm:ca-central-1:239030031457:certificate/af9150f1-4635-4154-8b19-e5449d57e971', }); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ21haWwtdmlld2VyLWNkay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImdtYWlsLXZpZXdlci1jZGsudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EsNkNBQWtDO0FBQ2xDLDBFQUFvRTtBQUVwRSxNQUFNLEdBQUcsR0FBRyxJQUFJLGlCQUFHLEVBQUUsQ0FBQztBQUV0QiwwRkFBMEY7QUFDMUYsK0RBQStEO0FBQy9ELE1BQU0sY0FBYyxHQUFHLEdBQUcsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLGdCQUFnQixDQUFDLEVBQUUsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBRTVFLElBQUksNENBQW1CLENBQUMsR0FBRyxFQUFFLHFCQUFxQixFQUFFO0lBQ2xELEdBQUcsRUFBRTtRQUNILE9BQU8sRUFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLG1CQUFtQjtRQUN4QyxNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0I7S0FDdkM7SUFDRCxjQUFjLEVBQUUsQ0FBQywwQkFBMEIsRUFBRywwQkFBMEIsQ0FBQztJQUN6RSxXQUFXLEVBQUUsUUFBUTtJQUNyQixVQUFVLEVBQUUsTUFBTTtJQUNsQixZQUFZLEVBQUUsTUFBTTtJQUNwQixpQkFBaUIsRUFBRSxtR0FBbUc7SUFDdEgsaUJBQWlCLEVBQUUsS0FBSztJQUN4QixtQkFBbUIsRUFBRSxpQkFBaUI7SUFDdEMsY0FBYyxFQUFFLHdGQUF3RjtDQUN6RyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXHJcbmltcG9ydCB7IEFwcCB9IGZyb20gJ2F3cy1jZGstbGliJztcclxuaW1wb3J0IHsgR21haWxWaWV3ZXJDZGtTdGFjayB9IGZyb20gJy4uL2xpYi9nbWFpbC12aWV3ZXItY2RrLXN0YWNrJztcclxuXHJcbmNvbnN0IGFwcCA9IG5ldyBBcHAoKTtcclxuXHJcbi8vIFRvIHNwZWNpZnkgc3VibmV0cywgdXNlIHRoZSAtLWNvbnRleHQgb3B0aW9uIHdpdGggYSBjb21tYS1zZXBhcmF0ZWQgbGlzdCBvZiBzdWJuZXQgSURzOlxyXG4vLyBjZGsgZGVwbG95IC1jIHByaXZhdGVTdWJuZXRzPXN1Ym5ldC14eHh4eHh4eCxzdWJuZXQteXl5eXl5eXlcclxuY29uc3QgcHJpdmF0ZVN1Ym5ldHMgPSBhcHAubm9kZS50cnlHZXRDb250ZXh0KCdwcml2YXRlU3VibmV0cycpPy5zcGxpdCgnLCcpO1xyXG5cclxubmV3IEdtYWlsVmlld2VyQ2RrU3RhY2soYXBwLCAnR21haWxWaWV3ZXJDZGtTdGFjaycsIHtcclxuICBlbnY6IHtcclxuICAgIGFjY291bnQ6IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX0FDQ09VTlQsXHJcbiAgICByZWdpb246IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX1JFR0lPTixcclxuICB9LFxyXG4gIHByaXZhdGVTdWJuZXRzOiBbJ3N1Ym5ldC0wZDgyOTBkMGVhNTc1MzMxYycsIFx0J3N1Ym5ldC0wMzIyOTA1NjlhN2VjZjIxMCddLFxyXG4gIGdpdEh1Yk93bmVyOiAnZHlhbmV0JyxcclxuICBnaXRIdWJSZXBvOiAnaW1hcCcsXHJcbiAgZ2l0SHViQnJhbmNoOiAnbWFpbicsXHJcbiAgY29kZUNvbm5lY3Rpb25Bcm46ICdhcm46YXdzOmNvZGVjb25uZWN0aW9uczpjYS1jZW50cmFsLTE6MjM5MDMwMDMxNDU3OmNvbm5lY3Rpb24vMjlmZTVjMTEtMjNkMi00YzEwLWE4MWQtNGU4ODBiMTJlMmM2JyxcclxuICB1c2VHaXRIdWJXZWJob29rczogZmFsc2UsXHJcbiAgYXBpQ3VzdG9tRG9tYWluTmFtZTogJ21haWwuZHlhbmV0LmNvbScsXHJcbiAgY2VydGlmaWNhdGVBcm46ICdhcm46YXdzOmFjbTpjYS1jZW50cmFsLTE6MjM5MDMwMDMxNDU3OmNlcnRpZmljYXRlL2FmOTE1MGYxLTQ2MzUtNDE1NC04YjE5LWU1NDQ5ZDU3ZTk3MScsXHJcbn0pO1xyXG4iXX0= \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ21haWwtdmlld2VyLWNkay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImdtYWlsLXZpZXdlci1jZGsudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EsNkNBQWtDO0FBQ2xDLDBFQUFvRTtBQUVwRSxNQUFNLEdBQUcsR0FBRyxJQUFJLGlCQUFHLEVBQUUsQ0FBQztBQUV0QiwwRkFBMEY7QUFDMUYsK0RBQStEO0FBQy9ELE1BQU0sY0FBYyxHQUFHLEdBQUcsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLGdCQUFnQixDQUFDLEVBQUUsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBRTVFLElBQUksNENBQW1CLENBQUMsR0FBRyxFQUFFLHFCQUFxQixFQUFFO0lBQ2xELEdBQUcsRUFBRTtRQUNILE9BQU8sRUFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLG1CQUFtQjtRQUN4QyxNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0I7S0FDdkM7SUFDRCxjQUFjLEVBQUUsQ0FBQywwQkFBMEIsRUFBRSwwQkFBMEIsQ0FBQztJQUN4RSxhQUFhLEVBQUUsQ0FBQywwQkFBMEIsRUFBRSwwQkFBMEIsQ0FBQztJQUN2RSxXQUFXLEVBQUUsUUFBUTtJQUNyQixVQUFVLEVBQUUsTUFBTTtJQUNsQixZQUFZLEVBQUUsTUFBTTtJQUNwQixpQkFBaUIsRUFBRSxtR0FBbUc7SUFDdEgsaUJBQWlCLEVBQUUsS0FBSztJQUN4QixtQkFBbUIsRUFBRSxpQkFBaUI7SUFDdEMsY0FBYyxFQUFFLHdGQUF3RjtDQUN6RyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXHJcbmltcG9ydCB7IEFwcCB9IGZyb20gJ2F3cy1jZGstbGliJztcclxuaW1wb3J0IHsgR21haWxWaWV3ZXJDZGtTdGFjayB9IGZyb20gJy4uL2xpYi9nbWFpbC12aWV3ZXItY2RrLXN0YWNrJztcclxuXHJcbmNvbnN0IGFwcCA9IG5ldyBBcHAoKTtcclxuXHJcbi8vIFRvIHNwZWNpZnkgc3VibmV0cywgdXNlIHRoZSAtLWNvbnRleHQgb3B0aW9uIHdpdGggYSBjb21tYS1zZXBhcmF0ZWQgbGlzdCBvZiBzdWJuZXQgSURzOlxyXG4vLyBjZGsgZGVwbG95IC1jIHByaXZhdGVTdWJuZXRzPXN1Ym5ldC14eHh4eHh4eCxzdWJuZXQteXl5eXl5eXlcclxuY29uc3QgcHJpdmF0ZVN1Ym5ldHMgPSBhcHAubm9kZS50cnlHZXRDb250ZXh0KCdwcml2YXRlU3VibmV0cycpPy5zcGxpdCgnLCcpO1xyXG5cclxubmV3IEdtYWlsVmlld2VyQ2RrU3RhY2soYXBwLCAnR21haWxWaWV3ZXJDZGtTdGFjaycsIHtcclxuICBlbnY6IHtcclxuICAgIGFjY291bnQ6IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX0FDQ09VTlQsXHJcbiAgICByZWdpb246IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX1JFR0lPTixcclxuICB9LFxyXG4gIHByaXZhdGVTdWJuZXRzOiBbJ3N1Ym5ldC0wZDgyOTBkMGVhNTc1MzMxYycsICdzdWJuZXQtMDMyMjkwNTY5YTdlY2YyMTAnXSxcclxuICBwdWJsaWNTdWJuZXRzOiBbJ3N1Ym5ldC0wNzEzYWRlMzBkZDhjMzA4YycsICdzdWJuZXQtMGI0ZGQ0NWI1MWJiODI1MmYnXSxcclxuICBnaXRIdWJPd25lcjogJ2R5YW5ldCcsXHJcbiAgZ2l0SHViUmVwbzogJ2ltYXAnLFxyXG4gIGdpdEh1YkJyYW5jaDogJ21haW4nLFxyXG4gIGNvZGVDb25uZWN0aW9uQXJuOiAnYXJuOmF3czpjb2RlY29ubmVjdGlvbnM6Y2EtY2VudHJhbC0xOjIzOTAzMDAzMTQ1Nzpjb25uZWN0aW9uLzI5ZmU1YzExLTIzZDItNGMxMC1hODFkLTRlODgwYjEyZTJjNicsXHJcbiAgdXNlR2l0SHViV2ViaG9va3M6IGZhbHNlLFxyXG4gIGFwaUN1c3RvbURvbWFpbk5hbWU6ICdtYWlsLmR5YW5ldC5jb20nLFxyXG4gIGNlcnRpZmljYXRlQXJuOiAnYXJuOmF3czphY206Y2EtY2VudHJhbC0xOjIzOTAzMDAzMTQ1NzpjZXJ0aWZpY2F0ZS9hZjkxNTBmMS00NjM1LTQxNTQtOGIxOS1lNTQ0OWQ1N2U5NzEnLFxyXG59KTtcclxuIl19 \ 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 a0292c4..b984745 100644 --- a/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.ts +++ b/examples/gmail-viewer-cdk/bin/gmail-viewer-cdk.ts @@ -13,7 +13,8 @@ new GmailViewerCdkStack(app, 'GmailViewerCdkStack', { account: process.env.CDK_DEFAULT_ACCOUNT, region: process.env.CDK_DEFAULT_REGION, }, - privateSubnets: ['subnet-0d8290d0ea575331c', 'subnet-032290569a7ecf210'], + privateSubnets: ['subnet-0d8290d0ea575331c', 'subnet-032290569a7ecf210'], + publicSubnets: ['subnet-0713ade30dd8c308c', 'subnet-0b4dd45b51bb8252f'], gitHubOwner: 'dyanet', gitHubRepo: 'imap', gitHubBranch: 'main', diff --git a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.d.ts b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.d.ts index 0f6ce4b..177ad7f 100644 --- a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.d.ts +++ b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.d.ts @@ -2,6 +2,7 @@ import * as cdk from 'aws-cdk-lib'; export interface GmailViewerCdkStackProps extends cdk.StackProps { readonly vpcId?: string; readonly privateSubnets?: string[]; + readonly publicSubnets?: string[]; readonly containerPort?: number | string; readonly containerCpu?: number | string; readonly containerMemory?: number | string; 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 ae921de..d3db55e 100644 --- a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.js +++ b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.js @@ -37,6 +37,7 @@ exports.GmailViewerCdkStack = void 0; const cdk = __importStar(require("aws-cdk-lib")); const apigatewayv2 = __importStar(require("aws-cdk-lib/aws-apigatewayv2")); const codebuild = __importStar(require("aws-cdk-lib/aws-codebuild")); +const cr = __importStar(require("aws-cdk-lib/custom-resources")); const ec2 = __importStar(require("aws-cdk-lib/aws-ec2")); const ecr = __importStar(require("aws-cdk-lib/aws-ecr")); const ecs = __importStar(require("aws-cdk-lib/aws-ecs")); @@ -44,6 +45,7 @@ const iam = __importStar(require("aws-cdk-lib/aws-iam")); 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; @@ -60,611 +62,185 @@ class GmailViewerCdkStack extends cdk.Stack { const toCpuUnits = (value) => { const parsed = toNumber(value, 0.5); const vcpu = parsed <= 4 ? parsed * 1024 : parsed; - const rounded = Math.round(vcpu / 256) * 256; - return Math.max(256, rounded); + return Math.max(256, Math.round(vcpu / 256) * 256); }; const repoParts = (process.env.GITHUB_REPOSITORY ?? '').split('/'); - const guessedGitHubOwner = repoParts[0] || 'dyanet'; - const guessedGitHubRepo = repoParts[1] || 'imap'; const containerPort = toNumber(props.containerPort, 3000); const containerCpu = toCpuUnits(props.containerCpu); const containerMemory = toNumber(props.containerMemory, 1024); const desiredCount = toNumber(props.desiredCount, 1); - const serviceDiscoveryNamespaceName = props.serviceDiscoveryNamespaceName ?? 'mail.local'; - const serviceDiscoveryTtl = toNumber(props.serviceDiscoveryTtl, 60); + const nsName = props.serviceDiscoveryNamespaceName ?? 'mail.local'; + const nsTtl = cdk.Duration.seconds(toNumber(props.serviceDiscoveryTtl, 60)); const ssmPrefixRaw = props.ssmPrefix ?? '/mail-example'; const ssmPrefix = '/' + ssmPrefixRaw.split('/').filter(p => p).join('/'); - const gitHubOwner = props.gitHubOwner ?? guessedGitHubOwner; - const gitHubRepo = props.gitHubRepo ?? guessedGitHubRepo; + const gitHubOwner = props.gitHubOwner ?? (repoParts[0] || 'dyanet'); + const gitHubRepo = props.gitHubRepo ?? (repoParts[1] || 'imap'); const gitHubBranch = props.gitHubBranch ?? process.env.GITHUB_REF_NAME ?? 'main'; const publicEnvValue = props.publicEnvValue ?? 'production'; - const googleClientId = props.googleClientId || 'not-set'; - const googleClientSecret = props.googleClientSecret ?? 'not-set'; - const sessionSecretValue = props.sessionSecretValue ?? 'not-set'; - const apiCustomDomainName = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'mail.dyanet.com'; - const certificateArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN; - const hostedZoneId = props.hostedZoneId ?? ''; - const hasCustomDomain = Boolean(certificateArn && apiCustomDomainName); - const hasHostedZone = Boolean(hostedZoneId); + 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 certArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN; + const zoneId = props.hostedZoneId ?? ''; + const hasCustomDomain = Boolean(certArn && apiDomain); + const hasHostedZone = Boolean(zoneId); + const usePublicSubnets = props.publicSubnets && props.publicSubnets.length > 0; + // VPC setup let vpc; - let privateSubnetIds; let serviceSubnetIds; - // If account/region are available, use Vpc.fromLookup to automatically discover subnets. - // Otherwise fall back to CloudFormation parameters which let the deployer pick values in the console. const useVpcParameter = this.node.tryGetContext('useVpcParameter'); if (!useVpcParameter && this.account && this.region) { vpc = ec2.Vpc.fromLookup(this, 'MailVpc', { tags: { Name: 'dya-vpc' } }); - if (props.privateSubnets && props.privateSubnets.length > 0) { - privateSubnetIds = props.privateSubnets; + if (usePublicSubnets) { + serviceSubnetIds = props.publicSubnets; + } + else if (props.privateSubnets && props.privateSubnets.length > 0) { + serviceSubnetIds = props.privateSubnets; } else { - // The service linked to the VPC Link may not be available in all Availability Zones. const badAz = 'cac1-az4'; let subnets = vpc.privateSubnets.filter(s => s.availabilityZone !== badAz); - if (subnets.length === 0) { - // No private subnets, try public subnets + if (subnets.length === 0) subnets = vpc.publicSubnets.filter(s => s.availabilityZone !== badAz); - } - privateSubnetIds = subnets.slice(0, 2).map(s => s.subnetId); - if (privateSubnetIds.length < 1) { - throw new Error(`VPC '${vpc.vpcId}' must have at least one private or public subnet in an allowed availability zone (not in ${badAz}).`); - } + serviceSubnetIds = subnets.slice(0, 2).map(s => s.subnetId); + if (serviceSubnetIds.length < 1) + throw new Error(`VPC must have at least one subnet in an allowed AZ.`); } - serviceSubnetIds = privateSubnetIds; } else { - // CloudFormation parameter for VPC selection (dropdown of VPC IDs in the console) - const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', { - type: 'AWS::EC2::VPC::Id', - default: props.vpcId ?? '', - }); - // Optional parameter for private subnet IDs (CommaDelimitedList) when synthesizing without account/region - const privateSubnetIdsParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', { - type: 'CommaDelimitedList', - default: props.privateSubnets && props.privateSubnets.length > 0 ? props.privateSubnets.join(',') : '', - }); - // Use the provided parameter values (deployer must supply private subnet ids when synthesizing without lookup) - vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', { - vpcId: vpcIdParam.valueAsString, - availabilityZones: cdk.Fn.getAzs(), - privateSubnetIds: privateSubnetIdsParam.valueAsList, - }); - privateSubnetIds = props.privateSubnets ?? privateSubnetIdsParam.valueAsList; - serviceSubnetIds = privateSubnetIds; - } - const apiVpcLinkSecurityGroup = new ec2.CfnSecurityGroup(this, 'ApiVpcLinkSecurityGroup', { - groupDescription: 'Egress from API Gateway VPC Link to service', - vpcId: vpc.vpcId, - securityGroupEgress: [ - { - ipProtocol: 'tcp', - fromPort: containerPort, - toPort: containerPort, - cidrIp: '0.0.0.0/0', - }, - ], - }); - apiVpcLinkSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const serviceSecurityGroup = new ec2.CfnSecurityGroup(this, 'ServiceSecurityGroup', { - groupDescription: 'Allow API Gateway VPC Link to reach Fargate tasks', - vpcId: vpc.vpcId, - securityGroupIngress: [ - { - ipProtocol: 'tcp', - fromPort: containerPort, - toPort: containerPort, - cidrIp: '0.0.0.0/0', - }, - ], - securityGroupEgress: [ - { - ipProtocol: '-1', - fromPort: 0, - toPort: 65535, - cidrIp: '0.0.0.0/0', - }, - ], - }); - serviceSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const appLogGroup = new logs.CfnLogGroup(this, 'AppLogGroup', { - logGroupName: '/ecs/mail-example', - retentionInDays: 30, - }); - appLogGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const mailCluster = new ecs.CfnCluster(this, 'MailCluster', { - clusterName: 'mail-cluster', - capacityProviders: ['FARGATE', 'FARGATE_SPOT'], - defaultCapacityProviderStrategy: [ - { - capacityProvider: 'FARGATE_SPOT', - weight: 4, - }, - { - capacityProvider: 'FARGATE', - weight: 1, - }, - ], - }); - mailCluster.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const mailExampleRepository = new ecr.CfnRepository(this, 'MailExampleRepository', { - repositoryName: 'mail-example', - imageTagMutability: 'MUTABLE', - encryptionConfiguration: { - encryptionType: 'AES256', - }, - lifecyclePolicy: { - lifecyclePolicyText: '{\n "rules": [\n {\n "rulePriority": 1,\n "description": "Expire images older than 30 days",\n "selection": {\n "tagStatus": "any",\n "countType": "sinceImagePushed",\n "countUnit": "days",\n "countNumber": 30\n },\n "action": { "type": "expire" }\n }\n ]\n}\n', - }, - }); - mailExampleRepository.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const mailHttpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', { - name: 'mail-example', - protocolType: 'HTTP', - }); - mailHttpApi.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const mailApiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', { - stageName: 'prod', - apiId: mailHttpApi.ref, - autoDeploy: true, - }); - mailApiStage.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const apiDomainName = hasCustomDomain - ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', { - domainName: apiCustomDomainName, - domainNameConfigurations: [ - { - certificateArn: certificateArn, - endpointType: 'REGIONAL', - securityPolicy: 'TLS_1_2', - }, - ], - }) - : undefined; - if (apiDomainName) { - apiDomainName.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - } - const apiMapping = hasCustomDomain && apiDomainName - ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', { - apiId: mailHttpApi.ref, - domainName: apiDomainName.ref, - stage: mailApiStage.ref, - apiMappingKey: 'examples', - }) - : undefined; - if (apiMapping) { - apiMapping.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - } - const apiDomainRecord = hasCustomDomain && hasHostedZone && apiDomainName - ? new route53.CfnRecordSet(this, 'ApiDomainRecord', { - hostedZoneId, - name: apiCustomDomainName, - type: 'A', - aliasTarget: { - dnsName: apiDomainName.attrRegionalDomainName, - hostedZoneId: apiDomainName.attrRegionalHostedZoneId, - }, - }) - : undefined; - if (apiDomainRecord) { - apiDomainRecord.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', { type: 'AWS::EC2::VPC::Id', default: props.vpcId ?? '' }); + const subnetParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', { type: 'CommaDelimitedList', default: props.privateSubnets?.join(',') ?? '' }); + vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', { vpcId: vpcIdParam.valueAsString, availabilityZones: cdk.Fn.getAzs(), privateSubnetIds: subnetParam.valueAsList }); + serviceSubnetIds = props.privateSubnets ?? subnetParam.valueAsList; } + // Security Groups (L2) + const apiVpcLinkSg = new ec2.SecurityGroup(this, 'ApiVpcLinkSecurityGroup', { vpc, description: 'Egress from API Gateway VPC Link', allowAllOutbound: false }); + apiVpcLinkSg.addEgressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(containerPort)); + const serviceSg = new ec2.SecurityGroup(this, 'ServiceSecurityGroup', { vpc, description: 'Allow traffic to Fargate tasks', allowAllOutbound: true }); + serviceSg.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(containerPort)); + // Log Group (L2) + const appLogGroup = new logs.LogGroup(this, 'AppLogGroup', { logGroupName: '/ecs/mail-example', retention: logs.RetentionDays.ONE_MONTH, removalPolicy: cdk.RemovalPolicy.DESTROY }); + // ECS Cluster (L2) + const cluster = new ecs.Cluster(this, 'MailCluster', { vpc, clusterName: 'public-cluster', enableFargateCapacityProviders: true }); + // ECR Repository - create if not exists using custom resource + const ecrRepoName = 'mail-example'; + const checkEcrRepo = new cr.AwsCustomResource(this, 'CheckEcrRepo', { + onCreate: { service: 'ECR', action: 'describeRepositories', parameters: { repositoryNames: [ecrRepoName] }, physicalResourceId: cr.PhysicalResourceId.of(ecrRepoName) }, + policy: cr.AwsCustomResourcePolicy.fromSdkCalls({ resources: cr.AwsCustomResourcePolicy.ANY_RESOURCE }), + }); + const createEcrRepo = new cr.AwsCustomResource(this, 'CreateEcrRepo', { + onCreate: { service: 'ECR', action: 'createRepository', parameters: { repositoryName: ecrRepoName, imageTagMutability: 'MUTABLE', encryptionConfiguration: { encryptionType: 'AES256' } }, physicalResourceId: cr.PhysicalResourceId.of(ecrRepoName), ignoreErrorCodesMatching: 'RepositoryAlreadyExistsException' }, + policy: cr.AwsCustomResourcePolicy.fromSdkCalls({ resources: cr.AwsCustomResourcePolicy.ANY_RESOURCE }), + }); + createEcrRepo.node.addDependency(checkEcrRepo); + const ecrRepo = ecr.Repository.fromRepositoryName(this, 'MailExampleRepositoryRef', ecrRepoName); + // API Gateway HTTP API (L1 - no L2 for HTTP API with VPC Link) + const httpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', { name: 'mail-example', protocolType: 'HTTP' }); + const apiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', { stageName: 'prod', apiId: httpApi.ref, autoDeploy: true }); + const apiDomainName = hasCustomDomain ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', { + domainName: apiDomain, domainNameConfigurations: [{ certificateArn: certArn, endpointType: 'REGIONAL', securityPolicy: 'TLS_1_2' }], + }) : undefined; + const apiMapping = hasCustomDomain && apiDomainName ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', { + apiId: httpApi.ref, domainName: apiDomainName.ref, stage: apiStage.ref, apiMappingKey: 'examples', + }) : undefined; if (apiMapping && apiDomainName) { apiMapping.addDependency(apiDomainName); - apiMapping.addDependency(mailApiStage); + apiMapping.addDependency(apiStage); } - if (apiDomainRecord && apiDomainName) { - apiDomainRecord.addDependency(apiDomainName); - } - const baseUrl = hasCustomDomain - ? `https://${apiCustomDomainName}/examples/gmail-viewer` - : cdk.Fn.join('', [mailHttpApi.attrApiEndpoint, '/', mailApiStage.ref, '/gmail-viewer']); - const serviceNamespace = new servicediscovery.CfnPrivateDnsNamespace(this, 'ServiceNamespace', { - name: serviceDiscoveryNamespaceName, - vpc: vpc.vpcId, - description: 'Namespace for mail services', - }); - serviceNamespace.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const apiVpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', { - name: 'mail-example-vpclink', - subnetIds: serviceSubnetIds, - securityGroupIds: [apiVpcLinkSecurityGroup.ref], - }); - apiVpcLink.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const appCloudMapService = new servicediscovery.CfnService(this, 'AppCloudMapService', { - name: 'mail-example', - namespaceId: serviceNamespace.ref, - dnsConfig: { - routingPolicy: 'WEIGHTED', - dnsRecords: [ - { - ttl: serviceDiscoveryTtl, - type: 'SRV', - }, - ], + const apiRecord = hasCustomDomain && hasHostedZone && apiDomainName ? new route53.CfnRecordSet(this, 'ApiDomainRecord', { + hostedZoneId: zoneId, name: apiDomain, type: 'A', aliasTarget: { dnsName: apiDomainName.attrRegionalDomainName, hostedZoneId: apiDomainName.attrRegionalHostedZoneId }, + }) : undefined; + if (apiRecord && apiDomainName) + apiRecord.addDependency(apiDomainName); + const baseUrl = hasCustomDomain ? `https://${apiDomain}/examples/gmail-viewer` : cdk.Fn.join('', [httpApi.attrApiEndpoint, '/', apiStage.ref, '/gmail-viewer']); + // Cloud Map Namespace (L2) + const namespace = new servicediscovery.PrivateDnsNamespace(this, 'ServiceNamespace', { name: nsName, vpc, description: 'Namespace for mail services' }); + // VPC Link (L1) + const vpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', { name: 'mail-example-vpclink', subnetIds: serviceSubnetIds, securityGroupIds: [apiVpcLinkSg.securityGroupId] }); + // Cloud Map Service (L2) + 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 + const appSecrets = new secretsmanager.Secret(this, 'AppSecrets', { + secretName: `${ssmPrefix}/secrets`, + description: 'Application secrets for mail-example', + secretObjectValue: { + GOOGLE_CLIENT_SECRET: cdk.SecretValue.unsafePlainText(googleClientSecret), + SESSION_SECRET: cdk.SecretValue.unsafePlainText(sessionSecretValue), }, - healthCheckCustomConfig: { - failureThreshold: 1, - }, - }); - appCloudMapService.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const taskExecutionRole = new iam.CfnRole(this, 'TaskExecutionRole', { - assumeRolePolicyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Principal: { - Service: 'ecs-tasks.amazonaws.com', - }, - Action: 'sts:AssumeRole', - }, - ], - }, - managedPolicyArns: ['arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy'], - policies: [ - { - policyName: 'AllowParameterReadForSecrets', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], - Resource: [ - `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, - `arn:aws:kms:${this.region}:${this.account}:key/*`, - ], - }, - ], - }, - }, - ], }); - taskExecutionRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const taskRole = new iam.CfnRole(this, 'TaskRole', { - assumeRolePolicyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Principal: { - Service: 'ecs-tasks.amazonaws.com', - }, - Action: 'sts:AssumeRole', - }, - ], + // IAM Roles (L2) + const taskExecRole = new iam.Role(this, 'TaskExecutionRole', { assumedBy: new iam.ServicePrincipal('ecs-tasks.amazonaws.com'), managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AmazonECSTaskExecutionRolePolicy')] }); + 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/*`] })); + taskExecRole.addToPolicy(new iam.PolicyStatement({ actions: ['secretsmanager:GetSecretValue'], resources: [appSecrets.secretArn] })); + 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', { + image: ecs.ContainerImage.fromEcrRepository(ecrRepo, 'latest'), portMappings: [{ containerPort }], + 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_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_SECRET'), + SESSION_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'SESSION_SECRET'), }, - policies: [ - { - policyName: 'AppRuntimeConfigRead', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], - Resource: [ - `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, - `arn:aws:kms:${this.region}:${this.account}:key/*`, - ], - }, - ], - }, - }, - ], - }); - taskRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const appTaskDefinition = new ecs.CfnTaskDefinition(this, 'AppTaskDefinition', { - family: 'mail-example', - cpu: containerCpu.toString(), - memory: containerMemory.toString(), - networkMode: 'awsvpc', - requiresCompatibilities: ['FARGATE'], - executionRoleArn: taskExecutionRole.attrArn, - taskRoleArn: taskRole.attrArn, - containerDefinitions: [ - { - name: 'mail-example', - image: `${mailExampleRepository.attrRepositoryUri}:latest`, - portMappings: [ - { - containerPort, - }, - ], - logConfiguration: { - logDriver: 'awslogs', - options: { - 'awslogs-group': appLogGroup.ref, - 'awslogs-region': this.region, - 'awslogs-stream-prefix': 'mail-example', - }, - }, - environment: [ - { - name: 'CONFIG_SSM_PREFIX', - value: ssmPrefix, - }, - { - name: 'PORT', - value: containerPort.toString(), - }, - { - name: 'BASE_URL', - value: baseUrl.toString(), - }, - { - name: 'NODE_ENV', - value: publicEnvValue, - }, - ], - secrets: [ - { - name: 'GOOGLE_CLIENT_ID', - valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/env/GOOGLE_CLIENT_ID`, - }, - { - name: 'GOOGLE_CLIENT_SECRET', - valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`, - }, - { - name: 'SESSION_SECRET', - valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/SESSION_SECRET`, - }, - ], - }, - ], }); - appTaskDefinition.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - new ecs.CfnService(this, 'AppService', { - serviceName: 'mail-example', - cluster: mailCluster.ref, - taskDefinition: appTaskDefinition.ref, - desiredCount, - capacityProviderStrategy: [ - { - capacityProvider: 'FARGATE_SPOT', - weight: 4, - }, - { - capacityProvider: 'FARGATE', - weight: 1, - }, - ], - deploymentConfiguration: { - maximumPercent: 200, - minimumHealthyPercent: 50, - }, - networkConfiguration: { - awsvpcConfiguration: { - assignPublicIp: 'DISABLED', - subnets: serviceSubnetIds, - securityGroups: [serviceSecurityGroup.ref], - }, - }, - serviceRegistries: [ - { - registryArn: appCloudMapService.attrArn, - containerName: 'mail-example', - containerPort: containerPort, - }, - ], - platformVersion: 'LATEST', - }); - const mailIntegration = new apigatewayv2.CfnIntegration(this, 'MailIntegration', { - apiId: mailHttpApi.ref, - integrationType: 'HTTP_PROXY', - integrationMethod: 'ANY', - integrationUri: appCloudMapService.attrArn, - connectionType: 'VPC_LINK', - connectionId: apiVpcLink.ref, - payloadFormatVersion: '1.0', - }); - new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', { - apiId: mailHttpApi.ref, - routeKey: 'ANY /gmail-viewer', - target: `integrations/${mailIntegration.ref}`, - }); - new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', { - apiId: mailHttpApi.ref, - routeKey: 'ANY /gmail-viewer/{proxy+}', - target: `integrations/${mailIntegration.ref}`, - }); - const ssmBaseUrl = new ssm.StringParameter(this, 'SsmBaseUrl', { - parameterName: `${ssmPrefix}/env/BASE_URL`, - stringValue: baseUrl, - type: ssm.ParameterType.STRING, - }); - ssmBaseUrl.node.defaultChild.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const ssmGoogleClientId = new ssm.StringParameter(this, 'SsmGoogleClientId', { - parameterName: `${ssmPrefix}/env/GOOGLE_CLIENT_ID`, - stringValue: googleClientId, - type: ssm.ParameterType.STRING, - }); - ssmGoogleClientId.node.defaultChild.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const ssmGoogleClientSecret = new ssm.StringParameter(this, 'SsmGoogleClientSecret', { - parameterName: `${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`, - stringValue: googleClientSecret, - type: ssm.ParameterType.STRING, - }); - ssmGoogleClientSecret.node.defaultChild.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const ssmPort = new ssm.StringParameter(this, 'SsmPort', { - parameterName: `${ssmPrefix}/env/PORT`, - stringValue: containerPort.toString(), - type: ssm.ParameterType.STRING, - }); - ssmPort.node.defaultChild.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const ssmPublicEnv = new ssm.StringParameter(this, 'SsmPublicEnv', { - parameterName: `${ssmPrefix}/env/NODE_ENV`, - stringValue: publicEnvValue, - type: ssm.ParameterType.STRING, - }); - ssmPublicEnv.node.defaultChild.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const ssmSessionSecret = new ssm.StringParameter(this, 'SsmSessionSecret', { - parameterName: `${ssmPrefix}/secrets/SESSION_SECRET`, - stringValue: sessionSecretValue, - type: ssm.ParameterType.STRING, - }); - ssmSessionSecret.node.defaultChild.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const codeBuildRole = new iam.CfnRole(this, 'CodeBuildRole', { - assumeRolePolicyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Principal: { - Service: 'codebuild.amazonaws.com', - }, - Action: 'sts:AssumeRole', - }, - ], - }, - policies: [ - { - policyName: 'BuildLogs', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'], - Resource: '*', - }, - ], - }, - }, - { - policyName: 'ECRPush', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: [ - 'ecr:GetAuthorizationToken', - 'ecr:BatchCheckLayerAvailability', - 'ecr:CompleteLayerUpload', - 'ecr:BatchGetImage', - 'ecr:DescribeRepositories', - 'ecr:InitiateLayerUpload', - 'ecr:PutImage', - 'ecr:UploadLayerPart', - ], - Resource: '*', - }, - ], - }, - }, - { - policyName: 'ECSDeploy', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'], - Resource: '*', - }, - ], - }, - }, - { - policyName: 'SSMReadForBuild', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'], - Resource: '*', - }, - ], - }, - }, - { - policyName: 'CodeConnectionAccess', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: [ - "codeconnections:GetConnection", - "codeconnections:GetConnectionToken", - "codeconnections:UseConnection" - ], - Resource: props.codeConnectionArn, - }, - ], - }, - } - ], - }); - codeBuildRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - // Use the higher-level Project construct so we can provide a typed BuildSpec - const codebuildRoleRef = iam.Role.fromRoleArn(this, 'CodeBuildRoleRef', codeBuildRole.attrArn); - const codeBuildProject = new codebuild.Project(this, 'CodeBuildProject', { - projectName: 'mail-example', - role: codebuildRoleRef, - environment: { - buildImage: codebuild.LinuxBuildImage.STANDARD_7_0, - computeType: codebuild.ComputeType.SMALL, - privileged: true, - environmentVariables: { - ECR_URI: { value: mailExampleRepository.attrRepositoryUri }, - CLUSTER_NAME: { value: mailCluster.ref }, - SERVICE_NAME: { value: 'mail-example' }, - }, - }, - source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch, }), - timeout: cdk.Duration.minutes(30), - queuedTimeout: cdk.Duration.minutes(30), - badge: true, - description: 'Build & deploy mail-example to ECR then force ECS deploy', - cache: codebuild.Cache.none(), + // ECS Service (L2) - associate with standalone Cloud Map service for API Gateway integration + const service = new ecs.FargateService(this, 'AppService', { + serviceName: 'mail-example', cluster, taskDefinition: taskDef, desiredCount, + capacityProviderStrategies: [{ capacityProvider: 'FARGATE_SPOT', weight: 4 }, { capacityProvider: 'FARGATE', weight: 1 }], + vpcSubnets: { subnets: serviceSubnetIds.map((id, i) => ec2.Subnet.fromSubnetId(this, `ServiceSubnet${i}`, id)) }, + securityGroups: [serviceSg], assignPublicIp: usePublicSubnets ?? false, minHealthyPercent: 50, maxHealthyPercent: 200, + }); + service.associateCloudMapService({ service: cloudMapService, containerPort, container: taskDef.defaultContainer }); + // API Gateway Integration (L1) + 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' }); + new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', { apiId: httpApi.ref, routeKey: 'ANY /gmail-viewer', target: `integrations/${integration.ref}` }); + new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', { apiId: httpApi.ref, routeKey: 'ANY /gmail-viewer/{proxy+}', target: `integrations/${integration.ref}` }); + // CodeBuild Role (L2) + const buildRole = new iam.Role(this, 'CodeBuildRole', { assumedBy: new iam.ServicePrincipal('codebuild.amazonaws.com') }); + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'], resources: ['*'] })); + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ecr:GetAuthorizationToken', 'ecr:BatchCheckLayerAvailability', 'ecr:CompleteLayerUpload', 'ecr:BatchGetImage', 'ecr:DescribeRepositories', 'ecr:InitiateLayerUpload', 'ecr:PutImage', 'ecr:UploadLayerPart'], resources: ['*'] })); + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'], resources: ['*'] })); + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'], resources: ['*'] })); + if (props.codeConnectionArn) + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['codeconnections:GetConnection', 'codeconnections:GetConnectionToken', 'codeconnections:UseConnection'], resources: [props.codeConnectionArn] })); + // CodeBuild Project (L2) + new codebuild.Project(this, 'CodeBuildProject', { + projectName: 'mail-example', role: buildRole, + 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' } } }, + source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch }), + timeout: cdk.Duration.minutes(30), queuedTimeout: cdk.Duration.minutes(30), badge: true, description: 'Build & deploy mail-example to ECR then force ECS deploy', buildSpec: codebuild.BuildSpec.fromObject({ version: '0.2', phases: { - 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}', - ] }, - build: { commands: [ - 'cd examples/gmail-viewer', - 'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .', - ] }, - 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', - ] }, + 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}'] }, + build: { commands: ['cd examples/gmail-viewer', 'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .'] }, + 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'] }, }, - artifacts: { files: [] }, - env: { shell: 'bash' }, + artifacts: { files: [] }, env: { shell: 'bash' }, }), }); - codeBuildProject.node.defaultChild.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - this.clusterName = mailCluster.ref; - new cdk.CfnOutput(this, 'CfnOutputClusterName', { - key: 'ClusterName', - value: this.clusterName.toString(), - }); - this.repositoryUri = mailExampleRepository.attrRepositoryUri; - new cdk.CfnOutput(this, 'CfnOutputRepositoryUri', { - key: 'RepositoryUri', - value: this.repositoryUri.toString(), - }); + // Outputs + this.clusterName = cluster.clusterName; + new cdk.CfnOutput(this, 'ClusterName', { value: this.clusterName }); + this.repositoryUri = ecrRepo.repositoryUri; + new cdk.CfnOutput(this, 'RepositoryUri', { value: this.repositoryUri }); this.apiInvokeUrl = baseUrl.toString(); - new cdk.CfnOutput(this, 'CfnOutputApiInvokeUrl', { - key: 'ApiInvokeUrl', - value: this.apiInvokeUrl, - }); - this.cloudMapServiceArn = appCloudMapService.attrArn; - new cdk.CfnOutput(this, 'CfnOutputCloudMapServiceArn', { - key: 'CloudMapServiceArn', - value: this.cloudMapServiceArn.toString(), - }); + new cdk.CfnOutput(this, 'ApiInvokeUrl', { value: this.apiInvokeUrl }); + this.cloudMapServiceArn = cloudMapService.serviceArn; + new cdk.CfnOutput(this, 'CloudMapServiceArn', { value: this.cloudMapServiceArn }); } } 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 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 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 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\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      const rounded = Math.round(vcpu / 256) * 256;\r\n      return Math.max(256, rounded);\r\n    };\r\n\r\n    const repoParts = (process.env.GITHUB_REPOSITORY ?? '').split('/');\r\n    const guessedGitHubOwner = repoParts[0] || 'dyanet';\r\n    const guessedGitHubRepo = repoParts[1] || 'imap';\r\n\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 serviceDiscoveryNamespaceName = props.serviceDiscoveryNamespaceName ?? 'mail.local';\r\n    const serviceDiscoveryTtl = 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 ?? guessedGitHubOwner;\r\n    const gitHubRepo = props.gitHubRepo ?? guessedGitHubRepo;\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 || 'not-set';\r\n    const googleClientSecret = props.googleClientSecret ?? 'not-set';\r\n    const sessionSecretValue = props.sessionSecretValue ?? 'not-set';\r\n    const apiCustomDomainName = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'mail.dyanet.com';\r\n    const certificateArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN;\r\n    const hostedZoneId = props.hostedZoneId ?? '';\r\n    const hasCustomDomain = Boolean(certificateArn && apiCustomDomainName);\r\n    const hasHostedZone = Boolean(hostedZoneId);\r\n\r\n    let vpc: ec2.IVpc;\r\n    let privateSubnetIds: string[];\r\n    let serviceSubnetIds: string[];\r\n\r\n    // If account/region are available, use Vpc.fromLookup to automatically discover subnets.\r\n    // Otherwise fall back to CloudFormation parameters which let the deployer pick values in the console.\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      \r\n      if (props.privateSubnets && props.privateSubnets.length > 0) {\r\n          privateSubnetIds = props.privateSubnets;\r\n      } else {\r\n          // The service linked to the VPC Link may not be available in all Availability Zones.\r\n          const badAz = 'cac1-az4';\r\n          \r\n          let subnets = vpc.privateSubnets.filter(s => s.availabilityZone !== badAz);\r\n          if (subnets.length === 0) {\r\n            // No private subnets, try public subnets\r\n            subnets = vpc.publicSubnets.filter(s => s.availabilityZone !== badAz);\r\n          }\r\n\r\n          privateSubnetIds = subnets.slice(0, 2).map(s => s.subnetId);\r\n          \r\n          if (privateSubnetIds.length < 1) {\r\n            throw new Error(`VPC '${vpc.vpcId}' must have at least one private or public subnet in an allowed availability zone (not in ${badAz}).`);\r\n          }\r\n      }\r\n      serviceSubnetIds = privateSubnetIds;\r\n    } else {\r\n      // CloudFormation parameter for VPC selection (dropdown of VPC IDs in the console)\r\n      const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', {\r\n        type: 'AWS::EC2::VPC::Id',\r\n        default: props.vpcId ?? '',\r\n      });\r\n\r\n      // Optional parameter for private subnet IDs (CommaDelimitedList) when synthesizing without account/region\r\n      const privateSubnetIdsParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', {\r\n        type: 'CommaDelimitedList',\r\n        default: props.privateSubnets && props.privateSubnets.length > 0 ? props.privateSubnets.join(',') : '',\r\n      });\r\n      // Use the provided parameter values (deployer must supply private subnet ids when synthesizing without lookup)\r\n      vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', {\r\n        vpcId: vpcIdParam.valueAsString,\r\n        availabilityZones: cdk.Fn.getAzs(),\r\n        privateSubnetIds: privateSubnetIdsParam.valueAsList,\r\n      });\r\n      privateSubnetIds = props.privateSubnets ?? privateSubnetIdsParam.valueAsList;\r\n      serviceSubnetIds = privateSubnetIds;\r\n    }\r\n\r\n    const apiVpcLinkSecurityGroup = new ec2.CfnSecurityGroup(this, 'ApiVpcLinkSecurityGroup', {\r\n      groupDescription: 'Egress from API Gateway VPC Link to service',\r\n      vpcId: vpc.vpcId,\r\n      securityGroupEgress: [\r\n        {\r\n          ipProtocol: 'tcp',\r\n          fromPort: containerPort,\r\n          toPort: containerPort,\r\n          cidrIp: '0.0.0.0/0',\r\n        },\r\n      ],\r\n    });\r\n    apiVpcLinkSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const serviceSecurityGroup = new ec2.CfnSecurityGroup(this, 'ServiceSecurityGroup', {\r\n      groupDescription: 'Allow API Gateway VPC Link to reach Fargate tasks',\r\n      vpcId: vpc.vpcId,\r\n      securityGroupIngress: [\r\n        {\r\n          ipProtocol: 'tcp',\r\n          fromPort: containerPort,\r\n          toPort: containerPort,\r\n          cidrIp: '0.0.0.0/0',\r\n        },\r\n      ],\r\n      securityGroupEgress: [\r\n        {\r\n          ipProtocol: '-1',\r\n          fromPort: 0,\r\n          toPort: 65535,\r\n          cidrIp: '0.0.0.0/0',\r\n        },\r\n      ],\r\n    });\r\n    serviceSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const appLogGroup = new logs.CfnLogGroup(this, 'AppLogGroup', {\r\n      logGroupName: '/ecs/mail-example',\r\n      retentionInDays: 30,\r\n    });\r\n    appLogGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const mailCluster = new ecs.CfnCluster(this, 'MailCluster', {\r\n      clusterName: 'mail-cluster',\r\n      capacityProviders: ['FARGATE', 'FARGATE_SPOT'],\r\n      defaultCapacityProviderStrategy: [\r\n        {\r\n          capacityProvider: 'FARGATE_SPOT',\r\n          weight: 4,\r\n        },\r\n        {\r\n          capacityProvider: 'FARGATE',\r\n          weight: 1,\r\n        },\r\n      ],\r\n    });\r\n    mailCluster.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const mailExampleRepository = new ecr.CfnRepository(this, 'MailExampleRepository', {\r\n      repositoryName: 'mail-example',\r\n      imageTagMutability: 'MUTABLE',\r\n      encryptionConfiguration: {\r\n        encryptionType: 'AES256',\r\n      },\r\n      lifecyclePolicy: {\r\n        lifecyclePolicyText:\r\n          '{\\n  \"rules\": [\\n    {\\n      \"rulePriority\": 1,\\n      \"description\": \"Expire images older than 30 days\",\\n      \"selection\": {\\n        \"tagStatus\": \"any\",\\n        \"countType\": \"sinceImagePushed\",\\n        \"countUnit\": \"days\",\\n        \"countNumber\": 30\\n      },\\n      \"action\": { \"type\": \"expire\" }\\n    }\\n  ]\\n}\\n',\r\n      },\r\n    });\r\n    mailExampleRepository.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const mailHttpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', {\r\n      name: 'mail-example',\r\n      protocolType: 'HTTP',\r\n    });\r\n    mailHttpApi.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const mailApiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', {\r\n      stageName: 'prod',\r\n      apiId: mailHttpApi.ref,\r\n      autoDeploy: true,\r\n    });\r\n    mailApiStage.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const apiDomainName = hasCustomDomain\r\n      ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', {\r\n          domainName: apiCustomDomainName,\r\n          domainNameConfigurations: [\r\n            {\r\n              certificateArn: certificateArn!,\r\n              endpointType: 'REGIONAL',\r\n              securityPolicy: 'TLS_1_2',\r\n            },\r\n          ],\r\n        })\r\n      : undefined;\r\n    if (apiDomainName) {\r\n      apiDomainName.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n    }\r\n\r\n    const apiMapping =\r\n      hasCustomDomain && apiDomainName\r\n        ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', {\r\n            apiId: mailHttpApi.ref,\r\n            domainName: apiDomainName.ref,\r\n            stage: mailApiStage.ref,\r\n            apiMappingKey: 'examples',\r\n          })\r\n        : undefined;\r\n    if (apiMapping) {\r\n      apiMapping.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n    }\r\n\r\n    const apiDomainRecord =\r\n      hasCustomDomain && hasHostedZone && apiDomainName\r\n        ? new route53.CfnRecordSet(this, 'ApiDomainRecord', {\r\n            hostedZoneId,\r\n            name: apiCustomDomainName,\r\n            type: 'A',\r\n            aliasTarget: {\r\n              dnsName: apiDomainName.attrRegionalDomainName,\r\n              hostedZoneId: apiDomainName.attrRegionalHostedZoneId,\r\n            },\r\n          })\r\n        : undefined;\r\n    if (apiDomainRecord) {\r\n      apiDomainRecord.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n    }\r\n    if (apiMapping && apiDomainName) {\r\n      apiMapping.addDependency(apiDomainName);\r\n      apiMapping.addDependency(mailApiStage);\r\n    }\r\n    if (apiDomainRecord && apiDomainName) {\r\n      apiDomainRecord.addDependency(apiDomainName);\r\n    }\r\n\r\n    const baseUrl = hasCustomDomain\r\n      ? `https://${apiCustomDomainName}/examples/gmail-viewer`\r\n      : cdk.Fn.join('', [mailHttpApi.attrApiEndpoint, '/', mailApiStage.ref, '/gmail-viewer']);\r\n\r\n    const serviceNamespace = new servicediscovery.CfnPrivateDnsNamespace(this, 'ServiceNamespace', {\r\n      name: serviceDiscoveryNamespaceName,\r\n      vpc: vpc.vpcId,\r\n      description: 'Namespace for mail services',\r\n    });\r\n    serviceNamespace.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const apiVpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', {\r\n      name: 'mail-example-vpclink',\r\n      subnetIds: serviceSubnetIds,\r\n      securityGroupIds: [apiVpcLinkSecurityGroup.ref],\r\n    });\r\n    apiVpcLink.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const appCloudMapService = new servicediscovery.CfnService(this, 'AppCloudMapService', {\r\n      name: 'mail-example',\r\n      namespaceId: serviceNamespace.ref,\r\n      dnsConfig: {\r\n        routingPolicy: 'WEIGHTED',\r\n        dnsRecords: [\r\n          {\r\n            ttl: serviceDiscoveryTtl,\r\n            type: 'SRV',\r\n          },\r\n        ],\r\n      },\r\n      healthCheckCustomConfig: {\r\n        failureThreshold: 1,\r\n      },\r\n    });\r\n    appCloudMapService.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const taskExecutionRole = new iam.CfnRole(this, 'TaskExecutionRole', {\r\n      assumeRolePolicyDocument: {\r\n        Version: '2012-10-17',\r\n        Statement: [\r\n          {\r\n            Effect: 'Allow',\r\n            Principal: {\r\n              Service: 'ecs-tasks.amazonaws.com',\r\n            },\r\n            Action: 'sts:AssumeRole',\r\n          },\r\n        ],\r\n      },\r\n      managedPolicyArns: ['arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy'],\r\n      policies: [\r\n        {\r\n          policyName: 'AllowParameterReadForSecrets',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'],\r\n                Resource: [\r\n                  `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`,\r\n                  `arn:aws:kms:${this.region}:${this.account}:key/*`,\r\n                ],\r\n              },\r\n            ],\r\n          },\r\n        },\r\n      ],\r\n    });\r\n    taskExecutionRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const taskRole = new iam.CfnRole(this, 'TaskRole', {\r\n      assumeRolePolicyDocument: {\r\n        Version: '2012-10-17',\r\n        Statement: [\r\n          {\r\n            Effect: 'Allow',\r\n            Principal: {\r\n              Service: 'ecs-tasks.amazonaws.com',\r\n            },\r\n            Action: 'sts:AssumeRole',\r\n          },\r\n        ],\r\n      },\r\n      policies: [\r\n        {\r\n          policyName: 'AppRuntimeConfigRead',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'],\r\n                Resource: [\r\n                  `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`,\r\n                  `arn:aws:kms:${this.region}:${this.account}:key/*`,\r\n                ],\r\n              },\r\n            ],\r\n          },\r\n        },\r\n      ],\r\n    });\r\n    taskRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const appTaskDefinition = new ecs.CfnTaskDefinition(this, 'AppTaskDefinition', {\r\n      family: 'mail-example',\r\n      cpu: containerCpu.toString(),\r\n      memory: containerMemory.toString(),\r\n      networkMode: 'awsvpc',\r\n      requiresCompatibilities: ['FARGATE'],\r\n      executionRoleArn: taskExecutionRole.attrArn,\r\n      taskRoleArn: taskRole.attrArn,\r\n      containerDefinitions: [\r\n        {\r\n          name: 'mail-example',\r\n          image: `${mailExampleRepository.attrRepositoryUri}:latest`,\r\n          portMappings: [\r\n            {\r\n              containerPort,\r\n            },\r\n          ],\r\n          logConfiguration: {\r\n            logDriver: 'awslogs',\r\n            options: {\r\n              'awslogs-group': appLogGroup.ref,\r\n              'awslogs-region': this.region,\r\n              'awslogs-stream-prefix': 'mail-example',\r\n            },\r\n          },\r\n          environment: [\r\n            {\r\n              name: 'CONFIG_SSM_PREFIX',\r\n              value: ssmPrefix,\r\n            },\r\n            {\r\n              name: 'PORT',\r\n              value: containerPort.toString(),\r\n            },\r\n            {\r\n              name: 'BASE_URL',\r\n              value: baseUrl.toString(),\r\n            },\r\n            {\r\n              name: 'NODE_ENV',\r\n              value: publicEnvValue,\r\n            },\r\n          ],\r\n          secrets: [\r\n            {\r\n              name: 'GOOGLE_CLIENT_ID',\r\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/env/GOOGLE_CLIENT_ID`,\r\n            },\r\n            {\r\n              name: 'GOOGLE_CLIENT_SECRET',\r\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`,\r\n            },\r\n            {\r\n              name: 'SESSION_SECRET',\r\n              valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/SESSION_SECRET`,\r\n            },\r\n          ],\r\n        },\r\n      ],\r\n    });\r\n    appTaskDefinition.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    new ecs.CfnService(this, 'AppService', {\r\n      serviceName: 'mail-example',\r\n      cluster: mailCluster.ref,\r\n      taskDefinition: appTaskDefinition.ref,\r\n      desiredCount,\r\n      capacityProviderStrategy: [\r\n        {\r\n          capacityProvider: 'FARGATE_SPOT',\r\n          weight: 4,\r\n        },\r\n        {\r\n          capacityProvider: 'FARGATE',\r\n          weight: 1,\r\n        },\r\n      ],\r\n      deploymentConfiguration: {\r\n        maximumPercent: 200,\r\n        minimumHealthyPercent: 50,\r\n      },\r\n      networkConfiguration: {\r\n        awsvpcConfiguration: {\r\n          assignPublicIp: 'DISABLED',\r\n          subnets: serviceSubnetIds,\r\n          securityGroups: [serviceSecurityGroup.ref],\r\n        },\r\n      },\r\n      serviceRegistries: [\r\n        {\r\n          registryArn: appCloudMapService.attrArn,\r\n          containerName: 'mail-example',\r\n          containerPort: containerPort,\r\n        },\r\n      ],\r\n      platformVersion: 'LATEST',\r\n    });\r\n\r\n    const mailIntegration = new apigatewayv2.CfnIntegration(this, 'MailIntegration', {\r\n      apiId: mailHttpApi.ref,\r\n      integrationType: 'HTTP_PROXY',\r\n      integrationMethod: 'ANY',\r\n      integrationUri: appCloudMapService.attrArn,\r\n      connectionType: 'VPC_LINK',\r\n      connectionId: apiVpcLink.ref,\r\n      payloadFormatVersion: '1.0',\r\n    });\r\n\r\n    new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', {\r\n      apiId: mailHttpApi.ref,\r\n      routeKey: 'ANY /gmail-viewer',\r\n      target: `integrations/${mailIntegration.ref}`,\r\n    });\r\n\r\n    new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', {\r\n      apiId: mailHttpApi.ref,\r\n      routeKey: 'ANY /gmail-viewer/{proxy+}',\r\n      target: `integrations/${mailIntegration.ref}`,\r\n    });\r\n\r\n    const ssmBaseUrl = new ssm.StringParameter(this, 'SsmBaseUrl', {\r\n      parameterName: `${ssmPrefix}/env/BASE_URL`,\r\n      stringValue: baseUrl,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmBaseUrl.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmGoogleClientId = new ssm.StringParameter(this, 'SsmGoogleClientId', {\r\n      parameterName: `${ssmPrefix}/env/GOOGLE_CLIENT_ID`,\r\n      stringValue: googleClientId,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmGoogleClientId.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmGoogleClientSecret = new ssm.StringParameter(this, 'SsmGoogleClientSecret', {\r\n      parameterName: `${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`,\r\n      stringValue: googleClientSecret,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmGoogleClientSecret.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmPort = new ssm.StringParameter(this, 'SsmPort', {\r\n      parameterName: `${ssmPrefix}/env/PORT`,\r\n      stringValue: containerPort.toString(),\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmPort.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmPublicEnv = new ssm.StringParameter(this, 'SsmPublicEnv', {\r\n      parameterName: `${ssmPrefix}/env/NODE_ENV`,\r\n      stringValue: publicEnvValue,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmPublicEnv.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const ssmSessionSecret = new ssm.StringParameter(this, 'SsmSessionSecret', {\r\n      parameterName: `${ssmPrefix}/secrets/SESSION_SECRET`,\r\n      stringValue: sessionSecretValue,\r\n      type: ssm.ParameterType.STRING,\r\n    });\r\n    (ssmSessionSecret.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    const codeBuildRole = new iam.CfnRole(this, 'CodeBuildRole', {\r\n      assumeRolePolicyDocument: {\r\n        Version: '2012-10-17',\r\n        Statement: [\r\n          {\r\n            Effect: 'Allow',\r\n            Principal: {\r\n              Service: 'codebuild.amazonaws.com',\r\n            },\r\n            Action: 'sts:AssumeRole',\r\n          },\r\n        ],\r\n      },\r\n      policies: [\r\n        {\r\n          policyName: 'BuildLogs',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'],\r\n                Resource: '*',\r\n              },\r\n            ],\r\n          },\r\n        },\r\n        {\r\n          policyName: 'ECRPush',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: [\r\n                  'ecr:GetAuthorizationToken',\r\n                  'ecr:BatchCheckLayerAvailability',\r\n                  'ecr:CompleteLayerUpload',\r\n                  'ecr:BatchGetImage',\r\n                  'ecr:DescribeRepositories',\r\n                  'ecr:InitiateLayerUpload',\r\n                  'ecr:PutImage',\r\n                  'ecr:UploadLayerPart',\r\n                ],\r\n                Resource: '*',\r\n              },\r\n            ],\r\n          },\r\n        },\r\n        {\r\n          policyName: 'ECSDeploy',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'],\r\n                Resource: '*',\r\n              },\r\n            ],\r\n          },\r\n        },\r\n        {\r\n          policyName: 'SSMReadForBuild',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'],\r\n                Resource: '*',\r\n              },\r\n            ],\r\n          },\r\n        },\r\n        {\r\n          policyName: 'CodeConnectionAccess',\r\n          policyDocument: {\r\n            Version: '2012-10-17',\r\n            Statement: [\r\n              {\r\n                Effect: 'Allow',\r\n                Action: [\r\n                \"codeconnections:GetConnection\",\r\n                \"codeconnections:GetConnectionToken\",\r\n                \"codeconnections:UseConnection\"\r\n                ],\r\n                Resource: props.codeConnectionArn,\r\n              },\r\n            ],\r\n          },\r\n        }\r\n      ],\r\n    });\r\n    codeBuildRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    // Use the higher-level Project construct so we can provide a typed BuildSpec\r\n    const codebuildRoleRef = iam.Role.fromRoleArn(this, 'CodeBuildRoleRef', codeBuildRole.attrArn);\r\n\r\n    const codeBuildProject = new codebuild.Project(this, 'CodeBuildProject', {\r\n      projectName: 'mail-example',\r\n      role: codebuildRoleRef,\r\n      environment: {\r\n        buildImage: codebuild.LinuxBuildImage.STANDARD_7_0,\r\n        computeType: codebuild.ComputeType.SMALL,\r\n        privileged: true,\r\n        environmentVariables: {\r\n          ECR_URI: { value: mailExampleRepository.attrRepositoryUri },\r\n          CLUSTER_NAME: { value: mailCluster.ref },\r\n          SERVICE_NAME: { value: 'mail-example' },\r\n        },\r\n      },\r\n      source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch, }),\r\n      timeout: cdk.Duration.minutes(30),\r\n      queuedTimeout: cdk.Duration.minutes(30),\r\n      badge: true,\r\n      description: 'Build & deploy mail-example to ECR then force ECS deploy',\r\n      cache: codebuild.Cache.none(),\r\n      buildSpec: codebuild.BuildSpec.fromObject({\r\n        version: '0.2',\r\n        phases: {\r\n          pre_build: { commands: [\r\n            'echo \"Logging in to ECR\"',\r\n            'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ECR_URI',\r\n            'IMAGE_TAG=${CODEBUILD_RESOLVED_SOURCE_VERSION:-latest}',\r\n          ]},\r\n          build: { commands: [\r\n            'cd examples/gmail-viewer',\r\n            'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .',\r\n          ] },\r\n          post_build: { commands: [\r\n            'docker push $ECR_URI:latest',\r\n            'docker push $ECR_URI:$IMAGE_TAG',\r\n            'aws ecs update-service --cluster $CLUSTER_NAME --service $SERVICE_NAME --force-new-deployment',\r\n          ]},\r\n        },\r\n        artifacts: { files: [] },\r\n        env: { shell: 'bash' },\r\n      }),\r\n    });\r\n    (codeBuildProject.node.defaultChild as codebuild.CfnProject).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE;\r\n\r\n    this.clusterName = mailCluster.ref;\r\n    new cdk.CfnOutput(this, 'CfnOutputClusterName', {\r\n      key: 'ClusterName',\r\n      value: this.clusterName.toString(),\r\n    });\r\n    this.repositoryUri = mailExampleRepository.attrRepositoryUri;\r\n    new cdk.CfnOutput(this, 'CfnOutputRepositoryUri', {\r\n      key: 'RepositoryUri',\r\n      value: this.repositoryUri.toString(),\r\n    });\r\n    this.apiInvokeUrl = baseUrl.toString();\r\n    new cdk.CfnOutput(this, 'CfnOutputApiInvokeUrl', {\r\n      key: 'ApiInvokeUrl',\r\n      value: this.apiInvokeUrl,\r\n    });\r\n    this.cloudMapServiceArn = appCloudMapService.attrArn;\r\n    new cdk.CfnOutput(this, 'CfnOutputCloudMapServiceArn', {\r\n      key: 'CloudMapServiceArn',\r\n      value: this.cloudMapServiceArn.toString(),\r\n    });\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 ?? '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 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 7b85065..caea0e5 100644 --- a/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.ts +++ b/examples/gmail-viewer-cdk/lib/gmail-viewer-cdk-stack.ts @@ -1,6 +1,7 @@ import * as cdk from 'aws-cdk-lib'; import * as apigatewayv2 from 'aws-cdk-lib/aws-apigatewayv2'; import * as codebuild from 'aws-cdk-lib/aws-codebuild'; +import * as cr from 'aws-cdk-lib/custom-resources'; import * as ec2 from 'aws-cdk-lib/aws-ec2'; import * as ecr from 'aws-cdk-lib/aws-ecr'; import * as ecs from 'aws-cdk-lib/aws-ecs'; @@ -8,11 +9,13 @@ import * as iam from 'aws-cdk-lib/aws-iam'; import * as logs from 'aws-cdk-lib/aws-logs'; import * as route53 from 'aws-cdk-lib/aws-route53'; import * as servicediscovery from 'aws-cdk-lib/aws-servicediscovery'; +import * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager'; import * as ssm from 'aws-cdk-lib/aws-ssm'; export interface GmailViewerCdkStackProps extends cdk.StackProps { readonly vpcId?: string; readonly privateSubnets?: string[]; + readonly publicSubnets?: string[]; readonly containerPort?: number | string; readonly containerCpu?: number | string; readonly containerMemory?: number | string; @@ -48,655 +51,203 @@ export class GmailViewerCdkStack extends cdk.Stack { const parsed = typeof candidate === 'number' ? candidate : Number(candidate); return Number.isFinite(parsed) ? parsed : fallback; }; - const toCpuUnits = (value: string | number | undefined): number => { const parsed = toNumber(value, 0.5); const vcpu = parsed <= 4 ? parsed * 1024 : parsed; - const rounded = Math.round(vcpu / 256) * 256; - return Math.max(256, rounded); + return Math.max(256, Math.round(vcpu / 256) * 256); }; const repoParts = (process.env.GITHUB_REPOSITORY ?? '').split('/'); - const guessedGitHubOwner = repoParts[0] || 'dyanet'; - const guessedGitHubRepo = repoParts[1] || 'imap'; - const containerPort = toNumber(props.containerPort, 3000); const containerCpu = toCpuUnits(props.containerCpu); const containerMemory = toNumber(props.containerMemory, 1024); const desiredCount = toNumber(props.desiredCount, 1); - const serviceDiscoveryNamespaceName = props.serviceDiscoveryNamespaceName ?? 'mail.local'; - const serviceDiscoveryTtl = toNumber(props.serviceDiscoveryTtl, 60); + const nsName = props.serviceDiscoveryNamespaceName ?? 'mail.local'; + const nsTtl = cdk.Duration.seconds(toNumber(props.serviceDiscoveryTtl, 60)); const ssmPrefixRaw = props.ssmPrefix ?? '/mail-example'; const ssmPrefix = '/' + ssmPrefixRaw.split('/').filter(p => p).join('/'); - const gitHubOwner = props.gitHubOwner ?? guessedGitHubOwner; - const gitHubRepo = props.gitHubRepo ?? guessedGitHubRepo; + const gitHubOwner = props.gitHubOwner ?? (repoParts[0] || 'dyanet'); + const gitHubRepo = props.gitHubRepo ?? (repoParts[1] || 'imap'); const gitHubBranch = props.gitHubBranch ?? process.env.GITHUB_REF_NAME ?? 'main'; const publicEnvValue = props.publicEnvValue ?? 'production'; - const googleClientId = props.googleClientId || 'not-set'; - const googleClientSecret = props.googleClientSecret ?? 'not-set'; - const sessionSecretValue = props.sessionSecretValue ?? 'not-set'; - const apiCustomDomainName = props.apiCustomDomainName ?? process.env.API_CUSTOM_DOMAIN ?? 'mail.dyanet.com'; - const certificateArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN; - const hostedZoneId = props.hostedZoneId ?? ''; - const hasCustomDomain = Boolean(certificateArn && apiCustomDomainName); - const hasHostedZone = Boolean(hostedZoneId); - + 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 certArn = props.certificateArn ?? this.node.tryGetContext('certificateArn') ?? process.env.CERTIFICATE_ARN; + const zoneId = props.hostedZoneId ?? ''; + const hasCustomDomain = Boolean(certArn && apiDomain); + const hasHostedZone = Boolean(zoneId); + const usePublicSubnets = props.publicSubnets && props.publicSubnets.length > 0; + + // VPC setup let vpc: ec2.IVpc; - let privateSubnetIds: string[]; let serviceSubnetIds: string[]; - - // If account/region are available, use Vpc.fromLookup to automatically discover subnets. - // Otherwise fall back to CloudFormation parameters which let the deployer pick values in the console. const useVpcParameter = this.node.tryGetContext('useVpcParameter'); if (!useVpcParameter && this.account && this.region) { vpc = ec2.Vpc.fromLookup(this, 'MailVpc', { tags: { Name: 'dya-vpc' } }); - - if (props.privateSubnets && props.privateSubnets.length > 0) { - privateSubnetIds = props.privateSubnets; + if (usePublicSubnets) { + serviceSubnetIds = props.publicSubnets!; + } else if (props.privateSubnets && props.privateSubnets.length > 0) { + serviceSubnetIds = props.privateSubnets; } else { - // The service linked to the VPC Link may not be available in all Availability Zones. - const badAz = 'cac1-az4'; - - let subnets = vpc.privateSubnets.filter(s => s.availabilityZone !== badAz); - if (subnets.length === 0) { - // No private subnets, try public subnets - subnets = vpc.publicSubnets.filter(s => s.availabilityZone !== badAz); - } - - privateSubnetIds = subnets.slice(0, 2).map(s => s.subnetId); - - if (privateSubnetIds.length < 1) { - throw new Error(`VPC '${vpc.vpcId}' must have at least one private or public subnet in an allowed availability zone (not in ${badAz}).`); - } + const badAz = 'cac1-az4'; + let subnets = vpc.privateSubnets.filter(s => s.availabilityZone !== badAz); + if (subnets.length === 0) subnets = vpc.publicSubnets.filter(s => s.availabilityZone !== badAz); + serviceSubnetIds = subnets.slice(0, 2).map(s => s.subnetId); + if (serviceSubnetIds.length < 1) throw new Error(`VPC must have at least one subnet in an allowed AZ.`); } - serviceSubnetIds = privateSubnetIds; } else { - // CloudFormation parameter for VPC selection (dropdown of VPC IDs in the console) - const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', { - type: 'AWS::EC2::VPC::Id', - default: props.vpcId ?? '', - }); - - // Optional parameter for private subnet IDs (CommaDelimitedList) when synthesizing without account/region - const privateSubnetIdsParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', { - type: 'CommaDelimitedList', - default: props.privateSubnets && props.privateSubnets.length > 0 ? props.privateSubnets.join(',') : '', - }); - // Use the provided parameter values (deployer must supply private subnet ids when synthesizing without lookup) - vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', { - vpcId: vpcIdParam.valueAsString, - availabilityZones: cdk.Fn.getAzs(), - privateSubnetIds: privateSubnetIdsParam.valueAsList, - }); - privateSubnetIds = props.privateSubnets ?? privateSubnetIdsParam.valueAsList; - serviceSubnetIds = privateSubnetIds; + const vpcIdParam = new cdk.CfnParameter(this, 'VpcIdParam', { type: 'AWS::EC2::VPC::Id', default: props.vpcId ?? '' }); + const subnetParam = new cdk.CfnParameter(this, 'PrivateSubnetIdsParam', { type: 'CommaDelimitedList', default: props.privateSubnets?.join(',') ?? '' }); + vpc = ec2.Vpc.fromVpcAttributes(this, 'MailVpc', { vpcId: vpcIdParam.valueAsString, availabilityZones: cdk.Fn.getAzs(), privateSubnetIds: subnetParam.valueAsList }); + serviceSubnetIds = props.privateSubnets ?? subnetParam.valueAsList; } - const apiVpcLinkSecurityGroup = new ec2.CfnSecurityGroup(this, 'ApiVpcLinkSecurityGroup', { - groupDescription: 'Egress from API Gateway VPC Link to service', - vpcId: vpc.vpcId, - securityGroupEgress: [ - { - ipProtocol: 'tcp', - fromPort: containerPort, - toPort: containerPort, - cidrIp: '0.0.0.0/0', - }, - ], - }); - apiVpcLinkSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + // Security Groups (L2) + const apiVpcLinkSg = new ec2.SecurityGroup(this, 'ApiVpcLinkSecurityGroup', { vpc, description: 'Egress from API Gateway VPC Link', allowAllOutbound: false }); + apiVpcLinkSg.addEgressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(containerPort)); - const serviceSecurityGroup = new ec2.CfnSecurityGroup(this, 'ServiceSecurityGroup', { - groupDescription: 'Allow API Gateway VPC Link to reach Fargate tasks', - vpcId: vpc.vpcId, - securityGroupIngress: [ - { - ipProtocol: 'tcp', - fromPort: containerPort, - toPort: containerPort, - cidrIp: '0.0.0.0/0', - }, - ], - securityGroupEgress: [ - { - ipProtocol: '-1', - fromPort: 0, - toPort: 65535, - cidrIp: '0.0.0.0/0', - }, - ], - }); - serviceSecurityGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + const serviceSg = new ec2.SecurityGroup(this, 'ServiceSecurityGroup', { vpc, description: 'Allow traffic to Fargate tasks', allowAllOutbound: true }); + serviceSg.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(containerPort)); - const appLogGroup = new logs.CfnLogGroup(this, 'AppLogGroup', { - logGroupName: '/ecs/mail-example', - retentionInDays: 30, - }); - appLogGroup.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + // Log Group (L2) + const appLogGroup = new logs.LogGroup(this, 'AppLogGroup', { logGroupName: '/ecs/mail-example', retention: logs.RetentionDays.ONE_MONTH, removalPolicy: cdk.RemovalPolicy.DESTROY }); - const mailCluster = new ecs.CfnCluster(this, 'MailCluster', { - clusterName: 'mail-cluster', - capacityProviders: ['FARGATE', 'FARGATE_SPOT'], - defaultCapacityProviderStrategy: [ - { - capacityProvider: 'FARGATE_SPOT', - weight: 4, - }, - { - capacityProvider: 'FARGATE', - weight: 1, - }, - ], - }); - mailCluster.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + // ECS Cluster (L2) + const cluster = new ecs.Cluster(this, 'MailCluster', { vpc, clusterName: 'public-cluster', enableFargateCapacityProviders: true }); - const mailExampleRepository = new ecr.CfnRepository(this, 'MailExampleRepository', { - repositoryName: 'mail-example', - imageTagMutability: 'MUTABLE', - encryptionConfiguration: { - encryptionType: 'AES256', - }, - lifecyclePolicy: { - lifecyclePolicyText: - '{\n "rules": [\n {\n "rulePriority": 1,\n "description": "Expire images older than 30 days",\n "selection": {\n "tagStatus": "any",\n "countType": "sinceImagePushed",\n "countUnit": "days",\n "countNumber": 30\n },\n "action": { "type": "expire" }\n }\n ]\n}\n', - }, - }); - mailExampleRepository.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const mailHttpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', { - name: 'mail-example', - protocolType: 'HTTP', + // ECR Repository - create if not exists using custom resource + const ecrRepoName = 'mail-example'; + const checkEcrRepo = new cr.AwsCustomResource(this, 'CheckEcrRepo', { + onCreate: { service: 'ECR', action: 'describeRepositories', parameters: { repositoryNames: [ecrRepoName] }, physicalResourceId: cr.PhysicalResourceId.of(ecrRepoName) }, + policy: cr.AwsCustomResourcePolicy.fromSdkCalls({ resources: cr.AwsCustomResourcePolicy.ANY_RESOURCE }), }); - mailHttpApi.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - - const mailApiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', { - stageName: 'prod', - apiId: mailHttpApi.ref, - autoDeploy: true, + const createEcrRepo = new cr.AwsCustomResource(this, 'CreateEcrRepo', { + onCreate: { service: 'ECR', action: 'createRepository', parameters: { repositoryName: ecrRepoName, imageTagMutability: 'MUTABLE', encryptionConfiguration: { encryptionType: 'AES256' } }, physicalResourceId: cr.PhysicalResourceId.of(ecrRepoName), ignoreErrorCodesMatching: 'RepositoryAlreadyExistsException' }, + policy: cr.AwsCustomResourcePolicy.fromSdkCalls({ resources: cr.AwsCustomResourcePolicy.ANY_RESOURCE }), }); - mailApiStage.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + createEcrRepo.node.addDependency(checkEcrRepo); + const ecrRepo = ecr.Repository.fromRepositoryName(this, 'MailExampleRepositoryRef', ecrRepoName); - const apiDomainName = hasCustomDomain - ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', { - domainName: apiCustomDomainName, - domainNameConfigurations: [ - { - certificateArn: certificateArn!, - endpointType: 'REGIONAL', - securityPolicy: 'TLS_1_2', - }, - ], - }) - : undefined; - if (apiDomainName) { - apiDomainName.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - } + // API Gateway HTTP API (L1 - no L2 for HTTP API with VPC Link) + const httpApi = new apigatewayv2.CfnApi(this, 'MailHttpApi', { name: 'mail-example', protocolType: 'HTTP' }); + const apiStage = new apigatewayv2.CfnStage(this, 'MailApiStage', { stageName: 'prod', apiId: httpApi.ref, autoDeploy: true }); - const apiMapping = - hasCustomDomain && apiDomainName - ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', { - apiId: mailHttpApi.ref, - domainName: apiDomainName.ref, - stage: mailApiStage.ref, - apiMappingKey: 'examples', - }) - : undefined; - if (apiMapping) { - apiMapping.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - } + const apiDomainName = hasCustomDomain ? new apigatewayv2.CfnDomainName(this, 'ApiDomainName', { + domainName: apiDomain, domainNameConfigurations: [{ certificateArn: certArn!, endpointType: 'REGIONAL', securityPolicy: 'TLS_1_2' }], + }) : undefined; - const apiDomainRecord = - hasCustomDomain && hasHostedZone && apiDomainName - ? new route53.CfnRecordSet(this, 'ApiDomainRecord', { - hostedZoneId, - name: apiCustomDomainName, - type: 'A', - aliasTarget: { - dnsName: apiDomainName.attrRegionalDomainName, - hostedZoneId: apiDomainName.attrRegionalHostedZoneId, - }, - }) - : undefined; - if (apiDomainRecord) { - apiDomainRecord.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - } - if (apiMapping && apiDomainName) { - apiMapping.addDependency(apiDomainName); - apiMapping.addDependency(mailApiStage); - } - if (apiDomainRecord && apiDomainName) { - apiDomainRecord.addDependency(apiDomainName); - } - - const baseUrl = hasCustomDomain - ? `https://${apiCustomDomainName}/examples/gmail-viewer` - : cdk.Fn.join('', [mailHttpApi.attrApiEndpoint, '/', mailApiStage.ref, '/gmail-viewer']); - - const serviceNamespace = new servicediscovery.CfnPrivateDnsNamespace(this, 'ServiceNamespace', { - name: serviceDiscoveryNamespaceName, - vpc: vpc.vpcId, - description: 'Namespace for mail services', - }); - serviceNamespace.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + const apiMapping = hasCustomDomain && apiDomainName ? new apigatewayv2.CfnApiMapping(this, 'ApiMapping', { + apiId: httpApi.ref, domainName: apiDomainName.ref, stage: apiStage.ref, apiMappingKey: 'examples', + }) : undefined; + if (apiMapping && apiDomainName) { apiMapping.addDependency(apiDomainName); apiMapping.addDependency(apiStage); } - const apiVpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', { - name: 'mail-example-vpclink', - subnetIds: serviceSubnetIds, - securityGroupIds: [apiVpcLinkSecurityGroup.ref], - }); - apiVpcLink.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + const apiRecord = hasCustomDomain && hasHostedZone && apiDomainName ? new route53.CfnRecordSet(this, 'ApiDomainRecord', { + hostedZoneId: zoneId, name: apiDomain, type: 'A', aliasTarget: { dnsName: apiDomainName.attrRegionalDomainName, hostedZoneId: apiDomainName.attrRegionalHostedZoneId }, + }) : undefined; + if (apiRecord && apiDomainName) apiRecord.addDependency(apiDomainName); - const appCloudMapService = new servicediscovery.CfnService(this, 'AppCloudMapService', { - name: 'mail-example', - namespaceId: serviceNamespace.ref, - dnsConfig: { - routingPolicy: 'WEIGHTED', - dnsRecords: [ - { - ttl: serviceDiscoveryTtl, - type: 'SRV', - }, - ], - }, - healthCheckCustomConfig: { - failureThreshold: 1, - }, - }); - appCloudMapService.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + const baseUrl = hasCustomDomain ? `https://${apiDomain}/examples/gmail-viewer` : cdk.Fn.join('', [httpApi.attrApiEndpoint, '/', apiStage.ref, '/gmail-viewer']); - const taskExecutionRole = new iam.CfnRole(this, 'TaskExecutionRole', { - assumeRolePolicyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Principal: { - Service: 'ecs-tasks.amazonaws.com', - }, - Action: 'sts:AssumeRole', - }, - ], - }, - managedPolicyArns: ['arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy'], - policies: [ - { - policyName: 'AllowParameterReadForSecrets', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], - Resource: [ - `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, - `arn:aws:kms:${this.region}:${this.account}:key/*`, - ], - }, - ], - }, - }, - ], - }); - taskExecutionRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + // Cloud Map Namespace (L2) + const namespace = new servicediscovery.PrivateDnsNamespace(this, 'ServiceNamespace', { name: nsName, vpc, description: 'Namespace for mail services' }); - const taskRole = new iam.CfnRole(this, 'TaskRole', { - assumeRolePolicyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Principal: { - Service: 'ecs-tasks.amazonaws.com', - }, - Action: 'sts:AssumeRole', - }, - ], - }, - policies: [ - { - policyName: 'AppRuntimeConfigRead', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath', 'kms:Decrypt'], - Resource: [ - `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}*`, - `arn:aws:kms:${this.region}:${this.account}:key/*`, - ], - }, - ], - }, - }, - ], - }); - taskRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; + // VPC Link (L1) + const vpcLink = new apigatewayv2.CfnVpcLink(this, 'ApiVpcLink', { name: 'mail-example-vpclink', subnetIds: serviceSubnetIds, securityGroupIds: [apiVpcLinkSg.securityGroupId] }); - const appTaskDefinition = new ecs.CfnTaskDefinition(this, 'AppTaskDefinition', { - family: 'mail-example', - cpu: containerCpu.toString(), - memory: containerMemory.toString(), - networkMode: 'awsvpc', - requiresCompatibilities: ['FARGATE'], - executionRoleArn: taskExecutionRole.attrArn, - taskRoleArn: taskRole.attrArn, - containerDefinitions: [ - { - name: 'mail-example', - image: `${mailExampleRepository.attrRepositoryUri}:latest`, - portMappings: [ - { - containerPort, - }, - ], - logConfiguration: { - logDriver: 'awslogs', - options: { - 'awslogs-group': appLogGroup.ref, - 'awslogs-region': this.region, - 'awslogs-stream-prefix': 'mail-example', - }, - }, - environment: [ - { - name: 'CONFIG_SSM_PREFIX', - value: ssmPrefix, - }, - { - name: 'PORT', - value: containerPort.toString(), - }, - { - name: 'BASE_URL', - value: baseUrl.toString(), - }, - { - name: 'NODE_ENV', - value: publicEnvValue, - }, - ], - secrets: [ - { - name: 'GOOGLE_CLIENT_ID', - valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/env/GOOGLE_CLIENT_ID`, - }, - { - name: 'GOOGLE_CLIENT_SECRET', - valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`, - }, - { - name: 'SESSION_SECRET', - valueFrom: `arn:aws:ssm:${this.region}:${this.account}:parameter${ssmPrefix}/secrets/SESSION_SECRET`, - }, - ], - }, - ], + // Cloud Map Service (L2) + const cloudMapService = new servicediscovery.Service(this, 'AppCloudMapService', { + name: 'mail-example', namespace, dnsRecordType: servicediscovery.DnsRecordType.SRV, dnsTtl: nsTtl, routingPolicy: servicediscovery.RoutingPolicy.WEIGHTED, customHealthCheck: { failureThreshold: 1 }, }); - appTaskDefinition.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - new ecs.CfnService(this, 'AppService', { - serviceName: 'mail-example', - cluster: mailCluster.ref, - taskDefinition: appTaskDefinition.ref, - desiredCount, - capacityProviderStrategy: [ - { - capacityProvider: 'FARGATE_SPOT', - weight: 4, - }, - { - capacityProvider: 'FARGATE', - weight: 1, - }, - ], - deploymentConfiguration: { - maximumPercent: 200, - minimumHealthyPercent: 50, + // Secrets Manager - single secret with GOOGLE_CLIENT_SECRET and SESSION_SECRET + const appSecrets = new secretsmanager.Secret(this, 'AppSecrets', { + secretName: `${ssmPrefix}/secrets`, + description: 'Application secrets for mail-example', + secretObjectValue: { + GOOGLE_CLIENT_SECRET: cdk.SecretValue.unsafePlainText(googleClientSecret), + SESSION_SECRET: cdk.SecretValue.unsafePlainText(sessionSecretValue), }, - networkConfiguration: { - awsvpcConfiguration: { - assignPublicIp: 'DISABLED', - subnets: serviceSubnetIds, - securityGroups: [serviceSecurityGroup.ref], - }, - }, - serviceRegistries: [ - { - registryArn: appCloudMapService.attrArn, - containerName: 'mail-example', - containerPort: containerPort, - }, - ], - platformVersion: 'LATEST', - }); - - const mailIntegration = new apigatewayv2.CfnIntegration(this, 'MailIntegration', { - apiId: mailHttpApi.ref, - integrationType: 'HTTP_PROXY', - integrationMethod: 'ANY', - integrationUri: appCloudMapService.attrArn, - connectionType: 'VPC_LINK', - connectionId: apiVpcLink.ref, - payloadFormatVersion: '1.0', - }); - - new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', { - apiId: mailHttpApi.ref, - routeKey: 'ANY /gmail-viewer', - target: `integrations/${mailIntegration.ref}`, - }); - - new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', { - apiId: mailHttpApi.ref, - routeKey: 'ANY /gmail-viewer/{proxy+}', - target: `integrations/${mailIntegration.ref}`, - }); - - const ssmBaseUrl = new ssm.StringParameter(this, 'SsmBaseUrl', { - parameterName: `${ssmPrefix}/env/BASE_URL`, - stringValue: baseUrl, - type: ssm.ParameterType.STRING, - }); - (ssmBaseUrl.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - - const ssmGoogleClientId = new ssm.StringParameter(this, 'SsmGoogleClientId', { - parameterName: `${ssmPrefix}/env/GOOGLE_CLIENT_ID`, - stringValue: googleClientId, - type: ssm.ParameterType.STRING, - }); - (ssmGoogleClientId.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - - const ssmGoogleClientSecret = new ssm.StringParameter(this, 'SsmGoogleClientSecret', { - parameterName: `${ssmPrefix}/secrets/GOOGLE_CLIENT_SECRET`, - stringValue: googleClientSecret, - type: ssm.ParameterType.STRING, - }); - (ssmGoogleClientSecret.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - - const ssmPort = new ssm.StringParameter(this, 'SsmPort', { - parameterName: `${ssmPrefix}/env/PORT`, - stringValue: containerPort.toString(), - type: ssm.ParameterType.STRING, - }); - (ssmPort.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - - const ssmPublicEnv = new ssm.StringParameter(this, 'SsmPublicEnv', { - parameterName: `${ssmPrefix}/env/NODE_ENV`, - stringValue: publicEnvValue, - type: ssm.ParameterType.STRING, }); - (ssmPublicEnv.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - const ssmSessionSecret = new ssm.StringParameter(this, 'SsmSessionSecret', { - parameterName: `${ssmPrefix}/secrets/SESSION_SECRET`, - stringValue: sessionSecretValue, - type: ssm.ParameterType.STRING, - }); - (ssmSessionSecret.node.defaultChild as ssm.CfnParameter).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - - const codeBuildRole = new iam.CfnRole(this, 'CodeBuildRole', { - assumeRolePolicyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Principal: { - Service: 'codebuild.amazonaws.com', - }, - Action: 'sts:AssumeRole', - }, - ], + // IAM Roles (L2) + const taskExecRole = new iam.Role(this, 'TaskExecutionRole', { assumedBy: new iam.ServicePrincipal('ecs-tasks.amazonaws.com'), managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AmazonECSTaskExecutionRolePolicy')] }); + 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/*`] })); + taskExecRole.addToPolicy(new iam.PolicyStatement({ actions: ['secretsmanager:GetSecretValue'], resources: [appSecrets.secretArn] })); + + 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', { + image: ecs.ContainerImage.fromEcrRepository(ecrRepo, 'latest'), portMappings: [{ containerPort }], + 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_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'GOOGLE_CLIENT_SECRET'), + SESSION_SECRET: ecs.Secret.fromSecretsManager(appSecrets, 'SESSION_SECRET'), }, - policies: [ - { - policyName: 'BuildLogs', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'], - Resource: '*', - }, - ], - }, - }, - { - policyName: 'ECRPush', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: [ - 'ecr:GetAuthorizationToken', - 'ecr:BatchCheckLayerAvailability', - 'ecr:CompleteLayerUpload', - 'ecr:BatchGetImage', - 'ecr:DescribeRepositories', - 'ecr:InitiateLayerUpload', - 'ecr:PutImage', - 'ecr:UploadLayerPart', - ], - Resource: '*', - }, - ], - }, - }, - { - policyName: 'ECSDeploy', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'], - Resource: '*', - }, - ], - }, - }, - { - policyName: 'SSMReadForBuild', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'], - Resource: '*', - }, - ], - }, - }, - { - policyName: 'CodeConnectionAccess', - policyDocument: { - Version: '2012-10-17', - Statement: [ - { - Effect: 'Allow', - Action: [ - "codeconnections:GetConnection", - "codeconnections:GetConnectionToken", - "codeconnections:UseConnection" - ], - Resource: props.codeConnectionArn, - }, - ], - }, - } - ], }); - codeBuildRole.cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - - // Use the higher-level Project construct so we can provide a typed BuildSpec - const codebuildRoleRef = iam.Role.fromRoleArn(this, 'CodeBuildRoleRef', codeBuildRole.attrArn); - const codeBuildProject = new codebuild.Project(this, 'CodeBuildProject', { - projectName: 'mail-example', - role: codebuildRoleRef, - environment: { - buildImage: codebuild.LinuxBuildImage.STANDARD_7_0, - computeType: codebuild.ComputeType.SMALL, - privileged: true, - environmentVariables: { - ECR_URI: { value: mailExampleRepository.attrRepositoryUri }, - CLUSTER_NAME: { value: mailCluster.ref }, - SERVICE_NAME: { value: 'mail-example' }, - }, - }, - source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch, }), - timeout: cdk.Duration.minutes(30), - queuedTimeout: cdk.Duration.minutes(30), - badge: true, - description: 'Build & deploy mail-example to ECR then force ECS deploy', - cache: codebuild.Cache.none(), + // ECS Service (L2) - associate with standalone Cloud Map service for API Gateway integration + const service = new ecs.FargateService(this, 'AppService', { + serviceName: 'mail-example', cluster, taskDefinition: taskDef, desiredCount, + capacityProviderStrategies: [{ capacityProvider: 'FARGATE_SPOT', weight: 4 }, { capacityProvider: 'FARGATE', weight: 1 }], + vpcSubnets: { subnets: serviceSubnetIds.map((id, i) => ec2.Subnet.fromSubnetId(this, `ServiceSubnet${i}`, id)) }, + securityGroups: [serviceSg], assignPublicIp: usePublicSubnets ?? false, minHealthyPercent: 50, maxHealthyPercent: 200, + }); + service.associateCloudMapService({ service: cloudMapService, containerPort, container: taskDef.defaultContainer! }); + + // API Gateway Integration (L1) + 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' }); + new apigatewayv2.CfnRoute(this, 'GmailViewerRoute', { apiId: httpApi.ref, routeKey: 'ANY /gmail-viewer', target: `integrations/${integration.ref}` }); + new apigatewayv2.CfnRoute(this, 'GmailViewerProxyRoute', { apiId: httpApi.ref, routeKey: 'ANY /gmail-viewer/{proxy+}', target: `integrations/${integration.ref}` }); + + // CodeBuild Role (L2) + const buildRole = new iam.Role(this, 'CodeBuildRole', { assumedBy: new iam.ServicePrincipal('codebuild.amazonaws.com') }); + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'], resources: ['*'] })); + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ecr:GetAuthorizationToken', 'ecr:BatchCheckLayerAvailability', 'ecr:CompleteLayerUpload', 'ecr:BatchGetImage', 'ecr:DescribeRepositories', 'ecr:InitiateLayerUpload', 'ecr:PutImage', 'ecr:UploadLayerPart'], resources: ['*'] })); + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ecs:UpdateService', 'ecs:DescribeServices', 'ecs:DescribeClusters'], resources: ['*'] })); + buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['ssm:GetParameter', 'ssm:GetParameters', 'ssm:GetParametersByPath'], resources: ['*'] })); + if (props.codeConnectionArn) buildRole.addToPolicy(new iam.PolicyStatement({ actions: ['codeconnections:GetConnection', 'codeconnections:GetConnectionToken', 'codeconnections:UseConnection'], resources: [props.codeConnectionArn] })); + + // CodeBuild Project (L2) + new codebuild.Project(this, 'CodeBuildProject', { + projectName: 'mail-example', role: buildRole, + 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' } } }, + source: codebuild.Source.gitHub({ owner: gitHubOwner, repo: gitHubRepo, branchOrRef: gitHubBranch }), + timeout: cdk.Duration.minutes(30), queuedTimeout: cdk.Duration.minutes(30), badge: true, description: 'Build & deploy mail-example to ECR then force ECS deploy', buildSpec: codebuild.BuildSpec.fromObject({ version: '0.2', phases: { - 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}', - ]}, - build: { commands: [ - 'cd examples/gmail-viewer', - 'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .', - ] }, - 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', - ]}, + 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}'] }, + build: { commands: ['cd examples/gmail-viewer', 'docker build -t $ECR_URI:latest -t $ECR_URI:$IMAGE_TAG .'] }, + 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'] }, }, - artifacts: { files: [] }, - env: { shell: 'bash' }, + artifacts: { files: [] }, env: { shell: 'bash' }, }), }); - (codeBuildProject.node.defaultChild as codebuild.CfnProject).cfnOptions.deletionPolicy = cdk.CfnDeletionPolicy.DELETE; - this.clusterName = mailCluster.ref; - new cdk.CfnOutput(this, 'CfnOutputClusterName', { - key: 'ClusterName', - value: this.clusterName.toString(), - }); - this.repositoryUri = mailExampleRepository.attrRepositoryUri; - new cdk.CfnOutput(this, 'CfnOutputRepositoryUri', { - key: 'RepositoryUri', - value: this.repositoryUri.toString(), - }); + // Outputs + this.clusterName = cluster.clusterName; + new cdk.CfnOutput(this, 'ClusterName', { value: this.clusterName }); + this.repositoryUri = ecrRepo.repositoryUri; + new cdk.CfnOutput(this, 'RepositoryUri', { value: this.repositoryUri }); this.apiInvokeUrl = baseUrl.toString(); - new cdk.CfnOutput(this, 'CfnOutputApiInvokeUrl', { - key: 'ApiInvokeUrl', - value: this.apiInvokeUrl, - }); - this.cloudMapServiceArn = appCloudMapService.attrArn; - new cdk.CfnOutput(this, 'CfnOutputCloudMapServiceArn', { - key: 'CloudMapServiceArn', - value: this.cloudMapServiceArn.toString(), - }); + new cdk.CfnOutput(this, 'ApiInvokeUrl', { value: this.apiInvokeUrl }); + this.cloudMapServiceArn = cloudMapService.serviceArn; + new cdk.CfnOutput(this, 'CloudMapServiceArn', { value: this.cloudMapServiceArn }); } } diff --git a/examples/gmail-viewer/package-lock.json b/examples/gmail-viewer/package-lock.json index bfb97ff..82025b0 100644 --- a/examples/gmail-viewer/package-lock.json +++ b/examples/gmail-viewer/package-lock.json @@ -9,6 +9,9 @@ "version": "0.5.2", "license": "MIT", "dependencies": { + "@aws-sdk/client-s3": "^3.0.0", + "@aws-sdk/client-secrets-manager": "^3.0.0", + "@aws-sdk/client-ssm": "^3.0.0", "@dyanet/config-aws": "^1.0.3", "@dyanet/imap": "^0.5.2", "express": "^4.21.0", @@ -16,50 +19,1781 @@ "helmet": "^8.0.0", "zod": "^3.22.0" }, - "devDependencies": { - "@types/express": "^5.0.0", - "@types/express-session": "^1.18.0", - "@types/node": "^20.10.0", - "typescript": "^5.3.0" + "devDependencies": { + "@types/express": "^5.0.0", + "@types/express-session": "^1.18.0", + "@types/node": "^20.10.0", + "typescript": "^5.3.0" + }, + "engines": { + "node": ">=20.0.0" + } + }, + "node_modules/@aws-crypto/crc32": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@aws-crypto/crc32/-/crc32-5.2.0.tgz", + "integrity": "sha512-nLbCWqQNgUiwwtFsen1AdzAtvuLRsQS8rYgMuxCrdKf9kOssamGLuPwyTY9wyYblNr9+1XM8v6zoDTPPSIeANg==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/util": "^5.2.0", + "@aws-sdk/types": "^3.222.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@aws-crypto/crc32c": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@aws-crypto/crc32c/-/crc32c-5.2.0.tgz", + "integrity": "sha512-+iWb8qaHLYKrNvGRbiYRHSdKRWhto5XlZUEBwDjYNf+ly5SVYG6zEoYIdxvf5R3zyeP16w4PLBn3rH1xc74Rag==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/util": "^5.2.0", + "@aws-sdk/types": "^3.222.0", + "tslib": "^2.6.2" + } + }, + "node_modules/@aws-crypto/sha1-browser": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@aws-crypto/sha1-browser/-/sha1-browser-5.2.0.tgz", + "integrity": "sha512-OH6lveCFfcDjX4dbAvCFSYUjJZjDr/3XJ3xHtjn3Oj5b9RjojQo8npoLeA/bNwkOkrSQ0wgrHzXk4tDRxGKJeg==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/supports-web-crypto": "^5.2.0", + "@aws-crypto/util": "^5.2.0", + "@aws-sdk/types": "^3.222.0", + "@aws-sdk/util-locate-window": "^3.0.0", + "@smithy/util-utf8": "^2.0.0", + "tslib": "^2.6.2" + } + }, + "node_modules/@aws-crypto/sha1-browser/node_modules/@smithy/is-array-buffer": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@smithy/is-array-buffer/-/is-array-buffer-2.2.0.tgz", + "integrity": "sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@aws-crypto/sha1-browser/node_modules/@smithy/util-buffer-from": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-buffer-from/-/util-buffer-from-2.2.0.tgz", + "integrity": "sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/is-array-buffer": "^2.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@aws-crypto/sha1-browser/node_modules/@smithy/util-utf8": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@smithy/util-utf8/-/util-utf8-2.3.0.tgz", + "integrity": "sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/util-buffer-from": "^2.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@aws-crypto/sha256-browser": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@aws-crypto/sha256-browser/-/sha256-browser-5.2.0.tgz", + "integrity": "sha512-AXfN/lGotSQwu6HNcEsIASo7kWXZ5HYWvfOmSNKDsEqC4OashTp8alTmaz+F7TC2L083SFv5RdB+qU3Vs1kZqw==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/sha256-js": "^5.2.0", + "@aws-crypto/supports-web-crypto": "^5.2.0", + "@aws-crypto/util": "^5.2.0", + "@aws-sdk/types": "^3.222.0", + "@aws-sdk/util-locate-window": "^3.0.0", + "@smithy/util-utf8": "^2.0.0", + "tslib": "^2.6.2" + } + }, + "node_modules/@aws-crypto/sha256-browser/node_modules/@smithy/is-array-buffer": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@smithy/is-array-buffer/-/is-array-buffer-2.2.0.tgz", + "integrity": "sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@aws-crypto/sha256-browser/node_modules/@smithy/util-buffer-from": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-buffer-from/-/util-buffer-from-2.2.0.tgz", + "integrity": "sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/is-array-buffer": "^2.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@aws-crypto/sha256-browser/node_modules/@smithy/util-utf8": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@smithy/util-utf8/-/util-utf8-2.3.0.tgz", + "integrity": "sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/util-buffer-from": "^2.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@aws-crypto/sha256-js": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@aws-crypto/sha256-js/-/sha256-js-5.2.0.tgz", + "integrity": "sha512-FFQQyu7edu4ufvIZ+OadFpHHOt+eSTBaYaki44c+akjg7qZg9oOQeLlk77F6tSYqjDAFClrHJk9tMf0HdVyOvA==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/util": "^5.2.0", + "@aws-sdk/types": "^3.222.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@aws-crypto/supports-web-crypto": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@aws-crypto/supports-web-crypto/-/supports-web-crypto-5.2.0.tgz", + "integrity": "sha512-iAvUotm021kM33eCdNfwIN//F77/IADDSs58i+MDaOqFrVjZo9bAal0NK7HurRuWLLpF1iLX7gbWrjHjeo+YFg==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + } + }, + "node_modules/@aws-crypto/util": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@aws-crypto/util/-/util-5.2.0.tgz", + "integrity": "sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "^3.222.0", + "@smithy/util-utf8": "^2.0.0", + "tslib": "^2.6.2" + } + }, + "node_modules/@aws-crypto/util/node_modules/@smithy/is-array-buffer": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@smithy/is-array-buffer/-/is-array-buffer-2.2.0.tgz", + "integrity": "sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@aws-crypto/util/node_modules/@smithy/util-buffer-from": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-buffer-from/-/util-buffer-from-2.2.0.tgz", + "integrity": "sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/is-array-buffer": "^2.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@aws-crypto/util/node_modules/@smithy/util-utf8": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@smithy/util-utf8/-/util-utf8-2.3.0.tgz", + "integrity": "sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/util-buffer-from": "^2.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=14.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", + "integrity": "sha512-I2/1McBZCcM3PfM4ck8D6gnZR3K7+yl1fGkwTq/3ThEn9tdLjNwcdgTbPfxfX6LoecLrH9Ekoo+D9nmQ0T261w==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/sha1-browser": "5.2.0", + "@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-bucket-endpoint": "3.957.0", + "@aws-sdk/middleware-expect-continue": "3.957.0", + "@aws-sdk/middleware-flexible-checksums": "3.957.0", + "@aws-sdk/middleware-host-header": "3.957.0", + "@aws-sdk/middleware-location-constraint": "3.957.0", + "@aws-sdk/middleware-logger": "3.957.0", + "@aws-sdk/middleware-recursion-detection": "3.957.0", + "@aws-sdk/middleware-sdk-s3": "3.957.0", + "@aws-sdk/middleware-ssec": "3.957.0", + "@aws-sdk/middleware-user-agent": "3.957.0", + "@aws-sdk/region-config-resolver": "3.957.0", + "@aws-sdk/signature-v4-multi-region": "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/eventstream-serde-browser": "^4.2.7", + "@smithy/eventstream-serde-config-resolver": "^4.3.7", + "@smithy/eventstream-serde-node": "^4.2.7", + "@smithy/fetch-http-handler": "^5.3.8", + "@smithy/hash-blob-browser": "^4.2.8", + "@smithy/hash-node": "^4.2.7", + "@smithy/hash-stream-node": "^4.2.7", + "@smithy/invalid-dependency": "^4.2.7", + "@smithy/md5-js": "^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-stream": "^4.5.8", + "@smithy/util-utf8": "^4.2.0", + "@smithy/util-waiter": "^4.2.7", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/client-secrets-manager": { + "version": "3.962.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/client-secrets-manager/-/client-secrets-manager-3.962.0.tgz", + "integrity": "sha512-UwXvVou6ACuYYEcdRjRVVq/M6w7X1kly80kavd0GdaPG0ZUhwGOfd3K5nqqwBM+meEcnQiAF2rTlHZ1yw/+e3w==", + "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-ssm": { + "version": "3.962.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/client-ssm/-/client-ssm-3.962.0.tgz", + "integrity": "sha512-UONnQGDp+02fh4M4ym44Zzato39O36JoJ+vvLKMsAfPJ70dhSbtO0xnMBhUtmNZ3d68JtCa6gBzXileSIKCNhw==", + "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", + "@smithy/util-waiter": "^4.2.7", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/client-sso": { + "version": "3.958.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/client-sso/-/client-sso-3.958.0.tgz", + "integrity": "sha512-6qNCIeaMzKzfqasy2nNRuYnMuaMebCcCPP4J2CVGkA8QYMbIVKPlkn9bpB20Vxe6H/r3jtCCLQaOJjVTx/6dXg==", + "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/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/core": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/core/-/core-3.957.0.tgz", + "integrity": "sha512-DrZgDnF1lQZv75a52nFWs6MExihJF2GZB6ETZRqr6jMwhrk2kbJPUtvgbifwcL7AYmVqHQDJBrR/MqkwwFCpiw==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@aws-sdk/xml-builder": "3.957.0", + "@smithy/core": "^3.20.0", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/property-provider": "^4.2.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/signature-v4": "^5.3.7", + "@smithy/smithy-client": "^4.10.2", + "@smithy/types": "^4.11.0", + "@smithy/util-base64": "^4.3.0", + "@smithy/util-middleware": "^4.2.7", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/crc64-nvme": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/crc64-nvme/-/crc64-nvme-3.957.0.tgz", + "integrity": "sha512-qSwSfI+qBU9HDsd6/4fM9faCxYJx2yDuHtj+NVOQ6XYDWQzFab/hUdwuKZ77Pi6goLF1pBZhJ2azaC2w7LbnTA==", + "license": "Apache-2.0", + "dependencies": { + "@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", + "integrity": "sha512-475mkhGaWCr+Z52fOOVb/q2VHuNvqEDixlYIkeaO6xJ6t9qR0wpLt4hOQaR6zR1wfZV0SlE7d8RErdYq/PByog==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.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-http": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-http/-/credential-provider-http-3.957.0.tgz", + "integrity": "sha512-8dS55QHRxXgJlHkEYaCGZIhieCs9NU1HU1BcqQ4RfUdSsfRdxxktqUKgCnBnOOn0oD3PPA8cQOCAVgIyRb3Rfw==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.0", + "@aws-sdk/types": "3.957.0", + "@smithy/fetch-http-handler": "^5.3.8", + "@smithy/node-http-handler": "^4.4.7", + "@smithy/property-provider": "^4.2.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/smithy-client": "^4.10.2", + "@smithy/types": "^4.11.0", + "@smithy/util-stream": "^4.5.8", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/credential-provider-ini": { + "version": "3.962.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-ini/-/credential-provider-ini-3.962.0.tgz", + "integrity": "sha512-h0kVnXLW2d3nxbcrR/Pfg3W/+YoCguasWz7/3nYzVqmdKarGrpJzaFdoZtLgvDSZ8VgWUC4lWOTcsDMV0UNqUQ==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.0", + "@aws-sdk/credential-provider-env": "3.957.0", + "@aws-sdk/credential-provider-http": "3.957.0", + "@aws-sdk/credential-provider-login": "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/credential-provider-imds": "^4.2.7", + "@smithy/property-provider": "^4.2.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/credential-provider-login": { + "version": "3.962.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-login/-/credential-provider-login-3.962.0.tgz", + "integrity": "sha512-kHYH6Av2UifG3mPkpPUNRh/PuX6adaAcpmsclJdHdxlixMCRdh8GNeEihq480DC0GmfqdpoSf1w2CLmLLPIS6w==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.0", + "@aws-sdk/nested-clients": "3.958.0", + "@aws-sdk/types": "3.957.0", + "@smithy/property-provider": "^4.2.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/credential-provider-node": { + "version": "3.962.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-node/-/credential-provider-node-3.962.0.tgz", + "integrity": "sha512-CS78NsWRxLa+nWqeWBEYMZTLacMFIXs1C5WJuM9kD05LLiWL32ksljoPsvNN24Bc7rCSQIIMx/U3KGvkDVZMVg==", + "license": "Apache-2.0", + "dependencies": { + "@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-process": "3.957.0", + "@aws-sdk/credential-provider-sso": "3.958.0", + "@aws-sdk/credential-provider-web-identity": "3.958.0", + "@aws-sdk/types": "3.957.0", + "@smithy/credential-provider-imds": "^4.2.7", + "@smithy/property-provider": "^4.2.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/credential-provider-process": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-process/-/credential-provider-process-3.957.0.tgz", + "integrity": "sha512-/KIz9kadwbeLy6SKvT79W81Y+hb/8LMDyeloA2zhouE28hmne+hLn0wNCQXAAupFFlYOAtZR2NTBs7HBAReJlg==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.0", + "@aws-sdk/types": "3.957.0", + "@smithy/property-provider": "^4.2.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/credential-provider-sso": { + "version": "3.958.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-sso/-/credential-provider-sso-3.958.0.tgz", + "integrity": "sha512-CBYHJ5ufp8HC4q+o7IJejCUctJXWaksgpmoFpXerbjAso7/Fg7LLUu9inXVOxlHKLlvYekDXjIUBXDJS2WYdgg==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/client-sso": "3.958.0", + "@aws-sdk/core": "3.957.0", + "@aws-sdk/token-providers": "3.958.0", + "@aws-sdk/types": "3.957.0", + "@smithy/property-provider": "^4.2.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/credential-provider-web-identity": { + "version": "3.958.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/credential-provider-web-identity/-/credential-provider-web-identity-3.958.0.tgz", + "integrity": "sha512-dgnvwjMq5Y66WozzUzxNkCFap+umHUtqMMKlr8z/vl9NYMLem/WUbWNpFFOVFWquXikc+ewtpBMR4KEDXfZ+KA==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.0", + "@aws-sdk/nested-clients": "3.958.0", + "@aws-sdk/types": "3.957.0", + "@smithy/property-provider": "^4.2.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@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", + "integrity": "sha512-iczcn/QRIBSpvsdAS/rbzmoBpleX1JBjXvCynMbDceVLBIcVrwT1hXECrhtIC2cjh4HaLo9ClAbiOiWuqt+6MA==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@aws-sdk/util-arn-parser": "3.957.0", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "@smithy/util-config-provider": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-expect-continue": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-expect-continue/-/middleware-expect-continue-3.957.0.tgz", + "integrity": "sha512-AlbK3OeVNwZZil0wlClgeI/ISlOt/SPUxBsIns876IFaVu/Pj3DgImnYhpcJuFRek4r4XM51xzIaGQXM6GDHGg==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-flexible-checksums": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-flexible-checksums/-/middleware-flexible-checksums-3.957.0.tgz", + "integrity": "sha512-iJpeVR5V8se1hl2pt+k8bF/e9JO4KWgPCMjg8BtRspNtKIUGy7j6msYvbDixaKZaF2Veg9+HoYcOhwnZumjXSA==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/crc32": "5.2.0", + "@aws-crypto/crc32c": "5.2.0", + "@aws-crypto/util": "5.2.0", + "@aws-sdk/core": "3.957.0", + "@aws-sdk/crc64-nvme": "3.957.0", + "@aws-sdk/types": "3.957.0", + "@smithy/is-array-buffer": "^4.2.0", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "@smithy/util-middleware": "^4.2.7", + "@smithy/util-stream": "^4.5.8", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-host-header": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-host-header/-/middleware-host-header-3.957.0.tgz", + "integrity": "sha512-BBgKawVyfQZglEkNTuBBdC3azlyqNXsvvN4jPkWAiNYcY0x1BasaJFl+7u/HisfULstryweJq/dAvIZIxzlZaA==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-location-constraint": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-location-constraint/-/middleware-location-constraint-3.957.0.tgz", + "integrity": "sha512-y8/W7TOQpmDJg/fPYlqAhwA4+I15LrS7TwgUEoxogtkD8gfur9wFMRLT8LCyc9o4NMEcAnK50hSb4+wB0qv6tQ==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-logger": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-logger/-/middleware-logger-3.957.0.tgz", + "integrity": "sha512-w1qfKrSKHf9b5a8O76yQ1t69u6NWuBjr5kBX+jRWFx/5mu6RLpqERXRpVJxfosbep7k3B+DSB5tZMZ82GKcJtQ==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-recursion-detection": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-recursion-detection/-/middleware-recursion-detection-3.957.0.tgz", + "integrity": "sha512-D2H/WoxhAZNYX+IjkKTdOhOkWQaK0jjJrDBj56hKjU5c9ltQiaX/1PqJ4dfjHntEshJfu0w+E6XJ+/6A6ILBBA==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@aws/lambda-invoke-store": "^0.2.2", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-sdk-s3": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-sdk-s3/-/middleware-sdk-s3-3.957.0.tgz", + "integrity": "sha512-5B2qY2nR2LYpxoQP0xUum5A1UNvH2JQpLHDH1nWFNF/XetV7ipFHksMxPNhtJJ6ARaWhQIDXfOUj0jcnkJxXUg==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.0", + "@aws-sdk/types": "3.957.0", + "@aws-sdk/util-arn-parser": "3.957.0", + "@smithy/core": "^3.20.0", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/signature-v4": "^5.3.7", + "@smithy/smithy-client": "^4.10.2", + "@smithy/types": "^4.11.0", + "@smithy/util-config-provider": "^4.2.0", + "@smithy/util-middleware": "^4.2.7", + "@smithy/util-stream": "^4.5.8", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-ssec": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-ssec/-/middleware-ssec-3.957.0.tgz", + "integrity": "sha512-qwkmrK0lizdjNt5qxl4tHYfASh8DFpHXM1iDVo+qHe+zuslfMqQEGRkzxS8tJq/I+8F0c6v3IKOveKJAfIvfqQ==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/middleware-user-agent": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/middleware-user-agent/-/middleware-user-agent-3.957.0.tgz", + "integrity": "sha512-50vcHu96XakQnIvlKJ1UoltrFODjsq2KvtTgHiPFteUS884lQnK5VC/8xd1Msz/1ONpLMzdCVproCQqhDTtMPQ==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.0", + "@aws-sdk/types": "3.957.0", + "@aws-sdk/util-endpoints": "3.957.0", + "@smithy/core": "^3.20.0", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/nested-clients": { + "version": "3.958.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/nested-clients/-/nested-clients-3.958.0.tgz", + "integrity": "sha512-/KuCcS8b5TpQXkYOrPLYytrgxBhv81+5pChkOlhegbeHttjM69pyUpQVJqyfDM/A7wPLnDrzCAnk4zaAOkY0Nw==", + "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/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/region-config-resolver": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/region-config-resolver/-/region-config-resolver-3.957.0.tgz", + "integrity": "sha512-V8iY3blh8l2iaOqXWW88HbkY5jDoWjH56jonprG/cpyqqCnprvpMUZWPWYJoI8rHRf2bqzZeql1slxG6EnKI7A==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@smithy/config-resolver": "^4.4.5", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/signature-v4-multi-region": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/signature-v4-multi-region/-/signature-v4-multi-region-3.957.0.tgz", + "integrity": "sha512-t6UfP1xMUigMMzHcb7vaZcjv7dA2DQkk9C/OAP1dKyrE0vb4lFGDaTApi17GN6Km9zFxJthEMUbBc7DL0hq1Bg==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/middleware-sdk-s3": "3.957.0", + "@aws-sdk/types": "3.957.0", + "@smithy/protocol-http": "^5.3.7", + "@smithy/signature-v4": "^5.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/token-providers": { + "version": "3.958.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/token-providers/-/token-providers-3.958.0.tgz", + "integrity": "sha512-UCj7lQXODduD1myNJQkV+LYcGYJ9iiMggR8ow8Hva1g3A/Na5imNXzz6O67k7DAee0TYpy+gkNw+SizC6min8Q==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/core": "3.957.0", + "@aws-sdk/nested-clients": "3.958.0", + "@aws-sdk/types": "3.957.0", + "@smithy/property-provider": "^4.2.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/types": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/types/-/types-3.957.0.tgz", + "integrity": "sha512-wzWC2Nrt859ABk6UCAVY/WYEbAd7FjkdrQL6m24+tfmWYDNRByTJ9uOgU/kw9zqLCAwb//CPvrJdhqjTznWXAg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/util-arn-parser": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/util-arn-parser/-/util-arn-parser-3.957.0.tgz", + "integrity": "sha512-Aj6m+AyrhWyg8YQ4LDPg2/gIfGHCEcoQdBt5DeSFogN5k9mmJPOJ+IAmNSWmWRjpOxEy6eY813RNDI6qS97M0g==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/util-endpoints": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/util-endpoints/-/util-endpoints-3.957.0.tgz", + "integrity": "sha512-xwF9K24mZSxcxKS3UKQFeX/dPYkEps9wF1b+MGON7EvnbcucrJGyQyK1v1xFPn1aqXkBTFi+SZaMRx5E5YCVFw==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@smithy/types": "^4.11.0", + "@smithy/url-parser": "^4.2.7", + "@smithy/util-endpoints": "^3.2.7", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/util-locate-window": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/util-locate-window/-/util-locate-window-3.957.0.tgz", + "integrity": "sha512-nhmgKHnNV9K+i9daumaIz8JTLsIIML9PE/HUks5liyrjUzenjW/aHoc7WJ9/Td/gPZtayxFnXQSJRb/fDlBuJw==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws-sdk/util-user-agent-browser": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/util-user-agent-browser/-/util-user-agent-browser-3.957.0.tgz", + "integrity": "sha512-exueuwxef0lUJRnGaVkNSC674eAiWU07ORhxBnevFFZEKisln+09Qrtw823iyv5I1N8T+wKfh95xvtWQrNKNQw==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/types": "3.957.0", + "@smithy/types": "^4.11.0", + "bowser": "^2.11.0", + "tslib": "^2.6.2" + } + }, + "node_modules/@aws-sdk/util-user-agent-node": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/util-user-agent-node/-/util-user-agent-node-3.957.0.tgz", + "integrity": "sha512-ycbYCwqXk4gJGp0Oxkzf2KBeeGBdTxz559D41NJP8FlzSej1Gh7Rk40Zo6AyTfsNWkrl/kVi1t937OIzC5t+9Q==", + "license": "Apache-2.0", + "dependencies": { + "@aws-sdk/middleware-user-agent": "3.957.0", + "@aws-sdk/types": "3.957.0", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "aws-crt": ">=1.0.0" + }, + "peerDependenciesMeta": { + "aws-crt": { + "optional": true + } + } + }, + "node_modules/@aws-sdk/xml-builder": { + "version": "3.957.0", + "resolved": "https://registry.npmjs.org/@aws-sdk/xml-builder/-/xml-builder-3.957.0.tgz", + "integrity": "sha512-Ai5iiQqS8kJ5PjzMhWcLKN0G2yasAkvpnPlq2EnqlIMdB48HsizElt62qcktdxp4neRMyGkFq4NzgmDbXnhRiA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "fast-xml-parser": "5.2.5", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@aws/lambda-invoke-store": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/@aws/lambda-invoke-store/-/lambda-invoke-store-0.2.2.tgz", + "integrity": "sha512-C0NBLsIqzDIae8HFw9YIrIBsbc0xTiOtt7fAukGPnqQ/+zZNaq+4jhuccltK0QuWHBnNm/a6kLIRA6GFiM10eg==", + "license": "Apache-2.0", + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@dyanet/config-aws": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@dyanet/config-aws/-/config-aws-1.0.3.tgz", + "integrity": "sha512-Y7Wr9MWU+5rRrwg8Wr/5bqufdZWZvo8l3OZd18oCetRT9935+aPVdNP0IuGwLocitQoz+pP91x4G/3bIe/PIOg==", + "license": "MIT", + "engines": { + "node": ">=20.0.0", + "npm": ">=10.0.0" + }, + "peerDependencies": { + "@aws-sdk/client-s3": "^3.0.0", + "@aws-sdk/client-secrets-manager": "^3.0.0", + "@aws-sdk/client-ssm": "^3.0.0", + "zod": "^3.22.0" + }, + "peerDependenciesMeta": { + "@aws-sdk/client-s3": { + "optional": true + }, + "@aws-sdk/client-secrets-manager": { + "optional": true + }, + "@aws-sdk/client-ssm": { + "optional": true + } + } + }, + "node_modules/@dyanet/imap": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/@dyanet/imap/-/imap-0.5.2.tgz", + "integrity": "sha512-VwxFsqCY4DKct4A+bHzLpBex7UrwWE4p0DsLrA6Bjuwftxmtm7HkkENuRwEZOlw020XDIAfbWMKXWd/n8M/N0A==", + "license": "MIT", + "engines": { + "node": ">=20.0.0" + } + }, + "node_modules/@smithy/abort-controller": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/abort-controller/-/abort-controller-4.2.7.tgz", + "integrity": "sha512-rzMY6CaKx2qxrbYbqjXWS0plqEy7LOdKHS0bg4ixJ6aoGDPNUcLWk/FRNuCILh7GKLG9TFUXYYeQQldMBBwuyw==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/chunked-blob-reader": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/@smithy/chunked-blob-reader/-/chunked-blob-reader-5.2.0.tgz", + "integrity": "sha512-WmU0TnhEAJLWvfSeMxBNe5xtbselEO8+4wG0NtZeL8oR21WgH1xiO37El+/Y+H/Ie4SCwBy3MxYWmOYaGgZueA==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/chunked-blob-reader-native": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/@smithy/chunked-blob-reader-native/-/chunked-blob-reader-native-4.2.1.tgz", + "integrity": "sha512-lX9Ay+6LisTfpLid2zZtIhSEjHMZoAR5hHCR4H7tBz/Zkfr5ea8RcQ7Tk4mi0P76p4cN+Btz16Ffno7YHpKXnQ==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/util-base64": "^4.3.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/config-resolver": { + "version": "4.4.5", + "resolved": "https://registry.npmjs.org/@smithy/config-resolver/-/config-resolver-4.4.5.tgz", + "integrity": "sha512-HAGoUAFYsUkoSckuKbCPayECeMim8pOu+yLy1zOxt1sifzEbrsRpYa+mKcMdiHKMeiqOibyPG0sFJnmaV/OGEg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/node-config-provider": "^4.3.7", + "@smithy/types": "^4.11.0", + "@smithy/util-config-provider": "^4.2.0", + "@smithy/util-endpoints": "^3.2.7", + "@smithy/util-middleware": "^4.2.7", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/core": { + "version": "3.20.0", + "resolved": "https://registry.npmjs.org/@smithy/core/-/core-3.20.0.tgz", + "integrity": "sha512-WsSHCPq/neD5G/MkK4csLI5Y5Pkd9c1NMfpYEKeghSGaD4Ja1qLIohRQf2D5c1Uy5aXp76DeKHkzWZ9KAlHroQ==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/middleware-serde": "^4.2.8", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "@smithy/util-base64": "^4.3.0", + "@smithy/util-body-length-browser": "^4.2.0", + "@smithy/util-middleware": "^4.2.7", + "@smithy/util-stream": "^4.5.8", + "@smithy/util-utf8": "^4.2.0", + "@smithy/uuid": "^1.1.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/credential-provider-imds": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/credential-provider-imds/-/credential-provider-imds-4.2.7.tgz", + "integrity": "sha512-CmduWdCiILCRNbQWFR0OcZlUPVtyE49Sr8yYL0rZQ4D/wKxiNzBNS/YHemvnbkIWj623fplgkexUd/c9CAKdoA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/node-config-provider": "^4.3.7", + "@smithy/property-provider": "^4.2.7", + "@smithy/types": "^4.11.0", + "@smithy/url-parser": "^4.2.7", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/eventstream-codec": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/eventstream-codec/-/eventstream-codec-4.2.7.tgz", + "integrity": "sha512-DrpkEoM3j9cBBWhufqBwnbbn+3nf1N9FP6xuVJ+e220jbactKuQgaZwjwP5CP1t+O94brm2JgVMD2atMGX3xIQ==", + "license": "Apache-2.0", + "dependencies": { + "@aws-crypto/crc32": "5.2.0", + "@smithy/types": "^4.11.0", + "@smithy/util-hex-encoding": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/eventstream-serde-browser": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/eventstream-serde-browser/-/eventstream-serde-browser-4.2.7.tgz", + "integrity": "sha512-ujzPk8seYoDBmABDE5YqlhQZAXLOrtxtJLrbhHMKjBoG5b4dK4i6/mEU+6/7yXIAkqOO8sJ6YxZl+h0QQ1IJ7g==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/eventstream-serde-universal": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/eventstream-serde-config-resolver": { + "version": "4.3.7", + "resolved": "https://registry.npmjs.org/@smithy/eventstream-serde-config-resolver/-/eventstream-serde-config-resolver-4.3.7.tgz", + "integrity": "sha512-x7BtAiIPSaNaWuzm24Q/mtSkv+BrISO/fmheiJ39PKRNH3RmH2Hph/bUKSOBOBC9unqfIYDhKTHwpyZycLGPVQ==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/eventstream-serde-node": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/eventstream-serde-node/-/eventstream-serde-node-4.2.7.tgz", + "integrity": "sha512-roySCtHC5+pQq5lK4be1fZ/WR6s/AxnPaLfCODIPArtN2du8s5Ot4mKVK3pPtijL/L654ws592JHJ1PbZFF6+A==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/eventstream-serde-universal": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/eventstream-serde-universal": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/eventstream-serde-universal/-/eventstream-serde-universal-4.2.7.tgz", + "integrity": "sha512-QVD+g3+icFkThoy4r8wVFZMsIP08taHVKjE6Jpmz8h5CgX/kk6pTODq5cht0OMtcapUx+xrPzUTQdA+TmO0m1g==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/eventstream-codec": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/fetch-http-handler": { + "version": "5.3.8", + "resolved": "https://registry.npmjs.org/@smithy/fetch-http-handler/-/fetch-http-handler-5.3.8.tgz", + "integrity": "sha512-h/Fi+o7mti4n8wx1SR6UHWLaakwHRx29sizvp8OOm7iqwKGFneT06GCSFhml6Bha5BT6ot5pj3CYZnCHhGC2Rg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/protocol-http": "^5.3.7", + "@smithy/querystring-builder": "^4.2.7", + "@smithy/types": "^4.11.0", + "@smithy/util-base64": "^4.3.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/hash-blob-browser": { + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/@smithy/hash-blob-browser/-/hash-blob-browser-4.2.8.tgz", + "integrity": "sha512-07InZontqsM1ggTCPSRgI7d8DirqRrnpL7nIACT4PW0AWrgDiHhjGZzbAE5UtRSiU0NISGUYe7/rri9ZeWyDpw==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/chunked-blob-reader": "^5.2.0", + "@smithy/chunked-blob-reader-native": "^4.2.1", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/hash-node": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/hash-node/-/hash-node-4.2.7.tgz", + "integrity": "sha512-PU/JWLTBCV1c8FtB8tEFnY4eV1tSfBc7bDBADHfn1K+uRbPgSJ9jnJp0hyjiFN2PMdPzxsf1Fdu0eo9fJ760Xw==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "@smithy/util-buffer-from": "^4.2.0", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/hash-stream-node": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/hash-stream-node/-/hash-stream-node-4.2.7.tgz", + "integrity": "sha512-ZQVoAwNYnFMIbd4DUc517HuwNelJUY6YOzwqrbcAgCnVn+79/OK7UjwA93SPpdTOpKDVkLIzavWm/Ck7SmnDPQ==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/invalid-dependency": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/invalid-dependency/-/invalid-dependency-4.2.7.tgz", + "integrity": "sha512-ncvgCr9a15nPlkhIUx3CU4d7E7WEuVJOV7fS7nnK2hLtPK9tYRBkMHQbhXU1VvvKeBm/O0x26OEoBq+ngFpOEQ==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/is-array-buffer": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@smithy/is-array-buffer/-/is-array-buffer-4.2.0.tgz", + "integrity": "sha512-DZZZBvC7sjcYh4MazJSGiWMI2L7E0oCiRHREDzIxi/M2LY79/21iXt6aPLHge82wi5LsuRF5A06Ds3+0mlh6CQ==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/md5-js": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/md5-js/-/md5-js-4.2.7.tgz", + "integrity": "sha512-Wv6JcUxtOLTnxvNjDnAiATUsk8gvA6EeS8zzHig07dotpByYsLot+m0AaQEniUBjx97AC41MQR4hW0baraD1Xw==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/middleware-content-length": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/middleware-content-length/-/middleware-content-length-4.2.7.tgz", + "integrity": "sha512-GszfBfCcvt7kIbJ41LuNa5f0wvQCHhnGx/aDaZJCCT05Ld6x6U2s0xsc/0mBFONBZjQJp2U/0uSJ178OXOwbhg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/middleware-endpoint": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/@smithy/middleware-endpoint/-/middleware-endpoint-4.4.1.tgz", + "integrity": "sha512-gpLspUAoe6f1M6H0u4cVuFzxZBrsGZmjx2O9SigurTx4PbntYa4AJ+o0G0oGm1L2oSX6oBhcGHwrfJHup2JnJg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/core": "^3.20.0", + "@smithy/middleware-serde": "^4.2.8", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@smithy/types": "^4.11.0", + "@smithy/url-parser": "^4.2.7", + "@smithy/util-middleware": "^4.2.7", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/middleware-retry": { + "version": "4.4.17", + "resolved": "https://registry.npmjs.org/@smithy/middleware-retry/-/middleware-retry-4.4.17.tgz", + "integrity": "sha512-MqbXK6Y9uq17h+4r0ogu/sBT6V/rdV+5NvYL7ZV444BKfQygYe8wAhDrVXagVebN6w2RE0Fm245l69mOsPGZzg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/node-config-provider": "^4.3.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/service-error-classification": "^4.2.7", + "@smithy/smithy-client": "^4.10.2", + "@smithy/types": "^4.11.0", + "@smithy/util-middleware": "^4.2.7", + "@smithy/util-retry": "^4.2.7", + "@smithy/uuid": "^1.1.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/middleware-serde": { + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/@smithy/middleware-serde/-/middleware-serde-4.2.8.tgz", + "integrity": "sha512-8rDGYen5m5+NV9eHv9ry0sqm2gI6W7mc1VSFMtn6Igo25S507/HaOX9LTHAS2/J32VXD0xSzrY0H5FJtOMS4/w==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/middleware-stack": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/middleware-stack/-/middleware-stack-4.2.7.tgz", + "integrity": "sha512-bsOT0rJ+HHlZd9crHoS37mt8qRRN/h9jRve1SXUhVbkRzu0QaNYZp1i1jha4n098tsvROjcwfLlfvcFuJSXEsw==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" }, "engines": { - "node": ">=20.0.0" + "node": ">=18.0.0" } }, - "node_modules/@dyanet/config-aws": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/@dyanet/config-aws/-/config-aws-1.0.3.tgz", - "integrity": "sha512-Y7Wr9MWU+5rRrwg8Wr/5bqufdZWZvo8l3OZd18oCetRT9935+aPVdNP0IuGwLocitQoz+pP91x4G/3bIe/PIOg==", - "license": "MIT", + "node_modules/@smithy/node-config-provider": { + "version": "4.3.7", + "resolved": "https://registry.npmjs.org/@smithy/node-config-provider/-/node-config-provider-4.3.7.tgz", + "integrity": "sha512-7r58wq8sdOcrwWe+klL9y3bc4GW1gnlfnFOuL7CXa7UzfhzhxKuzNdtqgzmTV+53lEp9NXh5hY/S4UgjLOzPfw==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/property-provider": "^4.2.7", + "@smithy/shared-ini-file-loader": "^4.4.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, "engines": { - "node": ">=20.0.0", - "npm": ">=10.0.0" + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/node-http-handler": { + "version": "4.4.7", + "resolved": "https://registry.npmjs.org/@smithy/node-http-handler/-/node-http-handler-4.4.7.tgz", + "integrity": "sha512-NELpdmBOO6EpZtWgQiHjoShs1kmweaiNuETUpuup+cmm/xJYjT4eUjfhrXRP4jCOaAsS3c3yPsP3B+K+/fyPCQ==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/abort-controller": "^4.2.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/querystring-builder": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" }, - "peerDependencies": { - "@aws-sdk/client-s3": "^3.0.0", - "@aws-sdk/client-secrets-manager": "^3.0.0", - "@aws-sdk/client-ssm": "^3.0.0", - "zod": "^3.22.0" + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/property-provider": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/property-provider/-/property-provider-4.2.7.tgz", + "integrity": "sha512-jmNYKe9MGGPoSl/D7JDDs1C8b3dC8f/w78LbaVfoTtWy4xAd5dfjaFG9c9PWPihY4ggMQNQSMtzU77CNgAJwmA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" }, - "peerDependenciesMeta": { - "@aws-sdk/client-s3": { - "optional": true - }, - "@aws-sdk/client-secrets-manager": { - "optional": true - }, - "@aws-sdk/client-ssm": { - "optional": true - } + "engines": { + "node": ">=18.0.0" } }, - "node_modules/@dyanet/imap": { - "version": "0.5.2", - "resolved": "https://registry.npmjs.org/@dyanet/imap/-/imap-0.5.2.tgz", - "integrity": "sha512-VwxFsqCY4DKct4A+bHzLpBex7UrwWE4p0DsLrA6Bjuwftxmtm7HkkENuRwEZOlw020XDIAfbWMKXWd/n8M/N0A==", - "license": "MIT", + "node_modules/@smithy/protocol-http": { + "version": "5.3.7", + "resolved": "https://registry.npmjs.org/@smithy/protocol-http/-/protocol-http-5.3.7.tgz", + "integrity": "sha512-1r07pb994I20dD/c2seaZhoCuNYm0rWrvBxhCQ70brNh11M5Ml2ew6qJVo0lclB3jMIXirD4s2XRXRe7QEi0xA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, "engines": { - "node": ">=20.0.0" + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/querystring-builder": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/querystring-builder/-/querystring-builder-4.2.7.tgz", + "integrity": "sha512-eKONSywHZxK4tBxe2lXEysh8wbBdvDWiA+RIuaxZSgCMmA0zMgoDpGLJhnyj+c0leOQprVnXOmcB4m+W9Rw7sg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "@smithy/util-uri-escape": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/querystring-parser": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/querystring-parser/-/querystring-parser-4.2.7.tgz", + "integrity": "sha512-3X5ZvzUHmlSTHAXFlswrS6EGt8fMSIxX/c3Rm1Pni3+wYWB6cjGocmRIoqcQF9nU5OgGmL0u7l9m44tSUpfj9w==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/service-error-classification": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/service-error-classification/-/service-error-classification-4.2.7.tgz", + "integrity": "sha512-YB7oCbukqEb2Dlh3340/8g8vNGbs/QsNNRms+gv3N2AtZz9/1vSBx6/6tpwQpZMEJFs7Uq8h4mmOn48ZZ72MkA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/shared-ini-file-loader": { + "version": "4.4.2", + "resolved": "https://registry.npmjs.org/@smithy/shared-ini-file-loader/-/shared-ini-file-loader-4.4.2.tgz", + "integrity": "sha512-M7iUUff/KwfNunmrgtqBfvZSzh3bmFgv/j/t1Y1dQ+8dNo34br1cqVEqy6v0mYEgi0DkGO7Xig0AnuOaEGVlcg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/signature-v4": { + "version": "5.3.7", + "resolved": "https://registry.npmjs.org/@smithy/signature-v4/-/signature-v4-5.3.7.tgz", + "integrity": "sha512-9oNUlqBlFZFOSdxgImA6X5GFuzE7V2H7VG/7E70cdLhidFbdtvxxt81EHgykGK5vq5D3FafH//X+Oy31j3CKOg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/is-array-buffer": "^4.2.0", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "@smithy/util-hex-encoding": "^4.2.0", + "@smithy/util-middleware": "^4.2.7", + "@smithy/util-uri-escape": "^4.2.0", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/smithy-client": { + "version": "4.10.2", + "resolved": "https://registry.npmjs.org/@smithy/smithy-client/-/smithy-client-4.10.2.tgz", + "integrity": "sha512-D5z79xQWpgrGpAHb054Fn2CCTQZpog7JELbVQ6XAvXs5MNKWf28U9gzSBlJkOyMl9LA1TZEjRtwvGXfP0Sl90g==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/core": "^3.20.0", + "@smithy/middleware-endpoint": "^4.4.1", + "@smithy/middleware-stack": "^4.2.7", + "@smithy/protocol-http": "^5.3.7", + "@smithy/types": "^4.11.0", + "@smithy/util-stream": "^4.5.8", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/types": { + "version": "4.11.0", + "resolved": "https://registry.npmjs.org/@smithy/types/-/types-4.11.0.tgz", + "integrity": "sha512-mlrmL0DRDVe3mNrjTcVcZEgkFmufITfUAPBEA+AHYiIeYyJebso/He1qLbP3PssRe22KUzLRpQSdBPbXdgZ2VA==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/url-parser": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/url-parser/-/url-parser-4.2.7.tgz", + "integrity": "sha512-/RLtVsRV4uY3qPWhBDsjwahAtt3x2IsMGnP5W1b2VZIe+qgCqkLxI1UOHDZp1Q1QSOrdOR32MF3Ph2JfWT1VHg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/querystring-parser": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-base64": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/@smithy/util-base64/-/util-base64-4.3.0.tgz", + "integrity": "sha512-GkXZ59JfyxsIwNTWFnjmFEI8kZpRNIBfxKjv09+nkAWPt/4aGaEWMM04m4sxgNVWkbt2MdSvE3KF/PfX4nFedQ==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/util-buffer-from": "^4.2.0", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-body-length-browser": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-body-length-browser/-/util-body-length-browser-4.2.0.tgz", + "integrity": "sha512-Fkoh/I76szMKJnBXWPdFkQJl2r9SjPt3cMzLdOB6eJ4Pnpas8hVoWPYemX/peO0yrrvldgCUVJqOAjUrOLjbxg==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-body-length-node": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/@smithy/util-body-length-node/-/util-body-length-node-4.2.1.tgz", + "integrity": "sha512-h53dz/pISVrVrfxV1iqXlx5pRg3V2YWFcSQyPyXZRrZoZj4R4DeWRDo1a7dd3CPTcFi3kE+98tuNyD2axyZReA==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-buffer-from": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-buffer-from/-/util-buffer-from-4.2.0.tgz", + "integrity": "sha512-kAY9hTKulTNevM2nlRtxAG2FQ3B2OR6QIrPY3zE5LqJy1oxzmgBGsHLWTcNhWXKchgA0WHW+mZkQrng/pgcCew==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/is-array-buffer": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-config-provider": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-config-provider/-/util-config-provider-4.2.0.tgz", + "integrity": "sha512-YEjpl6XJ36FTKmD+kRJJWYvrHeUvm5ykaUS5xK+6oXffQPHeEM4/nXlZPe+Wu0lsgRUcNZiliYNh/y7q9c2y6Q==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-defaults-mode-browser": { + "version": "4.3.16", + "resolved": "https://registry.npmjs.org/@smithy/util-defaults-mode-browser/-/util-defaults-mode-browser-4.3.16.tgz", + "integrity": "sha512-/eiSP3mzY3TsvUOYMeL4EqUX6fgUOj2eUOU4rMMgVbq67TiRLyxT7Xsjxq0bW3OwuzK009qOwF0L2OgJqperAQ==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/property-provider": "^4.2.7", + "@smithy/smithy-client": "^4.10.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-defaults-mode-node": { + "version": "4.2.19", + "resolved": "https://registry.npmjs.org/@smithy/util-defaults-mode-node/-/util-defaults-mode-node-4.2.19.tgz", + "integrity": "sha512-3a4+4mhf6VycEJyHIQLypRbiwG6aJvbQAeRAVXydMmfweEPnLLabRbdyo/Pjw8Rew9vjsh5WCdhmDaHkQnhhhA==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/config-resolver": "^4.4.5", + "@smithy/credential-provider-imds": "^4.2.7", + "@smithy/node-config-provider": "^4.3.7", + "@smithy/property-provider": "^4.2.7", + "@smithy/smithy-client": "^4.10.2", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-endpoints": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/@smithy/util-endpoints/-/util-endpoints-3.2.7.tgz", + "integrity": "sha512-s4ILhyAvVqhMDYREeTS68R43B1V5aenV5q/V1QpRQJkCXib5BPRo4s7uNdzGtIKxaPHCfU/8YkvPAEvTpxgspg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/node-config-provider": "^4.3.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-hex-encoding": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-hex-encoding/-/util-hex-encoding-4.2.0.tgz", + "integrity": "sha512-CCQBwJIvXMLKxVbO88IukazJD9a4kQ9ZN7/UMGBjBcJYvatpWk+9g870El4cB8/EJxfe+k+y0GmR9CAzkF+Nbw==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-middleware": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/util-middleware/-/util-middleware-4.2.7.tgz", + "integrity": "sha512-i1IkpbOae6NvIKsEeLLM9/2q4X+M90KV3oCFgWQI4q0Qz+yUZvsr+gZPdAEAtFhWQhAHpTsJO8DRJPuwVyln+w==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-retry": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/util-retry/-/util-retry-4.2.7.tgz", + "integrity": "sha512-SvDdsQyF5CIASa4EYVT02LukPHVzAgUA4kMAuZ97QJc2BpAqZfA4PINB8/KOoCXEw9tsuv/jQjMeaHFvxdLNGg==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/service-error-classification": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-stream": { + "version": "4.5.8", + "resolved": "https://registry.npmjs.org/@smithy/util-stream/-/util-stream-4.5.8.tgz", + "integrity": "sha512-ZnnBhTapjM0YPGUSmOs0Mcg/Gg87k503qG4zU2v/+Js2Gu+daKOJMeqcQns8ajepY8tgzzfYxl6kQyZKml6O2w==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/fetch-http-handler": "^5.3.8", + "@smithy/node-http-handler": "^4.4.7", + "@smithy/types": "^4.11.0", + "@smithy/util-base64": "^4.3.0", + "@smithy/util-buffer-from": "^4.2.0", + "@smithy/util-hex-encoding": "^4.2.0", + "@smithy/util-utf8": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-uri-escape": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-uri-escape/-/util-uri-escape-4.2.0.tgz", + "integrity": "sha512-igZpCKV9+E/Mzrpq6YacdTQ0qTiLm85gD6N/IrmyDvQFA4UnU3d5g3m8tMT/6zG/vVkWSU+VxeUyGonL62DuxA==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-utf8": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@smithy/util-utf8/-/util-utf8-4.2.0.tgz", + "integrity": "sha512-zBPfuzoI8xyBtR2P6WQj63Rz8i3AmfAaJLuNG8dWsfvPe8lO4aCPYLn879mEgHndZH1zQ2oXmG8O1GGzzaoZiw==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/util-buffer-from": "^4.2.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/util-waiter": { + "version": "4.2.7", + "resolved": "https://registry.npmjs.org/@smithy/util-waiter/-/util-waiter-4.2.7.tgz", + "integrity": "sha512-vHJFXi9b7kUEpHWUCY3Twl+9NPOZvQ0SAi+Ewtn48mbiJk4JY9MZmKQjGB4SCvVb9WPiSphZJYY6RIbs+grrzw==", + "license": "Apache-2.0", + "dependencies": { + "@smithy/abort-controller": "^4.2.7", + "@smithy/types": "^4.11.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@smithy/uuid": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@smithy/uuid/-/uuid-1.1.0.tgz", + "integrity": "sha512-4aUIteuyxtBUhVdiQqcDhKFitwfd9hqoSDYY2KRXiWtgoWJ9Bmise+KfEPDiVHWeJepvF8xJO9/9+WDIciMFFw==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.6.2" + }, + "engines": { + "node": ">=18.0.0" } }, "node_modules/@types/body-parser": { @@ -213,6 +1947,12 @@ "npm": "1.2.8000 || >= 1.4.16" } }, + "node_modules/bowser": { + "version": "2.13.1", + "resolved": "https://registry.npmjs.org/bowser/-/bowser-2.13.1.tgz", + "integrity": "sha512-OHawaAbjwx6rqICCKgSG0SAnT05bzd7ppyKLVUITZpANBaaMFBAsaNkto3LoQ31tyFP5kNujE8Cdx85G9VzOkw==", + "license": "MIT" + }, "node_modules/bytes": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", @@ -454,6 +2194,24 @@ "node": ">= 0.8.0" } }, + "node_modules/fast-xml-parser": { + "version": "5.2.5", + "resolved": "https://registry.npmjs.org/fast-xml-parser/-/fast-xml-parser-5.2.5.tgz", + "integrity": "sha512-pfX9uG9Ki0yekDHx2SiuRIyFdyAr1kMIMitPvb0YBo8SUfKvia7w7FIyd/l6av85pFYRhZscS75MwMnbvY+hcQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/NaturalIntelligence" + } + ], + "license": "MIT", + "dependencies": { + "strnum": "^2.1.0" + }, + "bin": { + "fxparser": "src/cli/cli.js" + } + }, "node_modules/finalhandler": { "version": "1.3.2", "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.2.tgz", @@ -979,6 +2737,18 @@ "node": ">= 0.8" } }, + "node_modules/strnum": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/strnum/-/strnum-2.1.2.tgz", + "integrity": "sha512-l63NF9y/cLROq/yqKXSLtcMeeyOfnSQlfMSlzFt/K73oIaD8DGaQWd7Z34X9GPiKqP5rbSh84Hl4bOlLcjiSrQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/NaturalIntelligence" + } + ], + "license": "MIT" + }, "node_modules/toidentifier": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", @@ -988,6 +2758,12 @@ "node": ">=0.6" } }, + "node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "license": "0BSD" + }, "node_modules/type-is": { "version": "1.6.18", "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", diff --git a/examples/gmail-viewer/package.json b/examples/gmail-viewer/package.json index 0ce9b40..20e4523 100644 --- a/examples/gmail-viewer/package.json +++ b/examples/gmail-viewer/package.json @@ -20,6 +20,9 @@ "author": "", "license": "MIT", "dependencies": { + "@aws-sdk/client-s3": "^3.0.0", + "@aws-sdk/client-secrets-manager": "^3.0.0", + "@aws-sdk/client-ssm": "^3.0.0", "@dyanet/config-aws": "^1.0.3", "@dyanet/imap": "^0.5.2", "express": "^4.21.0",