Files
fn_registry/functions/infra/rbac_middleware.md
T
egutierrez f46fde3656 feat: rbac_check (pure), jwt_middleware, rbac_middleware
Fase 5 del issue 0010 — RBAC y middlewares de auth.
- rbac_check es pura: solo recorre la matriz roles/permisos
- jwt_middleware extrae token del header Authorization: Bearer, valida e
  inyecta claims en el context con una key privada (jwtCtxKey struct{})
- rbac_middleware requiere jwt_middleware antes; lee role de claims.Custom
- Helper JWTClaimsFromContext para acceder a las claims desde handlers
- 401 claro si RBAC se usa sin JWT antes (code: no_claims)
2026-04-18 17:44:04 +02:00

45 lines
2.2 KiB
Markdown

---
name: rbac_middleware
kind: function
lang: go
domain: infra
version: "1.0.0"
purity: impure
signature: "func RBACMiddleware(roles []Role, required Permission) Middleware"
description: "Middleware HTTP que verifica que el usuario autenticado tenga un permiso concreto. Lee las claims del context (puestas por JWTMiddleware), extrae el rol de claims.Custom[role] y evalua con RBACCheck. Responde 403 si no tiene permiso, 401 si no hay claims."
tags: [rbac, auth, middleware, http, server, infra]
uses_functions: [rbac_check_go_infra, http_error_response_go_infra]
uses_types: [Role_go_infra, Permission_go_infra, Middleware_go_infra, HTTPError_go_infra]
returns: [Middleware_go_infra]
returns_optional: false
error_type: error_go_core
imports: [net/http]
params:
- name: roles
desc: "matriz de roles y sus permisos, usualmente hardcoded o leida de config al arrancar"
- name: required
desc: "permiso requerido para acceder al handler (Resource + Action)"
output: "Middleware que bloquea el request con 403 si el rol del usuario no tiene el permiso requerido"
tested: true
tests: ["pasa con rol que tiene el permiso", "403 con rol sin permiso", "403 con rol inexistente", "401 si no hay claims en el context"]
test_file_path: "functions/infra/rbac_middleware_test.go"
file_path: "functions/infra/rbac_middleware.go"
---
## Ejemplo
```go
roles := []Role{
{Name: "admin", Permissions: []Permission{{Resource: "users", Action: "delete"}}},
}
adminProtected := HTTPMiddlewareChain(
JWTMiddleware(secret),
RBACMiddleware(roles, Permission{Resource: "users", Action: "delete"}),
)
mux.Handle("DELETE /api/users/{id}", adminProtected(deleteUserHandler))
```
## Notas
Impura — depende del estado del request y de JWTMiddleware. El rol se lee de `claims.Custom["role"]` como string: si la app usa otro campo (ej: `claims.Custom["roles"]` como slice) conviene crear un middleware variante. Si el usuario tiene multiples roles, extenderse iterando `RBACCheck` sobre cada uno. Orden en la chain: SIEMPRE JWTMiddleware antes — si se olvida, el middleware retorna 401 con code `no_claims` para facilitar debugging. No cachea la evaluacion: RBACCheck es O(roles*permisos) pero para listas pequeñas (<100) es despreciable.