Files
ragflow/internal/handler/connector_test.go

326 lines
8.9 KiB
Go
Raw Permalink Normal View History

package handler
import (
"encoding/json"
"fmt"
"net/http"
"net/http/httptest"
"testing"
"time"
"github.com/gin-gonic/gin"
"ragflow/internal/common"
"ragflow/internal/entity"
"ragflow/internal/service"
)
type fakeConnectorService struct {
connector *entity.Connector
logs []*entity.ConnectorSyncLog
total int64
code common.ErrorCode
err error
}
func (s fakeConnectorService) ListConnectors(string) (*service.ListConnectorsResponse, error) {
return &service.ListConnectorsResponse{}, nil
}
func (s fakeConnectorService) TestConnector(string, string) error {
return s.err
}
func (s fakeConnectorService) CreateConnector(string, *service.CreateConnectorRequest) (*entity.Connector, error) {
if s.err != nil {
return nil, s.err
}
return s.connector, nil
}
func (s fakeConnectorService) GetConnector(string, string) (*entity.Connector, common.ErrorCode, error) {
if s.err != nil {
return nil, s.code, s.err
}
return s.connector, common.CodeSuccess, nil
}
func (s fakeConnectorService) UpdateConnector(string, string, *service.UpdateConnectorRequest) (*entity.Connector, common.ErrorCode, error) {
if s.err != nil {
return nil, s.code, s.err
}
return s.connector, common.CodeSuccess, nil
}
func (s fakeConnectorService) StartGoogleWebOAuth(string, string, *service.StartGoogleWebOAuthRequest) (*service.StartGoogleWebOAuthResponse, common.ErrorCode, error) {
if s.err != nil {
return nil, s.code, s.err
}
return &service.StartGoogleWebOAuthResponse{}, common.CodeSuccess, nil
}
func (s fakeConnectorService) GoogleWebOAuthCallback(string, string, string, string, string) string {
return ""
}
func (s fakeConnectorService) PollGoogleWebOAuthResult(string, string, *service.PollGoogleWebOAuthResultRequest) (*service.PollGoogleWebOAuthResultResponse, common.ErrorCode, error) {
if s.err != nil {
return nil, s.code, s.err
}
return &service.PollGoogleWebOAuthResultResponse{}, common.CodeSuccess, nil
}
func (s fakeConnectorService) ListLog(string, string, int, int) ([]*entity.ConnectorSyncLog, int64, common.ErrorCode, error) {
if s.err != nil {
return nil, 0, s.code, s.err
}
return s.logs, s.total, common.CodeSuccess, nil
}
func (s fakeConnectorService) DeleteConnector(string, string) (bool, common.ErrorCode, error) {
if s.err != nil {
return false, s.code, s.err
}
return true, common.CodeSuccess, nil
}
func (s fakeConnectorService) RebuildConnector(string, string, string) (bool, common.ErrorCode, error) {
if s.err != nil {
return false, s.code, s.err
}
return true, common.CodeSuccess, nil
}
func TestConnectorHandlerTestConnector(t *testing.T) {
gin.SetMode(gin.TestMode)
tests := []struct {
name string
err error
wantCode common.ErrorCode
}{
{
name: "success",
err: nil,
wantCode: common.CodeSuccess,
},
{
name: "not found",
err: service.ErrConnectorNotFound,
wantCode: common.CodeDataError,
},
{
name: "unauthorized",
err: service.ErrConnectorNoAuth,
wantCode: common.CodeAuthenticationError,
},
{
name: "unsupported source",
err: service.ErrConnectorTestUnsupported,
wantCode: common.CodeArgumentError,
},
{
name: "validation failure",
err: fmt.Errorf("connector credentials are missing"),
wantCode: common.CodeDataError,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
h := &ConnectorHandler{connectorService: fakeConnectorService{err: tt.err}}
router := gin.New()
router.POST("/api/v1/connectors/:connector_id/test", func(c *gin.Context) {
c.Set("user", &entity.User{ID: "tenant-1"})
h.TestConnector(c)
})
resp := httptest.NewRecorder()
req := httptest.NewRequest(http.MethodPost, "/api/v1/connectors/connector-1/test", nil)
router.ServeHTTP(resp, req)
if resp.Code != http.StatusOK {
t.Fatalf("status=%d body=%s", resp.Code, resp.Body.String())
}
var body map[string]interface{}
if err := json.Unmarshal(resp.Body.Bytes(), &body); err != nil {
t.Fatalf("unmarshal response: %v", err)
}
if body["code"] != float64(tt.wantCode) {
t.Fatalf("code=%v want=%v body=%v", body["code"], tt.wantCode, body)
}
})
}
}
func TestConnectorHandlerDeleteConnector(t *testing.T) {
gin.SetMode(gin.TestMode)
tests := []struct {
name string
service fakeConnectorService
wantCode common.ErrorCode
wantData interface{}
wantMsg string
}{
{
name: "success",
service: fakeConnectorService{},
wantCode: common.CodeSuccess,
wantData: true,
},
{
name: "unauthorized",
service: fakeConnectorService{code: common.CodeAuthenticationError, err: fmt.Errorf("No authorization.")},
wantCode: common.CodeAuthenticationError,
wantData: nil,
wantMsg: "No authorization.",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
h := &ConnectorHandler{connectorService: tt.service}
router := gin.New()
router.DELETE("/api/v1/connectors/:connector_id", func(c *gin.Context) {
c.Set("user", &entity.User{ID: "tenant-1"})
h.DeleteConnector(c)
})
resp := httptest.NewRecorder()
req := httptest.NewRequest(http.MethodDelete, "/api/v1/connectors/connector-1", nil)
router.ServeHTTP(resp, req)
if resp.Code != http.StatusOK {
t.Fatalf("status=%d body=%s", resp.Code, resp.Body.String())
}
var body map[string]interface{}
if err := json.Unmarshal(resp.Body.Bytes(), &body); err != nil {
t.Fatalf("unmarshal response: %v", err)
}
if body["code"] != float64(tt.wantCode) {
t.Fatalf("code=%v body=%v", body["code"], body)
}
if body["data"] != tt.wantData {
t.Fatalf("data=%v body=%v", body["data"], body)
}
if tt.wantMsg != "" && body["message"] != tt.wantMsg {
t.Fatalf("message=%v", body["message"])
}
})
}
}
func TestConnectorHandlerListLogs(t *testing.T) {
gin.SetMode(gin.TestMode)
startedAt := time.Date(2026, 5, 28, 8, 30, 0, 0, time.Local)
updatedAt := time.Date(2026, 5, 28, 9, 0, 0, 0, time.Local)
tests := []struct {
name string
service fakeConnectorService
wantCode common.ErrorCode
wantMsg string
wantTotal float64
wantLogID string
wantLogs int
}{
{
name: "success",
service: fakeConnectorService{
logs: []*entity.ConnectorSyncLog{{
ID: "log-1",
ConnectorID: "connector-1",
TaskType: "sync",
KbID: "kb-1",
UpdateDate: &updatedAt,
NewDocsIndexed: 2,
TotalDocsIndexed: 10,
DocsRemovedFromIndex: 1,
ErrorMsg: "",
ErrorCount: 0,
TimeStarted: &startedAt,
RefreshFreq: 5,
PruneFreq: 5,
KbName: "Docs",
Status: "3",
}},
total: 1,
},
wantCode: common.CodeSuccess,
wantTotal: 1,
wantLogID: "log-1",
wantLogs: 1,
},
{
name: "empty logs",
service: fakeConnectorService{
logs: nil,
total: 0,
},
wantCode: common.CodeSuccess,
wantTotal: 0,
wantLogs: 0,
},
{
name: "unauthorized",
service: fakeConnectorService{code: common.CodeAuthenticationError, err: fmt.Errorf("No authorization.")},
wantCode: common.CodeAuthenticationError,
wantMsg: "No authorization.",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
h := &ConnectorHandler{connectorService: tt.service}
router := gin.New()
router.GET("/api/v1/connectors/:connector_id/logs", func(c *gin.Context) {
c.Set("user", &entity.User{ID: "tenant-1"})
h.ListLogs(c)
})
resp := httptest.NewRecorder()
req := httptest.NewRequest(http.MethodGet, "/api/v1/connectors/connector-1/logs?page=2&page_size=5", nil)
router.ServeHTTP(resp, req)
if resp.Code != http.StatusOK {
t.Fatalf("status=%d body=%s", resp.Code, resp.Body.String())
}
var body map[string]interface{}
if err := json.Unmarshal(resp.Body.Bytes(), &body); err != nil {
t.Fatalf("unmarshal response: %v", err)
}
if body["code"] != float64(tt.wantCode) {
t.Fatalf("code=%v body=%v", body["code"], body)
}
if tt.wantMsg != "" && body["message"] != tt.wantMsg {
t.Fatalf("message=%v", body["message"])
}
if tt.wantLogID != "" {
data := body["data"].(map[string]interface{})
if data["total"] != tt.wantTotal {
t.Fatalf("total=%v body=%v", data["total"], body)
}
logs := data["logs"].([]interface{})
if len(logs) != tt.wantLogs {
t.Fatalf("logs=%v body=%v", logs, body)
}
if logs[0].(map[string]interface{})["id"] != tt.wantLogID {
t.Fatalf("logs=%v body=%v", logs, body)
}
}
if tt.wantLogID == "" && tt.wantMsg == "" {
data := body["data"].(map[string]interface{})
if data["total"] != tt.wantTotal {
t.Fatalf("total=%v body=%v", data["total"], body)
}
logs := data["logs"].([]interface{})
if len(logs) != tt.wantLogs {
t.Fatalf("logs=%v body=%v", logs, body)
}
}
})
}
}