In our microservices deployment, we release several monolithic applications that interact with each other to achieve a specific goal. The experts at Heroku have formulated a set of 12 fundamental principles tailored for ensuring a successful microservices deployment.
1.Codebase:
Maintain a single codebase per service.
Codebase(project) can contain one single repository or multiple repositories
Our microservice should have a single codebase for each service tracked using version control, using a version control system, such as Git. This makes it easier to deploy and rollback changes easily to each individual service, and it also makes it easier to collaborate on the development of the microservice when there are changes that need to be made to the code, like a bugfix, feature, migration, and so on.
Keeping code for each service in a seperate repo also helps us in preventing entire system to break and find the root cause, and fix the issue much faster.
Under each service, we can have multiple deployment environments, such as
- dev
- staging
- production
2.Dependencies
Explicitly declare and isolate dependencies.
Our microservice should explicitly declare and isolate its dependencies meaning for each external dependency, we should keep the version that we used when we were developing the service and those dependencies should be isolated from the rest of the system.
For example:
- requests=version_no
- Flask==version_no
- --extra-index-url https://download.pytorch.org/whl/cpu
And each service should have its own version, like maybe one service requires one version of flask and another service might require another version of flask.
If we dont maintain the version numbers, that means, every developer in our project might install a different version, and it might break our codebase, because of issues such as backwards incompatibility etc.
3.Configuration
Store configurations outside the code.
We should store all the configuration in environment variables, for example .env, or secret manager etc, rather than in the code itself. This makes it easier to deploy the microservice to different environments, and it also makes it easier to change the microservice's configuration without having to deploy a new version of the code.
Also this helps us prevent releasing confidential information in repositories or to the public.
4.Backing Services
Treat services like databases as attached resources.
Supporting components and resources that our services depend on, such as databases, message queues, caching systems, and other external services are called backing services which can store data, communicate with other service, or perform specific tasks.
Our services act as main actors in our project, and these backing services are crucial because our services depend on these for storing data, or facilitating communication between services.
If any of these backing services become unavailable, it can disrupt overall performance. So, ensuring that these backing services are reliable, scalable, and easily accessible is crucial in the world of microservices to keep your applications running smoothly.
So, we should be able to point our app to another instance which is running these backing services.
5.Build, Release, Run
Separate build, release, and run stages.
We should seperate our services into build, release, and run stages. This means that the microservice should be built into a deployable artifact, such as a Docker image. This artifact can then be released to production and run in a variety of environments. This separation of stages makes it easier to automate the deployment and release of the microservice.
The build stage is where you take your codebase and turn it into a runnable artifact. This could be a Docker image, or something else. The important thing is that the artifact is self-contained and can be run on any environment without any additional dependencies.
The release stage is where you combine your build artifact with the configuration settings that your microservice needs to run. This configuration could include things like database connection strings, API keys, and environment variables. The release stage should also produce a unique identifier for each release. This will help you to track which versions of your microservices are running in production.
The run stage is where you actually start and run your microservice. This could be done by deploying it to a container orchestration platform like Kubernetes, or by running it directly on a server.
The important thing is that the build, release, and run stages are strictly separated. This means that you should be able to build, release, and run your microservices without making any changes to the codebase. This makes it easier to deploy new versions of your microservices and to roll back to previous versions if necessary.
For example, in our Dockerfile, we can have different names for build, release, run etc. and push this to Artifact registry.
6.Processes
Execute the application as stateless processes.
Our microservice should be stateless and should be executed as one or more identical processes. A microservice should not maintain any state between requests. This makes it easier to scale the microservice horizontally by adding more instances of the process.
Everything should be store in an external service, such as database or a cache.
7.Port Binding
Export services via a port and keep it self-contained.
Using port binding expose its services on a specific port. Other microservices can then communicate with the microservice by sending requests to that port. This makes it easy to discover and consume the microservice's services.
8.Concurrency
Scale horizontally by adding processes.
Our microservice should scale horizontally by scaling out its processes to handle more concurrent requests by adding more servers. This makes it easier to handle spikes in traffic and to scale the microservice to meet the needs of a growing.
Also this helps us reduce latency and achieve faster response time.
9.Disposability
Maximize robustness with fast startup and graceful shutdown.
All the services of our microservices should be disposable, meaning that it can be started, stopped, and restarted quickly and efficiently. This is important for microservices because they are often deployed and scaled dynamically.
10.Dev/Prod Parity
Keep development and production environments as similar as possible.
Having deployed same version of aall our services in both Dev and Prod environments will help us test changes/features/bugfixes easier and deploy to production. With the help of CI/CD tools, we can easily achive this.
11.Logs
Treat logs as event streams.
Our microservice should have structured logs that are easy to collect, search, and analyze which can help in debugging and troubleshooting problems with the microservice.
12.Admin Processes
Run admin tasks as one-off processes.
Our microservice should expose administrative interfaces for monitoring and managing the microservice. This is important for troubleshooting problems with the microservice and for scaling the microservice up and down.
References:
- https://12factor.net/