mirror of
https://github.com/infiniflow/ragflow.git
synced 2026-06-29 23:41:12 +08:00
### What problem does this PR solve? The following API is available in go > /api/v1/connectors/google/oauth/web/start POST > /api/v1/connectors/gmail/oauth/web/callback GET > /api/v1/connectors/google-drive/oauth/web/callback GET > /api/v1/connectors/google/oauth/web/result POST ### Type of change - [x] New Feature (non-breaking change which adds functionality)
326 lines
8.9 KiB
Go
326 lines
8.9 KiB
Go
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)
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|