## Loki + Alloy re-enabled (code-146x)
- Uncommented helm_release.loki, helm_release.alloy, kubernetes_daemon_set_v1.sysctl-inotify,
kubernetes_config_map.loki_alert_rules, kubernetes_config_map.grafana_loki_datasource
- Reverses the documented "operational overhead vs benefit after node2 incident"
decision. Re-evaluated because wave 1 security detection layer (beads code-8ywc)
needs Loki + ruler + alert routing.
- SingleBinary mode, 2-4Gi mem, 50Gi proxmox-lvm PVC, 30-day retention, ruler enabled
pointed at prometheus-alertmanager.monitoring.svc:9093
- Alloy DaemonSet (4 pods on worker nodes) discovers pod logs via K8s API + pushes
to Loki
- Loki canaries running (4)
- Vault audit-tail sidecar logs now flowing to Loki: queried
{namespace="vault",container="audit-tail"} returns live audit JSON
## Wave 1 alert rules deployed (W1.3 partial)
Added "Security Wave 1" rule group to loki_alert_rules configmap:
- V1: VaultRootTokenCreated — auth/token/create with policies=[root]
- V2: VaultAuditDeviceModified — sys/audit/* create/delete/update
- V3: VaultSealChanged — sys/seal update
- V4: VaultPolicyModified — sys/policies/acl/* create/update/delete
- V5: VaultAuthFailureSpike — >10 permission denied/min
- V7: VaultViktorFromUnexpectedIP — auth as me@viktorbarzin.me from non-allowlist source IP
(allowlist: 10.0.20.0/22, 192.168.1.0/24, 10.10.0.0/16 pod, 10.96.0.0/12 svc, 100.64-127 tailnet)
- S1: PVEsshLoginFromUnexpectedIP — sshd "Accepted" from non-allowlist IP (rule defined,
fires once promtail/Alloy ships sshd journal with job=sshd-pve)
Verified rules visible via /loki/api/v1/rules. K2-K9 (K8s API audit) deferred to W1.1
which needs the audit policy + apiserver log shipping codified.
## #security Slack lane (Alertmanager)
- New `slack-security` receiver in prometheus_chart_values.tpl, channel #security
- Higher-priority route at top of routes list: matchers `lane = security` →
slack-security, continue: false (so wave 1 alerts never fall through to #alerts)
- Slack message format includes summary + description + runbook link annotation
- All wave 1 rules set `lane = "security"` label
## Resource summary
- 6 added: helm_release.loki, helm_release.alloy, kubernetes_config_map.grafana_loki_datasource,
kubernetes_config_map.loki_alert_rules, kubernetes_daemon_set_v1.sysctl-inotify,
+ 1 other
- 5 changed: helm_release.prometheus (alertmanager config — new receiver + route),
4 deployments (image tag drift from Keel-managed images, unrelated)
- 1 destroyed: null_resource grafana_admin_only_folder_acl["Finance (Personal)"]
(timestamp-triggered always recreates — not destructive)
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
Closes: code-146x
303 lines
12 KiB
HCL
303 lines
12 KiB
HCL
variable "nfs_server" { type = string }
|
|
|
|
# Loki + Alloy — re-enabled 2026-05-18 for wave 1 security audit logging
|
|
# (beads code-8ywc + code-146x). Original disable rationale was "operational
|
|
# overhead vs benefit after node2 incident" — re-evaluated because the wave 1
|
|
# detection layer (K8s audit, Vault audit, source-IP anomaly rules) needs Loki.
|
|
# Resource budget: SingleBinary mode, 2-4Gi memory, 50Gi proxmox-lvm PVC,
|
|
# 30-day retention, ruler enabled pointed at prometheus-alertmanager.
|
|
resource "helm_release" "loki" {
|
|
namespace = kubernetes_namespace.monitoring.metadata[0].name
|
|
create_namespace = true
|
|
name = "loki"
|
|
|
|
repository = "https://grafana.github.io/helm-charts"
|
|
chart = "loki"
|
|
|
|
values = [templatefile("${path.module}/loki.yaml", {})]
|
|
timeout = 600
|
|
|
|
depends_on = [kubernetes_config_map.loki_alert_rules]
|
|
}
|
|
|
|
resource "helm_release" "alloy" {
|
|
namespace = kubernetes_namespace.monitoring.metadata[0].name
|
|
create_namespace = true
|
|
name = "alloy"
|
|
|
|
repository = "https://grafana.github.io/helm-charts"
|
|
chart = "alloy"
|
|
|
|
values = [file("${path.module}/alloy.yaml")]
|
|
atomic = true
|
|
|
|
depends_on = [helm_release.loki]
|
|
}
|
|
|
|
# inotify limits raised for Alloy pod log tailing (one watch per container).
|
|
resource "kubernetes_daemon_set_v1" "sysctl-inotify" {
|
|
metadata {
|
|
name = "sysctl-inotify"
|
|
namespace = kubernetes_namespace.monitoring.metadata[0].name
|
|
labels = {
|
|
app = "sysctl-inotify"
|
|
}
|
|
}
|
|
spec {
|
|
selector {
|
|
match_labels = {
|
|
app = "sysctl-inotify"
|
|
}
|
|
}
|
|
template {
|
|
metadata {
|
|
labels = {
|
|
app = "sysctl-inotify"
|
|
}
|
|
}
|
|
spec {
|
|
init_container {
|
|
name = "sysctl"
|
|
image = "busybox:1.37"
|
|
command = [
|
|
"sh", "-c",
|
|
"sysctl -w fs.inotify.max_user_watches=1048576 && sysctl -w fs.inotify.max_user_instances=8192 && sysctl -w fs.inotify.max_queued_events=1048576"
|
|
]
|
|
security_context {
|
|
privileged = true
|
|
}
|
|
}
|
|
container {
|
|
name = "pause"
|
|
image = "registry.k8s.io/pause:3.10"
|
|
resources {
|
|
requests = {
|
|
cpu = "1m"
|
|
memory = "4Mi"
|
|
}
|
|
limits = {
|
|
cpu = "1m"
|
|
memory = "4Mi"
|
|
}
|
|
}
|
|
}
|
|
host_pid = true
|
|
toleration {
|
|
operator = "Exists"
|
|
}
|
|
dns_config {
|
|
option {
|
|
name = "ndots"
|
|
value = "2"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
lifecycle {
|
|
# KYVERNO_LIFECYCLE_V1: Kyverno admission webhook mutates dns_config with ndots=2
|
|
ignore_changes = [spec[0].template[0].spec[0].dns_config]
|
|
}
|
|
}
|
|
|
|
# resource "helm_release" "k8s-monitoring" {
|
|
# namespace = kubernetes_namespace.monitoring.metadata[0].name
|
|
# create_namespace = true
|
|
# name = "k8s-monitoring"
|
|
|
|
# repository = "https://grafana.github.io/helm-charts"
|
|
# chart = "k8s-monitoring"
|
|
|
|
# values = [templatefile("${path.module}/k8s-monitoring-values.yaml", {})]
|
|
# atomic = true
|
|
# }
|
|
|
|
resource "kubernetes_config_map" "loki_alert_rules" {
|
|
metadata {
|
|
name = "loki-alert-rules"
|
|
namespace = kubernetes_namespace.monitoring.metadata[0].name
|
|
}
|
|
data = {
|
|
"rules.yaml" = yamlencode({
|
|
groups = [
|
|
{
|
|
name = "Node Health"
|
|
rules = [
|
|
{
|
|
alert = "KernelOOMKiller"
|
|
expr = "sum by (node) (count_over_time({job=\"node-journal\"} |~ \"(?i)Out of memory.*Killed process\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = {
|
|
severity = "critical"
|
|
}
|
|
annotations = {
|
|
summary = "OOM killer active on {{ $labels.node }}"
|
|
}
|
|
},
|
|
{
|
|
alert = "KernelPanic"
|
|
expr = "sum by (node) (count_over_time({job=\"node-journal\"} |~ \"(?i)Kernel panic\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = {
|
|
severity = "critical"
|
|
}
|
|
annotations = {
|
|
summary = "Kernel panic on {{ $labels.node }}"
|
|
}
|
|
},
|
|
{
|
|
alert = "KernelHungTask"
|
|
expr = "sum by (node) (count_over_time({job=\"node-journal\"} |~ \"blocked for more than\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = {
|
|
severity = "warning"
|
|
}
|
|
annotations = {
|
|
summary = "Hung task detected on {{ $labels.node }}"
|
|
}
|
|
},
|
|
{
|
|
alert = "KernelSoftLockup"
|
|
expr = "sum by (node) (count_over_time({job=\"node-journal\"} |~ \"(?i)soft lockup\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = {
|
|
severity = "critical"
|
|
}
|
|
annotations = {
|
|
summary = "Soft lockup on {{ $labels.node }}"
|
|
}
|
|
},
|
|
{
|
|
alert = "ContainerdDown"
|
|
expr = "sum by (node) (count_over_time({job=\"node-journal\", unit=\"containerd.service\"} |~ \"(?i)(dead|failed|deactivating)\" [5m])) > 0"
|
|
for = "1m"
|
|
labels = {
|
|
severity = "critical"
|
|
}
|
|
annotations = {
|
|
summary = "containerd service unhealthy on {{ $labels.node }}"
|
|
}
|
|
},
|
|
]
|
|
},
|
|
{
|
|
# Wave 1 security alerts (beads code-8ywc). Routed via Loki ruler →
|
|
# prometheus-alertmanager → #security Slack receiver. Allowlist CIDRs:
|
|
# 10.0.20.0/22, 192.168.1.0/24, K8s pod CIDR 10.10.0.0/16, K8s service
|
|
# CIDR 10.96.0.0/12. Identity allowlist: me@viktorbarzin.me only.
|
|
# NOTE: K1 (cluster-admin grant) intentionally skipped.
|
|
name = "Security Wave 1"
|
|
rules = [
|
|
# V1: Root token created (Vault audit, vault-tail sidecar stream)
|
|
{
|
|
alert = "VaultRootTokenCreated"
|
|
expr = "sum(count_over_time({namespace=\"vault\",container=\"audit-tail\"} | json | request_path=\"auth/token/create\" |~ \"\\\"policies\\\":\\\\[\\\"root\\\"\\\\]\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = { severity = "critical", lane = "security" }
|
|
annotations = {
|
|
summary = "Vault root token created"
|
|
description = "A token with policies=[root] was issued via auth/token/create. Verify this is a planned bootstrap or break-glass; otherwise treat as critical compromise."
|
|
runbook = "docs/runbooks/security-incident.md#v1-root-token-created"
|
|
}
|
|
},
|
|
# V2: Audit device disabled/modified
|
|
{
|
|
alert = "VaultAuditDeviceModified"
|
|
expr = "sum(count_over_time({namespace=\"vault\",container=\"audit-tail\"} | json | request_path=~\"sys/audit/.+\" | operation=~\"(create|delete|update)\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = { severity = "critical", lane = "security" }
|
|
annotations = {
|
|
summary = "Vault audit device modified — attacker may be silencing visibility"
|
|
runbook = "docs/runbooks/security-incident.md#v2-audit-device-disabledmodified"
|
|
}
|
|
},
|
|
# V3: Seal status changed
|
|
{
|
|
alert = "VaultSealChanged"
|
|
expr = "sum(count_over_time({namespace=\"vault\",container=\"audit-tail\"} | json | request_path=\"sys/seal\" | operation=\"update\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = { severity = "critical", lane = "security" }
|
|
annotations = {
|
|
summary = "Vault seal status changed via API — confirm planned operation"
|
|
runbook = "docs/runbooks/security-incident.md#v3-seal-status-changed"
|
|
}
|
|
},
|
|
# V4: Policy modified
|
|
{
|
|
alert = "VaultPolicyModified"
|
|
expr = "sum(count_over_time({namespace=\"vault\",container=\"audit-tail\"} | json | request_path=~\"sys/policies/acl/.+\" | operation=~\"(create|update|delete)\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = { severity = "warning", lane = "security" }
|
|
annotations = {
|
|
summary = "Vault policy modified — verify Terraform-driven change"
|
|
runbook = "docs/runbooks/security-incident.md#v4-policy-modified"
|
|
}
|
|
},
|
|
# V5: Auth failure spike
|
|
{
|
|
alert = "VaultAuthFailureSpike"
|
|
expr = "sum(count_over_time({namespace=\"vault\",container=\"audit-tail\"} | json | type=\"response\" |~ \"\\\"error\\\":\\\"permission denied\\\"\" [1m])) > 10"
|
|
for = "1m"
|
|
labels = { severity = "warning", lane = "security" }
|
|
annotations = {
|
|
summary = "Vault permission-denied spike >10/min — possible brute force or CI rotation glitch"
|
|
runbook = "docs/runbooks/security-incident.md#v5-auth-failure-spike"
|
|
}
|
|
},
|
|
# V7: Viktor identity from non-allowlist source IP
|
|
# XFF trust enabled, so request.remote_address is the real client IP.
|
|
# Allowlist regex covers: 10.0.20.x, 192.168.1.x, pod CIDR 10.10.x.x,
|
|
# service CIDR 10.96-111.x.x, Headscale tailnet 100.64-127.x.x.
|
|
{
|
|
alert = "VaultViktorFromUnexpectedIP"
|
|
expr = "sum(count_over_time({namespace=\"vault\",container=\"audit-tail\"} | json | auth_metadata_username=\"me@viktorbarzin.me\" | request_remote_address!~\"^(10\\\\.0\\\\.2[0-3]\\\\.|192\\\\.168\\\\.1\\\\.|10\\\\.10\\\\.|10\\\\.(9[6-9]|1[01][0-9]|111)\\\\.|100\\\\.(6[4-9]|[7-9][0-9]|1[01][0-9]|12[0-7])\\\\.).*\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = { severity = "critical", lane = "security" }
|
|
annotations = {
|
|
summary = "Vault auth as me@viktorbarzin.me from non-allowlist source IP — possible stolen OIDC token"
|
|
runbook = "docs/runbooks/security-incident.md#v7-viktors-vault-identity-from-unexpected-source-ip"
|
|
}
|
|
},
|
|
# S1: PVE sshd auth success from non-allowlist IP.
|
|
# Conditional on the pve-sshd promtail unit being live on PVE host
|
|
# (deployed via stacks/infra/scripts — out of scope until W1.3 host
|
|
# piece lands). Rule is defined so it fires automatically once logs
|
|
# flow with job=sshd-pve.
|
|
{
|
|
alert = "PVEsshLoginFromUnexpectedIP"
|
|
expr = "sum(count_over_time({job=\"sshd-pve\"} |~ \"Accepted (publickey|password|keyboard-interactive)\" | regexp \"Accepted (?P<method>\\\\S+) for (?P<user>\\\\S+) from (?P<ip>\\\\S+) port\" | ip!~\"^(10\\\\.0\\\\.2[0-3]\\\\.|192\\\\.168\\\\.1\\\\.|100\\\\.(6[4-9]|[7-9][0-9]|1[01][0-9]|12[0-7])\\\\.).*\" [5m])) > 0"
|
|
for = "0m"
|
|
labels = { severity = "critical", lane = "security" }
|
|
annotations = {
|
|
summary = "PVE sshd login from non-allowlist source IP — possible stolen SSH key"
|
|
runbook = "docs/runbooks/security-incident.md#s1-pve-sshd-auth-success-from-unexpected-ip"
|
|
}
|
|
},
|
|
]
|
|
}
|
|
]
|
|
})
|
|
}
|
|
}
|
|
|
|
resource "kubernetes_config_map" "grafana_loki_datasource" {
|
|
metadata {
|
|
name = "grafana-loki-datasource"
|
|
namespace = kubernetes_namespace.monitoring.metadata[0].name
|
|
labels = {
|
|
grafana_datasource = "1"
|
|
}
|
|
}
|
|
data = {
|
|
"loki-datasource.yaml" = yamlencode({
|
|
apiVersion = 1
|
|
datasources = [{
|
|
name = "Loki"
|
|
type = "loki"
|
|
access = "proxy"
|
|
url = "http://loki.monitoring.svc.cluster.local:3100"
|
|
isDefault = false
|
|
}]
|
|
})
|
|
}
|
|
}
|