package admin import ( "bytes" "encoding/json" "net/http" "net/http/httptest" "testing" "github.com/Wei-Shaw/sub2api/internal/config" "github.com/Wei-Shaw/sub2api/internal/server/middleware" "github.com/Wei-Shaw/sub2api/internal/service" "github.com/gin-gonic/gin" ) type responseEnvelope struct { Code int `json:"code"` Message string `json:"message"` Data json.RawMessage `json:"data"` } func newOpsSystemLogTestRouter(handler *OpsHandler, withUser bool) *gin.Engine { gin.SetMode(gin.TestMode) r := gin.New() if withUser { r.Use(func(c *gin.Context) { c.Set(string(middleware.ContextKeyUser), middleware.AuthSubject{UserID: 99}) c.Next() }) } r.GET("/logs", handler.ListSystemLogs) r.POST("/logs/cleanup", handler.CleanupSystemLogs) r.GET("/logs/health", handler.GetSystemLogIngestionHealth) return r } func TestOpsSystemLogHandler_ListUnavailable(t *testing.T) { h := NewOpsHandler(nil) r := newOpsSystemLogTestRouter(h, false) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodGet, "/logs", nil) r.ServeHTTP(w, req) if w.Code != http.StatusServiceUnavailable { t.Fatalf("status=%d, want 503", w.Code) } } func TestOpsSystemLogHandler_ListInvalidUserID(t *testing.T) { svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, false) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodGet, "/logs?user_id=abc", nil) r.ServeHTTP(w, req) if w.Code != http.StatusBadRequest { t.Fatalf("status=%d, want 400", w.Code) } } func TestOpsSystemLogHandler_ListInvalidAccountID(t *testing.T) { svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, false) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodGet, "/logs?account_id=-1", nil) r.ServeHTTP(w, req) if w.Code != http.StatusBadRequest { t.Fatalf("status=%d, want 400", w.Code) } } func TestOpsSystemLogHandler_ListMonitoringDisabled(t *testing.T) { svc := service.NewOpsService(nil, nil, &config.Config{ Ops: config.OpsConfig{Enabled: false}, }, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, false) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodGet, "/logs", nil) r.ServeHTTP(w, req) if w.Code != http.StatusNotFound { t.Fatalf("status=%d, want 404", w.Code) } } func TestOpsSystemLogHandler_ListSuccess(t *testing.T) { svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, false) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodGet, "/logs?time_range=30m&page=1&page_size=20", nil) r.ServeHTTP(w, req) if w.Code != http.StatusOK { t.Fatalf("status=%d, want 200", w.Code) } var resp responseEnvelope if err := json.Unmarshal(w.Body.Bytes(), &resp); err != nil { t.Fatalf("unmarshal response: %v", err) } if resp.Code != 0 { t.Fatalf("unexpected response code: %+v", resp) } } func TestOpsSystemLogHandler_CleanupUnauthorized(t *testing.T) { svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, false) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/logs/cleanup", bytes.NewBufferString(`{"request_id":"r1"}`)) req.Header.Set("Content-Type", "application/json") r.ServeHTTP(w, req) if w.Code != http.StatusUnauthorized { t.Fatalf("status=%d, want 401", w.Code) } } func TestOpsSystemLogHandler_CleanupInvalidPayload(t *testing.T) { svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, true) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/logs/cleanup", bytes.NewBufferString(`{bad-json`)) req.Header.Set("Content-Type", "application/json") r.ServeHTTP(w, req) if w.Code != http.StatusBadRequest { t.Fatalf("status=%d, want 400", w.Code) } } func TestOpsSystemLogHandler_CleanupInvalidTime(t *testing.T) { svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, true) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/logs/cleanup", bytes.NewBufferString(`{"start_time":"bad","request_id":"r1"}`)) req.Header.Set("Content-Type", "application/json") r.ServeHTTP(w, req) if w.Code != http.StatusBadRequest { t.Fatalf("status=%d, want 400", w.Code) } } func TestOpsSystemLogHandler_CleanupInvalidEndTime(t *testing.T) { svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, true) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/logs/cleanup", bytes.NewBufferString(`{"end_time":"bad","request_id":"r1"}`)) req.Header.Set("Content-Type", "application/json") r.ServeHTTP(w, req) if w.Code != http.StatusBadRequest { t.Fatalf("status=%d, want 400", w.Code) } } func TestOpsSystemLogHandler_CleanupServiceUnavailable(t *testing.T) { svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, true) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/logs/cleanup", bytes.NewBufferString(`{"request_id":"r1"}`)) req.Header.Set("Content-Type", "application/json") r.ServeHTTP(w, req) if w.Code != http.StatusServiceUnavailable { t.Fatalf("status=%d, want 503", w.Code) } } func TestOpsSystemLogHandler_CleanupMonitoringDisabled(t *testing.T) { svc := service.NewOpsService(nil, nil, &config.Config{ Ops: config.OpsConfig{Enabled: false}, }, nil, nil, nil, nil, nil, nil, nil, nil) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, true) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/logs/cleanup", bytes.NewBufferString(`{"request_id":"r1"}`)) req.Header.Set("Content-Type", "application/json") r.ServeHTTP(w, req) if w.Code != http.StatusNotFound { t.Fatalf("status=%d, want 404", w.Code) } } func TestOpsSystemLogHandler_Health(t *testing.T) { sink := service.NewOpsSystemLogSink(nil) svc := service.NewOpsService(nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, sink) h := NewOpsHandler(svc) r := newOpsSystemLogTestRouter(h, false) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodGet, "/logs/health", nil) r.ServeHTTP(w, req) if w.Code != http.StatusOK { t.Fatalf("status=%d, want 200", w.Code) } } func TestOpsSystemLogHandler_HealthUnavailableAndMonitoringDisabled(t *testing.T) { h := NewOpsHandler(nil) r := newOpsSystemLogTestRouter(h, false) w := httptest.NewRecorder() req := httptest.NewRequest(http.MethodGet, "/logs/health", nil) r.ServeHTTP(w, req) if w.Code != http.StatusServiceUnavailable { t.Fatalf("status=%d, want 503", w.Code) } svc := service.NewOpsService(nil, nil, &config.Config{ Ops: config.OpsConfig{Enabled: false}, }, nil, nil, nil, nil, nil, nil, nil, nil) h = NewOpsHandler(svc) r = newOpsSystemLogTestRouter(h, false) w = httptest.NewRecorder() req = httptest.NewRequest(http.MethodGet, "/logs/health", nil) r.ServeHTTP(w, req) if w.Code != http.StatusNotFound { t.Fatalf("status=%d, want 404", w.Code) } }