Install on AWS
Self-managed Materialize requires: a Kubernetes (v1.31+) cluster; PostgreSQL as a metadata database; and blob storage.
The tutorial deploys Materialize to AWS Elastic Kubernetes Service (EKS) with a PostgreSQL RDS database as the metadata database and AWS S3 for blob storage. The tutorial uses Materialize on AWS Terraform module to:
- Set up the AWS Kubernetes environment.
- Call terraform-helm-materialize module to deploy Materialize Operator and Materialize instances to that EKS cluster.
The Terraform modules used in this tutorial are intended for evaluation/demonstration purposes and for serving as a template when building your own production deployment. The modules should not be directly relied upon for production deployments: future releases of the modules will contain breaking changes. Instead, to use as a starting point for your own production deployment, either:
- 
Fork the repo and pin to a specific version; or 
- 
Use the code as a reference when developing your own deployment. 
When operating in AWS, we recommend the following instances:
| EC2 Instances | 
|---|
| r8g,r7g, andr6gfamilies when running without local disk. | 
| r7gdandr6gdfamilies (andr8gdonce available) when running with local disk.  Recommended for production. | 
Starting in v0.3.1, the Materialize on AWS Terraform uses ["r7gd.2xlarge"] as
the default node_group_instance_types.
See Appendix: AWS Deployment guidelines for more information.
Prerequisites
Terraform
If you don’t have Terraform installed, install Terraform.
AWS CLI
If you do not have the AWS CLI installed, install. For details, see the AWS documentation.
kubectl
If you do not have kubectl, install. See the Amazon EKS: install kubectl
documentation
for details.
Helm 3.2.0+
If you do not have Helm 3.2.0+, install. For details, see the Helm documentation.
Set up AWS Kubernetes environment and install Materialize
The Terraform modules used in this tutorial are intended for evaluation/demonstration purposes and for serving as a template when building your own production deployment. The modules should not be directly relied upon for production deployments: future releases of the modules will contain breaking changes. Instead, to use as a starting point for your own production deployment, either:
- 
Fork the repo and pin to a specific version; or 
- 
Use the code as a reference when developing your own deployment. 
For simplicity, this tutorial stores various secrets in a file as well as prints them to the terminal. In practice, refer to your organization’s official security and Terraform/infrastructure practices.
Materialize on AWS Terraform
module
deploys a sample infrastructure on AWS (region us-east-1) with the following
components:
| Component | Version | 
|---|---|
| Kubernetes (EKS) cluster | All | 
| Dedicated VPC | All | 
| S3 for blob storage | All | 
| RDS PostgreSQL cluster and database | All | 
| Materialize Operator | All | 
| Materialize instances (Deployed during subsequent runs after the Operator is running) | All | 
| AWS Load Balancer Controller and Network Load Balancers for each Materialize instance | v0.3.0+ | 
| OpenEBS and NVMe instance storage to enable spill-to-disk | v0.3.1+ | 
| cert-managerand a self-signedClusterIssuer.ClusterIssueris
deployed on subsequent runs after thecert-manageris running. | v0.4.0+ | 
The tutorial uses the main.tf found in the examples/simple/ directory, which
requires minimal user input. For details on the  examples/simple/
infrastructure configuration (such as the node instance type, etc.), see the
examples/simple/main.tf.
For more configuration options, you can use the main.tf file at the root of
the repository
instead. When running with the root main.tf, see AWS required
configuration.
| Terraform version | Notable changes | 
|---|---|
| v0.4.4 | 
 | 
| v0.4.3 | 
 | 
| v0.4.0 | 
 | 
| v0.3.1 | 
 | 
| v0.3.0 | 
 | 
| v0.2.7 | 
 | 
- 
Open a Terminal window. 
- 
Configure AWS CLI with your AWS credentials. For details, see the AWS documentation. 
- 
Fork the Materialize’s sample Terraform repo. 
- 
Set MY_ORGANIZATIONto your github organization name, substituting your organization’s name for<enter-your-organization>:MY_ORGANIZATION=<enter-your-organization>
- 
Clone your forked repo and checkout the v0.4.5tag. For example,- 
If cloning via SSH (replace YOUR_ORGANIZATIONwith your organization’s name):git clone --depth 1 -b v0.4.5 git@github.com:${MY_ORGANIZATION}/terraform-aws-materialize.git
- 
If cloning via HTTPS (replace YOUR_ORGANIZATIONwith your organization’s name):git clone --depth 1 -b v0.4.5 https://github.com/${MY_ORGANIZATION}/terraform-aws-materialize.git
 
- 
- 
Go to the examples/simplefolder in the Materialize Terraform repo directory.cd terraform-aws-materialize/examples/simple💡 Tip:The tutorial uses the main.tffound in theexamples/simple/directory, which requires minimal user input. For details on theexamples/simple/infrastructure configuration (such as the node instance type, etc.), see the examples/simple/main.tf.For more configuration options, you can use the main.tffile at the root of the repository instead. When running with the rootmain.tf, see AWS required configuration.
- 
Create a terraform.tfvarsfile (you can copy from theterraform.tfvars.examplefile) and specify the following variables:Variable Description namespaceA namespace (e.g., my-demo) that will be used to form part of the prefix for your AWS resources.
 Requirements:
 - Maximum of 12 characters
 - Must start with a lowercase letter
 - Must be lowercase alphanumeric and hyphens onlyenvironmentAn environment name (e.g., dev,test) that will be used to form part of the prefix for your AWS resources.
 Requirements:
 - Maximum of 8 characters
 - Must be lowercase alphanumeric only# The namespace and environment variables are used to construct the names of the resources # e.g. ${namespace}-${environment}-storage, ${namespace}-${environment}-db etc. namespace = "enter-namespace" // maximum 12 characters, start with a letter, contain lowercase alphanumeric and hyphens only (e.g. my-demo) environment = "enter-environment" // maximum 8 characters, lowercase alphanumeric only (e.g., dev, test)💡 Tip:The tutorial uses the main.tffound in theexamples/simple/directory, which requires minimal user input. For details on theexamples/simple/infrastructure configuration (such as the node instance type, etc.), see the examples/simple/main.tf.For more configuration options, you can use the main.tffile at the root of the repository instead. When running with the rootmain.tf, see AWS required configuration.
- 
Initialize the terraform directory. terraform init
- 
Use terraform plan to review the changes to be made. terraform plan
- 
If you are satisfied with the changes, apply. terraform applyTo approve the changes and apply, enter yes.Upon successful completion, various fields and their values are output: Apply complete! Resources: 89 added, 0 changed, 0 destroyed. Outputs: cluster_certificate_authority_data = <sensitive> database_endpoint = "my-demo-dev-db.abcdefg8dsto.us-east-1.rds.amazonaws.com:5432" eks_cluster_endpoint = "https://0123456789A00BCD000E11BE12345A01.gr7.us-east-1.eks.amazonaws.com" eks_cluster_name = "my-demo-dev-eks" materialize_s3_role_arn = "arn:aws:iam::000111222333:role/my-demo-dev-mz-role" metadata_backend_url = <sensitive> nlb_details = [] oidc_provider_arn = "arn:aws:iam::000111222333:oidc-provider/oidc.eks.us-east-1.amazonaws.com/id/7D14BCA3A7AA896A836782D96A24F958" persist_backend_url = "s3://my-demo-dev-storage-f2def2a9/dev:serviceaccount:materialize-environment:12345678-1234-1234-1234-12345678912" s3_bucket_name = "my-demo-dev-storage-f2def2a9" vpc_id = "vpc-0abc000bed1d111bd"
- 
Note your specific values for the following fields: - eks_cluster_name(Used to configure- kubectl)
 
- 
Configure kubectlto connect to your EKS cluster, replacing:- 
<your-eks-cluster-name>with the name of your EKS cluster. Your cluster name has the form{namespace}-{environment}-eks; e.g.,my-demo-dev-eks.
- 
<your-region>with the region of your EKS cluster. The simple example usesus-east-1.
 aws eks update-kubeconfig --name <your-eks-cluster-name> --region <your-region>To verify that you have configured correctly, run the following command: kubectl get nodesFor help with kubectlcommands, see kubectl Quick reference.
- 
- 
By default, the example Terraform installs the Materialize Operator and, starting in v0.4.0, a cert-manager. Verify the installation and check the status:Verify the installation and check the status: kubectl get all -n materializeWait for the components to be in the Runningstate:NAME READY STATUS RESTARTS AGE pod/my-demo-dev-materialize-operator-84ff4b4648-brjhl 1/1 Running 0 12sNAME READY UP-TO-DATE AVAILABLE AGE deployment.apps/my-demo-dev-materialize-operator 1/1 1 1 12s NAME DESIRED CURRENT READY AGE replicaset.apps/my-demo-dev-materialize-operator-84ff4b4648 1 1 1 12s Verify the installation and check the status: kubectl get all -n cert-managerWait for the components to be in the Runningstate:NAME READY STATUS RESTARTS AGE pod/cert-manager-cainjector-686546c9f7-v9hwp 1/1 Running 0 4m20s pod/cert-manager-d6746cf45-cdmb5 1/1 Running 0 4m20s pod/cert-manager-webhook-5f79cd6f4b-rcjbq 1/1 Running 0 4m20s NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE service/cert-manager ClusterIP 172.20.2.136 <none> 9402/TCP 4m20s service/cert-manager-cainjector ClusterIP 172.20.154.137 <none> 9402/TCP 4m20s service/cert-manager-webhook ClusterIP 172.20.63.217 <none> 443/TCP,9402/TCP 4m20s NAME READY UP-TO-DATE AVAILABLE AGE deployment.apps/cert-manager 1/1 1 1 4m20s deployment.apps/cert-manager-cainjector 1/1 1 1 4m20s deployment.apps/cert-manager-webhook 1/1 1 1 4m20s NAME DESIRED CURRENT READY AGE replicaset.apps/cert-manager-cainjector-686546c9f7 1 1 1 4m20s replicaset.apps/cert-manager-d6746cf45 1 1 1 4m20s replicaset.apps/cert-manager-webhook-5f79cd6f4b 1 1 1 4m20sIf you run into an error during deployment, refer to the Troubleshooting guide. 
- 
Once the Materialize operator is deployed and running, you can deploy the Materialize instances. To deploy Materialize instances, create a mz_instances.tfvarsfile with the Materialize instance configuration.For example, the following specifies the configuration for a demoinstance.cat <<EOF > mz_instances.tfvars materialize_instances = [ { name = "demo" namespace = "materialize-environment" database_name = "demo_db" cpu_request = "1" memory_request = "2Gi" memory_limit = "2Gi" } ] EOFStarting in v0.3.0, the Materialize on AWS Terraform module also deploys, by default, Network Load Balancers (NLBs) for each Materialize instance (i.e., the create_nlbflag defaults totrue). The NLBs, by default, are configured to be internal (i.e., theinternal_nlbflag defaults totrue). Seematerialize_instancesfor the Materialize instance configuration options.Starting in v0.4.0, a self-signed ClusterIssueris deployed by default. TheClusterIssueris deployed on subsequent after thecert-manageris running.💡 Tip:If upgrading from a deployment that was set up using an earlier version of the Terraform modules, additional considerations may apply when using an updated Terraform modules to your existing deployments. See Materialize on AWS releases for notable changes. 
- 
Run terraform planwith both.tfvarsfiles and review the changes to be made.terraform plan -var-file=terraform.tfvars -var-file=mz_instances.tfvarsThe plan should show the changes to be made, with a summary similar to the following: Plan: 17 to add, 1 to change, 0 to destroy.
- 
If you are satisfied with the changes, apply. terraform apply -var-file=terraform.tfvars -var-file=mz_instances.tfvarsTo approve the changes and apply, enter yes.Upon successful completion, you should see output with a summary similar to the following: Apply complete! Resources: 17 added, 1 changed, 0 destroyed. Outputs: cluster_certificate_authority_data = <sensitive> database_endpoint = "my-demo-dev-db.abcdefg8dsto.us-east-1.rds.amazonaws.com:5432" eks_cluster_endpoint = "https://0123456789A00BCD000E11BE12345A01.gr7.us-east-1.eks.amazonaws.com" eks_cluster_name = "my-demo-dev-eks" materialize_s3_role_arn = "arn:aws:iam::000111222333:role/my-demo-dev-mz-role" metadata_backend_url = <sensitive> nlb_details = [ "demo" = { "arn" = "arn:aws:elasticloadbalancing:us-east-1:000111222333:loadbalancer/net/my-demo-dev/aeae3d936afebcfe" "dns_name" = "my-demo-dev-aeae3d936afebcfe.elb.us-east-1.amazonaws.com" } ] oidc_provider_arn = "arn:aws:iam::000111222333:oidc-provider/oidc.eks.us-east-1.amazonaws.com/id/7D14BCA3A7AA896A836782D96A24F958" persist_backend_url = "s3://my-demo-dev-storage-f2def2a9/dev:serviceaccount:materialize-environment:12345678-1234-1234-1234-12345678912" s3_bucket_name = "my-demo-dev-storage-f2def2a9" vpc_id = "vpc-0abc000bed1d111bd"The Network Load Balancer (NLB) details nlb_detailsare available when running the Terraform module v0.3.0+.
- 
Verify the installation and check the status: kubectl get all -n materialize-environmentWait for the components to be in the Runningstate.NAME READY STATUS RESTARTS AGE pod/create-db-demo-db-6swk7 0/1 Completed 0 33s pod/mzutd2fbabf5-balancerd-6c9755c498-28kcw 1/1 Running 0 11s pod/mzutd2fbabf5-cluster-s2-replica-s1-gen-1-0 1/1 Running 0 11s pod/mzutd2fbabf5-cluster-u1-replica-u1-gen-1-0 1/1 Running 0 11s pod/mzutd2fbabf5-console-57f94b4588-6lg2x 1/1 Running 0 4s pod/mzutd2fbabf5-console-57f94b4588-v65lk 1/1 Running 0 4s pod/mzutd2fbabf5-environmentd-1-0 1/1 Running 0 16s NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE service/mzutd2fbabf5-balancerd ClusterIP None <none> 6876/TCP,6875/TCP 11s service/mzutd2fbabf5-cluster-s2-replica-s1-gen-1 ClusterIP None <none> 2100/TCP,2103/TCP,2101/TCP,2102/TCP,6878/TCP 12s service/mzutd2fbabf5-cluster-u1-replica-u1-gen-1 ClusterIP None <none> 2100/TCP,2103/TCP,2101/TCP,2102/TCP,6878/TCP 12s service/mzutd2fbabf5-console ClusterIP None <none> 8080/TCP 4s service/mzutd2fbabf5-environmentd ClusterIP None <none> 6875/TCP,6876/TCP,6877/TCP,6878/TCP 11s service/mzutd2fbabf5-environmentd-1 ClusterIP None <none> 6875/TCP,6876/TCP,6877/TCP,6878/TCP 16s service/mzutd2fbabf5-persist-pubsub-1 ClusterIP None <none> 6879/TCP 16s NAME READY UP-TO-DATE AVAILABLE AGE deployment.apps/mzutd2fbabf5-balancerd 1/1 1 1 11s deployment.apps/mzutd2fbabf5-console 2/2 2 2 4s NAME DESIRED CURRENT READY AGE replicaset.apps/mzutd2fbabf5-balancerd-6c9755c498 1 1 1 11s replicaset.apps/mzutd2fbabf5-console-57f94b4588 2 2 2 4s NAME READY AGE statefulset.apps/mzutd2fbabf5-cluster-s2-replica-s1-gen-1 1/1 12s statefulset.apps/mzutd2fbabf5-cluster-u1-replica-u1-gen-1 1/1 11s statefulset.apps/mzutd2fbabf5-environmentd-1 1/1 16s NAME STATUS COMPLETIONS DURATION AGE job.batch/create-db-demo-db Complete 1/1 11s 33sIf you run into an error during deployment, refer to the Troubleshooting. 
- 
Open the Materialize Console in your browser: Starting in v0.3.0, for each Materialize instance, Materialize on AWS Terraform module also deploys AWS Network Load Balancers (by default, internal) with the following listeners, including a listener on port 8080 for the Materialize Console: Port Description 6875 For SQL connections to the database 6876 For HTTP(S) connections to the database 8080 For HTTP(S) connections to Materialize Console The Network Load Balancer (NLB) details are found in the nlb_detailsin the Terraform output.The example uses a self-signed ClusterIssuer. As such, you may encounter a warning with regards to the certificate. In production, run with certificates from an official Certificate Authority (CA) rather than self-signed certificates. - 
Find your console service name. MZ_SVC_CONSOLE=$(kubectl -n materialize-environment get svc \ -o custom-columns="NAME:.metadata.name" --no-headers | grep console) echo $MZ_SVC_CONSOLE
- 
Port forward the Materialize Console service to your local machine:1 ( while true; do kubectl port-forward svc/$MZ_SVC_CONSOLE 8080:8080 -n materialize-environment 2>&1 | tee /dev/stderr | grep -q "portforward.go" && echo "Restarting port forwarding due to an error." || break; done; ) &The command is run in background. 
 - To list the background jobs, usejobs.
 - To bring back to foreground, usefg %<job-number>.
 - To kill the background job, usekill %<job-number>.
- 
Open a browser and navigate to https://localhost:8080 (or, if you have not enabled TLS, http://localhost:8080). The example uses a self-signed ClusterIssuer. As such, you may encounter a warning with regards to the certificate. In production, run with certificates from an official Certificate Authority (CA) rather than self-signed certificates. 
 
 - 
The port forwarding command uses a while loop to handle a known Kubernetes issue 78446, where interrupted long-running requests through a standard port-forward cause the port forward to hang. The command automatically restarts the port forwarding if an error occurs, ensuring a more stable connection. It detects failures by monitoring for “portforward.go” error messages. ↩︎ 
 💡 Tip: If you experience long loading screens or unresponsiveness in the Materialize Console, we recommend increasing the size of themz_catalog_servercluster. Refer to the Troubleshooting Console Unresponsiveness guide.
- 
Next steps
- 
From the Console, you can get started with the Quickstart. 
- 
To start ingesting your own data from an external system like Kafka, MySQL or PostgreSQL, check the documentation for sources. 
Cleanup
To delete the whole sample infrastructure and deployment (including the Materialize operator and Materialize instances and data), run from the Terraform directory:
terraform destroy
When prompted to proceed, type yes to confirm the deletion.
- 
To delete your S3 bucket, you may need to empty the S3 bucket first. If the terraform destroycommand is unable to delete the S3 bucket and does not progress beyond “Still destroying…”, empty the S3 bucket first and rerun theterraform destroycommand.- Upon successful destroy, you may receive some informational messages with regards to CustomResourceDefinition(CRD). You may safely ignore these messages as your whole deployment has been destroyed, including the CRDs.