DEV Community

jesus manrique
jesus manrique

Posted on • Originally published at guayoyo.tech

Zero to Kubernetes Part 5: Two Frontends + Complete Stack

K8s Terraform ArgoCD — Header

Series: Zero to Kubernetes — Part 1 · Part 2 · Part 3 · Part 4 · Part 5


You've made it to the final part. You now have a real Kubernetes cluster, automatic TLS, GitOps with ArgoCD, PostgreSQL, and a FastAPI backend running with 2 replicas. Only what your users actually see is missing: the frontends.

We'll deploy two independent applications that consume the same API. One is an admin dashboard (React + Vite). The other is a public app (Vue 3). Both packaged with Docker, served with nginx for production, and exposed with ingress + TLS. At the end, we'll run a full integration test.


Final Architecture

┌────────────────────────────────────────────────────────────┐
│                      INTERNET                               │
│                         │                                   │
│    ┌────────────────────┼────────────────────┐              │
│    │                    │                    │              │
│    ▼                    ▼                    ▼              │
│ ┌──────────┐    ┌──────────────┐    ┌──────────────┐       │
│ │ admin.   │    │ api.         │    │ app.         │       │
│ │yourdomain│    │yourdomain.com│    │yourdomain.com│       │
│ └────┬─────┘    └──────┬───────┘    └──────┬───────┘       │
│      │                 │                   │               │
│      ▼                 ▼                   ▼               │
│ ┌──────────┐    ┌──────────────┐    ┌──────────────┐       │
│ │ React    │    │ FastAPI      │    │ Vue 3        │       │
│ │ Dashboard│───▶│ Backend      │◀───│ Landing Page │       │
│ │ (Vite)   │    │ (2 replicas) │    │ (Vite)       │       │
│ └──────────┘    └──────┬───────┘    └──────────────┘       │
│                        │                                   │
│                        ▼                                   │
│               ┌────────────────┐                           │
│               │ PostgreSQL 16  │                           │
│               │ (CloudNativePG)│                           │
│               └────────────────┘                           │
│                                                            │
│ ☸️ Kubernetes (kubeadm + containerd + Calico)               │
│ 🌐 nginx ingress + cert-manager (Let's Encrypt TLS)        │
│ 🔄 ArgoCD — everything synced from GitHub                  │
└────────────────────────────────────────────────────────────┘
Enter fullscreen mode Exit fullscreen mode

Part A: Frontend #1 — Admin Dashboard (React + Vite)

npm create vite@latest frontend-admin -- --template react-ts
cd frontend-admin && npm install
Enter fullscreen mode Exit fullscreen mode

src/App.tsx (simplified):

import { useEffect, useState } from 'react'

const API_URL = import.meta.env.VITE_API_URL || 'https://api.yourdomain.com:30443'

interface Item {
  id: number; name: string; description: string; created_at: string
}

export default function App() {
  const [items, setItems] = useState<Item[]>([])
  const [name, setName] = useState('')
  const [desc, setDesc] = useState('')
  const [loading, setLoading] = useState(true)
  const [dbStatus, setDbStatus] = useState('')

  useEffect(() => {
    fetch(`${API_URL}/health`).then(r => r.json()).then(d => setDbStatus(d.db))
    loadItems()
  }, [])

  const loadItems = () => {
    fetch(`${API_URL}/items`).then(r => r.json()).then(setItems).finally(() => setLoading(false))
  }

  const createItem = async () => {
    await fetch(`${API_URL}/items`, {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ name, description: desc })
    })
    setName(''); setDesc(''); loadItems()
  }

  const deleteItem = async (id: number) => {
    await fetch(`${API_URL}/items/${id}`, { method: 'DELETE' })
    loadItems()
  }

  return (
    <div className="dashboard">
      <header>
        <h1>📊 Admin Dashboard</h1>
        <span className={`status ${dbStatus === 'connected' ? 'ok' : 'err'}`}>
          DB: {dbStatus}
        </span>
      </header>
      <section className="create-form">
        <input value={name} onChange={e => setName(e.target.value)} placeholder="Item name" />
        <input value={desc} onChange={e => setDesc(e.target.value)} placeholder="Description" />
        <button onClick={createItem}>Create Item</button>
      </section>
      <section className="items-table">
        <h2>Items ({items.length})</h2>
        {loading ? <p>Loading...</p> : (
          <table>
            <thead><tr><th>ID</th><th>Name</th><th>Description</th><th>Created</th><th></th></tr></thead>
            <tbody>
              {items.map(item => (
                <tr key={item.id}>
                  <td>{item.id}</td><td>{item.name}</td><td>{item.description}</td>
                  <td>{new Date(item.created_at).toLocaleString()}</td>
                  <td><button onClick={() => deleteItem(item.id)}>🗑️</button></td>
                </tr>
              ))}
            </tbody>
          </table>
        )}
      </section>
    </div>
  )
}
Enter fullscreen mode Exit fullscreen mode

Part B: Frontend #2 — Public Landing (Vue 3)

npm create vite@latest frontend-app -- --template vue-ts
cd frontend-app && npm install
Enter fullscreen mode Exit fullscreen mode

src/App.vue (simplified):

<script setup lang="ts">
import { ref, onMounted } from 'vue'

const API_URL = import.meta.env.VITE_API_URL || 'https://api.yourdomain.com:30443'
const items = ref<any[]>([])
const loading = ref(true)
const form = ref({ name: '', description: '' })

const loadItems = async () => {
  const r = await fetch(`${API_URL}/items`)
  items.value = await r.json()
  loading.value = false
}

const createItem = async () => {
  await fetch(`${API_URL}/items`, {
    method: 'POST',
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify(form.value)
  })
  form.value = { name: '', description: '' }
  loadItems()
}

onMounted(loadItems)
</script>

<template>
  <main class="landing">
    <h1>🚀 Guayoyo Items</h1>
    <p class="subtitle">Create and share items instantly</p>
    <form @submit.prevent="createItem" class="create">
      <input v-model="form.name" placeholder="What do you want to create?" required />
      <textarea v-model="form.description" placeholder="Describe your item..." rows="2" />
      <button type="submit">Create</button>
    </form>
    <div class="grid">
      <div v-for="item in items" :key="item.id" class="card">
        <h3>{{ item.name }}</h3>
        <p>{{ item.description }}</p>
        <small>{{ new Date(item.created_at).toLocaleDateString() }}</small>
      </div>
    </div>
    <p v-if="loading">Loading...</p>
  </main>
</template>
Enter fullscreen mode Exit fullscreen mode

Part C: Docker + nginx for Production

Both frontends use the exact same multi-stage Docker pattern:

Dockerfile:

FROM node:20-alpine AS builder
WORKDIR /app
COPY package*.json ./
RUN npm ci
COPY . .
RUN npm run build

FROM nginx:alpine
COPY --from=builder /app/dist /usr/share/nginx/html
COPY nginx.conf /etc/nginx/conf.d/default.conf
EXPOSE 80
CMD ["nginx", "-g", "daemon off;"]
Enter fullscreen mode Exit fullscreen mode

nginx.conf:

server {
    listen 80;
    server_name _;
    root /usr/share/nginx/html;
    index index.html;

    location / {
        try_files $uri $uri/ /index.html;
    }

    location ~* \.(js|css|png|jpg|jpeg|gif|ico|svg|woff2?)$ {
        expires 1y;
        add_header Cache-Control "public, immutable";
    }

    location /health {
        return 200 "OK";
        add_header Content-Type text/plain;
    }
}
Enter fullscreen mode Exit fullscreen mode

Build and push for both:

docker build -t ghcr.io/your-org/frontend-admin:v1.0.0 ./frontend-admin
docker push ghcr.io/your-org/frontend-admin:v1.0.0

docker build -t ghcr.io/your-org/frontend-app:v1.0.0 ./frontend-app
docker push ghcr.io/your-org/frontend-app:v1.0.0
Enter fullscreen mode Exit fullscreen mode

Part D: Kubernetes Manifests for Both Frontends

Both use the same structure. Only the image, ingress host, and API variable change.

gitops/apps/frontend-app1/deployment.yaml (Admin Dashboard):

apiVersion: apps/v1
kind: Deployment
metadata:
  name: frontend-admin
  namespace: production
spec:
  replicas: 2
  selector:
    matchLabels:
      app: frontend-admin
  template:
    metadata:
      labels:
        app: frontend-admin
    spec:
      containers:
        - name: nginx
          image: ghcr.io/your-org/frontend-admin:v1.0.0
          ports:
            - containerPort: 80
          resources:
            requests:
              memory: "64Mi"
              cpu: "50m"
            limits:
              memory: "128Mi"
              cpu: "200m"
          livenessProbe:
            httpGet:
              path: /health
              port: 80
            initialDelaySeconds: 5
            periodSeconds: 15
          readinessProbe:
            httpGet:
              path: /health
              port: 80
            initialDelaySeconds: 3
            periodSeconds: 10
Enter fullscreen mode Exit fullscreen mode

Ingress (admin):

apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
  name: frontend-admin
  namespace: production
  annotations:
    cert-manager.io/cluster-issuer: letsencrypt-production
    nginx.ingress.kubernetes.io/ssl-redirect: "true"
spec:
  ingressClassName: nginx
  tls:
    - hosts:
        - admin.yourdomain.com
      secretName: admin-tls
  rules:
    - host: admin.yourdomain.com
      http:
        paths:
          - path: /
            pathType: Prefix
            backend:
              service:
                name: frontend-admin
                port:
                  number: 80
Enter fullscreen mode Exit fullscreen mode

Repeat the same pattern for frontend-app (Vue), changing admin to app everywhere.


Part E: ArgoCD Syncs Everything

On git push with both frontend files, ArgoCD auto-detects the new directories in gitops/apps/ (thanks to directory.recurse: true in app-of-apps). The UI shows:

🌳 app-of-apps
  ├── 🟢 database        (Healthy · Synced)
  ├── 🟢 backend         (Healthy · Synced)  2 pods
  ├── 🟢 frontend-app1   (Healthy · Synced)  2 pods
  └── 🟢 frontend-app2   (Healthy · Synced)  2 pods
Enter fullscreen mode Exit fullscreen mode

Four applications, two replicas each (except DB), TLS on all ingresses. Everything from a git push.


Part F: Full Integration Test

1. Create an item from the public app

Open https://app.yourdomain.com, fill in "Coffee time" and create. The backend responds 201 Created.

2. See it in the admin dashboard

Open https://admin.yourdomain.com. The table shows:

ID Name Description Created
1 Coffee time Best coffee in town 5/20/2026

The dashboard also shows DB: connected in green.

3. Create another item from the dashboard

"Create Item" → "Morning meeting notes" → Enter.

4. Verify the public app sees it

Refresh https://app.yourdomain.com. Now there are 2 cards. Both apps share data through the same API.

5. Delete from dashboard

🗑️ on "Coffee time" → disappears from both apps.


Rollback in Action

# 1. Revert the change in git
cd gitops
git revert HEAD --no-edit
git push

# 2. ArgoCD detects the change and reverts
argocd app diff backend  # shows image going back to v1.0.0

# 3. Verify
curl https://api.yourdomain.com/health
# {"status":"ok","db":"connected"}
Enter fullscreen mode Exit fullscreen mode

Total rollback time: however long git revert && git push takes. Under 30 seconds.


What You Built Across These 5 Parts

Part Deliverable
Part 1 3 Hetzner VMs, kubeadm, containerd, Calico — real cluster
Part 2 Namespaces, nginx ingress, cert-manager (auto TLS), storage, RBAC
Part 3 ArgoCD, GitOps, app-of-apps, auto sync with selfHeal + prune
Part 4 PostgreSQL with CloudNativePG, FastAPI backend with health checks and migrations
Part 5 Two frontends (React + Vue, each with its own ingress + TLS), backend ↔ frontend communication

Complete tech stack: Hetzner Cloud → Terraform → kubeadm → containerd → Calico → nginx ingress → cert-manager → ArgoCD → CloudNativePG → FastAPI → React → Vue → Docker → GitHub


Next Steps (On Your Own)

What you built is the foundation. On top of this you can add:

  • CI with GitHub Actions: Automatic Docker build on push, push to registry, update tag in GitOps repo
  • Monitoring: Prometheus + Grafana (kube-prometheus-stack installs in 5 minutes)
  • Centralized Logs: Loki + Grafana
  • HA: Second PostgreSQL instance with replication, multiple control-planes
  • Service Mesh: Linkerd for automatic mTLS between services
  • Image Updater: ArgoCD Image Updater for auto-updating images on new tags
  • Real Secrets: Migrate from plain text Secret to SealedSecrets or ExternalSecrets + Vault

At Guayoyo Tech, we build complete architectures like this for companies that need serious infrastructure. From the Terraform that provisions the VMs to the last button on the React dashboard. Kubernetes, GitOps, CI/CD, monitoring — all integrated, documented, with knowledge transfer included. Talk to the team free for 15 minutes and we'll show you how to take your infrastructure to the next level.

Top comments (0)