PowerDNS lua-records (`ifurlup`, `pickclosest`, `ifportup`) cover everything k8gb was doing — geo-aware response selection, health-checked failover, weighted round-robin — at the authoritative DNS layer. Eliminates a separate K8s controller, CRD set, and CoreDNS plugin from every Sovereign. Changes: - platform/k8gb/ deleted (Chart.yaml, values.yaml, blueprint.yaml never authored — only README existed) - products/catalyst/bootstrap/ui/public/component-logos/k8gb.svg deleted - componentGroups.ts: remove k8gb component (PowerDNS already there) - componentLogos.tsx: drop logo_k8gb + k8gb map entry - model.ts DEFAULT_COMPONENT_GROUPS spine: replace k8gb with powerdns - StepInfrastructure.tsx: copy refers to PowerDNS lua-records, not k8gb - provision.html: replace k8gb tile and edges with powerdns - catalog.generated.ts regenerated (now includes bp-powerdns) - docs sweep — every k8gb reference in PLATFORM-TECH-STACK, NAMING- CONVENTION, SOVEREIGN-PROVISIONING, SRE, ARCHITECTURE, GLOSSARY, COMPONENT-LOGOS, IMPLEMENTATION-STATUS, BUSINESS-STRATEGY, TECHNOLOGY-FORECAST, README, infra/hetzner/README, platform READMEs (cilium, external-dns, failover-controller, litmus, flux, opentofu) rewritten to point at PowerDNS lua-records / MULTI-REGION-DNS.md. Historical entries in VALIDATION-LOG.md preserved as audit trail. - New docs/MULTI-REGION-DNS.md — canonical reference for the lua-record patterns (ifurlup all/pickclosest/pickfirst, ifportup, pickwhashed), Application Placement → lua-record selector mapping, when to add a second Sovereign region, operational checks. Closes #171. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> |
||
|---|---|---|
| .. | ||
| README.md | ||
OpenTofu
Bootstrap Infrastructure as Code (one-shot). Used by Catalyst Phase 0 (see docs/SOVEREIGN-PROVISIONING.md §3) to provision the initial cloud resources for a new Sovereign — VPC, host nodes, load balancers, DNS records, object storage. After Phase 0, OpenTofu state is archived; Crossplane (running inside the new Sovereign) takes over all subsequent infrastructure changes. OpenTofu is NOT installed on host clusters at runtime — it lives on the always-on catalyst-provisioner only.
Drop-in replacement for Terraform with MPL 2.0 license.
Status: Accepted | Updated: 2026-04-27
Overview
OpenTofu is a Linux Foundation / CNCF fork of Terraform, created after HashiCorp changed Terraform's license from MPL 2.0 to the Business Source License (BSL 1.1). OpenTofu retains the MPL 2.0 license and is fully compatible with all Terraform providers and HCL syntax.
OpenTofu provisions the initial infrastructure for Kubernetes clusters. After bootstrap, Crossplane handles all day-2 cloud resource provisioning.
flowchart LR
subgraph Bootstrap["Bootstrap (OpenTofu)"]
TF[OpenTofu]
VMs[VMs + Network]
K8s[K8s Cluster]
end
subgraph Day2["Day-2 (Crossplane)"]
XP[Crossplane]
Cloud[Cloud Resources]
end
TF --> VMs
VMs --> K8s
K8s --> XP
XP --> Cloud
Why OpenTofu
| Factor | Detail |
|---|---|
| License | MPL 2.0 (open source) |
| Compatibility | 100% compatible with Terraform providers and HCL |
| Governance | Linux Foundation / CNCF project |
| Migration | Drop-in replacement, no code changes required |
| Reason for switch | HashiCorp changed Terraform to BSL 1.1 |
Supported Providers
OpenOva supports multiple cloud providers. Each provider has a corresponding Crossplane provider for day-2 operations.
| Provider | Crossplane Provider |
|---|---|
| Hetzner Cloud | provider-hcloud |
| Huawei Cloud | provider-huaweicloud |
| Oracle Cloud (OCI) | provider-oci |
| AWS | provider-aws |
| GCP | provider-gcp |
| Azure | provider-azure |
Directory Structure
opentofu/
├── modules/
│ ├── <provider>-vm/ # Provider-specific VPS provisioning
│ ├── k3s-cluster/ # K3s installation
│ └── dns-failover/ # PowerDNS authoritative + lua-records (see docs/MULTI-REGION-DNS.md)
├── environments/
│ ├── <provider>-<region>/ # Per-environment configs
│ └── ...
└── README.md
Quick Start
cd environments/<provider>-<region>
# Catalyst bootstrap (Phase 0) handles credentials interactively
# Creates tofu.tfvars (not committed to Git)
# Initialize and apply
tofu init
tofu plan -var-file=tofu.tfvars
tofu apply -var-file=tofu.tfvars
Provider Configuration
Generic Provider Setup
terraform {
required_providers {
<provider> = {
source = "<provider-source>"
version = "~> <version>"
}
}
}
provider "<provider>" {
# Credentials via tofu.tfvars or environment variables
}
resource "<provider>_server" "k8s_node" {
count = 3
name = "<org>-k8s-${count.index + 1}"
# Provider-specific configuration
}
resource "<provider>_network" "k8s_network" {
name = "<org>-network"
ip_range = "10.0.0.0/16"
}
Note: The
terraform {}block name is retained for HCL compatibility. OpenTofu uses the same HCL syntax.
K3s Cluster Configuration
Disabled Components
| Component | Reason |
|---|---|
| traefik | Gateway API (Cilium) handles ingress |
| servicelb | Cloud LB + PowerDNS lua-records for cross-region failover |
| local-storage | App-level replication |
| flannel | Cilium CNI |
Optimization Parameters
| Parameter | Value | Purpose |
|---|---|---|
| node-monitor-period | 5s | Faster health detection |
| node-monitor-grace-period | 20s | Faster failover |
| default-watch-cache-size | 50 | Memory optimization |
| quota-backend-bytes | 1GB | etcd limit |
| max-pods | 50 | Per-node limit |
Multi-Region Architecture
flowchart TB
subgraph Region1["Region 1"]
K8s1[K8s Cluster]
LB1[Cloud LB]
end
subgraph Region2["Region 2"]
K8s2[K8s Cluster]
LB2[Cloud LB]
end
subgraph DNS["DNS (PowerDNS authoritative + lua-records + ExternalDNS)"]
GSLB[GSLB — ifurlup / pickclosest]
end
LB1 --> K8s1
LB2 --> K8s2
GSLB --> LB1
GSLB --> LB2
Secrets Management
No SOPS: All secrets handled via interactive bootstrap.
- Catalyst Bootstrap (Phase 0) prompts for cloud credentials.
- Creates
tofu.tfvarslocally on the provisioner (never committed to Git). - Provisions infrastructure.
- Initializes OpenBao with generated unseal keys (saved offline by the sovereign-admin).
- After Phase 0, all secret writes go to the primary OpenBao region only; replicas pick up via async perf replication. See
docs/SECURITY.md§5.
See External Secrets README for full details.
Post-Bootstrap
After OpenTofu provisioning:
- Install Cilium CNI
- Bootstrap Flux (from Gitea)
- Flux deploys remaining components
All subsequent cloud resources are managed by Crossplane, not OpenTofu.
Consequences
Positive:
- Multi-cloud support
- Crossplane support for Day-2
- Native LoadBalancer support (where available)
- Multi-region capable
- MPL 2.0 license (fully open source)
- Full compatibility with existing Terraform providers and modules
Negative:
- Provider-specific modules required
- Some providers have limited managed services
Part of OpenOva