Kubernetes Adoption Journey - DigitalOcean
Kubernetes Adoption Journey - DigitalOcean
Kubernetes Adoption Journey - DigitalOcean
contents Development 8
CI/CD 15
Production 24
Scale 40
The decision to adopt Kubernetes shouldn’t be taken lightly. It’s a complex system that requires hard
work and dedication. In fact, when it comes to using Kubernetes, the best value comes from adopting a
DevOps culture. Kubernetes isn’t just installing something on a virtual machine (VM), it’s a transformative
way of viewing IT and enabling your teams. Automation is a direct path to becoming more productive.
Kubernetes can make every aspect of your applications and tools accessible to automation, including
role-based access controls, ports, databases, storage, networking, container builds, security scanning,
and more. Kubernetes gives teams a well-documented and proven system that they can use to automate
so that they can focus on outcomes.
Startups and small-to-medium-sized businesses (SMBs) are uniquely positioned in their journeys.
Unlike enterprises, they typically don’t have dedicated departments for operations or security, and they
usually run lean development teams. At the same time, SMBs are ultra-efficient, sometimes going from
discovery phases to production in a matter of two or three weeks. When startups, SMBs, and individual
developers consider adopting Kubernetes, they need to retain their ability to be flexible and pivot quickly
while benefiting from the scalability and automation provided by Kubernetes. To achieve both without
large and ongoing maintenance overhead, SMBs may consider the hands-free operations provided by
Managed Kubernetes offerings.
In this guide, you’ll find recommendations for every phase of the Kubernetes adoption journey.
From Discovery and Development through Staging, Production, and ultimately Scale, we’ll offer a simple
yet comprehensive approach to getting the most out of Kubernetes. We’ll walk you through some best
practices for any Kubernetes deployment, plus recommendations for tools and how to get started
using DigitalOcean Kubernetes, a managed Kubernetes offering.
If you would like to experience hands-on tutorials from day-1 to production, please follow the Kubernetes
Adoption Journey. It provides detailed step-by-step guidance on how to adopt Kubernetes, irrespective
of the phase of your development cycle.
If you are new to Kubernetes, we highly recommend the Kubernetes starter kit for
DigitalOcean. It provides a comprehensive tutorial and automation stack for
getting to production-ready Kubernetes.
Breaking down the
phases of adoption
Discovery
Explore setups, resources, and options for implementing
Kubernetes for your business.
Development
Deploy and test iteratively in a Kubernetes environment
without spending time in operational activities.
CI/CD
With continuous integration and continuous delivery,
set up an automated pipeline for building and
deploying application code to Kubernetes.
Production
Create a production-ready Kubernetes cluster.
Scale
Utilize more sophisticated tools to optimize Kubernetes
deployments for continued reliability and scale.
3
Discovery
Discover what you need to be successful with Kubernetes
During the Discovery phase of the Kubernetes adoption journey, teams will explore their options for
using Kubernetes and begin to plan their infrastructure on a block level or by grouping together one or
more infrastructure resources. The Discovery phase of the Kubernetes adoption journey is unique to you
and your business, and you’ll decide how your application will best map to a Kubernetes infrastructure.
With effective planning, you can move from Discovery to Production and Scale in as little as 4-6 weeks.
4
Plan infrastructure components
As your team begins to plan infrastructure inclusive of Kubernetes, find out what tech stack you need to
be most successful, both now and in the future. Understanding the reason you want to use Kubernetes
is key to deciding what types of components you’ll need to evaluate. Whether you’re a developer
interested in hassle-free Kubernetes management, a startup needing a faster time to market, or an SMB
that needs dynamic scaling capabilities, managed Kubernetes offerings like DigitalOcean Kubernetes can
help you achieve your goals.
Kubernetes installation
Consider if your team will use a managed or self-hosted version of Kubernetes. While there are times
when it makes sense to self-host Kubernetes, maintaining self-hosted clusters is time and resource
intensive and requires a level of expertise that many small businesses lack. Managed Kubernetes like
DigitalOcean Kubernetes bears the burden of managing the control plane, ensuring high availability
with the latest security patches and release updates, providing autoscaling capabilities, and more.
Businesses leveraging Managed Kubernetes often have a faster time to market.
“DigitalOcean Managed Kubernetes has totally changed the way I think about
deploying applications. In an ad push, we’d have to know ahead of time to spin up
servers and it could take a whole afternoon. Now, I can click a button and it’s done.”
Joshua Verdehem, Co-Founder, Loot.tv
Workload distribution
Do all workloads run on Kubernetes or are they distributed across Kubernetes, VMs, and other compute
platforms? For example, you may want to keep frontend running on Droplets while moving backend jobs
to Kubernetes and adding a load balancer and a managed MongoDB.
Kubernetes provides a consistently proven system across clouds that can lessen the technical debt
of multi-cloud environments. If you include Infrastructure as Code (IaC), tools like Terraform, Ansible,
Pulumi, and many more have further simplified provisioning across clouds, all of which can be fully
automated through DevOps workflows.
5
Sizing your cluster
An oversized cluster is inefficient with its resources and costs more, but an undersized cluster running
at full CPU or memory suffers performance issues and can crash under stress. Look at your node size
and count to determine the overall CPU, RAM, and storage of your cluster. Larger nodes are easier to
manage, in many cases are more cost-efficient, and can run more demanding applications; however,
they require more pod management and cause a larger impact if they fail.
Managed Databases
While a large business can usually afford to hire employees with a deep knowledge of databases, smaller
teams usually have fewer resources available. This makes tasks like replication, migrations, and backups
more difficult and time-consuming. Managed Databases handle common database administration tasks
such as setup, backups, and updates so you can focus more on your app and less on the database.
1-Click applications
Certain cloud providers have 1-click applications. For example, DigitalOcean Marketplace provides a
variety of apps and stacks that you can install to run pre-configured container images on Kubernetes
clusters. Most Kubernetes 1-Click apps are Helm charts under the hood and require Helm 3 package
manager to run.
6
Loot.tv
Loot.tv is an engaging video platform where creators and communities can earn rewards for uploading
and interacting with their favorite content. DigitalOcean’s low bandwidth costs and SLA guarantees
allow Loot.tv to effectively serve its customers while paying a fraction of the cost for bandwidth that
they would pay on hyperscalers. DigitalOcean Managed Kubernetes allows the Loot.tv team to focus on
building their application rather than maintaining infrastructure. DigitalOcean Kubernetes also provides
them with reliable scaling to match demand. Month to month Loot.tv can see three petabytes of data
transfer and average 8,000 viewers at any one time.
Cloudflare
Network
Amazon S3
for Nightly Backups
Digital Ocean
Load Balancer
Loot.tv MongoDB
Kubernetes Atlas Database
Cluster
Redis Cache
7
Development
Testing and deploying applications iteratively in a
Kubernetes environment
Prerequisites: Discovery is complete. You have identified the resources, tools and setup needed to
incorporate kubernetes for your application deployment.
Goals: Create a kubernetes cluster for development work. Install and configure necessary tools for
developer efficiency.
The Development phase of the Kubernetes adoption journey focuses on developer efficiency.
Significant operational complexities come into play from the time the developer commits their code
until it’s deployed in a Kubernetes cluster. Understanding an optimal developer workflow can help
teams build, test, and deploy faster, ultimately enabling the business to increase productivity and
innovate faster. In order to move into the Development phase, your team needs to have the basic
tooling for Kubernetes in place, and the application should be running in a development environment.
8
Development workflow in Kubernetes
A typical developer workflow consists of several cycles of an inner development loop and an outer
development loop. An Inner Loop includes coding, building, running, and testing the application. It takes
place on the developer’s machine before committing the code to the source code repository. It’s the
phase before the developer shares the changes done locally with others. An Outer Loop is made up of
several cycles, including code merge, code review, build artifact, test execution, and deployment in a
shared dev/staging/production environment. Everything in this loop should be automated as much as
possible. The developer’s “commit and push” to the source code repository from the inner development
loop triggers the outer development loop.
As a developer, if you code six hours per day (360 minutes) in a traditional local iterative development
loop of eight minutes, you can expect to make about 45 iterations of your code per day. When you bring
“containerization” in, it takes additional four minutes to
Building and developing containers can slow down this inner development loop that impacts the
cycle time, a “hidden tax” in the container-native inner loop of application development.
The default inner loop in Kubernetes takes a long time and impacts developer productivity. Due to this,
the CI/CD pipeline should not be an inner loop for development. The primary focus of the development
phase is to create a development setup that makes the inner loop faster, just like you are building an
application and running it on your laptop.
In order to achieve the inner loop of application development, teams need to focus on three areas.
First, creating the development cluster, including local and remote setup. Next, installing basic tools for
working efficiently with the cluster, and finally, setting up development tooling and workflow for rapid
application development.
9
Create a development cluster
Teams can use either a local or remote development cluster. There’s no hard and fast rule, and
Kubernetes users can be successful either way. Consider your use case and resources when deciding
which is right for your application.
If you’re already using a local cluster of your choice and it’s working well, don’t change it. If that’s not the
case, use Docker desktop for entry-level developers on Kubernetes as it is one of the easiest, fastest,
and most secure ways to build containerized applications. It’s a native application designed by Docker
for both Windows and MAC users. Built on top of the Docker engine, it’s a perfect tool for beginner-level
developers who are looking to leverage the Docker ecosystem to increase development velocity and
delivery of modern container applications. Docker Desktop packages Docker Engine, Docker CLI client,
Docker Compose, Kubernetes, and Credential Helper into one single big component.
Key capabilities like a simple graphical UI, easily mapped Storage Volumes, managed kernel patches
and security updates of the linuxkit VM, ready-to-operate single node Kubernetes cluster, and tailored
support of multi-architecture image building with fairly large community support make it a strong
differentiator amongst others.
10
Remote development cluster
While there is an indispensable benefit in application development in the local Kubernetes cluster, it is
still limited by the CPU, Memory, and disk constraints of the local environment. You would benefit from a
remote Kubernetes cluster for your application development for two main reasons
• Remotely distributed teams can contribute to the same project to the same Kubernetes cluster
• On-the-fly changes are possible that do not hinge on a local machine’s capabilities
Digital Ocean’s Managed Kubernetes provides easy-to-run Kubernetes in the cloud without the pain of
setting up and maintaining the control plane. It takes away the management overhead of configuring,
maintaining, and scaling the cluster so that software teams can focus on optimizing their software
applications. The control plane is also free, which helps keep the cost lower.
“The fact that it’s so easy to configure, administrate and scale with DigitalOcean is
something which I love. I’ve worked with Kubernetes before, hands-on, self-hosted,
but the DigitalOcean integration and provision of Kubernetes has been the most
seamless that I’ve experienced so far.”
James Hooker, CTO, Hack The Box
If you need multiple developers sharing a single remote development cluster, we recommend
Telepresence. See the tooling section below for details on Telepresence.
• kubectl: To communicate with the Kubernetes Control Plane, i.e., API server using the K8s API.
• Kubectx & kubens: Kubectx to switch context between multiple Kubernetes clusters & Kubens to
switch between namespaces.
• Stern: To tail multiple pods on Kubernetes and various containers within the pod color coded for
quicker debugging.
• k9s: Terminal-based UI to navigate, observe and manage your deployed applications. If you like
vi, you will love k9s. Alternatively, you can use Lens or Kubernetes dashboard.
• k8syaml: Web-based UI to help developers build YAML files for Kubernetes cluster deployments.
• Kustomize: Kustomize introduces a template-free way to customize application configuration that
simplifies the use of off-the-shelf applications.
• Helm: An Omnipresent Kubernetes package manager to install and uninstall off-the-shelf apps.
11
Development tooling for Inner Loop
There are several popular Kubernetes development tools for automatic build, push and deploy.
• Tilt
• Skaffold
• DevSpace
• Telepresence
If you’re already using a tool, such as Okteto, Gitpod, or another choice, keep using it. If not, we have
the following recommendations for local vs. remote clusters.
We recommend Tilt for teams with little-to-no Kubernetes experience. Tilt simplifies the Kubernetes
workflow by automating and abstracting away the process of building and deploying container images
in a Kubernetes Cluster.
Tilt helps in creating continuous delivery pipelines for both your local and remote development
environments. It keeps track of the changes made to the Dockerfile and automatically builds, pushes,
and deploys them to the live containers in the Kubernetes clusters in real-time. The Heads-up display in
the terminal and the Web-UI are the two key features that differentiate Tilt from its competitors, which
provide in-depth information about your service, health status, build and run-time logs, and more.
12
We recommend Telepresence for multiple developers sharing a single remote DigitalOcean Kubernetes
cluster. Telepresence is an open source CNCF project that lets you set up a local development
environment that connects to a remote Kubernetes cluster. You can make code changes locally and
immediately see the impact of those changes in the remote Kubernetes cluster.
The key feature of Telepresence is the bidirectional network proxy that is created between the process
that runs in the development machine and the pod in the remote Kubernetes cluster. This way, it
magically transforms your laptop into a pod that runs in a remote Kubernetes cluster, making it look like
your local environment is inside a pod. You can launch your application locally and tell Telepresence to
intercept and route all the traffic to your local port.
13
Hack the Box
Hack The Box Team transitioned to Kubernetes, completely replacing the Docker and Docker Swarm
environments with DigitalOcean’s Managed Kubernetes offering. Hack The Box Challenge environments
need to work quickly and also provide strict network policies for users, enabling them to access parts
of the machine required to complete challenges but preventing access to outside networks. With
DigitalOcean’s managed Kubernetes, the team can use Kubernetes to handle the application and
required network policies while scaling to meet traffic needs.
Utilizing Kubernetes has enabled the Hack The Box team to tremendously scale Capture The Flag,
a platform that allows users to compete in challenges with other members. Recently, Hack The Box
hosted Cyber Apocalypse, their first-ever global community Capture The Flag event. That event saw
9,900 players sign up, compared to their typical few hundred users. Even with thousands of users and
61 challenges to choose from, DigitalOcean Kubernetes seamlessly scaled up backend clusters and
supported containers needed to host the event.
Elastic Cluster
Digital Ocean
Firewall
Elastic Stack Pod
User Digital Ocean
Elastic Stack Pod volumes
...
Web ...
Backend
Challenge Cluster
Challenge Pod
Digital Ocean Digital Ocean
Spaces Challenge Pod volumes
...
...
14
CI/CD:
The code-to-delivery pipeline
Set up an automated pipeline for building and deploying
application code to Kubernetes.
What you need to get started: You have already created a development setup and the necessary
client tools. Your application is running in a development environment.
Now that the application is running in a development environment, you’re ready to plan for the
production deployment. CI/CD (Continuous Integration and Deployment) readiness is a journey that’s
more about DevOps and automation practices than the specific tooling. Teams need to have good
test coverage, for example, before they can automate releases. A good CI/CD process should enable
teams to recreate their applications anywhere (minus the user data) in a matter of minutes. Consider
the needs of your team as you explore CI/CD options. Depending on the nature of your application
and shipping frequency, you may not even need a full-fledged CI/CD.
CI/CD Overview
Let us go over the high-level overview of the code-to-delivery pipeline and the typical challenges you
will encounter. Refer to the diagram below.
Container
registry
Update Application
Manifest, Helm Chart, etc.
Users
Update Application
Manifest, Helm Chart, etc.
Update Application
Manifest, Helm Chart, etc.
Kubernetes
clusters
15
There are broadly four stages in going from code to delivery.
Git Repository: Committing application code to the git release branch kickstarts the pipeline.
1 The most important decision you will make is your branching strategy. There are various types
of branching options.
• GitHub flow: Use GitHub flow to create a branch for your feature work. When your
feature is ready, submit a pull request (PR) to the main branch. GitHub flow works great
for small teams.
• GitLab flow: With GitLab flow, you can maintain separate branches for different
environments (pre-prod, prod, dev) and promote code as needed. This works well
when you want to control when you deploy code to a certain environment.
• Trunk-based: Developers using trunk-based branching are continuously committing
code to the main branch. No branching, no PR. This works well if you have seasoned
developers and mature testing and rollback processes.
If you already have a preferred branching structure, stick to that. Otherwise, we recommend
GitHub flow. It is well-understood and simple to start. Pull Requests are the source of truth for
the application history, so it’s important to keep the PR well-scoped and self-explanatory.
Build Pipeline: At any time, you have one or more application repositories with multiple
2 branches and ongoing development. As you continue to update your code, you need to build
new images for testing. This is done using a build pipeline. You can trigger an image build for
every PR or decide to do it once a week or based on any other trigger. It comes down to your
business needs. The issue with delaying the build is that you will discover issues much later,
and it will be hard to troubleshoot. So it is a recommended practice to build and publish a new
image for a reasonable set of code changes (PR). A well-scoped PR makes it much easier.
For building images, there are many tools (managed or self-hosted). We recommend using
a managed provider for the image build pipeline. For the image registry, use DigitalOcean
Container Registry unless you already have a working registry like Docker Hub.
16
Application Manifests: The CI pipeline has pushed the new application image (with a new
3 tag) to the image registry. Now, you need the application image to be deployed onto the
cluster. A Kubernetes cluster needs application metadata (YAML file, Helm chart, etc.).
You must store and version the application metadata in the git repository. You may decide
to have the same repo for both applications and metadata or use separate repositories.
We recommend using the same repo for both applications and configuration.
Another important aspect is promoting your code changes to the staging and production
environments. When you have a new image available, you must first deploy it to the
staging environment. Deployment to the staging environment should be automated for
every image push. So the continuous integration (CI) pipeline should update the metadata
for the staging environment. For production, we recommend manually approving changes,
meaning an individual manually submits a PR to update the application manifest for the
production environment.
Rollout: The issue with Kubernetes deployment kubectl apply -f is that you’re just overwriting
4 the existing application, and all the users will have to interact with the new application. In case
of an issue, you will have to do additional work to roll it back. That’s why the general practice
is to roll out the new version of the application slowly. It can be one replica at a time, cluster
at a time, blue-green deployment (rollout a copy of the new replica first), canary deployment
(rollout one replica and redirect a subset of the production traffic), or a mix of these. The
whole idea is to reduce the surface area and roll back quickly if the new version does not
work well in production. We don’t recommend this at the start. Your first focus should be to
get the build pipeline working, enable proper automated testing in the CI pipeline, and set up
automated deployment to the staging environment.
17
Challenges in CI/CD
CI/CD readiness is a journey. DevOps and automation practices make a larger impact on success than
the tooling that’s chosen. To achieve your goals, identify and implement the right level of automation
needed at the current stage of your business. Always start by deciding your objective for the CI/CD.
This should be driven by your current business needs and skillset. As you build the appropriate level of
automation for your needs, challenges may arise with structuring your environments, test coverage,
and handling Drift, rollbacks, permissions, and secrets.
CI/CD Spectrum
• If you start allocating one remote dev cluster for each developer, it will quickly increase costs.
Refer to the Development stage for setting up the dev clusters.
• Your application likely uses load balancers, block storage, object storage, managed databases,
external SaaS APIs, and other tools. You will need to customize the development manifests for
different environments. This is a one-time effort.
• You may likely have many production clusters, meaning you’re looking at many copies of
the application manifest with different levels of customizations. Structuring this right is very
important. This will avoid manifest sprawl and improve maintainability.
Test coverage.
This is the hard part of a good pipeline. This is no different for Kubernetes than any other application.
Teams need to invest the time in writing down unit tests and any other tests to reduce manual testing
time for every iteration of the application.
18
Rollback.
When you keep your application metadata in the Git repository, rollback to a prior version of the
application may be tricky sometimes. While it sounds pretty straightforward to bring the Git folder to a
prior version and let the cluster synchronize the metadata, complications can occur.
As an example, consider a team using WordPress version 1.18 via GitOps that upgrades the WordPress
manifests to 1.19 and lets the cluster synchronize. If they realize something broke along the way, they’d
be required to switch back to 1.18. This can easily be done by changing the Git folder manifests, but
what if the application fails while applying 1.18 manifests? Scenarios like these are why it’s important to
test properly before deploying to production and why it’s important to keep the PRs atomic.
Permissions.
Some businesses may demand different permissions structures. For example, it may be necessary to
separate access rights for different employees for infrastructure, applications, and more. In this case,
teams must start with the business need and then build the repo structures, with the goal of avoiding
manifest code duplication and ensuring that they can quickly roll out new clusters with the same
applications and configurations.
Secrets.
Cloud credentials, third-party SaaS credentials, DigitalOcean Spaces tokens, and private keys are all
examples of secrets. Secrets are normally referenced in a YAML file in Kubernetes, which is a base64
encoded text, as is plain text. Secrets shouldn’t be exposed outside the cluster, particularly in a Git
repository, creating a challenge because the manifests are kept in the Git repository. There are many
tools available to encrypt the secrets outside the cluster—we recommend one of the following:
• An external secrets manager using a Kubernetes CRD (External Secrets operator) to interface.
If you’re already using something like Vault, AWS/GCP/Azure secrets manager, or 1password,
then setting up an external secrets operator on your cluster is all that’s required.
• A sealed secrets controller to enable encryption of secrets outside the cluster. This is popular in
the GitOps world.
Our recommendation is to choose the external secrets operator as the first option because it has
better security and less management overhead. See this section of the tutorial for the external
secrets operator setup with Vault. If you prefer to use sealed secrets, then refer to this tutorial from
the starter kit.
19
Recommended steps
There are several approaches to setting up CI/CD, and which approach you choose will depend on the
needs of your business and the expertise of your team. We recommend starting with the following steps.
• Keep your infrastructure configuration as code (IaC) in the same Git repository. Every manifest
applied to Kubernetes is a code (YAML file). This includes application metadata and even the
infrastructure components like prometheus stack, velero, nginx ingress, and others. As you’re
setting up the Git repository, CI pipeline, and GitOps, we recommend keeping IaC in the same Git
repository. This will allow you to roll out a new Kubernetes infrastructure in minutes.
Development
clusters
Container
Git -
registry
Application
Code
Staging Cluster
Users
Git - Infra
as Code
Prod Cluster - APAC
Prod Cluster - NA
• Use any CI (continuous integration) tool. For this tutorial, we used Github and Github actions.
• Use GitOps. For our guide, we’ll use ArgoCD. Flux is equally popular—feel free to pick whatever
you are comfortable with.
20
• Structure your environments. Structure your environments and keep separate accounts for
development, staging, and production. You can use the team structure to hold different accounts.
Give employees access to different teams as needed.
Owner
• Lock down your production cluster. The default kubeconfig for a Kubernetes cluster has admin
rights. While it makes sense for a development cluster (and sometimes staging), the production
cluster should be locked down. It means no doctl access to production accounts, and you grant
limited access to Kubernetes clusters using client certificates. If you have a small team, feel free
to skip this step for now and maybe limit production account access to a few developers.
• Plan the staging environment. We recommend a 24x7 staging environment.
• Choose a Git folder layout. This is an important decision you will make. Here’s a recommended
practice to structure your git folder layout.
21
My_git_repo
Clusters
ArgoCD
Prod - APAC Kustomize manifests (Prod)
Controller
ArgoCD
Prod - EMEA Kustomize manifests (Prod)
Controller
Infra
Ingress Manifests
Kube-prom-stack manifests
Velero manifests
App1
Source Code
Dockerfile
Base manifests
App2
Source Code
Dockerfile
Base manifests
In the above diagram, kustomize manifests are the ones that get synced to the clusters. We have four
clusters (staging, prod-NA, prod-APAC, prod-EMEA). Kustomize manifests refer to the base manifests in
the same repo or another repo. Likewise, you may directly refer to the upstream helm chart for monitoring
stack (eg. kube-prom-stack) and customize it directly. Kustomize is built into Kubernetes tooling and offers
a lot of flexibility.
While the above example shows all apps and infrastructure configurations in a single repo, you may
separate those into different repositories.
• Write your CI program to trigger on PR to the application source code folder. This CI program
should run necessary verifications, build the docker image and publish it to the registry with an
appropriate tag. If successful, it should update the kustomize manifest of the staging cluster with
the image tag as the newly created one. That will automatically trigger the Argo CD controller
running on the staging cluster. Argo CD will deploy the new image on the staging cluster.
• Set up manual promotion between environments. While deployment to staging should be automatic,
we do not recommend automatic deployment to production clusters. Promoting the production
should be done manually by someone updating the kustomize manifests for the production clusters
and creating the PR. This allows you to control when to ship a feature to production.
22
Bright Data
Bright Data has over 9,000 nodes running on DigitalOcean. As a team dedicated to efficiency and
innovation, they prioritize finding the best possible solution. As they continued to grow, the team
realized that they were underutilizing their Droplets. They needed the ability to scale up to meet demand
but wanted to efficiently utilize resources. In order to optimize performance, the team switched to
DigitalOcean Kubernetes.
Bright Data containerized one of their flagship products, their data platform, adjusting it to work for
Kubernetes, a technique known as lift and shift. The team had their minimal viable product (MVP) after
aboutu three weeks and was ready to serve customers soon after. Bright Data moved its servers to
Kubernetes and automated provisioning with Terraform.
Build
9,000 nodes Image
DigitalOcean DigitalOcean
Container Registry volumes
Admin
ns-prod
commit
WebBrowser WebBrowser WebBrowser
Version CICD
SVC Bright Data ns-test
control LB
WebBrowser WebBrowser WebBrowser
23
Production
Create a robust, production-ready Kubernetes cluster
that’s future-proof and ready to serve end-users.
What you need to get started: A ready staging environment, CI/CD tooling in place,
and a promotion strategy.
Getting a Kubernetes cluster production-ready means more than just serving traffic. Commercial-ready
clusters should be future-proof—prepared for scale, optimization, security, and disaster recovery.
The below section focuses on features available in DigitalOcean Kubernetes, but many of these steps
will also be applicable to other managed Kubernetes providers or to a self-hosted Kubernetes. In order
to run microservices on a Kubernetes stack in production, you must:
24
Choose the right node types
Different applications need different compute/memory footprint. Depending on your application type, it
is important to start with the right node type and size for your production clusters. These decisions will
ultimately guide you in choosing the right Kubernetes plan.
DigitalOcean offers Horizontal Pod Autoscaler (HPA) in conjunction with Cluster auto-scaler (CA) to
autoscale your pods and worker nodes, respectively. When HPA decides to increase the pod limit to
maintain resource utilization thresholds, CA will add more worker nodes to the cluster to maintain the
cluster size needed for those pods.
Check out the Scale section to learn more about when to use auto-scaling.
25
Set up ingress
While building web applications on microservices is a popular solution, exposing and managing external
access to multiple software components can be hard to manage. Traditionally, teams employ load balancers
provided by the cloud provider to fulfill these needs, but load balancers can be rigid in nature and get more
expensive as the application grows. This is where an ingress comes in. The main difference between ingress
and load balancers is that load balancers can route only to a single service staying external to the cluster
while Ingresses are native objects inside the cluster that can route to multiple services.
An ingress Kubernetes object along with its associated controller fundamentally lets you to route
the web traffic traffic from outside the Kubernetes cluster to one or more services inside the cluster
acting as a single point of entry for all the incoming traffic. This makes it the best option for the production
environments. It can also handle much more than just load balancing web applications,
and can scale up easily.
Cluster pod
Service 1 pod
pod
Routing
Ingress Controller
Ingress policy
managed
Clients DO LB pod
Service 2 pod
pod
The main benefits of using an Ingress controller are:
There are many ingress implementations available for Kubernetes, including NGINX ingress controller,
Traefik Kubernetes Ingress provider, Ambassador API Gateway, Contour, an Envoy based ingress
controller, Istio ingress, and HAProxy ingress.
While we see growing adoption of all of the above, we recommend NGINX ingress for users who do not
have an existing preference. NGINX is well established and has a massive user base.
26
Getting started with NGINX ingress controller
The primary reason developers set up an ingress is to receive external traffic to the services running on
Kubernetes clusters. Ideally, a DNS would be mapped to the ingress controller load balancer IP. This can be
done via a DNS provider with the domain name you own. The load balancer receives traffic on HTTP/HTTPS
ports 80/443 and forwards it to the Ingress Controller Pod. The Ingress Controller will then route the traffic
to the appropriate backend Service.
Here’s one way to set up the NGINX ingress controller on DigitalOcean Kubernetes. You may need to modify
the installation process based on your unique requirements. For a hands-on walkthrough of the below steps
check out this tutorial.
The ingress controller will create a LoadBalancer Service that provisions a DigitalOcean Load
Balancer to which all external traffic will be directed. For higher capacity, teams can configure
the load balancer node counts using annotations. There are a number of customizations
available. Always customize the load balancer from Kubernetes NGINX service definition and
never directly from places like cloud console or CLI. Sometimes, you may need to use the
existing load balancer in your cloud account. This is a fairly straightforward process.
27
Set up Domain Name Mapping
3 Developers can choose to map a single domain name (www.sample.com) to the A record
of the Load balancer or map a wildcard domain (*.sample.com) to the A record of the load
balancer. Through Single DNS Mapping, developers can map only one domain to the ingress
controller and use multiple path-based routing to route to the backend services. For example:
If opting for wild card DNS Mapping, you can have multiple dynamic subdomains through the
single ingress controller and each sub-domain can have its own path-based routing.
For example:
The main advantage of using wildcard DNS is that you can have dynamic DNS endpoints
through ingress. Once you add the wildcard entry in the DNS records, you need to mention
the required DNS in the ingress object and the ingress controller will take care of routing it to
the required service endpoint.
Some use cases require multiple replicas of applications running across different clusters.
For these use cases, use DNS load balancing. This creates multiple A records for the
same domain name that point to different load balancers to distribute the application load,
avoiding cluster downtimes. DigitalOcean allows developers to set up DNS load balancing.
By doing this, developers can expose multiple endpoints to the outside world to serve their
applications. This set up protects through cluster failovers and increases overall throughput,
giving you the ability to scale for high availability and high performance.
28
Terminate TLS on DigitalOcean Load Balancer or NGINX Ingress
6 If you have just one service running on DigitalOcean Kubernetes and you need to expose the
service outside the cluster, there’s no need to use NGINX. Developers can directly expose
the service via a DigitalOcean Load Balancer. In these situations, terminate the TLS on
DigitalOcean Load Balancer. You can scale this model (1:1 mapping between LB and
DigitalOcean Kubernetes service) for a few services. After that, cost will play a role.
If you have multiple backend services that need to be exposed to outside, you definitely need
an Ingress. For example, if you have x.app.com and y.app.com subdomains pointing to different
services, then you need “app.com” to point to DigitalOcean Load Balancer, which then directs
the connection to NGINX ingress. We recommend terminating TLS on NGINX whenever you use
one. This is done using the “TLS passthrough” option on DigitalOcean Load Balancer.
Finally, note that DigitalOcean load balancers do not support path-based routing, so if you
have multiple services that share a subdomain—like app.com/x and app.com/y are handled by
different Kubernetes services—you must use an ingress.
When you terminate TLS on the DO LB, use the X-Forwarded-For HTTP header. DigitalOcean
load balancers automatically add this header. This option works only when the entry and
target protocols are HTTP or HTTP/2.
If you’re using sub-domains, use wildcard certificates from cert-manager. Keep in mind that
deleting/re-creating NGINX resources triggers new certificate negotiation. Since Let's Encrypt
has rate limits, if not monitored, you may get blocked on the application development. When
you hit the rate limit, there’s no way to reset it. You will need to wait until the rate limit expires
after a week. They have a rate-limiting form that can be used to request a higher rate limit,
but it is no faster than its natural resetting time. You can opt for paid issuers to save yourself
from these rate limits.
29
Optimize for performance
9 Now that the production setup is ready, you can look for the below tuning parameters to
optimize for the maximum performance
• Keep-alive-requests: 10,000
• Upstream-keepalive-requests: 1000
• Worker-processes: defaults to auto
• Max-worker-connections: defaults to 512; 65535 on high concurrency
Use wrk2, a modern HTTP benchmarking tool to measure the NGINX performance that can
produce a constant throughput load and accurate latency details to the high 9s.
30
Set up observability
When running a production Kubernetes cluster, it’s important to have complete visibility into the health
of the Kubernetes cluster and application workloads. Observability helps teams understand what’s
happening across environments and technologies so they can detect and resolve issues and keep
systems efficient and reliable. Without effective observability, any downtime in the cluster may result in
increased time to recovery, resulting in customer dissatisfaction and a cascading impact on the business.
Observability is made up of the following main areas: monitoring, logging, alerting, tracing, and chaos
experiments. Start with monitoring, logging, and alerting.
There are three aspects of monitoring you need to be aware of when running microservices in
Kubernetes:
1. Cluster monitoring which includes node availability and node resource utilization.
2. Pod Monitoring which includes pod instance monitoring and container resource utilization.
3. Application and service monitoring, which includes service-to-service communication and
ingress/egress traffic monitoring.
Maintaining a robust Kubernetes monitoring and alerting solution gives teams the ability to diagnose
the issues that occurred and the location they occurred, optimize hardware utilization based on metrics
alerting, optimize costs by scraping off underutilized resources, detect malicious ingress and egress
traffic, and predict imminent problems.
There are several popular open source Application Performance Monitoring (APM) tools as well as
paid enterprise solutions that could fit your needs, including Datadog, Newrelic, Splunk, Prometheus,
and Grafana stack.
If you’re already using a SaaS or home-grown solution for Kubernetes monitoring, please continue to use
what works for you. If you’re not currently using a robust solution, DigitalOcean strongly recommends
Kubernetes Monitoring Stack, the most popular Prometheus, Grafana, and Alertmanager stack for
monitoring Kubernetes clusters, node/pod resource utilizations, and application/service deployments.
31
The stack is deployed with kube-state-metrics and node-exporter to expose cluster-level Kubernetes
object metrics as well as machine-level metrics like CPU, disk I/O, and memory usage. The kube-
prometheus-stack includes six main components:
Please follow this tutorial to incorporate the monitoring stack to your Kubernetes cluster.
32
Enable logging
For a production Kubernetes Cluster, creating a continuing record of cluster and application events
through Logging and Eventing is essential to debug, trace back and troubleshoot any pod/container
performance issue, audit for compliance needs, and identify event patterns for performance/scale
forecasting. The Kubernetes environment has grown significantly, and the sheer volume of data
generated by the components involved is tremendous, warranting a strong need for centralized Logging
and eventing that brings significant value to customers and to your business.
1. Cluster logging
2. Node logging
3. Container logging
4. Application / Service logging
Popular open source solutions for logging include Graylog, ELK Stack (ElasticSearch, Logstash, and
Kibana), EFK Stack (ElasticSearch, Fluentd, and Kibana), Grafana Loki, Kubewatch, and Zebrium.
If you’re already using a SaaS or home-grown solution for Kubernetes logging, please continue to
the solution that works for you, if you’re not currently using a working solution, DigitalOcean
recommends Grafana-Loki-based logging stack, a multi-tenant log aggregation system. The stack
has three components:
• Promtail: The agent responsible to collect worker node and application logs to send to Loki
• Loki: The main server that stores logs and processes queries
• Grafana: The visualization platform to query and display the logs
This stack uses DigitalOcean block storage as Persistent Volumes starting from 5 GB for Loki Time
Series database.
Follow this tutorial to incorporate the logging stack into your Kubernetes cluster.
33
Enable event logging
In a production environment that operates at scale, you will observe several node/pod resource
reallocations, rescheduling, and state transitions triggering Kubernetes events. These events are entirely
different from the logs generated by the Kubernetes components of the control plane. Kubernetes events
are not regular objects like logs. They are transient and persist for less than one hour. There is no built-in
mechanism in Kubernetes to store events for a long period of time.
• Easy to Setup alerting by routing specific events to tools like Slack and PagerDuty
• Easy to Log all the events in Elastic Search or Syslog or Splunk
• Easier to conduct post-mortem analysis after any event
DigitalOcean strongly recommends Kubernetes Events Explorer, an open-source event logging tool that
exports all these events for observability of alerting purposes.
Teams using GitOps (explained in CI/CD) are already in a position to bootstrap a new cluster and get the
applications running with just a few commands. For disaster recovery, consider the following challenges:
You have a working setup with a DO load balancer and external DNS names. When recovering the
cluster, it’s necessary to preserve the load balancer and associated DNS records. Otherwise, it will take
days for the DNS information to be synchronized among your users, impacting the end-user experience.
You have applications running on your cluster, which may be using persistent volumes (PV).
When an application crashes and needs restored, teams need a last working volume copy of that
application. Otherwise, the newly created application will start in an empty state. There are multiple
tools including velero/trilio, volumesnapshot, and backup/restore offered by the application itself.
It’s important to go with the tested and recommended path for your specific application.
You’re using many certificates for various domains and an issuer like letsencrypt. Depending on your
usage, you may run into certificate rate limit issues when you are trying to restore too many applications.
Under the hood, the restore process will trigger new certificate creation. Solving this (to backup and
restore the certs/keys) is a bit tricky, and you must test it beforehand.
You’re encrypting the secrets outside the cluster. In a new cluster, you will need to recover the secrets.
Otherwise, GitOps will work fine, but you won’t have the secrets you need in the new cluster.
34
The following sequence is recommended for restoring any application or the entire cluster.
Give preference to the recommended guidelines for your specific application like Elasticsearch, Kafka,
and more, which may be different from the above.
Disaster recovery is hard and needs practice. We recommend testing your procedure from time to time by
deleting apps or the entire cluster.
Refer to this section of the hands-on tutorial for an example of disaster recovery.
35
Security Hygiene
Security for Kubernetes is a fairly broad topic, and security practices for a large enterprise can be
substantially different from security best practices for a small organization. In a large organization, there
are dedicated security teams. In an SMB or startup the developers often do everything.
1. Are the components in your kubernetes clusters up-to-date, and hardened based in industry best
practices (eg. NSA/CISA Kubernetes hardening, CIS Kubernetes and Docker controls)?
2. Do you know who is accessing your cluster? Have you enforced the right role based access control
(RBAC) for different users/entities?
3. How are you ensuring that safe images with reasonably safe configurations are deployed to
your cluster?
4. How are you protecting against run-time attacks like web application attacks, malicious container
deployment, and more.
Keeping Kubernetes clusters up-to-date should be covered by the managed Kubernetes provider.
Knowing who is accessing your cluster and ensuring safe images are deployed to your cluster provide
maximum ROI for your team and are comparatively easier to automate. Protecting against run-time
attacks is hard, especially for ongoing maintenance.
Registry
End Users
Developer Security Scanning on RBAC. Ongoing Scans for nodes
Dockerfile, manifests and containers. Image admission
and source code. control. Security compliance
checks. Network policy.
Image and Manifests Secure secrets. Audit logs.
Signing.
ACL for Kube API.
Outbound threat detection
(eg. Crypto, Command and Control).
Run-time threat detection.
36
The Kubernetes and containers developer workflow is ideal for bringing security into the development
process. As part of the development workflow, developers should build container images, set up
configuration files and keep those in registry and git repositories. Given that they have to automate this
process, it makes sense to incorporate necessary security controls along the way.
Developers should use an IDE like VScode for development. Using a plugin for code/library/configuration
scanning for vulnerabilities can help in fixing the issues at the earliest. Once they commit the code to the
git repository the CI pipeline is triggered. At this stage, it’s recommended to incorporate the necessary
security checks. This includes vulnerability scanning, configuration checks, and more.
Next, the the image is built and stored in the registry and continuous scanning is necessary because
new vulnerabilities are detected over time. Now, the application is run on the cluster. Assuming you’ve
done the necessary hardening of the cluster, some tasks like image admission control, network policies,
and others, can be performed with less effort. This becomes important if you have a larger team and
less visibility. Once end users have access to the application, it’s important to secure the ingress with
appropriate configuration.
For security tools, the challenge is not creation, but ongoing update of the tool. There are many tools
available and in addition to considering the ongoing updates and maintenance of the tool, consider the
target persona of the tool and use products built with developers in mind.
Misconfiguration:
Datree, Conftest Registry
End Users
Developer
Runtime: Sysdig falco,
Optimize Docker image: Cilium network policy
multistage dockerfile, editor, kubescape,
docker-slim kube-hunter, kube-bench.
Most of these tools have an open-source and free-tier component. It is important to adopt a few tools
and implement them well.
37
For supply-chain security consider the following recommendations:
• Try Snyk or Kubescape, both have an open source CLI tool that can be integrated into the
CI pipeline.
• Reduce Docker image file size, which improves security, performance, and productivity while
reducing cost.
• Remember to follow the recommendations from your source-code repositories and invest in one or
more good tools to identify vulnerabilities and insecure configurations upfront and resolve those
prior to deployment.
• Lock down the deployment to accept infrastructure-as-code (IaC) using GitOps.
• Use a managed build pipeline, eg. GitHub, GitLab, CircleCI etc.
• Implement periodic scanning as additional vulnerability information is learned every week.
• If you have a reasonable team size with a continuously used supply chain, lock down
authorizations, sign images and artifacts, and verify those in deployment.
38
NitroPack
NitroPack’s product scans a webpage and applies techniques to solve performance problems using its
own programs and logic. The application needs to be able to handle hundreds of requests per second
and needs to be able to scale up and down to meet demand seamlessly. Originally, the NitroPack
team was managing servers by hand. As a small startup with a lean team, they wanted a solution
that would allow them to spend more time developing their product and less time managing
infrastructure while still supporting their need to scale. DigitalOcean Kubernetes provides the
automated infrastructure they need.
NitroPack set up one cluster using a local environment for development featuring over 150 nodes. They
use a microservices architecture that automates the delivery of over 120,000 sites with over five million
pages. They enabled High Availabilty (HA) for their DigitalOcean Kubernetes clusters to ensure peak
performance and reliability. Using the High Availability feature Nitropack doesn’t worry about losing the
control plane under stress like their rapid growth.
ns ns
ns
CPU ElasticDB
Optimized
node
Proxy
Load Redis
User SVC
Balancers Memory
Optimized
Proxy node Data
ns
Spaces
CDN CPU
Optimized
node
39
Scale
Utilize sophisticated tools to optimize Kubernetes
deployments for continued reliability and scale.
What you need to get started: You’ll need an application running in production and experiencing growth.
As an application in production begins to experience growth, developers need to scale the existing
cluster or multiple clusters/regions, enforce appropriate policies, and optimize for resource consumption.
Optimizing Kubernetes and applications that run in Kubernetes is similar to scaling applications outside of
Kubernetes. Optimization starts by developing a thorough understanding of the performance characteristics
of an application and the environment in which the processes are run. Measure memory, CPU, and I/O.
Focus measurements and alerts on the four golden signals: latency, traffic, errors, and saturation.
Access Control
If you’re using DigitalOcean Managed Kubernetes, all users in the DigitalOcean team have administrative
access to the clusters provisioned in the team. Provision clusters in a team account for the Kubernetes
administrators and use certificates for regular Kubernetes users and RBAC for users that need to
operate applications running in the cluster. Be sure that only the users that should have administrative
access are in the team that hosts the Kubernetes cluster.
Develop and refine the runbooks for your applications and infrastructure by holding disaster recovery
drills periodically.
40
Limits
Optimizing Kubernetes deployments for scaling requires considering failure modes, the account, resilience,
network, and CPU load, storage needs, dynamic runtime conditions, and deployment best practices.
If you’re using DigitalOcean Kubernetes, all DigitalOcean accounts have some level of limits; newer
accounts have tighter limits than established accounts. Kubernetes users should consider Droplet, load
balancer, firewall, firewall rule, and Volume limits.
Teams can review their current Droplet limits on the respective team page and can request the limits to
be increased there as well. Limits on other resources are not exposed but can be adjusted by support.
As Kubernetes customers enable autoscaling, they may need to have their account limits adjusted to allow
their cluster to scale to their maximum expected size. When considering account limits, consider:
• Given the Droplet account limits and autoscaling configuration, will your nodes be able to scale out
enough to handle the expected maximum load? Consider the max_nodes setting of each node pool
and the account’s Droplet limit.
• There is a total volumes limit and a hard limit of four volumes per Droplet: can your applications
scale to their maximum respective limits given the total volumes limit and the hard volume limit
per Droplet? Pay careful attention to the affinity and anti-affinity rules in statefulsets that have
persistent volume claim templates.
• Load balancer limits may need to be adjusted to allow a Kubernetes service of type LoadBalancer to
be created.
• Two firewalls per Kubernetes cluster are provisioned for all clusters. Services of type NodePort add
firewall rules by default.
Probes
Add useful readiness probes to your pods. Readiness probes should signal ready only when the pod is
ready to accept traffic. When a container is not yet ready for traffic (e.g., not fully initialized) or should not
receive additional traffic (e.g., traffic is blocked by a synchronous process that needs to complete before
handling more requests), it is desirable to direct traffic elsewhere. In these cases, the readiness check
should return a failure, causing traffic to be directed elsewhere.
Most applications do not need liveness probes. Liveness probes cause a container to be restarted without
causing the pod to be rescheduled/replaced. In general, liveness probes should be used sparingly. A typical
use case for liveness probes is for a temporary mutation of a detectable condition that can be handled via a
restart (e.g. memory corruption or a memory leak) until the root cause of the problem can be found.
Graceful Shutdown
Cluster upgrades is the most common reason for node shutdown. DigitalOcean Kubernetes performs a
surge upgrade by default by creating up to ten worker nodes in a pool at a time and recycling existing
nodes. This can mean disruption to the workloads. Make sure to implement graceful shutdown for your
pods. For stateful workloads that need time to drain from a node, DigitalOcean Kubernetes waits for up to
30 minutes for a given node.
41
Affinity
Add pod anti-affinity preferences to pod templates to cause pod replicas to be spread across nodes for
resilience to node failures.
Set accurate CPU requests and never set CPU limits. CPU requests guarantee that processes will
receive at least the amount of CPU requested. By omitting CPU limits, processes will be able to use
any additional CPU capacity available on the node. When CPU limits are set, applications may be CPU
throttled even when the node has otherwise idle CPU capacity.
refs:
• https://twitter.com/thockin/status/1134193838841401345
• https://home.robusta.dev/blog/stop-using-cpu-limits/
Node Pools
Use node pools and node affinity rules on pods to cause workloads to be distributed to nodes of the
appropriate type given the work of the pod.
All nodes in a node pool will have two labels by default that can be used for scheduling:
Additional labels, as well as taints, can be configured via the DigitalOcean public API. Note that any
labels or taints applied directly to a node via the Kubernetes API will not be persisted across node
recycles or upgrades.
DNS Load
DNS load can be mitigated by using NodeLocalDNSCache daemonsets. Using NodeLocalDNSCache
will require that pods have dnsConfig.nameserver set to the address to which the NodeLocalDNSCache
binds. A 169.254.0.0/16 address is a good choice for the address. The NodeLocalDNSCache can be
configured to use TCP for upstream requests while listening on UDP for client requests.
42
Resilience to API Latency and Failure
The Kubernetes control plane is a distributed system with many failure modes. Anything that interacts with
kubernetes via requests to kube-apiserver (e.g. operators or controllers) should employ best practices:
retries, backoffs, and circuit breaking. Handling all the distributed computing concerns can be difficult at
first. Start with retries and implement backoffs and circuit breaking later.
Be prepared for HTTP failure responses. Typical examples include 5xx and 4xx errors, especially when
kube-apiserver requests to etcd fail or are rejected.
The other type of Persistent Volume required in Kubernetes is NFS Volume, a shared file system that can be
accessed by multiple pods at the same time. We recommend OpenEBS Dynamic PV NFS provisioner (follow
this link for helm chart install) which can be used to dynamically provision NFS Volumes using the block
storage available on the Kubernetes nodes. The NFS provisioner runs an NFS server pod for each shared
storage volume. It uses DigitalOcean block storage. Below are the high-level set-up recommendations.
Note: The NFS Server pod is a single point of failure. It takes about 90 seconds for the NFS server to be
back up if the pod goes down. It takes about 7 minutes for the volume to be fully functional if the node
hosting the NFS server pod goes down. So, create your application deployment minding these downtimes.
Check out this step-by-step tutorial to incorporate the NFS Volume in your Kubernetes infrastructure.
43
Egress Gateway
While deploying containerized applications in production, it is a common developer interest to have some
sort of gateway so that multiple nodes can use the same IP address for the egress traffic for whitelisting
purposes. It is also a common security requirement to route all cluster traffic through a specific set
of dedicated nodes. If you’re using DigitalOcean Kubernetes, it does not have a home-grown egress
resource to manage the outgoing traffic. This is handled in a few ways:
DigitalOcean does not provide a managed NAT gateway. We recommend setting up a droplet as a
NAT gateway in the VPC. Refer to this guide for setting up an egress gateway in two steps. This way,
the egress gateway translates the source address (i.e SNATs) of the outgoing connections such
that for the external services outside the cluster, the traffic appears to be coming from the single IP
address of the egress gateway. Note that the NAT gateway in this case is a single point of failure for
the routed egress traffic.
Please check out this guide to incorporate egress gateway in your infrastructure.
Canary
Leverage canary testing to validate changes to a limited set of users and/or a small percentage of
requests. Canaries can take a couple of different forms. Kubernetes services can direct traffic to pods
from multiple deployments. By leveraging multiple deployments whose pods match the selector in a
Kubernetes service, Kubernetes users can direct connections to multiple variations of a handler and
thereby control the proportion of requests that are handled by each variant.
Security
All users in the DigitalOcean team will have administrative access to the clusters provisioned in the team.
Therefore, provision clusters in a team account for the Kubernetes administrators and use certificates
for regular Kubernetes users and RBAC for users that need to operate applications running in the
cluster. Be sure that only the users that should have administrative access are in the team that hosts the
Kubernetes cluster.
44
About DigitalOcean
DigitalOcean simplifies cloud computing so developers and businesses
can spend more time building software that changes the world. With its
mission-critical infrastructure and fully managed offerings, DigitalOcean
helps developers, startups, and small- and medium-sized businesses
(SMBs) rapidly build, deploy, and scale applications to accelerate innovation
and increase productivity and agility. DigitalOcean combines the power of
simplicity, community, open source, and customer support so customers
can spendless time managing their infrastructure and more time building
innovative applications that drive business growth.