234 lines
7.4 KiB
Go
234 lines
7.4 KiB
Go
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)
|
|
}
|
|
}
|