2025-11-27 17:38:42 +08:00
|
|
|
|
package ragflow
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"context"
|
2025-12-10 09:50:54 +08:00
|
|
|
|
"net"
|
2025-11-27 17:38:42 +08:00
|
|
|
|
"net/http"
|
|
|
|
|
|
"net/url"
|
|
|
|
|
|
"strings"
|
2025-12-09 17:55:08 +08:00
|
|
|
|
"sync"
|
2025-12-09 09:20:44 +08:00
|
|
|
|
"time"
|
2025-11-27 17:38:42 +08:00
|
|
|
|
|
2025-12-06 18:04:29 +08:00
|
|
|
|
"github.com/gogf/gf/v2/encoding/gjson"
|
|
|
|
|
|
"github.com/gogf/gf/v2/errors/gerror"
|
2025-12-03 09:59:40 +08:00
|
|
|
|
"github.com/gogf/gf/v2/frame/g"
|
2025-11-27 17:38:42 +08:00
|
|
|
|
"github.com/gogf/gf/v2/net/gclient"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2025-12-03 09:59:40 +08:00
|
|
|
|
var (
|
2025-12-09 17:55:08 +08:00
|
|
|
|
// globalClient 全局 RAGFlow 客户端(单例,延迟初始化)
|
2025-12-03 09:59:40 +08:00
|
|
|
|
globalClient *Client
|
2025-12-09 17:55:08 +08:00
|
|
|
|
clientOnce sync.Once
|
2025-12-03 09:59:40 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
2025-12-09 17:55:08 +08:00
|
|
|
|
// initClient 延迟初始化客户端
|
|
|
|
|
|
func initClient() {
|
|
|
|
|
|
clientOnce.Do(func() {
|
|
|
|
|
|
ctx := context.Background()
|
2025-11-27 17:38:42 +08:00
|
|
|
|
|
2025-12-09 17:55:08 +08:00
|
|
|
|
// 读取配置
|
|
|
|
|
|
baseURL, apiKey := loadConfig(ctx)
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-12-09 17:55:08 +08:00
|
|
|
|
// 如果配置不完整,跳过初始化
|
|
|
|
|
|
if baseURL == "" || apiKey == "" {
|
|
|
|
|
|
g.Log().Warning(ctx, "⚠️ RAGFlow 配置未找到,请在项目 config.yml 中添加 ragflow.base_url 和 ragflow.api_key")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
2025-12-03 09:59:40 +08:00
|
|
|
|
|
2025-12-10 09:50:54 +08:00
|
|
|
|
// 自定义 Transport,增大连接池(解决并发连接不足导致的超时)
|
|
|
|
|
|
transport := &http.Transport{
|
|
|
|
|
|
Proxy: http.ProxyFromEnvironment,
|
|
|
|
|
|
DialContext: (&net.Dialer{
|
|
|
|
|
|
Timeout: 30 * time.Second,
|
|
|
|
|
|
KeepAlive: 30 * time.Second,
|
|
|
|
|
|
}).DialContext,
|
|
|
|
|
|
MaxIdleConns: 200, // 最大空闲连接数
|
|
|
|
|
|
MaxIdleConnsPerHost: 100, // 每个 host 最大空闲连接数(关键!默认只有 2)
|
|
|
|
|
|
MaxConnsPerHost: 100, // 每个 host 最大连接数
|
|
|
|
|
|
IdleConnTimeout: 90 * time.Second, // 空闲连接超时
|
|
|
|
|
|
TLSHandshakeTimeout: 10 * time.Second,
|
|
|
|
|
|
ExpectContinueTimeout: 1 * time.Second,
|
|
|
|
|
|
ResponseHeaderTimeout: 180 * time.Second, // 等待响应头超时(关键!)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-09 17:55:08 +08:00
|
|
|
|
// 初始化全局客户端
|
|
|
|
|
|
httpClient := gclient.New()
|
2025-12-10 09:50:54 +08:00
|
|
|
|
httpClient.SetBrowserMode(false)
|
2025-12-09 17:55:08 +08:00
|
|
|
|
httpClient.SetHeader("Authorization", "Bearer "+apiKey)
|
|
|
|
|
|
httpClient.SetHeader("Content-Type", "application/json")
|
|
|
|
|
|
httpClient.SetTimeout(180 * time.Second) // RAGFlow AI 推理需要较长时间
|
2025-12-03 09:59:40 +08:00
|
|
|
|
|
2025-12-10 09:50:54 +08:00
|
|
|
|
// 设置自定义 Transport
|
|
|
|
|
|
httpClient.Client.Transport = transport
|
|
|
|
|
|
|
2025-12-09 17:55:08 +08:00
|
|
|
|
globalClient = &Client{
|
|
|
|
|
|
BaseURL: strings.TrimSuffix(baseURL, "/"),
|
|
|
|
|
|
APIKey: apiKey,
|
|
|
|
|
|
HTTPClient: httpClient,
|
|
|
|
|
|
}
|
2025-12-03 09:59:40 +08:00
|
|
|
|
|
2025-12-09 17:55:08 +08:00
|
|
|
|
g.Log().Infof(ctx, "✅ RAGFlow 全局客户端初始化成功: baseURL=%s", baseURL)
|
|
|
|
|
|
})
|
2025-12-03 09:59:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// loadConfig 从配置文件加载 RAGFlow 配置
|
|
|
|
|
|
func loadConfig(ctx context.Context) (baseURL, apiKey string) {
|
2025-12-03 10:09:00 +08:00
|
|
|
|
// 使用 GoFrame 全局配置(从项目的 config.yml 读取)
|
|
|
|
|
|
baseURL = g.Cfg().MustGet(ctx, "ragflow.base_url", "").String()
|
|
|
|
|
|
apiKey = g.Cfg().MustGet(ctx, "ragflow.api_key", "").String()
|
2025-12-06 10:13:38 +08:00
|
|
|
|
return
|
2025-12-03 09:59:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-09 17:55:08 +08:00
|
|
|
|
// GetGlobalClient 获取全局客户端(延迟初始化)
|
2025-12-03 09:59:40 +08:00
|
|
|
|
// 使用示例:client := ragflow.GetGlobalClient()
|
|
|
|
|
|
func GetGlobalClient() *Client {
|
2025-12-09 17:55:08 +08:00
|
|
|
|
initClient()
|
2025-12-03 09:59:40 +08:00
|
|
|
|
return globalClient
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Client RAGFlow API 客户端
|
|
|
|
|
|
type Client struct {
|
|
|
|
|
|
BaseURL string
|
|
|
|
|
|
APIKey string
|
|
|
|
|
|
HTTPClient *gclient.Client // HTTP 客户端
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// CommonResponse 通用响应结构
|
|
|
|
|
|
type CommonResponse struct {
|
2025-11-27 18:03:01 +08:00
|
|
|
|
Code int `json:"code"`
|
|
|
|
|
|
Message string `json:"message"`
|
2025-11-27 17:38:42 +08:00
|
|
|
|
Data interface{} `json:"data,omitempty"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// IsSuccess 检查响应是否成功
|
|
|
|
|
|
func (r *CommonResponse) IsSuccess() bool {
|
|
|
|
|
|
return r.Code == 0
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// request 发送 HTTP 请求
|
2025-12-06 18:04:29 +08:00
|
|
|
|
func (c *Client) request(ctx context.Context, method, path string, body interface{}, result interface{}) (err error) {
|
2025-11-27 17:38:42 +08:00
|
|
|
|
fullURL := c.BaseURL + path
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-12-06 18:04:29 +08:00
|
|
|
|
var reqBody string
|
2025-11-27 17:38:42 +08:00
|
|
|
|
if body != nil {
|
2025-12-06 18:04:29 +08:00
|
|
|
|
jsonData, err := gjson.Encode(body)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
if err != nil {
|
2025-12-06 18:04:29 +08:00
|
|
|
|
return gerror.Newf("marshal request body failed: %v", err)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
2025-12-06 18:04:29 +08:00
|
|
|
|
reqBody = string(jsonData)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-12-10 09:50:54 +08:00
|
|
|
|
// 设置 180 秒超时(RAGFlow AI 推理需要较长时间)
|
|
|
|
|
|
reqCtx, cancel := context.WithTimeout(ctx, 180*time.Second)
|
|
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
2025-11-27 17:38:42 +08:00
|
|
|
|
var resp *gclient.Response
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-11-27 17:38:42 +08:00
|
|
|
|
switch method {
|
|
|
|
|
|
case "GET":
|
2025-12-10 09:50:54 +08:00
|
|
|
|
resp, err = c.HTTPClient.Get(reqCtx, fullURL)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
case "POST":
|
2025-12-10 09:50:54 +08:00
|
|
|
|
resp, err = c.HTTPClient.Post(reqCtx, fullURL, reqBody)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
case "PUT":
|
2025-12-10 09:50:54 +08:00
|
|
|
|
resp, err = c.HTTPClient.Put(reqCtx, fullURL, reqBody)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
case "DELETE":
|
2025-12-10 09:50:54 +08:00
|
|
|
|
resp, err = c.HTTPClient.Delete(reqCtx, fullURL, reqBody)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
default:
|
2025-12-06 18:04:29 +08:00
|
|
|
|
return gerror.Newf("unsupported method: %s", method)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-11-27 17:38:42 +08:00
|
|
|
|
if err != nil {
|
2025-12-10 09:50:54 +08:00
|
|
|
|
g.Log().Errorf(ctx, "[RAGFlow HTTP] 请求失败: method=%s, url=%s, error=%v", method, fullURL, err)
|
|
|
|
|
|
return gerror.Newf("request failed: %v", err)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
defer resp.Close()
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-12-10 09:50:54 +08:00
|
|
|
|
respBody := resp.ReadAll()
|
|
|
|
|
|
|
|
|
|
|
|
// 打印响应详情
|
|
|
|
|
|
g.Log().Debugf(ctx, "[RAGFlow HTTP] 响应: status=%d, body=%s", resp.StatusCode, string(respBody))
|
|
|
|
|
|
|
2025-11-27 17:38:42 +08:00
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
2025-12-10 09:50:54 +08:00
|
|
|
|
g.Log().Errorf(ctx, "[RAGFlow HTTP] 非200响应: status=%d, body=%s", resp.StatusCode, string(respBody))
|
|
|
|
|
|
return gerror.Newf("http status %d: %s", resp.StatusCode, string(respBody))
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-12-06 18:04:29 +08:00
|
|
|
|
if err = gjson.DecodeTo(respBody, result); err != nil {
|
2025-12-10 09:50:54 +08:00
|
|
|
|
g.Log().Errorf(ctx, "[RAGFlow HTTP] 解析响应失败: body=%s, error=%v", string(respBody), err)
|
2025-12-06 18:04:29 +08:00
|
|
|
|
return gerror.Newf("unmarshal response failed: %v", err)
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-12-06 18:04:29 +08:00
|
|
|
|
return
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// buildQueryString 构建查询字符串
|
|
|
|
|
|
func buildQueryString(params map[string]interface{}) string {
|
|
|
|
|
|
if len(params) == 0 {
|
|
|
|
|
|
return ""
|
|
|
|
|
|
}
|
2025-11-27 18:03:01 +08:00
|
|
|
|
|
2025-12-06 18:04:29 +08:00
|
|
|
|
parts := make([]string, 0, len(params))
|
2025-11-27 17:38:42 +08:00
|
|
|
|
for k, v := range params {
|
2025-12-06 18:04:29 +08:00
|
|
|
|
parts = append(parts, url.QueryEscape(k)+"="+url.QueryEscape(g.NewVar(v).String()))
|
2025-11-27 17:38:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
return strings.Join(parts, "&")
|
|
|
|
|
|
}
|