Skip to content

dominodatalab/terraform-aws-eks

Repository files navigation

Terraform Module for EKS Setup. Optimized for Domino Installation

CircleCI

⚠️ Important: If you are currently using a version of the module <= v3.24.2 and want to update to >= v3.25.0 you should take a look at the bin/module-update/ utility. There are changes related to the EFS mount_point which may cause downtime if not addressed properly.

⚠️ Important: If you are currently using a version of the module >= v3.0.0 and want to update to >= v3.5.0 you should take a look at the bin/module-update/ utility.

⚠️ Important: If you have existing infrastructure created with a version of this module < v3.0.0 you will need to migrate your state before updating the module to versions >= v3.0.0. See state-migration for more details.

⚠️ Important: Starting from version v2.0.0, this module has KMS enabled by default. If you utilized a prior version without setting the kms.enabled variable, ensure you define kms.enabled = false to preserve your existing state. Failing to do so will default to kms.enabled = true, potentially causing the destruction of your existing infrastructure and possible data loss.

Repository Structure

  • examples/:

    • Purpose: Acts as an intuitive guide for module users.
    • Contents: Features end-user Terraform configurations along with illustrative tfvars samples, showcasing potential setups.
  • modules/:

    • Purpose: Houses the primary modules that orchestrate the provisioning process.
    • Contents: These modules act as the main entry points for the setup of the underlying infrastructure beneath the EKS cluster and its associated components.
  • tests/:

    • Purpose: Ensures the integrity and functionality of the module.
    • Contents: Contains automation-driven tests intended for validation and continuous integration (CI) checks.
  • bin/state-migration/:

    • Purpose: Contains automation to perform terraform state migration, from a monolithic module to a multi-module structure.
    • Contents: Script and documentation to perform terraform state migration.
  • bin/module-update/:

    • A helper script to update the module version. Compatible only with modules on version >= v3.0.0.
    • Contents: The directory includes the script and associated documentation for performing module updates.

Always refer to each section's respective README or documentation for detailed information and usage guidelines.

Prerequisites

Bootstrap module

We first need to setup the module structure.

1. Set your desired module version and deployment directory:

Set up the following environment variables. Update the following values(Using v3.0.0 and domino-deploy as an example):

MOD_VERSION='v3.0.0'
DEPLOY_DIR='domino-deploy'

⚠️ Ensure the DEPLOY_DIR does not exist or is currently empty.

Create the DEPLOY_DIR and use terraform to bootstrap the module from source.

mkdir -p "$DEPLOY_DIR"
terraform -chdir="$DEPLOY_DIR" init -backend=false -from-module="github.com/dominodatalab/terraform-aws-eks.git//examples/deploy?ref=${MOD_VERSION}"

Ignore this message:

Terraform initialized in an empty directory!

The directory has no Terraform configuration files. You may begin working
with Terraform immediately by creating Terraform configuration files.

✅ If successful, you should get a structure similar to this:

domino-deploy
├── README.md
├── meta.sh
├── set-mod-version.sh
├── terraform
│   ├── cluster
│   │   ├── README.md
│   │   ├── main.tf
│   │   ├── outputs.tf
│   │   └── variables.tf
│   ├── cluster.tfvars
│   ├── infra
│   │   ├── README.md
│   │   ├── main.tf
│   │   ├── outputs.tf
│   │   └── variables.tf
│   ├── infra.tfvars
│   ├── nodes
│   │   ├── README.md
│   │   ├── main.tf
│   │   ├── outputs.tf
│   │   └── variables.tf
│   └── nodes.tfvars
└── tf.sh

Note: It's recommended to go through the README.md within the DEPLOY_DIR for further details.

2. Update modules version

You can update the modules version using a script or manually.

Using script

You can update the modules version using the set-mod-version.sh

Prerequisites

Command

./set-mod-version.sh "$MOD_VERSION"

Manually

Update the modules' source with the MOD_VERSION value.

For example if MOD_VERSION=v3.0.0

  • infra/main.tf : Update module.infra.source from "./../../../../modules/infra" to github.com/dominodatalab/terraform-aws-eks.git//modules/infra?ref=v3.0.0
  • cluster/main.tf : Update module.eks.source from "./../../../../modules/eks" to github.com/dominodatalab/terraform-aws-eks.git//modules/eks?ref=v3.0.0
  • nodes/main.tf : Update module.nodes.source from "./../../../../modules/nodes" to github.com/dominodatalab/terraform-aws-eks.git//modules/nodes?ref=v3.0.0

3. Review and Configure tfvars

Consult available variables within each of the modules variables.tf

  • domino-deploy/terraform/infra/variables.tf

    • deploy_id
    • region
    • tags
    • network
    • default_node_groups
    • additional_node_groups
    • storage
    • kms
    • eks
    • ssh_pvt_key_path
    • route53_hosted_zone_name
    • bastion
  • domino-deploy/terraform/cluster/variables.tf

    • eks
    • kms_info: ⚠️ Variable is only intended for migrating infrastructure, it is not recommended to set it.
  • domino-deploy/terraform/nodes/variables.tf

    • default_node_groups
    • additional_node_groups

Configure terraform variables at:

  • domino-deploy/terraform/infra.tfvars
  • domino-deploy/terraform/cluster.tfvars
  • domino-deploy/terraform/nodes.tfvars

NOTE: The eks configuration is required in both the infra and cluster modules because the Kubernetes version is used for installing the kubectl binary on the bastion host. Similarly, default_node_groups and additional_node_groups must be defined in both the infra and nodes modules, as the availability zones for the nodes are necessary for setting up the network infrastructure.

4. Create SSH Key pair

The deployment requires an SSH key. Update the ssh_pvt_key_path variable in domino-deploy/terraform/infra.tfvars with the full path of your key (we recommend you place your key under the domino-deploy/terraform directory).

If you don't have an SSH key, you can create one using:

 ssh-keygen -q -P '' -t rsa -b 4096 -m PEM -f domino.pem && chmod 600 domino.pem

5. Deploy

1. Set AWS credentials and verify.

aws sts get-caller-identity

2. Change into domino-deploy(or whatever your DEPLOY_DIR is)

cd domino-deploy

3. Plan and Apply.

⚠️ It is recommended to become familiar with the tf.sh usage.

At this point all requirements should be set to provision the infrastructure.

For each of the modules, run init, plan, inspect the plan, then apply in the following order:

  1. infra
  2. cluster
  3. nodes

Note: You can use all instead but it is recommended that the plan and apply be done one at a time, so that the plans can be carefully examined.

  1. Init all
./tf.sh all init
  1. infra plan.
./tf.sh infra plan
  1. ❗ Carefully inspect the actions detailed in the infra plan for correctness, before proceeding.

  2. infra apply

./tf.sh infra apply
  1. cluster plan
./tf.sh cluster plan
  1. ❗ Carefully inspect the actions detailed in the cluster plan for correctness, before proceeding.

  2. cluster apply

./tf.sh cluster apply
  1. nodes plan
./tf.sh nodes plan
  1. ❗ Carefully inspect the actions detailed in the nodes plan for correctness, before proceeding.

  2. nodes apply

./tf.sh nodes apply

At this point the infrastructure has been created.

Interacting with Kubernetes

To interact with the EKS Control Plane using kubectl or helm commands, you'll need to set up both the appropriate AWS credentials and the KUBECONFIG environment variable. If your EKS cluster is private, you can use mechanisms provided by this module to establish an SSH tunnel through a Bastion host. However, if your EKS endpoint is publicly accessible, you only need to follow steps 1-3 below.

For ease of setup, use the k8s-functions.sh script, which contains helper functions for cluster configuration.

Steps

  1. Verify AWS Credentials: Ensure your AWS credentials are properly configured by running the following command:
aws sts get-caller-identity
  1. Import Functions: Source the k8s-functions.sh script to import its functions into your current shell.
source k8s-functions.sh
  1. Set KUBECONFIG: Use the check_kubeconfig function to set the KUBECONFIG environment variable appropriately.
check_kubeconfig
  1. Open SSH Tunnel (Optional): If your EKS cluster is private, open an SSH tunnel through the Bastion host by executing:
open_ssh_tunnel_to_k8s_api
  1. Close SSH Tunnel: To close the SSH tunnel, run:
close_ssh_tunnel_to_k8s_api

Retrieve Configuration Values for domino.yaml.

Run the command below to generate a list of infrastructure values. These values are necessary for configuring the domino.yaml file, which is in turn used for installing the Domino product.

./tf.sh infra output domino_config_values

This command will output a set of key-value pairs, extracted from the infrastructure setup, that can be used as inputs in the domino.yaml configuration file.

Domino Backups

If you would like to increase the safety of data stored in AWS S3 and EFS by backing them up into another account (Accounts under same AWS Organization), use the terraform-aws-domino-backup module:

  1. Define another provider for the backup account in main.tf for infra module.

Location

domino-deploy
├── terraform
│   ├── infra
│   │   ├── main.tf

Content

provider "aws" {
  alias   = "domino-backup"
  region  = <<Backup Account Region>>
}
  1. Add the following content
module "backups" {
  count  = 1
  source = "github.com/dominodatalab/terraform-aws-domino-backup.git?ref=v1.0.12"
  providers = {
    aws.dst = aws.domino-backup
  }
}

Karpenter

This module supports running workloads on Karpenter instead of EKS managed node groups.

If you have an existing deployment and want to migrate to using karpenter, see karpenter-migration.

To create a new deployment that uses Karpenter

  1. Complete all standard steps
  2. Add the following to your infra.tfvars and nodes.tfvars. See Note on selecting AZs karpenter-availability_zone_ids
karpenter_node_groups = {
  karpenter = {
    availability_zone_ids = ["usw2-az1", "usw2-az2", "usw2-az3", "usw2-az4"]
    single_nodegroup = true
  }
}

default_node_groups = null
additional_node_groups = null
  1. Add the following to the cluster.tfvars:
# Consult the karpenter variable for additional options.
karpenter = {
  enabled = true
}
  1. Plan and Apply changes
./tf.sh all plan
./tf.sh all apply
  1. See and karpenter-configurations on how to configure ec2nodeclasses and nodepools.

About

Terraform module for deploying a Domino on EKS

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Contributors 24

Languages