{"value":"*This post also had contributions from Jiwon Yeom, Solutions Architect, AWS.*\n\n### **Introduction**\nAWS customers use infrastructure as code (IaC) to create cloud resources in a repeatable and predictable manner. IaC is especially helpful in managing environments with identical stacks, which is a common occurrence in active-active multi-region systems. Instead of managing each regional deployment individually, teams avoid infrastructure sprawl by managing infrastructure in both regions using code and pipelines.\n\nTeams need fewer resources to manage and troubleshoot infrastructure when their system configuration is identical across regions or accounts. Using IaC tools like [AWS CloudFormation](https://aws.amazon.com/cloudformation/), [Terraform](https://www.terraform.io/), [Pulumi](https://www.pulumi.com/), and [AWS Cloud Development Kit(https://aws.amazon.com/cdk) (CDK), operators create and manage multiple environments from a single codebase and avoid system failures caused by ad hoc changes. IaC also makes it easier to rollback when a change fails.\n\nSimilarly, the [GitOps](https://www.weave.works/blog/what-is-gitops-really) approach allows you to externalize your Kubernetes cluster configuration. GitOps tools like [ArgoCD](https://argo-cd.readthedocs.io/en/stable/) allow you to store your Kubernetes cluster state as code in a Git repository. If you manage multiple replica clusters, which is a common scenario in multi-region deployments, then you can control their configuration by pointing your GitOps operator to the same Git repository.\n\n#### **Unified infrastructure and workload deployments**\nCustomers that want to manage their infrastructure and Kubernetes resources using the same code library with a variety of available solutions. Teams already using [Terraform](https://www.terraform.io/) can package their applications using the [Helm](https://learn.hashicorp.com/tutorials/terraform/helm-provider?in=terraform/kubernetes) and [Kubernetes](https://registry.terraform.io/providers/hashicorp/kubernetes/latest/docs) provider. Pulumi users have [Pulumi Kubernetes Provider](https://www.pulumi.com/registry/packages/kubernetes/).\n\nSimilarly, [CDK](https://docs.aws.amazon.com/cdk/?id=docs_gateway) and [cdk8s](https://cdk8s.io/docs/latest/) allow you to code both the cloud infrastructure and Kubernetes resources using familiar programming languages. Customers can declare their Virtual Private Clouds (VPCs), [Subnets](https://docs.aws.amazon.com/eks/?id=docs_gateway), Amazon EKS clusters, Kubernetes namespaces, role-based access control (RBAC) policies, deployments, services, ingresses, and other tasks in the same code base. The Git repository that hosts the application code also becomes the source of truth for infrastructure and workload configuration.\n\nThis post shows you how to create [Amazon EKS](https://docs.aws.amazon.com/eks/?id=docs_gateway) clusters in multiple AWS Regions using [CDK](https://docs.aws.amazon.com/cdk/?id=docs_gateway) and create a continuous deployment pipeline for infrastructure and application changes.\n\n#### **Use cases**\nImagine you work for an Independent Software Vendor (ISV) with a security sensitive product that can’t be multi-tenant. The system design requires one [Amazon EKS](https://docs.aws.amazon.com/eks/?id=docs_gateway) cluster per tenant. When the product becomes successful, your organization may be responsible for operating a multitude of replica clusters.\n\nIn scenarios where the cluster lifecycle is closely tied to the application’s, you can store the definition of your infrastructure, [Amazon EKS](https://aws.amazon.com/eks/) cluster, and application resources in the same codebase. To deploy a new stack, you rerun the [CDK](https://aws.amazon.com/cdk/) code, which creates the infrastructure resources required to run the application and then deploys the application in an [Amazon EKS](https://aws.amazon.com/eks/) cluster.\n\nDay 2 operations, such as deploying new versions of applications and upgrading clusters, are also managed using [CDK](https://aws.amazon.com/eks/). We recommend checking code using a tool like [cdk-nag](https://github.com/cdklabs/cdk-nag), along with using standardization and parametrization to minimize the variability between regions.\n\n#### **Multi-region architecture**\nThe code included in this post creates two proof-of-concept [Amazon EKS](https://aws.amazon.com/eks/) clusters and the supporting infrastructure in two AWS Regions.\n\n![image.png](https://dev-media.amazoncloud.cn/080af6defe8e43da9e9c5fd44c60da2a_image.png)\n\n[CDK](https://aws.amazon.com/cdk/) also creates a stack with CodePipeline to build a container image from the sample application’s source code. Users trigger an automated deployment to the primary AWS Region by checking in code to the [AWS CodeCommit](https://aws.amazon.com/codecommit/) Git repository [CDK](https://aws.amazon.com/cdk/) creates. CodePipeline triggers [AWS CodeBuild](https://aws.amazon.com/codebuild/) when new changes are checked-in to the Git repository and the result is an image pushed to an [Amazon Elastic Container Registry](https://aws.amazon.com/ecr/) (Amazon ECR) repository.\n\n![image.png](https://dev-media.amazoncloud.cn/a7548cbffae9425f90ea1066d31ca41f_image.png)\n\nThe [CDK](https://aws.amazon.com/cdk/) code this post provides doesn’t include multi-region traffic routing. Customers can add a Global Accelerator, as explained [in this post](https://aws.amazon.com/blogs/containers/operating-a-multi-regional-stateless-application-using-amazon-eks/), or use Domain Name System (DNS) to route traffic between regions.\n\n### **Solution overview**\n#### **Prerequisites**\nYou will need the following to complete the steps in this post:\n\n- [AWS CLI version 2](https://docs.aws.amazon.com/cli/latest/userguide/install-cliv2.html)\n- [AWS CDK](https://docs.aws.amazon.com/cdk/v2/guide/getting_started.html#getting_started_install) version 2.19.0 or later\n- [Node](https://nodejs.org/en/download/current/) version 17.8.0 or later\n- [NPM](https://docs.npmjs.com/downloading-and-installing-node-js-and-npm) version 8.5.5 or later\n- [Kubectl](https://docs.aws.amazon.com/eks/latest/userguide/install-kubectl.html)\n- [Docker-cli](https://docs.docker.com/get-docker/)\n- [Git](https://git-scm.com/download)\n- [Jq](https://stedolan.github.io/jq/download/)\n\n\nLet’s start by setting a few environment variables:\n\n```\nACCOUNT_ID=$(aws sts get-caller-identity --query 'Account' --output text)\nAWS_PRIMARY_REGION=us-east-2\nAWS_SECONDARY_REGION=eu-west-2\n```\n\nClone the sample repository and install dependency packages. This repository contains CDK v2 code written in TypeScript.\n\n```\ngit clone https://github.com/aws-samples/containers-blog-maelstrom\ncd containers-blog-maelstrom/aws-cdk-eks-multi-region-skeleton/\nnpm install\n```\n\nThe [Amazon EKS](https://aws.amazon.com/eks/) cluster definition is stored in ```lib/cluster-stack.ts```. The following is the snippet of the code:\n\n```\n const cluster = new eks.Cluster(this, 'demoeks--cluster', {\n clusterName: `demoeks`,\n mastersRole: clusterAdmin,\n version: eks.KubernetesVersion.V1_21,\n defaultCapacity: 2,\n defaultCapacityInstance: new ec2.InstanceType(props.onDemandInstanceType)\n });\n```\n\nAs defined in ```bin/multi-cluster-ts.ts```, the code creates two clusters and deploys the supporting infrastructure, the\n\ncontinuous integration and continuous delivery/continuous deployment (CI/CD) stack (CodePipeline, [CodeBuild](https://aws.amazon.com/codebuild/), and [CodeCommit](https://aws.amazon.com/codecommit/)), [Amazon EKS](https://aws.amazon.com/eks/) clusters, and a sample web application.\n\nYou can customize the ```multi-cluster-ts.ts``` file to change regions and instance types based on your requirements.\n\n\n```\nconst primaryRegion = {account: account, region: 'us-east-2'};\nconst secondaryRegion = {account: account, region: 'eu-west-2'};\nconst primaryOnDemandInstanceType = 'r5.2xlarge';\nconst secondaryOnDemandInstanceType = 'm5.2xlarge';\n```\n\n#### **Bootstrap AWS Regions**\nThe first step to any [CDK](https://aws.amazon.com/cdk/) deployment is bootstrapping the environment. ```cdk bootstrap``` is a tool in the AWS CDK command-line interface ([AWS CLI](https://docs.aws.amazon.com/cli/?id=docs_gateway)) responsible for preparing the environment (i.e., a combination of AWS account and AWS Region) with resources required by CDK to perform deployments into that environment. If you already use [CDK](https://aws.amazon.com/cdk/) in a region, you don’t need to repeat the bootstrapping process.\n\nExecute the commands below to bootstrap the AWS environment in ```us-east-2``` and ```eu-west-2```:\n\n```\ncdk bootstrap aws://$ACCOUNT_ID/$AWS_PRIMARY_REGION\ncdk bootstrap aws://$ACCOUNT_ID/$AWS_SECONDARY_REGION\n```\n\nThe CDK code creates five stacks:\n\n- A ```cluster-stack``` that creates an [Amazon EKS](https://aws.amazon.com/eks/) cluster for primary region ```us-east-2```\n- A ```container-stack``` that deploys sample nginx containers to [Amazon EKS](https://aws.amazon.com/eks/) cluster for primary region ```us-east-2```\n- A ```cluster-stack``` that creates an [Amazon EKS](https://aws.amazon.com/eks/) cluster for secondary region ```eu-west-2```\n- A ```container-stack``` that deploys sample nginx containers to [Amazon EKS](https://aws.amazon.com/eks/) cluster or secondary region ```eu-west-2```\n- A ```cicd-stack``` that creates a cicd pipeline using [AWS CodePipeline](https://docs.aws.amazon.com/codepipeline/?id=docs_gateway) and [AWS CodeSuites](https://aws.amazon.com/products/developer-tools/) to containerize and deploy a sample flask application that crosses regions of [Amazon EKS](https://aws.amazon.com/eks/)\n\n\nRun ```cdk list``` to see the list of the stacks to be created:\n\n```\n$ cdk list\nClusterStack-us-east-2\nClusterStack-eu-west-2\nContainerStack-us-east-2\nContainerStack-eu-west-2\nCicdStack\n```\n\nYou can use ```cdk diff``` to get a list of resources [CDK](https://aws.amazon.com/cdk/) will create:\n\n```cdk diff```\n\n*If the result doesn’t match the output below, then refer to [this step](https://catalog.us-east-1.prod.workshops.aws/workshops/c15012ac-d05d-46b1-8a4a-205e7c9d93c9/en-US/40-deploy-clusters/100-clone-base-repo/) and check if npm run watch is running in the background.*\n\nYou should see the following output:\n```\nStack ClusterStack-ap-southeast-2IAM Statement Changes\n```\n\n```\n┌───┬────────────────────────┬────────┬────────────────────────┬────────────────────────┬───────────┐\n│ │ Resource │ Effect │ Action │ Principal │ Condition │\n├───┼────────────────────────┼────────┼────────────────────────┼────────────────────────┼───────────┤\n│ + │ ${AdminRole.Arn} │ Allow │ sts\\:AssumeRole │ AWS\\:arn:${AWS::Partiti│ │\n│ │ │ │ │ on}\\:iam::<<ACCOUNT_ID>│ │\n│ │ │ │ │ root │ │\n...\n\n...\nResources\n[+] AWS::IAM::Role AdminRole AdminRole38563C57 \n[+] AWS::EC2::VPC demogo-cluster/DefaultVpc demogoclusterDefaultVpc0F0EA8D6 \n[+] AWS::EC2::Subnet demogo-cluster/DefaultVpc/PublicSubnet1/Subnet demogoclusterDefaultVpcPublicSubnet1Subnet9B5D84CC \n[+] AWS::EC2::RouteTable demogo-cluster/DefaultVpc/PublicSubnet1/RouteTable demogoclusterDefaultVpcPublicSubnet1RouteTableA9719167 \n[+] AWS::EC2::SubnetRouteTableAssociation demogo-cluster/DefaultVpc/PublicSubnet1/RouteTableAssociation demogoclusterDefaultVpcPublicSubnet1RouteTableAssociationF6BCC682 \n[+] AWS::EC2::Route demogo-cluster/DefaultVpc/PublicSubnet1/DefaultRoute demogoclusterDefaultVpcPublicSubnet1DefaultRoute0A0FDBF1 \n[+] AWS::EC2::EIP demogo-cluster/DefaultVpc/PublicSubnet1/EIP demogoclusterDefaultVpcPublicSubnet1EIP42D57092 \n[+] AWS::EC2::NatGateway demogo-cluster/DefaultVpc/PublicSubnet1/NATGateway\n```\n\nThe code supports region-specific customization. It looks for Kubernetes deployment files in region-specific directories. The sample code contains directories for primary and secondary regions (called ```yaml-us-east-2``` and ```yaml-eu-west-2```). [CDK](https://aws.amazon.com/cdk/) will apply the deployment files in these folders to deploy the Kubernetes application in the two regions.\n\nIf your regions differ, then please create directories manually and copy the Kubernetes deployment manifest in each directory:\n\n```\n# mkdir yaml-<region-name> for primary and secondary regions\nmkdir yaml-$AWS_PRIMARY_REGION\nmkdir yaml-$AWS_SECONDARY_REGION\n\ncp yaml-us-east-2/00_ap_nginx.yaml yaml-$AWS_PRIMARY_REGION/\ncp yaml-us-east-2/00_ap_nginx.yaml yaml-$AWS_SECONDARY_REGION/\n```\n\n#### **Create the clusters**\nIf you are following the tutorial using different AWS Regions than the ones used in the post, please update the values of primary and secondary region in ```bin/multi-cluster-ts.ts``` before proceeding.\n\nThe ```cdk deploy``` subcommand deploys the specified stack(s) to your AWS account. Let’s deploy all five stacks:\n\n```\ncdk deploy \"*\"\n```\n\nUpon completion, you can check the pipeline in the [CodePipeline Console](http://console.aws.amazon.com/codesuite/codepipeline/home). There is no master branch in [CodeCommit](https://aws.amazon.com/codecommit/) yet, thus the pipeline will be on a failed status.\n\n[CDK](https://aws.amazon.com/cdk/) creates the following resources:\n\n- [Amazon EKS](https://aws.amazon.com/eks/) clusters in two regions and the required infrastructure including VPCs, 3x public and private subnets, Internet Gateways, Network Address Translation (NAT) Gateways.\n- [Amazon EC2](https://docs.aws.amazon.com/ec2/?id=docs_gateway) instances: r5.2xlarge, mv5.2xlarge\n- CI/CD Pipeline to deploy sample application to [Amazon EKS](https://aws.amazon.com/eks/) clusters in ```us-east-2``` and ```eu-west-2```\n- [Amazon ECR](https://docs.aws.amazon.com/ecr/?id=docs_gateway) Repository in both to store sample application container images\n- A Kubernetes namespace in each cluster\n\n\n#### **Deploy sample app to multiple regions**\nNavigate to sample application in the cloned repo and register the created [CodeCommit](https://aws.amazon.com/codecommit/) repository in the application project with the following commands:\n\n```\nmkdir ../../sample-app\ncp -R ../aws-cdk-multi-region-sample-app ../../sample-app\ncd ../../sample-app/aws-cdk-multi-region-sample-app\nEKS_CDK_CODECOMMIT_REPO=$(aws codecommit list-repositories --region $AWS_PRIMARY_REGION --query \"repositories[?starts_with(repositoryName,'hello-py')].repositoryName\" --output text)\nEKS_CDK_CODECOMMIT_REPO_URL=$(aws codecommit get-repository --region $AWS_PRIMARY_REGION --repository-name $EKS_CDK_CODECOMMIT_REPO --query'repositoryMetadata.cloneUrlHttp' --output text)\ngit init\ngit remote add codecommit $EKS_CDK_CODECOMMIT_REPO_URL\n```\n\nBefore checking code into the [CodeCommit](https://aws.amazon.com/codecommit/) repository, we have to configure the Git client with [CodeCommit](https://aws.amazon.com/codecommit/) credentials. This step requires using the [AWS Management Console](https://docs.aws.amazon.com/awsconsolehelpdocs/latest/gsg/learn-whats-new.html).\n\nNavigate to the AWS Identity & Access Management [(IAM) User](https://console.aws.amazon.com/iam/home?region=us-west-2) console and create the HTTPS Git credentials for AWS [CodeCommit](https://aws.amazon.com/codecommit/) for the IAM user your terminal is currently configured to use. For more information, please refer to [AWS documentation](https://docs.aws.amazon.com/codecommit/latest/userguide/setting-up-gc.html?icmpid=docs_acc_console_connect_np):\n\nPush the code in the directory to [CodeCommit](https://aws.amazon.com/codecommit/) with the following command:\n\n```\ngit add .\ngit commit -am \"initial commit\"\ngit push codecommit master\n```\n\nIf this is your first time using [CodeCommit](https://aws.amazon.com/codecommit/), you will be prompted for username and password to push code to the [CodeCommit](https://aws.amazon.com/codecommit/) Git repository. You can use the [CodeCommit](https://aws.amazon.com/codecommit/) credentials downloaded in the prior step.\n\nAfter you push to the repository, you can see in the CodePipeline console that the pipeline is triggered as follows:\n\n![image.png](https://dev-media.amazoncloud.cn/9442008aea36403e968382109ed1ce34_image.png)\n\nOnce the pipeline deploys the sample application to the [Amazon EKS](https://aws.amazon.com/eks/) cluster in the primary region, it waits for a manual review before proceeding. Let’s check the deployment’s health in the primary cluster. Configure your kubectl to use your primary cluster:\n\n```\nCLUSTER1_KUBECONFIG_COMMAND=$(aws cloudformation describe-stacks \\\n --stack-name \"ClusterStack-$AWS_PRIMARY_REGION\" \\\n --region $AWS_PRIMARY_REGION \\\n --query 'Stacks[0].Outputs[?starts_with(OutputKey,`demoeksclusterConfig`)].OutputValue' \\\n --output text)\n$(echo $CLUSTER1_KUBECONFIG_COMMAND)\n```\n\nSee running services:\n\n```\nkubectl get svc\n```\n\n![image.png](https://dev-media.amazoncloud.cn/54d5991f9ada483997d94134818168e5_image.png)\n\nLet’s curl the hello-py service:\n\n```\ncurl $(kubectl get service hello-py -o \\\n jsonpath='{.status.loadBalancer.ingress[*].hostname}') && echo \"\"\n```\n\n![image.png](https://dev-media.amazoncloud.cn/eeee7b2b511540caa9211ad33a21a5a0_image.png)\n\nThe service responds with a **Hello World from <AWS Region>** message to the curl request. Since the service is functional in the primary region, this deployment is assumed successful. As a result, the pipeline can proceed with deploying the application into the primary region.\n\nBefore we resume the pipeline, let’s verify that ```hello-py``` service doesn’t exist in the secondary Region.\n\nSet your ```kubectl's``` current context to the [Amazon EKS](https://aws.amazon.com/eks/) cluster in the secondary region:\n\n```\nCLUSTER2_KUBECONFIG_COMMAND=$(aws cloudformation describe-stacks \\\n --stack-name \"ClusterStack-$AWS_SECONDARY_REGION\" \\\n --region $AWS_SECONDARY_REGION \\\n --query 'Stacks[0].Outputs[?starts_with(OutputKey,`demoeksclusterConfig`)].OutputValue' \\\n --output text)\n$(echo $CLUSTER2_KUBECONFIG_COMMAND)\n```\n\nVerify that ```hello-py``` service is not running in the secondary cluster:\n\n```\nkubectl get service hello-py\n```\n\nYou should get an error:\n\n```\nError from server (NotFound): services \"hello-py\" not found\n```\n\n\nNow we know there is no ```hello-py``` service in the secondary region. Let’s return to CodePipeline and approve the manual step.\n\nReturn to the CodePipeline [AWS Management Console](https://signin.aws.amazon.com/signin?redirect_uri=https%3A%2F%2Fus-east-1.console.aws.amazon.com%2Fiam%2Fhome%3Fregion%3Dus-west-2%26state%3DhashArgs%2523%26isauthcode%3Dtrue&client_id=arn%3Aaws%3Aiam%3A%3A015428540659%3Auser%2Fiam&forceMobileApp=0&code_challenge=zBijXuawh90K_1v8K8Xn7UXlGgTnItRRKDVxhlrnFSo&code_challenge_method=SHA-256) and select Review and select Approve in the next prompt. CodePipeline API [PutApprovalResult](https://docs.aws.amazon.com/codepipeline/latest/APIReference/API_PutApprovalResult.html) can be used to automate this approval process.\n\n\n![image.png](https://dev-media.amazoncloud.cn/c22637226d2b4aa5802d06707b7e8a54_image.png)\n\n![image.png](https://dev-media.amazoncloud.cn/8b8d641a22db45beb17cac9841897f0d_image.png)\n\n\nWhen the pipeline gets the approval to deploy to the secondary region, it applies the deployment manifest (```app-deployment.yaml```) to the second [Amazon EKS](https://aws.amazon.com/eks/) cluster.\n\nOnce the pipeline finishes successfully, check if ```hello-py``` service is functional:\n\n```\nkubectl get service hello-py\n```\n\n\nOnce the service has an application load balancer (ALB) attached, then send a request to verify that the service responds with the secondary region’s name in the response.\n\n```\ncurl $(kubectl get service hello-py -o \\\n jsonpath='{.status.loadBalancer.ingress[*].hostname}') && echo \"\"\n```\n\nThis completes deployment process. hello-py service is deployed and functional in both regions.\n\n#### **Upgrades**\nAs new code is checked-in to the [CodeCommit](https://aws.amazon.com/codecommit/) repository, the pipeline will automatically attempt to create a new deployments. When operators verify the changes are successful in the primary region, they can approve the deployment to the secondary region.\n\nThe file lib/cluster-stack.ts contains the declaration of your [Amazon EKS](https://aws.amazon.com/eks/) cluster settings. You can modify it to control cluster properties and upgrade clusters.\n\nAs you onboard newer services into these clusters, you can replicate the CI/CD stack created in this post for each service.\n\n### **Cleanup**\nYou continue to incur cost until deleting the infrastructure that you created for this post. Use the following manual process and commands to clean up the created AWS resources for this post:\n\n```\n$(echo $CLUSTER1_KUBECONFIG_COMMAND)\nkubectl delete svc hello-py\n$(echo $CLUSTER2_KUBECONFIG_COMMAND)\nkubectl delete svc hello-py\ncd ../../containers-blog-maelstrom/aws-cdk-eks-multi-region-skeleton\nAWS_ECR_REPO=$(aws ecr describe-repositories --query \"repositories[].[repositoryName]\" --region $AWS_PRIMARY_REGION | grep 'cicdstack-ecrforhellopy' | sed -e 's/^[[:space:]]*//' | sed -e 's/^\"//' -e 's/\"$//')\nAWS_ECR_IMAGES_TO_DELETE=$(aws ecr list-images --region $AWS_PRIMARY_REGION --repository-name $AWS_ECR_REPO --query 'imageIds[*]' --output json )\naws ecr batch-delete-image --region $AWS_PRIMARY_REGION --repository-name $AWS_ECR_REPO --image-ids \"$AWS_ECR_IMAGES_TO_DELETE\" || true\ncdk destroy \"*\" \n```\n\n\nCDK will ask you ```Are you sure you want to delete: CicdStack, ContainerStack-<primary region>, ContainerStack-<secondary region>, ClusterStack-<primary region>, ClusterStack<secondary region> (y/n)?``` and enter ```y``` to delete.\n\n### **Conclusion**\nThis post demonstrated the procedure to create continuous deployment pipeline to deploy applications in multiple [Amazon EKS](https://aws.amazon.com/eks/) clusters running in different regions. The accompanying CDK code creates EKS clusters and the CI/CD stack to continuously deploy application to the clusters.\n\n[CDK](https://aws.amazon.com/cdk/) helps you define your cloud infrastructure, Kubernetes resources, and application in the same codebase. If your codebase is in a language that [CDK](https://aws.amazon.com/cdk/) supports, your applications, configuration, and the underlying infrastructure can be coded in the same language. If you must generate Kubernetes manifests using code, you can also use [cdk8s](https://cdk8s.io/) to define Kubernetes resource programmatically.\n\nAs a next step, you can also explore [EKS Blueprints](https://aws.amazon.com/blogs/containers/bootstrapping-clusters-with-eks-blueprints/), which is a collection of IaC modules, to help you configure and deploy consistent and batteries-included [Amazon EKS](https://aws.amazon.com/eks/) clusters across accounts and Regions. You can use [EKS Blueprints](https://aws.amazon.com/blogs/containers/bootstrapping-clusters-with-eks-blueprints/) to easily bootstrap an [Amazon EKS](https://aws.amazon.com/eks/) cluster with [Amazon EKS add-ons](https://docs.aws.amazon.com/eks/latest/userguide/eks-add-ons.html). A wide range of popular open-source add-ons, including [Prometheus](https://prometheus.io/), [Karpenter](https://karpenter.sh/), [Nginx](https://www.nginx.com/), [Traefik](https://traefik.io/), [AWS Load Balancer Controller](https://docs.aws.amazon.com/eks/latest/userguide/aws-load-balancer-controller.html), [Fluentbit](https://fluentbit.io/), [Keda](https://keda.sh/), [Argo CD](https://argoproj.github.io/cd/), and more.\n\nThis post is also available as a [workshop](https://catalog.us-east-1.prod.workshops.aws/workshops/c15012ac-d05d-46b1-8a4a-205e7c9d93c9/en-US/) that uses CDK v1. You can find the changes between AWS CDK v1 and v2 [here](https://docs.aws.amazon.com/cdk/v2/guide/migrating-v2.html).\n\n![image.png](https://dev-media.amazoncloud.cn/f7b52a0f5dd54febb5d01070a29cdf5f_image.png)\n\n**Elamaran Shanmugam**\n\nElamaran (Ela) Shanmugam is a Sr. Partner Container Specialist Solutions Architect with Amazon Web Services. Ela is a Container, Observability and Multi-Account Architecture SME and helps AWS customers to design and build scalable, secure and optimized container workloads on AWS. His passion is building and automating Infrastructure to allow customers to focus more on their business. He is based out of Tampa, Florida and you can reach him on twitter @IamElaShan\n\n![8a5f26145878487abb6cf354fa9ca388_image1.png](1)\n**Re Alvarez-Parmar**\n\nRe Alvarez-Parmar is a Container Specialist Solutions Architect at Amazon Web Services. He helps customers use AWS container services to design scalable and secure applications. He is based out of Seattle and uses Twitter, sparingly, @realz","render":"<p><em>This post also had contributions from Jiwon Yeom, Solutions Architect, AWS.</em></p>\n<h3><a id=\"Introduction_2\"></a><strong>Introduction</strong></h3>\n<p>AWS customers use infrastructure as code (IaC) to create cloud resources in a repeatable and predictable manner. IaC is especially helpful in managing environments with identical stacks, which is a common occurrence in active-active multi-region systems. Instead of managing each regional deployment individually, teams avoid infrastructure sprawl by managing infrastructure in both regions using code and pipelines.</p>\n<p>Teams need fewer resources to manage and troubleshoot infrastructure when their system configuration is identical across regions or accounts. Using IaC tools like <a href=\"https://aws.amazon.com/cloudformation/\" target=\"_blank\">AWS CloudFormation</a>, <a href=\"https://www.terraform.io/\" target=\"_blank\">Terraform</a>, <a href=\"https://www.pulumi.com/\" target=\"_blank\">Pulumi</a>, and [AWS Cloud Development Kit(https://aws.amazon.com/cdk) (CDK), operators create and manage multiple environments from a single codebase and avoid system failures caused by ad hoc changes. IaC also makes it easier to rollback when a change fails.</p>\n<p>Similarly, the <a href=\"https://www.weave.works/blog/what-is-gitops-really\" target=\"_blank\">GitOps</a> approach allows you to externalize your Kubernetes cluster configuration. GitOps tools like <a href=\"https://argo-cd.readthedocs.io/en/stable/\" target=\"_blank\">ArgoCD</a> allow you to store your Kubernetes cluster state as code in a Git repository. If you manage multiple replica clusters, which is a common scenario in multi-region deployments, then you can control their configuration by pointing your GitOps operator to the same Git repository.</p>\n<h4><a id=\"Unified_infrastructure_and_workload_deployments_9\"></a><strong>Unified infrastructure and workload deployments</strong></h4>\n<p>Customers that want to manage their infrastructure and Kubernetes resources using the same code library with a variety of available solutions. Teams already using <a href=\"https://www.terraform.io/\" target=\"_blank\">Terraform</a> can package their applications using the <a href=\"https://learn.hashicorp.com/tutorials/terraform/helm-provider?in=terraform/kubernetes\" target=\"_blank\">Helm</a> and <a href=\"https://registry.terraform.io/providers/hashicorp/kubernetes/latest/docs\" target=\"_blank\">Kubernetes</a> provider. Pulumi users have <a href=\"https://www.pulumi.com/registry/packages/kubernetes/\" target=\"_blank\">Pulumi Kubernetes Provider</a>.</p>\n<p>Similarly, <a href=\"https://docs.aws.amazon.com/cdk/?id=docs_gateway\" target=\"_blank\">CDK</a> and <a href=\"https://cdk8s.io/docs/latest/\" target=\"_blank\">cdk8s</a> allow you to code both the cloud infrastructure and Kubernetes resources using familiar programming languages. Customers can declare their Virtual Private Clouds (VPCs), <a href=\"https://docs.aws.amazon.com/eks/?id=docs_gateway\" target=\"_blank\">Subnets</a>, Amazon EKS clusters, Kubernetes namespaces, role-based access control (RBAC) policies, deployments, services, ingresses, and other tasks in the same code base. The Git repository that hosts the application code also becomes the source of truth for infrastructure and workload configuration.</p>\n<p>This post shows you how to create <a href=\"https://docs.aws.amazon.com/eks/?id=docs_gateway\" target=\"_blank\">Amazon EKS</a> clusters in multiple AWS Regions using <a href=\"https://docs.aws.amazon.com/cdk/?id=docs_gateway\" target=\"_blank\">CDK</a> and create a continuous deployment pipeline for infrastructure and application changes.</p>\n<h4><a id=\"Use_cases_16\"></a><strong>Use cases</strong></h4>\n<p>Imagine you work for an Independent Software Vendor (ISV) with a security sensitive product that can’t be multi-tenant. The system design requires one <a href=\"https://docs.aws.amazon.com/eks/?id=docs_gateway\" target=\"_blank\">Amazon EKS</a> cluster per tenant. When the product becomes successful, your organization may be responsible for operating a multitude of replica clusters.</p>\n<p>In scenarios where the cluster lifecycle is closely tied to the application’s, you can store the definition of your infrastructure, <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster, and application resources in the same codebase. To deploy a new stack, you rerun the <a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> code, which creates the infrastructure resources required to run the application and then deploys the application in an <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster.</p>\n<p>Day 2 operations, such as deploying new versions of applications and upgrading clusters, are also managed using <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">CDK</a>. We recommend checking code using a tool like <a href=\"https://github.com/cdklabs/cdk-nag\" target=\"_blank\">cdk-nag</a>, along with using standardization and parametrization to minimize the variability between regions.</p>\n<h4><a id=\"Multiregion_architecture_23\"></a><strong>Multi-region architecture</strong></h4>\n<p>The code included in this post creates two proof-of-concept <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> clusters and the supporting infrastructure in two AWS Regions.</p>\n<p><img src=\"https://dev-media.amazoncloud.cn/080af6defe8e43da9e9c5fd44c60da2a_image.png\" alt=\"image.png\" /></p>\n<p><a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> also creates a stack with CodePipeline to build a container image from the sample application’s source code. Users trigger an automated deployment to the primary AWS Region by checking in code to the <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">AWS CodeCommit</a> Git repository <a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> creates. CodePipeline triggers <a href=\"https://aws.amazon.com/codebuild/\" target=\"_blank\">AWS CodeBuild</a> when new changes are checked-in to the Git repository and the result is an image pushed to an <a href=\"https://aws.amazon.com/ecr/\" target=\"_blank\">Amazon Elastic Container Registry</a> (Amazon ECR) repository.</p>\n<p><img src=\"https://dev-media.amazoncloud.cn/a7548cbffae9425f90ea1066d31ca41f_image.png\" alt=\"image.png\" /></p>\n<p>The <a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> code this post provides doesn’t include multi-region traffic routing. Customers can add a Global Accelerator, as explained <a href=\"https://aws.amazon.com/blogs/containers/operating-a-multi-regional-stateless-application-using-amazon-eks/\" target=\"_blank\">in this post</a>, or use Domain Name System (DNS) to route traffic between regions.</p>\n<h3><a id=\"Solution_overview_34\"></a><strong>Solution overview</strong></h3>\n<h4><a id=\"Prerequisites_35\"></a><strong>Prerequisites</strong></h4>\n<p>You will need the following to complete the steps in this post:</p>\n<ul>\n<li><a href=\"https://docs.aws.amazon.com/cli/latest/userguide/install-cliv2.html\" target=\"_blank\">AWS CLI version 2</a></li>\n<li><a href=\"https://docs.aws.amazon.com/cdk/v2/guide/getting_started.html#getting_started_install\" target=\"_blank\">AWS CDK</a> version 2.19.0 or later</li>\n<li><a href=\"https://nodejs.org/en/download/current/\" target=\"_blank\">Node</a> version 17.8.0 or later</li>\n<li><a href=\"https://docs.npmjs.com/downloading-and-installing-node-js-and-npm\" target=\"_blank\">NPM</a> version 8.5.5 or later</li>\n<li><a href=\"https://docs.aws.amazon.com/eks/latest/userguide/install-kubectl.html\" target=\"_blank\">Kubectl</a></li>\n<li><a href=\"https://docs.docker.com/get-docker/\" target=\"_blank\">Docker-cli</a></li>\n<li><a href=\"https://git-scm.com/download\" target=\"_blank\">Git</a></li>\n<li><a href=\"https://stedolan.github.io/jq/download/\" target=\"_blank\">Jq</a></li>\n</ul>\n<p>Let’s start by setting a few environment variables:</p>\n<pre><code class=\"lang-\">ACCOUNT_ID=$(aws sts get-caller-identity --query 'Account' --output text)\nAWS_PRIMARY_REGION=us-east-2\nAWS_SECONDARY_REGION=eu-west-2\n</code></pre>\n<p>Clone the sample repository and install dependency packages. This repository contains CDK v2 code written in TypeScript.</p>\n<pre><code class=\"lang-\">git clone https://github.com/aws-samples/containers-blog-maelstrom\ncd containers-blog-maelstrom/aws-cdk-eks-multi-region-skeleton/\nnpm install\n</code></pre>\n<p>The <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster definition is stored in <code>lib/cluster-stack.ts</code>. The following is the snippet of the code:</p>\n<pre><code class=\"lang-\"> const cluster = new eks.Cluster(this, 'demoeks--cluster', {\n clusterName: `demoeks`,\n mastersRole: clusterAdmin,\n version: eks.KubernetesVersion.V1_21,\n defaultCapacity: 2,\n defaultCapacityInstance: new ec2.InstanceType(props.onDemandInstanceType)\n });\n</code></pre>\n<p>As defined in <code>bin/multi-cluster-ts.ts</code>, the code creates two clusters and deploys the supporting infrastructure, the</p>\n<p>continuous integration and continuous delivery/continuous deployment (CI/CD) stack (CodePipeline, <a href=\"https://aws.amazon.com/codebuild/\" target=\"_blank\">CodeBuild</a>, and <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a>), <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> clusters, and a sample web application.</p>\n<p>You can customize the <code>multi-cluster-ts.ts</code> file to change regions and instance types based on your requirements.</p>\n<pre><code class=\"lang-\">const primaryRegion = {account: account, region: 'us-east-2'};\nconst secondaryRegion = {account: account, region: 'eu-west-2'};\nconst primaryOnDemandInstanceType = 'r5.2xlarge';\nconst secondaryOnDemandInstanceType = 'm5.2xlarge';\n</code></pre>\n<h4><a id=\"Bootstrap_AWS_Regions_90\"></a><strong>Bootstrap AWS Regions</strong></h4>\n<p>The first step to any <a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> deployment is bootstrapping the environment. <code>cdk bootstrap</code> is a tool in the AWS CDK command-line interface (<a href=\"https://docs.aws.amazon.com/cli/?id=docs_gateway\" target=\"_blank\">AWS CLI</a>) responsible for preparing the environment (i.e., a combination of AWS account and AWS Region) with resources required by CDK to perform deployments into that environment. If you already use <a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> in a region, you don’t need to repeat the bootstrapping process.</p>\n<p>Execute the commands below to bootstrap the AWS environment in <code>us-east-2</code> and <code>eu-west-2</code>:</p>\n<pre><code class=\"lang-\">cdk bootstrap aws://$ACCOUNT_ID/$AWS_PRIMARY_REGION\ncdk bootstrap aws://$ACCOUNT_ID/$AWS_SECONDARY_REGION\n</code></pre>\n<p>The CDK code creates five stacks:</p>\n<ul>\n<li>A <code>cluster-stack</code> that creates an <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster for primary region <code>us-east-2</code></li>\n<li>A <code>container-stack</code> that deploys sample nginx containers to <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster for primary region <code>us-east-2</code></li>\n<li>A <code>cluster-stack</code> that creates an <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster for secondary region <code>eu-west-2</code></li>\n<li>A <code>container-stack</code> that deploys sample nginx containers to <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster or secondary region <code>eu-west-2</code></li>\n<li>A <code>cicd-stack</code> that creates a cicd pipeline using <a href=\"https://docs.aws.amazon.com/codepipeline/?id=docs_gateway\" target=\"_blank\">AWS CodePipeline</a> and <a href=\"https://aws.amazon.com/products/developer-tools/\" target=\"_blank\">AWS CodeSuites</a> to containerize and deploy a sample flask application that crosses regions of <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a></li>\n</ul>\n<p>Run <code>cdk list</code> to see the list of the stacks to be created:</p>\n<pre><code class=\"lang-\">$ cdk list\nClusterStack-us-east-2\nClusterStack-eu-west-2\nContainerStack-us-east-2\nContainerStack-eu-west-2\nCicdStack\n</code></pre>\n<p>You can use <code>cdk diff</code> to get a list of resources <a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> will create:</p>\n<p><code>cdk diff</code></p>\n<p><em>If the result doesn’t match the output below, then refer to <a href=\"https://catalog.us-east-1.prod.workshops.aws/workshops/c15012ac-d05d-46b1-8a4a-205e7c9d93c9/en-US/40-deploy-clusters/100-clone-base-repo/\" target=\"_blank\">this step</a> and check if npm run watch is running in the background.</em></p>\n<p>You should see the following output:</p>\n<pre><code class=\"lang-\">Stack ClusterStack-ap-southeast-2IAM Statement Changes\n</code></pre>\n<pre><code class=\"lang-\">┌───┬────────────────────────┬────────┬────────────────────────┬────────────────────────┬───────────┐\n│ │ Resource │ Effect │ Action │ Principal │ Condition │\n├───┼────────────────────────┼────────┼────────────────────────┼────────────────────────┼───────────┤\n│ + │ ${AdminRole.Arn} │ Allow │ sts\\:AssumeRole │ AWS\\:arn:${AWS::Partiti│ │\n│ │ │ │ │ on}\\:iam::<<ACCOUNT_ID>│ │\n│ │ │ │ │ root │ │\n...\n\n...\nResources\n[+] AWS::IAM::Role AdminRole AdminRole38563C57 \n[+] AWS::EC2::VPC demogo-cluster/DefaultVpc demogoclusterDefaultVpc0F0EA8D6 \n[+] AWS::EC2::Subnet demogo-cluster/DefaultVpc/PublicSubnet1/Subnet demogoclusterDefaultVpcPublicSubnet1Subnet9B5D84CC \n[+] AWS::EC2::RouteTable demogo-cluster/DefaultVpc/PublicSubnet1/RouteTable demogoclusterDefaultVpcPublicSubnet1RouteTableA9719167 \n[+] AWS::EC2::SubnetRouteTableAssociation demogo-cluster/DefaultVpc/PublicSubnet1/RouteTableAssociation demogoclusterDefaultVpcPublicSubnet1RouteTableAssociationF6BCC682 \n[+] AWS::EC2::Route demogo-cluster/DefaultVpc/PublicSubnet1/DefaultRoute demogoclusterDefaultVpcPublicSubnet1DefaultRoute0A0FDBF1 \n[+] AWS::EC2::EIP demogo-cluster/DefaultVpc/PublicSubnet1/EIP demogoclusterDefaultVpcPublicSubnet1EIP42D57092 \n[+] AWS::EC2::NatGateway demogo-cluster/DefaultVpc/PublicSubnet1/NATGateway\n</code></pre>\n<p>The code supports region-specific customization. It looks for Kubernetes deployment files in region-specific directories. The sample code contains directories for primary and secondary regions (called <code>yaml-us-east-2</code> and <code>yaml-eu-west-2</code>). <a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> will apply the deployment files in these folders to deploy the Kubernetes application in the two regions.</p>\n<p>If your regions differ, then please create directories manually and copy the Kubernetes deployment manifest in each directory:</p>\n<pre><code class=\"lang-\"># mkdir yaml-<region-name> for primary and secondary regions\nmkdir yaml-$AWS_PRIMARY_REGION\nmkdir yaml-$AWS_SECONDARY_REGION\n\ncp yaml-us-east-2/00_ap_nginx.yaml yaml-$AWS_PRIMARY_REGION/\ncp yaml-us-east-2/00_ap_nginx.yaml yaml-$AWS_SECONDARY_REGION/\n</code></pre>\n<h4><a id=\"Create_the_clusters_165\"></a><strong>Create the clusters</strong></h4>\n<p>If you are following the tutorial using different AWS Regions than the ones used in the post, please update the values of primary and secondary region in <code>bin/multi-cluster-ts.ts</code> before proceeding.</p>\n<p>The <code>cdk deploy</code> subcommand deploys the specified stack(s) to your AWS account. Let’s deploy all five stacks:</p>\n<pre><code class=\"lang-\">cdk deploy "*"\n</code></pre>\n<p>Upon completion, you can check the pipeline in the <a href=\"http://console.aws.amazon.com/codesuite/codepipeline/home\" target=\"_blank\">CodePipeline Console</a>. There is no master branch in <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> yet, thus the pipeline will be on a failed status.</p>\n<p><a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> creates the following resources:</p>\n<ul>\n<li><a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> clusters in two regions and the required infrastructure including VPCs, 3x public and private subnets, Internet Gateways, Network Address Translation (NAT) Gateways.</li>\n<li><a href=\"https://docs.aws.amazon.com/ec2/?id=docs_gateway\" target=\"_blank\">Amazon EC2</a> instances: r5.2xlarge, mv5.2xlarge</li>\n<li>CI/CD Pipeline to deploy sample application to <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> clusters in <code>us-east-2</code> and <code>eu-west-2</code></li>\n<li><a href=\"https://docs.aws.amazon.com/ecr/?id=docs_gateway\" target=\"_blank\">Amazon ECR</a> Repository in both to store sample application container images</li>\n<li>A Kubernetes namespace in each cluster</li>\n</ul>\n<h4><a id=\"Deploy_sample_app_to_multiple_regions_185\"></a><strong>Deploy sample app to multiple regions</strong></h4>\n<p>Navigate to sample application in the cloned repo and register the created <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> repository in the application project with the following commands:</p>\n<pre><code class=\"lang-\">mkdir ../../sample-app\ncp -R ../aws-cdk-multi-region-sample-app ../../sample-app\ncd ../../sample-app/aws-cdk-multi-region-sample-app\nEKS_CDK_CODECOMMIT_REPO=$(aws codecommit list-repositories --region $AWS_PRIMARY_REGION --query "repositories[?starts_with(repositoryName,'hello-py')].repositoryName" --output text)\nEKS_CDK_CODECOMMIT_REPO_URL=$(aws codecommit get-repository --region $AWS_PRIMARY_REGION --repository-name $EKS_CDK_CODECOMMIT_REPO --query'repositoryMetadata.cloneUrlHttp' --output text)\ngit init\ngit remote add codecommit $EKS_CDK_CODECOMMIT_REPO_URL\n</code></pre>\n<p>Before checking code into the <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> repository, we have to configure the Git client with <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> credentials. This step requires using the <a href=\"https://docs.aws.amazon.com/awsconsolehelpdocs/latest/gsg/learn-whats-new.html\" target=\"_blank\">AWS Management Console</a>.</p>\n<p>Navigate to the AWS Identity & Access Management <a href=\"https://console.aws.amazon.com/iam/home?region=us-west-2\" target=\"_blank\">(IAM) User</a> console and create the HTTPS Git credentials for AWS <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> for the IAM user your terminal is currently configured to use. For more information, please refer to <a href=\"https://docs.aws.amazon.com/codecommit/latest/userguide/setting-up-gc.html?icmpid=docs_acc_console_connect_np\" target=\"_blank\">AWS documentation</a>:</p>\n<p>Push the code in the directory to <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> with the following command:</p>\n<pre><code class=\"lang-\">git add .\ngit commit -am "initial commit"\ngit push codecommit master\n</code></pre>\n<p>If this is your first time using <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a>, you will be prompted for username and password to push code to the <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> Git repository. You can use the <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> credentials downloaded in the prior step.</p>\n<p>After you push to the repository, you can see in the CodePipeline console that the pipeline is triggered as follows:</p>\n<p><img src=\"https://dev-media.amazoncloud.cn/9442008aea36403e968382109ed1ce34_image.png\" alt=\"image.png\" /></p>\n<p>Once the pipeline deploys the sample application to the <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster in the primary region, it waits for a manual review before proceeding. Let’s check the deployment’s health in the primary cluster. Configure your kubectl to use your primary cluster:</p>\n<pre><code class=\"lang-\">CLUSTER1_KUBECONFIG_COMMAND=$(aws cloudformation describe-stacks \\\n --stack-name "ClusterStack-$AWS_PRIMARY_REGION" \\\n --region $AWS_PRIMARY_REGION \\\n --query 'Stacks[0].Outputs[?starts_with(OutputKey,`demoeksclusterConfig`)].OutputValue' \\\n --output text)\n$(echo $CLUSTER1_KUBECONFIG_COMMAND)\n</code></pre>\n<p>See running services:</p>\n<pre><code class=\"lang-\">kubectl get svc\n</code></pre>\n<p><img src=\"https://dev-media.amazoncloud.cn/54d5991f9ada483997d94134818168e5_image.png\" alt=\"image.png\" /></p>\n<p>Let’s curl the hello-py service:</p>\n<pre><code class=\"lang-\">curl $(kubectl get service hello-py -o \\\n jsonpath='{.status.loadBalancer.ingress[*].hostname}') && echo ""\n</code></pre>\n<p><img src=\"https://dev-media.amazoncloud.cn/eeee7b2b511540caa9211ad33a21a5a0_image.png\" alt=\"image.png\" /></p>\n<p>The service responds with a <strong>Hello World from <AWS Region></strong> message to the curl request. Since the service is functional in the primary region, this deployment is assumed successful. As a result, the pipeline can proceed with deploying the application into the primary region.</p>\n<p>Before we resume the pipeline, let’s verify that <code>hello-py</code> service doesn’t exist in the secondary Region.</p>\n<p>Set your <code>kubectl's</code> current context to the <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster in the secondary region:</p>\n<pre><code class=\"lang-\">CLUSTER2_KUBECONFIG_COMMAND=$(aws cloudformation describe-stacks \\\n --stack-name "ClusterStack-$AWS_SECONDARY_REGION" \\\n --region $AWS_SECONDARY_REGION \\\n --query 'Stacks[0].Outputs[?starts_with(OutputKey,`demoeksclusterConfig`)].OutputValue' \\\n --output text)\n$(echo $CLUSTER2_KUBECONFIG_COMMAND)\n</code></pre>\n<p>Verify that <code>hello-py</code> service is not running in the secondary cluster:</p>\n<pre><code class=\"lang-\">kubectl get service hello-py\n</code></pre>\n<p>You should get an error:</p>\n<pre><code class=\"lang-\">Error from server (NotFound): services "hello-py" not found\n</code></pre>\n<p>Now we know there is no <code>hello-py</code> service in the secondary region. Let’s return to CodePipeline and approve the manual step.</p>\n<p>Return to the CodePipeline <a href=\"https://signin.aws.amazon.com/signin?redirect_uri=https%3A%2F%2Fus-east-1.console.aws.amazon.com%2Fiam%2Fhome%3Fregion%3Dus-west-2%26state%3DhashArgs%2523%26isauthcode%3Dtrue&client_id=arn%3Aaws%3Aiam%3A%3A015428540659%3Auser%2Fiam&forceMobileApp=0&code_challenge=zBijXuawh90K_1v8K8Xn7UXlGgTnItRRKDVxhlrnFSo&code_challenge_method=SHA-256\" target=\"_blank\">AWS Management Console</a> and select Review and select Approve in the next prompt. CodePipeline API <a href=\"https://docs.aws.amazon.com/codepipeline/latest/APIReference/API_PutApprovalResult.html\" target=\"_blank\">PutApprovalResult</a> can be used to automate this approval process.</p>\n<p><img src=\"https://dev-media.amazoncloud.cn/c22637226d2b4aa5802d06707b7e8a54_image.png\" alt=\"image.png\" /></p>\n<p><img src=\"https://dev-media.amazoncloud.cn/8b8d641a22db45beb17cac9841897f0d_image.png\" alt=\"image.png\" /></p>\n<p>When the pipeline gets the approval to deploy to the secondary region, it applies the deployment manifest (<code>app-deployment.yaml</code>) to the second <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster.</p>\n<p>Once the pipeline finishes successfully, check if <code>hello-py</code> service is functional:</p>\n<pre><code class=\"lang-\">kubectl get service hello-py\n</code></pre>\n<p>Once the service has an application load balancer (ALB) attached, then send a request to verify that the service responds with the secondary region’s name in the response.</p>\n<pre><code class=\"lang-\">curl $(kubectl get service hello-py -o \\\n jsonpath='{.status.loadBalancer.ingress[*].hostname}') && echo ""\n</code></pre>\n<p>This completes deployment process. hello-py service is deployed and functional in both regions.</p>\n<h4><a id=\"Upgrades_300\"></a><strong>Upgrades</strong></h4>\n<p>As new code is checked-in to the <a href=\"https://aws.amazon.com/codecommit/\" target=\"_blank\">CodeCommit</a> repository, the pipeline will automatically attempt to create a new deployments. When operators verify the changes are successful in the primary region, they can approve the deployment to the secondary region.</p>\n<p>The file lib/cluster-stack.ts contains the declaration of your <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster settings. You can modify it to control cluster properties and upgrade clusters.</p>\n<p>As you onboard newer services into these clusters, you can replicate the CI/CD stack created in this post for each service.</p>\n<h3><a id=\"Cleanup_307\"></a><strong>Cleanup</strong></h3>\n<p>You continue to incur cost until deleting the infrastructure that you created for this post. Use the following manual process and commands to clean up the created AWS resources for this post:</p>\n<pre><code class=\"lang-\">$(echo $CLUSTER1_KUBECONFIG_COMMAND)\nkubectl delete svc hello-py\n$(echo $CLUSTER2_KUBECONFIG_COMMAND)\nkubectl delete svc hello-py\ncd ../../containers-blog-maelstrom/aws-cdk-eks-multi-region-skeleton\nAWS_ECR_REPO=$(aws ecr describe-repositories --query "repositories[].[repositoryName]" --region $AWS_PRIMARY_REGION | grep 'cicdstack-ecrforhellopy' | sed -e 's/^[[:space:]]*//' | sed -e 's/^"//' -e 's/"$//')\nAWS_ECR_IMAGES_TO_DELETE=$(aws ecr list-images --region $AWS_PRIMARY_REGION --repository-name $AWS_ECR_REPO --query 'imageIds[*]' --output json )\naws ecr batch-delete-image --region $AWS_PRIMARY_REGION --repository-name $AWS_ECR_REPO --image-ids "$AWS_ECR_IMAGES_TO_DELETE" || true\ncdk destroy "*" \n</code></pre>\n<p>CDK will ask you <code>Are you sure you want to delete: CicdStack, ContainerStack-<primary region>, ContainerStack-<secondary region>, ClusterStack-<primary region>, ClusterStack<secondary region> (y/n)?</code> and enter <code>y</code> to delete.</p>\n<h3><a id=\"Conclusion_325\"></a><strong>Conclusion</strong></h3>\n<p>This post demonstrated the procedure to create continuous deployment pipeline to deploy applications in multiple <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> clusters running in different regions. The accompanying CDK code creates EKS clusters and the CI/CD stack to continuously deploy application to the clusters.</p>\n<p><a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> helps you define your cloud infrastructure, Kubernetes resources, and application in the same codebase. If your codebase is in a language that <a href=\"https://aws.amazon.com/cdk/\" target=\"_blank\">CDK</a> supports, your applications, configuration, and the underlying infrastructure can be coded in the same language. If you must generate Kubernetes manifests using code, you can also use <a href=\"https://cdk8s.io/\" target=\"_blank\">cdk8s</a> to define Kubernetes resource programmatically.</p>\n<p>As a next step, you can also explore <a href=\"https://aws.amazon.com/blogs/containers/bootstrapping-clusters-with-eks-blueprints/\" target=\"_blank\">EKS Blueprints</a>, which is a collection of IaC modules, to help you configure and deploy consistent and batteries-included <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> clusters across accounts and Regions. You can use <a href=\"https://aws.amazon.com/blogs/containers/bootstrapping-clusters-with-eks-blueprints/\" target=\"_blank\">EKS Blueprints</a> to easily bootstrap an <a href=\"https://aws.amazon.com/eks/\" target=\"_blank\">Amazon EKS</a> cluster with <a href=\"https://docs.aws.amazon.com/eks/latest/userguide/eks-add-ons.html\" target=\"_blank\">Amazon EKS add-ons</a>. A wide range of popular open-source add-ons, including <a href=\"https://prometheus.io/\" target=\"_blank\">Prometheus</a>, <a href=\"https://karpenter.sh/\" target=\"_blank\">Karpenter</a>, <a href=\"https://www.nginx.com/\" target=\"_blank\">Nginx</a>, <a href=\"https://traefik.io/\" target=\"_blank\">Traefik</a>, <a href=\"https://docs.aws.amazon.com/eks/latest/userguide/aws-load-balancer-controller.html\" target=\"_blank\">AWS Load Balancer Controller</a>, <a href=\"https://fluentbit.io/\" target=\"_blank\">Fluentbit</a>, <a href=\"https://keda.sh/\" target=\"_blank\">Keda</a>, <a href=\"https://argoproj.github.io/cd/\" target=\"_blank\">Argo CD</a>, and more.</p>\n<p>This post is also available as a <a href=\"https://catalog.us-east-1.prod.workshops.aws/workshops/c15012ac-d05d-46b1-8a4a-205e7c9d93c9/en-US/\" target=\"_blank\">workshop</a> that uses CDK v1. You can find the changes between AWS CDK v1 and v2 <a href=\"https://docs.aws.amazon.com/cdk/v2/guide/migrating-v2.html\" target=\"_blank\">here</a>.</p>\n<p><img src=\"https://dev-media.amazoncloud.cn/f7b52a0f5dd54febb5d01070a29cdf5f_image.png\" alt=\"image.png\" /></p>\n<p><strong>Elamaran Shanmugam</strong></p>\n<p>Elamaran (Ela) Shanmugam is a Sr. Partner Container Specialist Solutions Architect with Amazon Web Services. Ela is a Container, Observability and Multi-Account Architecture SME and helps AWS customers to design and build scalable, secure and optimized container workloads on AWS. His passion is building and automating Infrastructure to allow customers to focus more on their business. He is based out of Tampa, Florida and you can reach him on twitter @IamElaShan</p>\n<p><img src=\"\" alt=\"8a5f26145878487abb6cf354fa9ca388_image1.png\" rel=\"1\" /><br />\n<strong>Re Alvarez-Parmar</strong></p>\n<p>Re Alvarez-Parmar is a Container Specialist Solutions Architect at Amazon Web Services. He helps customers use AWS container services to design scalable and secure applications. He is based out of Seattle and uses Twitter, sparingly, @realz</p>\n"}