* fix(k8s): expose / without auth so homepage is publicly reachable
Adds a second Ingress (api-public) for the exact path / with no
forward-auth middleware. Traefik prefers the Exact match for the root,
while the Prefix ingress (with auth) still protects all other routes.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* fix: homepage renders correctly at / for unauthenticated visitors
Two fixes:
1. Added parseStandalone() helper — parseTmpl() roots on "" but ParseFS()
stores standalone (no {{define}}) files under their base filename, so
Execute() ran the empty root and returned Content-Length: 0.
2. Added router.priority: 100 annotation to api-public ingress so Traefik
picks the Exact / rule over the Prefix / rule (Traefik ranks by rule
string length by default, which made PathPrefix beat Path).
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* fix(k8s): remove forward-auth middleware from finance ingress
The app now handles its own auth at /auth/login — Traefik no longer
needs to forward-auth requests, which was causing redirects to
auth.homelab.local instead of finance.homelab.local.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(auth): harden authentication for cloud deployment
1. Secure cookie flag — set when BASE_URL starts with https://
2. SameSite=Strict on session cookie (was Lax)
3. Rate limiter — per-IP, 10 failures → 15-min lockout, auto-cleanup goroutine
4. Session rotation on login — old session deleted before issuing new one
(prevents session fixation attacks)
5. bcrypt cost 12 (was DefaultCost/10, OWASP minimum for cloud)
6. Security headers middleware on all responses:
X-Content-Type-Options, X-Frame-Options, Referrer-Policy,
Permissions-Policy, Content-Security-Policy, HSTS (when HTTPS)
7. Structured audit logging — login success/failure/lockout with IP + email
8. Google OAuth state cookie gets Secure flag too
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(infra): Gitea self-hosted CI/CD + MongoDB PVC + registry pipeline
- Add Gitea Helm deployment (git hosting, container registry, Gitea Actions)
- Add act runner with DinD sidecar for Docker builds in-cluster
- Add RBAC so act runner can kubectl-deploy to finance namespace
- Fix MongoDB StatefulSet: add volumeClaimTemplates (data was lost on restart)
- Configure k3d containerd to mirror git.homelab.local → Gitea NodePort 30002
- Add .gitea/workflows/finance-api.yml: test → build/push → rolling deploy
- Update finance-api deployment: Gitea registry image, imagePullPolicy Always
- Extract finance-api secrets (SESSION_SECRET, Google OAuth) into Terraform
- Add variables.tf for Gitea admin password and runner token
All changes testable on local k3d before the VPS exists.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
---------
Co-authored-by: Gonçalo Rodrigues <guga@Goncalos-MacBook-Pro.local>
Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
138 lines
2.9 KiB
HCL
138 lines
2.9 KiB
HCL
resource "random_password" "mongodb" {
|
|
length = 24
|
|
special = false
|
|
}
|
|
|
|
resource "kubernetes_secret" "mongodb" {
|
|
metadata {
|
|
name = "mongodb"
|
|
namespace = kubernetes_namespace.domains["infrastructure"].metadata[0].name
|
|
}
|
|
|
|
data = {
|
|
MONGO_INITDB_ROOT_PASSWORD = random_password.mongodb.result
|
|
MONGO_URI = "mongodb://root:${random_password.mongodb.result}@mongodb.infrastructure.svc:27017"
|
|
MONGO_DB = "homelab"
|
|
}
|
|
}
|
|
|
|
resource "kubernetes_secret" "mongodb_shared_config" {
|
|
for_each = toset(local.namespaces)
|
|
metadata {
|
|
name = "mongodb-shared-config"
|
|
namespace = kubernetes_namespace.domains[each.key].metadata[0].name
|
|
}
|
|
|
|
data = {
|
|
MONGO_URI = "mongodb://root:${random_password.mongodb.result}@mongodb.infrastructure.svc:27017"
|
|
MONGO_DB = "homelab"
|
|
}
|
|
}
|
|
|
|
resource "kubernetes_service" "mongodb" {
|
|
metadata {
|
|
name = "mongodb"
|
|
namespace = kubernetes_namespace.domains["infrastructure"].metadata[0].name
|
|
}
|
|
|
|
spec {
|
|
selector = {
|
|
app = "mongodb"
|
|
}
|
|
port {
|
|
port = 27017
|
|
target_port = 27017
|
|
}
|
|
}
|
|
}
|
|
|
|
resource "kubernetes_stateful_set" "mongodb" {
|
|
depends_on = [kubernetes_secret.mongodb, kubernetes_service.mongodb]
|
|
|
|
wait_for_rollout = false
|
|
|
|
metadata {
|
|
name = "mongodb"
|
|
namespace = kubernetes_namespace.domains["infrastructure"].metadata[0].name
|
|
}
|
|
|
|
spec {
|
|
service_name = "mongodb"
|
|
replicas = 1
|
|
|
|
selector {
|
|
match_labels = {
|
|
app = "mongodb"
|
|
}
|
|
}
|
|
|
|
template {
|
|
metadata {
|
|
labels = {
|
|
app = "mongodb"
|
|
}
|
|
}
|
|
|
|
spec {
|
|
container {
|
|
name = "mongodb"
|
|
image = "mongo:8"
|
|
|
|
env {
|
|
name = "MONGO_INITDB_ROOT_USERNAME"
|
|
value = "root"
|
|
}
|
|
env {
|
|
name = "MONGO_INITDB_ROOT_PASSWORD"
|
|
value_from {
|
|
secret_key_ref {
|
|
name = "mongodb"
|
|
key = "MONGO_INITDB_ROOT_PASSWORD"
|
|
}
|
|
}
|
|
}
|
|
env {
|
|
name = "MONGO_INITDB_DATABASE"
|
|
value = "homelab"
|
|
}
|
|
|
|
volume_mount {
|
|
name = "mongodb-data"
|
|
mount_path = "/data/db"
|
|
}
|
|
|
|
port {
|
|
container_port = 27017
|
|
}
|
|
|
|
resources {
|
|
requests = {
|
|
cpu = "200m"
|
|
memory = "256Mi"
|
|
}
|
|
limits = {
|
|
cpu = "500m"
|
|
memory = "512Mi"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
volume_claim_template {
|
|
metadata {
|
|
name = "mongodb-data"
|
|
}
|
|
spec {
|
|
access_modes = ["ReadWriteOnce"]
|
|
storage_class_name = "local-path"
|
|
resources {
|
|
requests = {
|
|
storage = "5Gi"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|