Files
Pulse/internal/api/router_state_test.go
rcourtman 9072b8eaa8 feat: enhance API router with multi-tenant authorization
Router & Middleware:
- Add auth context middleware for user/token extraction
- Add tenant middleware with authorization checking
- Refactor middleware chain ordering for proper isolation
- Add router helpers for common patterns

Authentication & SSO:
- Enhance auth with tenant-aware context
- Update OIDC, SAML, and SSO handlers for multi-tenant
- Add RBAC handler improvements
- Add security enhancements

New Test Coverage:
- API foundation tests
- Auth and authorization tests
- Router state and general tests
- SSO handler CRUD tests
- WebSocket isolation tests
- Resource handler tests
2026-01-24 22:42:23 +00:00

78 lines
2.1 KiB
Go

package api
import (
"encoding/json"
"net/http"
"net/http/httptest"
"testing"
"github.com/rcourtman/pulse-go-rewrite/internal/config"
"github.com/rcourtman/pulse-go-rewrite/internal/mock"
"github.com/rcourtman/pulse-go-rewrite/internal/models"
"github.com/rcourtman/pulse-go-rewrite/internal/monitoring"
"github.com/rcourtman/pulse-go-rewrite/pkg/auth"
"github.com/stretchr/testify/assert"
)
func TestRouter_HandleState_MockIsolation(t *testing.T) {
// Enable mock mode to control GetState output
t.Setenv("PULSE_MOCK_MODE", "true")
mock.SetEnabled(true)
defer mock.SetEnabled(false)
dataPath := t.TempDir()
hp, _ := auth.HashPassword("password")
cfg := &config.Config{
DataPath: dataPath,
MultiTenantEnabled: true,
AuthUser: "admin",
AuthPass: hp,
}
// Initialize persistent stores to avoid permission issues with /etc/pulse
InitSessionStore(dataPath)
InitCSRFStore(dataPath)
// Create a router with a real monitor
// Since monitor.GetState() checks IsMockEnabled(), it will return mock data.
monitor, _ := monitoring.New(cfg)
router := &Router{
config: cfg,
monitor: monitor,
persistence: config.NewConfigPersistence(dataPath),
}
t.Run("basic state access", func(t *testing.T) {
req := httptest.NewRequest("GET", "/api/state", nil)
// Set credentials for CheckAuth (Basic Auth)
req.SetBasicAuth("admin", "password")
w := httptest.NewRecorder()
router.handleState(w, req)
assert.Equal(t, http.StatusOK, w.Code)
var state models.StateFrontend
err := json.Unmarshal(w.Body.Bytes(), &state)
assert.NoError(t, err)
// Verify we got the mock state
assert.NotEmpty(t, state.Nodes)
assert.Greater(t, state.LastUpdate, int64(0))
})
t.Run("invalid method", func(t *testing.T) {
req := httptest.NewRequest("POST", "/api/state", nil)
w := httptest.NewRecorder()
router.handleState(w, req)
assert.Equal(t, http.StatusMethodNotAllowed, w.Code)
})
t.Run("unauthorized", func(t *testing.T) {
req := httptest.NewRequest("GET", "/api/state", nil)
w := httptest.NewRecorder()
router.handleState(w, req)
assert.Equal(t, http.StatusUnauthorized, w.Code)
})
}