2026-03-04 19:17:16 +08:00
|
|
|
//
|
|
|
|
|
// Copyright 2026 The InfiniFlow Authors. All Rights Reserved.
|
|
|
|
|
//
|
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
|
//
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
//
|
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
|
// limitations under the License.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
package handler
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"net/http"
|
2026-05-06 10:41:58 +08:00
|
|
|
"ragflow/internal/common"
|
2026-03-04 19:17:16 +08:00
|
|
|
"ragflow/internal/server"
|
2026-03-24 20:08:36 +08:00
|
|
|
"ragflow/internal/service"
|
2026-03-04 19:17:16 +08:00
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// SystemHandler system handler
|
|
|
|
|
type SystemHandler struct {
|
|
|
|
|
systemService *service.SystemService
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewSystemHandler create system handler
|
|
|
|
|
func NewSystemHandler(systemService *service.SystemService) *SystemHandler {
|
|
|
|
|
return &SystemHandler{
|
|
|
|
|
systemService: systemService,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Ping health check endpoint
|
|
|
|
|
// @Summary Ping
|
|
|
|
|
// @Description Simple ping endpoint
|
|
|
|
|
// @Tags system
|
|
|
|
|
// @Produce plain
|
|
|
|
|
// @Success 200 {string} string "pong"
|
|
|
|
|
// @Router /v1/system/ping [get]
|
|
|
|
|
func (h *SystemHandler) Ping(c *gin.Context) {
|
|
|
|
|
c.String(http.StatusOK, "pong")
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-07 19:07:47 +08:00
|
|
|
// Health check
|
2026-03-30 18:40:58 +08:00
|
|
|
func (h *SystemHandler) Health(c *gin.Context) {
|
|
|
|
|
c.JSON(200, gin.H{
|
|
|
|
|
"status": "ok",
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
Go: implement system healthz API (#15307)
## Summary
- Add Go REST support for `GET /api/v1/system/healthz`.
- Return Python-compatible `ok`/`nok` dependency fields for DB, Redis,
document engine, and storage.
- Return HTTP 200 only when all checks pass; otherwise return HTTP 500
with `_meta` failure details.
- Add focused service coverage for the unhealthy dependency response
when Go dependencies are not initialized.
## Scope
This is a small, isolated slice of #15240. It avoids current open
connector PRs (#15274, #15300, #15265, #15264), tenant/member PRs
(#15295, #15301, #15276), MCP PRs (#15281, #15253, #15254, #15260,
#15261, #15262), and the memory-message PR (#15256).
Refs #15240
2026-05-27 19:30:22 -10:00
|
|
|
// Healthz reports dependency health in the Python-compatible format.
|
|
|
|
|
func (h *SystemHandler) Healthz(c *gin.Context) {
|
|
|
|
|
result, allOK := h.systemService.Healthz(c.Request.Context())
|
|
|
|
|
statusCode := http.StatusOK
|
|
|
|
|
if !allOK {
|
|
|
|
|
statusCode = http.StatusInternalServerError
|
|
|
|
|
}
|
|
|
|
|
c.JSON(statusCode, result)
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-04 19:17:16 +08:00
|
|
|
// GetConfig get system configuration
|
|
|
|
|
// @Summary Get System Configuration
|
|
|
|
|
// @Description Get system configuration including register enabled status
|
|
|
|
|
// @Tags system
|
|
|
|
|
// @Accept json
|
|
|
|
|
// @Produce json
|
|
|
|
|
// @Success 200 {object} map[string]interface{}
|
|
|
|
|
// @Router /v1/system/config [get]
|
|
|
|
|
func (h *SystemHandler) GetConfig(c *gin.Context) {
|
|
|
|
|
config, err := h.systemService.GetConfig()
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.JSON(http.StatusInternalServerError, gin.H{
|
|
|
|
|
"code": 500,
|
|
|
|
|
"message": "Failed to get system configuration",
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
|
|
|
|
"message": "success",
|
|
|
|
|
"data": config,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// GetConfigs get all system configurations
|
|
|
|
|
// @Summary Get All System Configurations
|
|
|
|
|
// @Description Get all system configurations from globalConfig
|
|
|
|
|
// @Tags system
|
|
|
|
|
// @Accept json
|
|
|
|
|
// @Produce json
|
|
|
|
|
// @Success 200 {object} config.Config
|
|
|
|
|
// @Router /v1/system/configs [get]
|
|
|
|
|
func (h *SystemHandler) GetConfigs(c *gin.Context) {
|
|
|
|
|
cfg := server.GetConfig()
|
|
|
|
|
if cfg == nil {
|
|
|
|
|
c.JSON(http.StatusInternalServerError, gin.H{
|
|
|
|
|
"code": 500,
|
|
|
|
|
"message": "Configuration not initialized",
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
|
|
|
|
"message": "success",
|
|
|
|
|
"data": cfg,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-29 10:12:12 +08:00
|
|
|
// GetStatus get RAGFlow status
|
|
|
|
|
func (h *SystemHandler) GetStatus(c *gin.Context) {
|
|
|
|
|
_, errorCode, errorMessage := GetUser(c)
|
|
|
|
|
if errorCode != common.CodeSuccess {
|
|
|
|
|
jsonError(c, errorCode, errorMessage)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
status, err := h.systemService.GetStatus()
|
|
|
|
|
if err != nil {
|
2026-06-12 05:09:10 -03:00
|
|
|
jsonInternalError(c, err)
|
2026-05-29 10:12:12 +08:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": common.CodeSuccess,
|
|
|
|
|
"data": status,
|
|
|
|
|
"message": "success",
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-04 19:17:16 +08:00
|
|
|
// GetVersion get RAGFlow version
|
|
|
|
|
// @Summary Get RAGFlow Version
|
|
|
|
|
// @Description Get the current version of the application
|
|
|
|
|
// @Tags system
|
|
|
|
|
// @Accept json
|
|
|
|
|
// @Produce json
|
|
|
|
|
// @Security ApiKeyAuth
|
|
|
|
|
// @Success 200 {object} map[string]interface{}
|
|
|
|
|
// @Router /v1/system/version [get]
|
|
|
|
|
func (h *SystemHandler) GetVersion(c *gin.Context) {
|
|
|
|
|
version, err := h.systemService.GetVersion()
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.JSON(http.StatusInternalServerError, gin.H{
|
|
|
|
|
"code": 500,
|
|
|
|
|
"message": "Failed to get version",
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
|
|
|
|
"message": "success",
|
|
|
|
|
"data": version.Version,
|
|
|
|
|
})
|
|
|
|
|
}
|
2026-03-30 18:40:58 +08:00
|
|
|
|
2026-06-23 16:21:46 +08:00
|
|
|
// GetLogLevel returns the current log level. The response uses the
|
|
|
|
|
// {"level": <value>} shape — the same shape the admin handler's
|
|
|
|
|
// /admin/log_level endpoint returns — so the two log endpoints stay
|
|
|
|
|
// in lockstep. Per-package level entries that the old pkgLevels
|
|
|
|
|
// table carried (e.g. "peewee", "pdfminer") were inert for the Go
|
|
|
|
|
// side and are no longer returned.
|
2026-03-30 18:40:58 +08:00
|
|
|
func (h *SystemHandler) GetLogLevel(c *gin.Context) {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
|
|
|
|
"message": "success",
|
2026-06-23 16:21:46 +08:00
|
|
|
"data": gin.H{"level": common.GetLevel()},
|
2026-03-30 18:40:58 +08:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-23 16:21:46 +08:00
|
|
|
// SetLogLevelRequest set log level request. PkgName is accepted for
|
|
|
|
|
// backward compatibility with clients that previously targeted
|
|
|
|
|
// per-package levels; it is silently ignored. Only the global level
|
|
|
|
|
// can be set on the Go side.
|
2026-03-30 18:40:58 +08:00
|
|
|
type SetLogLevelRequest struct {
|
2026-06-23 16:21:46 +08:00
|
|
|
PkgName string `json:"pkg_name"`
|
2026-05-29 19:32:21 +08:00
|
|
|
Level string `json:"level" binding:"required"`
|
2026-03-30 18:40:58 +08:00
|
|
|
}
|
|
|
|
|
|
2026-06-23 16:21:46 +08:00
|
|
|
// SetLogLevel sets the log level at runtime.
|
|
|
|
|
//
|
|
|
|
|
// The "pkg_name and level are required" error message is preserved
|
|
|
|
|
// verbatim from the pre-Go-port handler so existing clients that
|
|
|
|
|
// inspect `message` on the missing-field path keep working. On the
|
|
|
|
|
// Go side `pkg_name` is no longer required (per-package filtering
|
|
|
|
|
// is gone), but the message wording is unchanged for backward
|
|
|
|
|
// compatibility — only `level` is enforced by binding; `pkg_name`
|
|
|
|
|
// is accepted but ignored.
|
2026-03-30 18:40:58 +08:00
|
|
|
func (h *SystemHandler) SetLogLevel(c *gin.Context) {
|
|
|
|
|
var req SetLogLevelRequest
|
|
|
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
2026-05-29 19:32:21 +08:00
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": common.CodeDataError,
|
|
|
|
|
"message": "pkg_name and level are required",
|
2026-03-30 18:40:58 +08:00
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-23 16:21:46 +08:00
|
|
|
if err := common.SetLevel(req.Level); err != nil {
|
2026-05-29 19:32:21 +08:00
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": common.CodeDataError,
|
|
|
|
|
"message": "Invalid log level: " + req.Level,
|
2026-03-30 18:40:58 +08:00
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-23 16:21:46 +08:00
|
|
|
if config := server.GetConfig(); config != nil {
|
2026-05-29 19:32:21 +08:00
|
|
|
config.Log.Level = common.GetLevel()
|
|
|
|
|
}
|
2026-04-08 19:32:53 +08:00
|
|
|
|
2026-03-30 18:40:58 +08:00
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
2026-05-29 19:32:21 +08:00
|
|
|
"message": "success",
|
2026-06-23 16:21:46 +08:00
|
|
|
"data": gin.H{"level": req.Level},
|
2026-03-30 18:40:58 +08:00
|
|
|
})
|
|
|
|
|
}
|
2026-06-25 20:36:50 +08:00
|
|
|
|
|
|
|
|
// ListVariables handle list variables
|
|
|
|
|
func (h *SystemHandler) ListVariables(c *gin.Context) {
|
|
|
|
|
variables, err := h.systemService.ListAllVariables()
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 500,
|
|
|
|
|
"message": err.Error(),
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
|
|
|
|
"message": "success",
|
|
|
|
|
"data": variables,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SetVariableHTTPRequest set variable request
|
|
|
|
|
type SetVariableHTTPRequest struct {
|
|
|
|
|
VarName string `json:"var_name" binding:"required"`
|
|
|
|
|
VarValue string `json:"var_value" binding:"required"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SetVariable handle set variable
|
|
|
|
|
// Python logic: update or create a system setting with the given name and value
|
|
|
|
|
func (h *SystemHandler) SetVariable(c *gin.Context) {
|
|
|
|
|
var req SetVariableHTTPRequest
|
|
|
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 400,
|
|
|
|
|
"message": "Var name is required",
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if req.VarName == "" {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 400,
|
|
|
|
|
"message": "Var name is required",
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if req.VarValue == "" {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 400,
|
|
|
|
|
"message": "Var value is required",
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := h.systemService.SetVariable(req.VarName, req.VarValue); err != nil {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 500,
|
|
|
|
|
"message": err.Error(),
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
|
|
|
|
"message": "SUCCESS",
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-26 13:51:56 +08:00
|
|
|
func (h *SystemHandler) ShowVariable(c *gin.Context) {
|
|
|
|
|
encodedVarName := c.Param("var_name")
|
|
|
|
|
|
|
|
|
|
varName, err := common.DecodeFromBase64(encodedVarName)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 400,
|
|
|
|
|
"message": err.Error(),
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if varName == "" {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 400,
|
|
|
|
|
"message": "Var name is required",
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
variable, err := h.systemService.ShowVariable(varName)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 500,
|
|
|
|
|
"message": err.Error(),
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
|
|
|
|
"message": "SUCCESS",
|
|
|
|
|
"data": variable,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-25 20:36:50 +08:00
|
|
|
// ListEnvironments handle list environments
|
|
|
|
|
func (h *SystemHandler) ListEnvironments(c *gin.Context) {
|
|
|
|
|
environments, err := h.systemService.ListEnvironments()
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 500,
|
|
|
|
|
"message": err.Error(),
|
|
|
|
|
})
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
|
|
|
"code": 0,
|
|
|
|
|
"message": "success",
|
|
|
|
|
"data": environments,
|
|
|
|
|
})
|
|
|
|
|
}
|