Serie: De Cero a Kubernetes — Parte 1 · Parte 2 · Parte 3 · Parte 4 · Parte 5
Llegaste a la última parte. Ya tienes un cluster Kubernetes real, TLS automático, GitOps con ArgoCD, PostgreSQL y un backend FastAPI corriendo con 2 réplicas. Solo falta lo que tus usuarios realmente ven: los frontends.
Vamos a desplegar dos aplicaciones independientes que consumen la misma API. Una es un dashboard administrativo (React + Vite). La otra es una app pública (Vue 3). Ambas empaquetadas con Docker, servidas con nginx para producción, y expuestas con ingress + TLS. Al final, haremos una prueba de integración completa.
Arquitectura final del stack completo
┌────────────────────────────────────────────────────────────┐
│ INTERNET │
│ │ │
│ ┌────────────────────┼────────────────────┐ │
│ │ │ │ │
│ ▼ ▼ ▼ │
│ ┌──────────┐ ┌──────────────┐ ┌──────────────┐ │
│ │ admin. │ │ api. │ │ app. │ │
│ │tudominio │ │tudominio.com │ │tudominio.com │ │
│ │.com │ │ │ │ │ │
│ └────┬─────┘ └──────┬───────┘ └──────┬───────┘ │
│ │ │ │ │
│ ▼ ▼ ▼ │
│ ┌──────────┐ ┌──────────────┐ ┌──────────────┐ │
│ │ React │ │ FastAPI │ │ Vue 3 │ │
│ │ Dashboard│───▶│ Backend │◀───│ Landing Page │ │
│ │ (Vite) │ │ (2 réplicas) │ │ (Vite) │ │
│ └──────────┘ └──────┬───────┘ └──────────────┘ │
│ │ │
│ ▼ │
│ ┌────────────────┐ │
│ │ PostgreSQL 16 │ │
│ │ (CloudNativePG)│ │
│ └────────────────┘ │
│ │
│ ☸️ Kubernetes (kubeadm + containerd + Calico) │
│ 🌐 nginx ingress + cert-manager (Let's Encrypt TLS) │
│ 🔄 ArgoCD — todo sincronizado desde GitHub │
└────────────────────────────────────────────────────────────┘
Parte A: Frontend #1 — Dashboard Administrativo (React + Vite)
Creamos el proyecto:
npm create vite@latest frontend-admin -- --template react-ts
cd frontend-admin
npm install
Lógica de la app: Un dashboard sencillo que consume GET /items del backend y los muestra en una tabla con estadísticas. Permite crear y eliminar items.
src/App.tsx (simplificado):
import { useEffect, useState } from 'react'
const API_URL = import.meta.env.VITE_API_URL || 'https://api.tudominio.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>📊 Dashboard Admin</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="Nombre del item" />
<input value={desc} onChange={e => setDesc(e.target.value)}
placeholder="Descripción (opcional)" />
<button onClick={createItem}>Crear Item</button>
</section>
<section className="items-table">
<h2>Items ({items.length})</h2>
{loading ? <p>Cargando...</p> : (
<table>
<thead><tr><th>ID</th><th>Nombre</th><th>Descripción</th><th>Creado</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>
)
}
.env.production:
VITE_API_URL=https://api.tudominio.com
Parte B: Frontend #2 — Landing Pública (Vue 3)
npm create vite@latest frontend-app -- --template vue-ts
cd frontend-app
npm install
src/App.vue (simplificado):
<script setup lang="ts">
import { ref, onMounted } from 'vue'
const API_URL = import.meta.env.VITE_API_URL || 'https://api.tudominio.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">Crea y comparte items al instante</p>
<form @submit.prevent="createItem" class="create">
<input v-model="form.name" placeholder="¿Qué quieres crear?" required />
<textarea v-model="form.description" placeholder="Describe tu item..." rows="2" />
<button type="submit">Crear</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">Cargando...</p>
</main>
</template>
Parte C: Docker + nginx para producción
Ambos frontends usan exactamente el mismo patrón Docker multi-stage:
frontend-admin/Dockerfile (idéntico para ambos, solo cambia el nombre de la app):
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;"]
frontend-admin/nginx.conf:
server {
listen 80;
server_name _;
root /usr/share/nginx/html;
index index.html;
# SPA routing — todas las rutas van a index.html
location / {
try_files $uri $uri/ /index.html;
}
# Cache estática
location ~* .(js|css|png|jpg|jpeg|gif|ico|svg|woff2?)$ {
expires 1y;
add_header Cache-Control "public, immutable";
}
# Health check para Kubernetes
location /health {
return 200 "OK";
add_header Content-Type text/plain;
}
}
Build y push para ambos:
# Admin
docker build -t ghcr.io/tu-org/frontend-admin:v1.0.0 ./frontend-admin
docker push ghcr.io/tu-org/frontend-admin:v1.0.0
# App pública
docker build -t ghcr.io/tu-org/frontend-app:v1.0.0 ./frontend-app
docker push ghcr.io/tu-org/frontend-app:v1.0.0
Parte D: Manifiestos Kubernetes para ambos frontends
Ambos usan la misma estructura. Solo cambian la imagen, el host del ingress y la variable de API.
gitops/apps/frontend-app1/deployment.yaml (Dashboard Admin):
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/tu-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
Service e Ingress para ambos:
# Service (idéntico, cambia name y selector)
apiVersion: v1
kind: Service
metadata:
name: frontend-admin # o frontend-app
namespace: production
spec:
selector:
app: frontend-admin
ports:
- port: 80
targetPort: 80
type: ClusterIP
---
# Ingress (cambia host)
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.tudominio.com # o app.tudominio.com
secretName: admin-tls
rules:
- host: admin.tudominio.com
http:
paths:
- path: /
pathType: Prefix
backend:
service:
name: frontend-admin
port:
number: 80
Variables de entorno en build time:
Como estamos sirviendo estáticos con nginx, la URL de la API se inyecta en el build de Vite (VITE_API_URL). Esto significa que API_URL queda hardcodeada en el bundle de JavaScript. Es intencional: no necesitas runtime injection para valores que no cambian entre entornos.
Si quisieras runtime injection sin rebuild, usarías un ConfigMap montado en nginx que exponga un /config.js con window.__CONFIG__. Pero para este tutorial, el build-time injection es suficiente.
Parte E: ArgoCD sincroniza todo
Al hacer git push con los archivos de ambos frontends, ArgoCD detecta los nuevos directorios en gitops/apps/ automáticamente (gracias a directory.recurse: true en app-of-apps). La UI muestra:
🌳 app-of-apps
├── 🟢 database (Healthy · Synced)
├── 🟢 backend (Healthy · Synced) 2 pods
├── 🟢 frontend-app1 (Healthy · Synced) 2 pods
└── 🟢 frontend-app2 (Healthy · Synced) 2 pods
Cuatro aplicaciones, dos réplicas cada una (excepto DB), TLS en todos los ingress. Todo desde un git push.
Parte F: Prueba de integración completa
1. Verificar que todo está verde
argocd app list
# NAME HEALTH STATUS
# database Healthy Synced
# backend Healthy Synced
# frontend-app1 Healthy Synced
# frontend-app2 Healthy Synced
2. Crear un item desde la app pública (Vue)
Abrimos https://app.tudominio.com, llenamos el formulario con “Café guayoyo” y creamos. El backend responde 201 Created.
3. Verlo en el dashboard admin (React)
Abrimos https://admin.tudominio.com. La tabla muestra:
| ID | Nombre | Descripción | Creado |
|---|---|---|---|
| 1 | Café guayoyo | El mejor café de Caracas | 2026-05-20, 10:15:00 AM |
El dashboard también muestra DB: connected en verde.
4. Crear otro item desde el dashboard
Botón “Crear Item” → “Tequeños con guasacaca” → Enter.
5. Verificar que la app pública lo ve
Recargamos https://app.tudominio.com. Ahora hay 2 cards:
🚀 Guayoyo Items
Crea y comparte items al instante
┌──────────────────────┐ ┌──────────────────────────┐
│ Tequeños con │ │ Café guayoyo │
│ guasacaca │ │ │
│ El mejor acompañante │ │ El mejor café de Caracas │
│ 5/20/2026 │ │ 5/20/2026 │
└──────────────────────┘ └──────────────────────────┘
6. Borrar un item desde el dashboard
🗑️ en “Café guayoyo” → desaparece de ambas apps.
Rollback en acción
Imagina que la versión v1.1.0 del backend tiene un bug. Así se revierte:
# 1. Revertir el cambio en git
cd gitops
git revert HEAD --no-edit
git push
# 2. ArgoCD detecta el cambio y revierte el deployment
argocd app diff backend
# (muestra que la imagen volverá a v1.0.0)
# 3. Verificar
curl https://api.tudominio.com/health
# {"status":"ok","db":"connected"}
Tiempo total de rollback: lo que tardes en hacer git revert && git push. Menos de 30 segundos.
Si el rollback de urgencia no puede esperar el reconciliation loop (3 min), fuerzas el sync manual:
argocd app sync backend
Lo que construiste en estas 5 partes
| Parte | Entregable |
|---|---|
| Parte 1 | 3 VMs Hetzner, kubeadm, containerd, Calico — cluster real |
| Parte 2 | Namespaces, nginx ingress, cert-manager (TLS automático), storage, RBAC |
| Parte 3 | ArgoCD, GitOps, app-of-apps, sync automático con selfHeal + prune |
| Parte 4 | PostgreSQL con CloudNativePG, backend FastAPI con health checks y migraciones |
| Parte 5 | Dos frontends (React + Vue, cada uno en su ingress con TLS), comunicación backend ↔ ambos |
Stack tecnológico completo: Hetzner Cloud → Terraform → kubeadm → containerd → Calico → nginx ingress → cert-manager → ArgoCD → CloudNativePG → FastAPI → React → Vue → Docker → GitHub
Próximos pasos (por tu cuenta)
Lo que construiste es la base. Sobre esto puedes agregar:
- CI con GitHub Actions: Build de Docker automático en cada push, push a registry, actualizar tag en repo GitOps
- Monitoreo: Prometheus + Grafana (el operator kube-prometheus-stack se instala en 5 minutos)
- Logs centralizados: Loki + Grafana (o Elastic Cloud on Kubernetes)
- HA: Segunda instancia de PostgreSQL con replicación, múltiples control-planes
- Service Mesh: Linkerd para mTLS automático entre servicios
- Image Updater: ArgoCD Image Updater para auto-actualizar imágenes cuando hay nuevo tag
- Secrets de verdad: Migrar de Secret en texto plano a SealedSecrets o ExternalSecrets + Vault
En Guayoyo Tech armamos arquitecturas completas como esta para empresas que necesitan infraestructura seria. Desde el Terraform que levanta las VMs hasta el último botón del dashboard React. Kubernetes, GitOps, CI/CD, monitoreo — todo integrado, documentado y con transferencia de conocimiento incluida. Habla con el equipo gratis 15 minutos y te mostramos cómo llevamos tu infraestructura al siguiente nivel.

