Mend.io Vulnerability Database
The largest open source vulnerability database
What is a Vulnerability ID?
New vulnerability? Tell us about it!
CVE-2026-47726
Published:June 09, 2026
Updated:June 15, 2026
"internal/api/audit.go:12" — "handleGetAuditLog" does no admin check. The route is bearer-auth gated only; any operator API key returns the full audit log via "store.ListAuditEntries" (up to limit=1000). This includes cross-tenant actor names, host/CA/operator IDs, action timestamps, and masked-IP entries from rate-limit refusals — enough surface for a tenant to enumerate the server's activity, infer staffing patterns, or identify high-value targets. Affected All released versions up to v0.3.1. Reproducer curl -H "Authorization: Bearer <any-operator-key>" https://server/api/v1/audit-log?limit=1000 Suggested fix Two options, either acceptable: 1. "if !actorIsAdmin(ctx) { 403 }" — strictest; matches the "operator management is admin-only" stance. 2. Scope to actor: filter "store.ListAuditEntries" by "actor.Username" plus a subquery of CA IDs the actor owns. Operators see their own audit entries plus entries against their CA's resources. Recommend option 1 unless the UI needs per-operator audit views. Suggested patch Verified locally: "go vet", "go test -race -count=1 ./...", "golangci-lint v2.12" all clean. diff --git a/internal/api/audit.go b/internal/api/audit.go index 3236631..57b57ce 100644 --- a/internal/api/audit.go +++ b/internal/api/audit.go @@ -10,6 +10,10 @@ import ( const defaultAuditLimit = 100 func (s *Server) handleGetAuditLog(w http.ResponseWriter, r *http.Request) { + if !actorIsAdmin(r.Context()) { + writeError(w, http.StatusForbidden, "audit log access requires the admin role") + return + } filter := store.AuditFilter{ Action: r.URL.Query().Get("action"), Limit: defaultAuditLimit, diff --git a/internal/api/audit_admin_test.go b/internal/api/audit_admin_test.go new file mode 100644 index 0000000..47e1ca4 --- /dev/null +++ b/internal/api/audit_admin_test.go @@ -0,0 +1,62 @@ +package api + +import ( + "context" + "crypto/sha256" + "encoding/hex" + "net/http" + "net/http/httptest" + "testing" + + "github.com/google/uuid" + "github.com/juev/nebula-mesh/internal/models" +) + +// TestHandleGetAuditLog_NonAdminForbidden confirms a non-admin operator +// API key cannot read the audit log. The legacy config-key path stays +// admin and is covered by the happy-path test elsewhere. +func TestHandleGetAuditLog_NonAdminForbidden(t *testing.T) { + srv, _ := newTestServer(t) + + nonAdminKey := uuid.New().String() + keyHash := sha256.Sum256([]byte(nonAdminKey)) + if err := srv.store.CreateOperator(context.Background(), &models.Operator{ + ID: uuid.New().String(), Username: "non-admin", PasswordHash: "x", + Role: "user", Status: models.OperatorStatusActive, + }); err != nil { + t.Fatal(err) + } + op, err := srv.store.GetOperatorByUsername(context.Background(), "non-admin") + if err != nil { + t.Fatal(err) + } + if err := srv.store.CreateOperatorAPIKey(context.Background(), &models.OperatorAPIKey{ + ID: uuid.New().String(), OperatorID: op.ID, KeyHash: hex.EncodeToString(keyHash[:]), + }); err != nil { + t.Fatal(err) + } + + req := httptest.NewRequest("GET", "/api/v1/audit-log", nil) + req.Header.Set("Authorization", "Bearer "+nonAdminKey) + rec := httptest.NewRecorder() + srv.ServeHTTP(rec, req) + + if rec.Code != http.StatusForbidden { + t.Errorf("non-admin audit-log status = %d, want 403", rec.Code) + } +} + +// TestHandleGetAuditLog_LegacyKeyAllowed confirms the legacy config-key +// path still reaches the handler (preserves backward compatibility). +func TestHandleGetAuditLog_LegacyKeyAllowed(t *testing.T) { + srv, _ := newTestServer(t) + + req := httptest.NewRequest("GET", "/api/v1/audit-log", nil) + req.Header.Set("Authorization", "Bearer "+testAPIKey) + rec := httptest.NewRecorder() + srv.ServeHTTP(rec, req) + + if rec.Code == http.StatusForbidden { + t.Errorf("legacy key rejected with 403; want pass-through") + } +}
Affected Packages
https://github.com/juev/nebula-mesh.git (GITHUB):
Affected version(s) >=v0.1.1 <v0.3.2
Fix Suggestion:
Update to version v0.3.2
github.com/juev/nebula-mesh (GO):
Affected version(s) >=v0.1.0 <v0.3.2
Fix Suggestion:
Update to version v0.3.2
Do you need more information?
Contact Us
CVSS v4
Base Score:
7.1
Attack Vector
NETWORK
Attack Complexity
LOW
Attack Requirements
NONE
Privileges Required
LOW
User Interaction
NONE
Vulnerable System Confidentiality
HIGH
Vulnerable System Integrity
NONE
Vulnerable System Availability
NONE
Subsequent System Confidentiality
NONE
Subsequent System Integrity
NONE
Subsequent System Availability
NONE
CVSS v3
Base Score:
6.5
Attack Vector
NETWORK
Attack Complexity
LOW
Privileges Required
LOW
User Interaction
NONE
Scope
UNCHANGED
Confidentiality
HIGH
Integrity
NONE
Availability
NONE
Weakness Type (CWE)
Improper Authorization