AWS Cloud Operations Blog
Application configuration deployment to container workloads using AWS AppConfig
UPDATE (15 Dec 22): AWS AppConfig released an Agent for containers (EKS, ECS, Docker, Kubernetes) in December 2022, which makes calling AppConfig much simpler from containerized applications. We recommend using the AppConfig Agent for containers instead of the method below. Read the Agent documentation.
AWS AppConfig is a capability of AWS Systems Manager that you can use to create, manage, and quickly deploy application configurations at runtime. With AWS AppConfig, you can validate your configuration data to check for any errors, and define deployment strategies to control the rate at which deployments occur. You can also set monitors to watch for any alarms during deployments. In case an error occurs, AWS AppConfig rolls back the deployment to the previous version to avoid application outages.
You can use AWS AppConfig to roll out application configurations across applications hosted on Amazon Elastic Compute Cloud (Amazon EC2) instances, containers, AWS Lambda, mobile apps, IoT devices, and on-premises servers in a validated, controlled, and monitored way.
In the last two posts, we showed you how to use AWS AppConfig in a serverless environment using an AWS AppConfig Lambda extension. We also showed you how to deploy configurations across environments using the AWS CodePipeline integration with AWS AppConfig. In this post, we explore how to use AWS AppConfig to deploy application configurations to complex containerized applications.
Specifically, we cover how to:
- Separate application configuration from application code for a containerized application.
- Use AWS AppConfig to manage and deploy the application configuration.
- Automate and efficiently manage application configurations in a containerized application.
Customers often use Amazon Elastic Container Service (Amazon ECS) or Amazon Elastic Kubernetes Service (Amazon EKS) to manage their application configurations running in container applications. A common practice is to either bake the configurations in the container image, or through the use of environment variables. Customers also retrieve the configuration dynamically using solutions like external key-value stores, Kubernetes ConfigMaps, or AWS Systems Manager Parameter Store. Managing application configuration externally not only allows you to store complex and dynamic configurations, but it allows makes the versioning of the configuration easier. Decoupling application configuration from code also means you can deploy changes without restarting the application or taking the application out of service. This is how AWS AppConfig improves configuration management.
AWS AppConfig lets you validate the configuration before you deploy it to make sure it is free of errors. As soon as the configuration is deployed, it is immediately available to the application and the updates can be consumed by the application at runtime. AWS AppConfig also allows you to use feedback loops using Amazon CloudWatch. It enables controlled rolling updates and automatic rollbacks of new configurations without degrading the application.
AWS AppConfig also provides rollout controls so you can apply advanced deployment techniques like feature toggling for fast, controlled launches of new features.
Solution architecture
data:image/s3,"s3://crabby-images/d0647/d0647262dbe901da5ef8464b8228db36ee412b4c" alt="Architecture shows how a user interacts with a running task in an AWS Fargate container in a VPC/subnet and how it fetches the application configuration from AWS AppConfig."
Figure 1: Architecture of the proposed solution
- To illustrate the integration between containers and AWS AppConfig, we work with a simple Java microservices application. It has a REST API that returns a limited number of movies based on the feature parameter and number of movies to be returned. This application is containerized, pushed to Amazon Elastic Container Registry (Amazon ECR), and deployed to AWS Fargate.
- The feature parameter and number of movies limit are stored as an AWS AppConfig hosted configuration in AWS AppConfig. The application code reads the application configuration from AWS AppConfig using AWS SDK API operations.
- The application has a built-in caching layer to cache the responses from AWS AppConfig. Subsequent calls to fetch the configuration check the cache first and return the response from the cache. If the cache is empty, the application calls the AWS AppConfig API to fetch the configuration. The cache expiry is based on the TTL set in the properties.
- We then change the movie limit in AWS AppConfig and deploy the configuration so that it is available for the container application to read. When we call the REST API to retrieve the list of movies, only the updated number of movies are returned based on the new number of results defined in the AWS AppConfig configuration profile.
Prerequisites
- The AWS CLI installed and configured. Use version 1.7 or later.
- IAM permissions to create resources in AWS AppConfig, Amazon ECR, and AWS Fargate.
- Familiarity with Java, Docker, containers, Amazon ECR, and Amazon ECS.
For the purposes of this post, we deployed a sample Java application in AWS Fargate. For additional information, check the prescriptive guidance for deploying Java microservices to AWS Fargate.
To implement this solution, we will:
- Create the application, environments, and configuration profile in AWS AppConfig.
- Set up the base application with Amazon ECS and Amazon ECR and the associated network components with AWS CloudFormation.
- Clone the code repository, create a Docker container, and publish to Amazon ECR.
- Create a Fargate task and deploy the container application into Amazon ECS on AWS Fargate using AWS CloudFormation.
- Verify the deployed application, update the AppConfig configuration data, and deploy the updated configuration.
Create application, environments, and configuration profile in AWS AppConfig
- Open the AWS Systems Manager console.
- In the left navigation pane, choose AWS AppConfig.
- If the AWS AppConfig welcome page appears, click Create configuration data. Otherwise, click Create application.
- For Name, enter a name for the application. (
MyContainerApplication
) You can add an optional description and apply tags to the application. Choose Create application.
data:image/s3,"s3://crabby-images/30222/30222a00baf86bfa30ea9833a27f362540b2b57e" alt="The Create application page provides text boxes for the application name and description. It also includes a Tags section."
Figure 2: Create application in AWS AppConfig
- After the application is created, you are directed to a page with Environments and Configuration Profiles Choose Create environment, and then enter a name (
MyContainerApplicationProductionEnvironment
) and optional description for the environment. You can also optionally add tags and configure Amazon CloudWatch alarms for this environment. - In the top navigation, choose the application name, and on the Configuration Profiles tab, choose Create configuration profile.
- Enter a name (
MyContainerApplicationConfigurationProfile
) and optional description for the configuration profile. - Under Configuration source, choose AWS AppConfig hosted configuration.Under Content, choose JSON, paste the following content, and then choose Next.
Note: As we see in this snippet, we are controlling the number of movies that we display through the intItemLimit
. We are controlling if that configuration is enabled or disabled through the boolEnableFeature
. As a default, we keep this configuration enabled but limit the movies to 5
.
data:image/s3,"s3://crabby-images/4efe2/4efe2c8cfe247cf0ba2a37e37731fab1ed601fe2" alt="Build configuration profile provides fields for the user to enter a configuration profile name, optional description, and configuration sources. In this example, for Configuration source provider, AWS AppConfig hosted configuration is selected. For the hosted configuration content, JSON is selected."
Figure 3: Build configuration profile in AWS AppConfig
- (Optional) You can add validators to validate the configuration. For information, check about validators in the AWS AppConfig documentation.
- Choose Create configuration profile.
- Choose Start deployment.
- Choose the environment, hosted configuration version, deployment strategy, and an optional description to start the deployment process.
- To create a custom deployment strategy, choose Create Deployment Strategy. Or choose one of the predefined deployment strategies provided by AWS AppConfig. For more information, check creating a deployment strategy in the AWS AppConfig documentation. For the purposes of this post, we chose the
AppConfig.Linear50PercentEvery30Seconds
deployment strategy.
data:image/s3,"s3://crabby-images/f811c/f811cb8be01caba6de1a410b47bf417717aba916" alt="Start deployment provides boxes for entering the environment to deploy to, hosted configuration version, deployment strategy, and an optional deployment description"
Figure 4: Deployment details in AWS AppConfig
Note: Depending on the deployment strategy you selected, this operation might take few minutes to complete. The configuration is available to the application as soon as the deployment state is Complete.
data:image/s3,"s3://crabby-images/849b9/849b948a2a1c18077f566bf87050a9fd94d346ed" alt="The deployment details page includes sections for deployment status, deployment details, monitors, and deployment strategy"
Figure 5: Completed deployment in AWS AppConfig
Set up the base application with Amazon ECS and Amazon ECR and associated network components using AWS CloudFormation
- We are using AWS CloudFormation to set up the base application. This includes an Amazon Elastic Container Service cluster, Amazon Elastic Container Registry, IAM roles, and required networking components like a VPC, subnets, and routes.
- Use the following code block to save a template as a local file (
.yml
file extension) on your computer.
- Open the AWS CloudFormation console, and choose Create stack.
- On the Specify template page, choose Upload a template file, and choose the file you just saved. Choose Next.
- On the Specify stack details page, enter a name for the stack (for example,
ECSCluster-dev
). - In the Parameters section, use
dev
as the value for the Environment and choose Next. - (Optional) Define your tags, and then choose Next.
- On the Review page, select the check box in the Capabilities section, and then choose Create stack.
Wait until the status displayed for the stacks is Create Complete.
Clone the code repository, create a Docker container, and publish to Amazon ECR
- Clone the code from the GitHub repository.
- Navigate to Amazon Elastic Container Registry console , click on the repository that you created and click View push commands.
- Navigate to the code repository in the command prompt and execute the push commands to upload the project.
data:image/s3,"s3://crabby-images/e85d5/e85d5dba81d281ff89f4ce94d8dbb96aa6b2e42d" alt="Push commands into Amazon ECR specify the AWS CLI commands to sign in to the Amazon ECR repository, build the Docker image, tag the image, and push the image into the Amazon ECR repository."
Figure 6: Push commands for the Docker image in Amazon ECR
Note: This operation might take few minutes to complete. When it is complete, a success message appears.
- When the upload is complete, copy the URL of the image in the repository. Use this URL as a parameter (
ImageUrl
) to the AWS CloudFormation template mentioned in the next section.
Create a Fargate task and deploy the container application into Amazon ECS on AWS Fargate using AWS CloudFormation
We are using AWS CloudFormation to set up the application container. It includes Amazon ECS, an AWS Fargate task definition, and a load balancer in front of Amazon ECS.
- Use the following code block to save a template as a local file (
.yml
file extension) on your computer.
- Open the AWS CloudFormation console, and choose Create stack.
- On the Specify template page, choose Upload template file, and choose the file you just saved. Choose Next.
- On the Specify stack details page, enter a name for the stack (for example,
fargate-task-dev
). - In the Parameters section, Use the image URL from the previous step for the ImageUrl parameter and use
dev
as the value for the Environment. Leave the rest of the parameters at their defaults. Choose Next. - (Optional) Define your tags, and then choose Next.
- On the Review page, select the check box in the Capabilities section, and choose Create stack.
Wait until the status displayed for the stacks is Create Complete.
Verify the deployed application, update the AppConfig configuration data, and deploy the updated configuration
- In the AWS CloudFormation console,open the fargate-task-dev stack you created
- Choose Outputs, and then copy the external URL for the load balancer.
- Use the
ExternalUrl
for the load balancer to verify the application.
http://ExternalUrl/movies/getMovies
data:image/s3,"s3://crabby-images/8991c/8991c869d7f7f84a529576069d062ca37f859c65" alt="The success response from the application displays the movie ID and movie name based on the number of movies set in the application configuration"
Figure 7: Success response from the application
Next, change the configuration value in the AWS AppConfig configuration to see how it is reflected in the container application.
- Open the AWS AppConfig console, and choose your application.
- On the Configuration Profiles tab, choose the configuration profile you created.
- Under Hosted configuration versions, choose Create.
- Edit the configuration value and choose Create hosted configuration version.
data:image/s3,"s3://crabby-images/84a8c/84a8c08acb2fac1024281434f618d4258729c570" alt="Create hosted configuration version provides options for the user to specify content in JSON, YAML, or plaintext"
Figure 8: Create hosted configuration version
- Choose Start Deployment, and then choose the environment, latest hosted configuration version, deployment strategy, and an optional description to start the deployment process.
- After the deployment is complete, visit the application URL again to check that the changes reflected immediately.
data:image/s3,"s3://crabby-images/83879/838799e652eed8384045df170f120398317a9d83" alt="Updated success response from the application displays the movie ID and movie name based on the number of movies changed in the application configuration."
Figure 9: Successful response from the application
Best practices
AWS AppConfig uses the value of the ClientConfigurationVersion parameter to identify the configuration version on your clients. If you don’t send ClientConfigurationVersion with each call to GetConfiguration, your clients receive the current configuration. You are charged each time your clients receive a configuration. To avoid excessive charges, we recommend that you include the ClientConfigurationVersion value with every call to GetConfiguration. This value must be saved on your client. Subsequent calls to GetConfiguration must pass this value by using the ClientConfigurationVersion parameter.
For more information, see AWS AppConfig best practices.
Cleanup
To avoid ongoing charges, delete the resources you created.
AWS AppConfig
In the AWS Systems Manager console, delete the following resources:
- Hosted configuration
- Configuration profile
- Environment
- Application
Base container application and Fargate task
- In the AWS CloudFormation console,choose the
fargate-task-dev
stack, and then choose Delete. - Choose the
ECSCluster-dev
stack, and then choose Delete.
Conclusion
In this post, we showed you how to integrate container applications with AWS AppConfig to retrieve and use an application configuration. AWS AppConfig not only makes application configuration management for your container workloads seamless. It also decouples configuration from your application code. These application configuration changes can be easily tracked outside of your software release cycle and are readily available for your containers to consume.
For more information, see the AWS AppConfig documentation.
About the authors
Luis Gómez is a Solutions Architect with AWS who works for the public sector in Spain. He has several years of experience building and operating cloud environments and applying DevOps practices. He works with customers to understand their goals and challenges and offers prescriptive guidance for achieving their objectives with AWS offerings.
Venugopalan Vasudevan is a Senior Technical Account Manager with AWS based in Denver, Colorado. Venu works with AWS customers to solve architectural, operational, and cost optimization challenges and help them build on AWS. In his spare time, he enjoys playing with his two kids.
Vinni Satija is a Product Manager for AWS AppConfig and is based in Washington D.C. She is passionate about taking a working backward approach with customers. She enjoys using technology to create solutions that address customer needs.