import { Stack, Duration, RemovalPolicy, CfnOutput } from 'aws-cdk-lib'; import { Construct } from 'constructs'; import { aws_ec2 as ec2, aws_ecs as ecs, aws_s3 as s3, aws_iam as iam, aws_logs as logs, aws_elasticloadbalancingv2 as elb, aws_cloudfront as cloudfront, aws_cloudfront_origins as origins, aws_s3_deployment as s3_deployment } from 'aws-cdk-lib'; import { CloudFrontToS3 } from '@aws-solutions-constructs/aws-cloudfront-s3'; import { CfnDistribution, Distribution } from 'aws-cdk-lib/aws-cloudfront'; import { NodejsBuild } from 'deploy-time-build'; interface WebProps { cluster:ecs.Cluster alb:elb.IApplicationLoadBalancer; albSG:ec2.SecurityGroup; } export class Web extends Construct { readonly distribution; constructor(scope: Construct, id: string, props:WebProps) { super(scope, id) const commonBucketProps: s3.BucketProps = { blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL, encryption: s3.BucketEncryption.S3_MANAGED, autoDeleteObjects: true, removalPolicy: RemovalPolicy.DESTROY, objectOwnership: s3.ObjectOwnership.OBJECT_WRITER, enforceSSL: true, }; // CDKにて 静的WebサイトをホストするためのAmazon S3バケットを作成 const websiteBucket = new s3.Bucket(this, 'LangflowWebsiteBucket', commonBucketProps); const originAccessIdentity = new cloudfront.OriginAccessIdentity( this, 'OriginAccessIdentity', { comment: 'langflow-distribution-originAccessIdentity', } ); const webSiteBucketPolicyStatement = new iam.PolicyStatement({ actions: ['s3:GetObject'], effect: iam.Effect.ALLOW, principals: [ new iam.CanonicalUserPrincipal( originAccessIdentity.cloudFrontOriginAccessIdentityS3CanonicalUserId ), ], resources: [`${websiteBucket.bucketArn}/*`], }); websiteBucket.addToResourcePolicy(webSiteBucketPolicyStatement); websiteBucket.grantRead(originAccessIdentity); const s3SpaOrigin = new origins.S3Origin(websiteBucket); const ApiSpaOrigin = new origins.LoadBalancerV2Origin(props.alb,{ protocolPolicy: cloudfront.OriginProtocolPolicy.HTTP_ONLY }); const albBehaviorOptions = { origin: ApiSpaOrigin, allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL, viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.ALLOW_ALL, cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED, originRequestPolicy: cloudfront.OriginRequestPolicy.ALL_VIEWER_EXCEPT_HOST_HEADER } const cloudFrontWebDistribution = new cloudfront.Distribution(this, 'distribution', { comment: 'langflow-distribution', defaultRootObject: 'index.html', errorResponses: [ { httpStatus: 403, responseHttpStatus: 200, responsePagePath: '/index.html', }, { httpStatus: 404, responseHttpStatus: 200, responsePagePath: '/index.html', }, ], defaultBehavior: { origin: s3SpaOrigin }, additionalBehaviors: { '/api/v1/*': albBehaviorOptions, '/health' : albBehaviorOptions, }, enableLogging: true, // ログ出力設定 logBucket: new s3.Bucket(this, 'LogBucket',commonBucketProps), logFilePrefix: 'distribution-access-logs/', logIncludesCookies: true, }); this.distribution = cloudFrontWebDistribution; new NodejsBuild(this, 'BuildFrontEnd', { assets: [ { path: '../../src/frontend', exclude: [ '.git', '.github', '.gitignore', '.prettierignore', 'build', 'node_modules' ], }, ], nodejsVersion:20, destinationBucket: websiteBucket, distribution: cloudFrontWebDistribution, outputSourceDirectory: 'build', buildCommands: ['npm install', 'npm run build'], buildEnvironment: { // VITE_AXIOS_BASE_URL: `https://${this.distribution.domainName}` }, }); // distribution から backendへのinbound 許可 const alb_listen_port=80 props.albSG.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(alb_listen_port)) const alb_listen_port_443=443 props.albSG.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(alb_listen_port_443)) new CfnOutput(this, 'URL', { value: `https://${this.distribution.domainName}`, }); } }