mirror of
https://github.com/infiniflow/ragflow.git
synced 2026-05-22 08:58:23 +08:00
Add REST APIs to dynamically query and modify log levels at runtime for
both Python (Flask) and Go servers.
Changes:
- common/log_utils.py: add set_log_level() and get_log_levels()
functions
- admin/server/routes.py: add GET/PUT /api/v1/admin/log_levels endpoints
- api/apps/system_app.py: add GET/PUT /api/{version}/system/log_levels
endpoints
- internal/logger/logger.go: add GetLevel() and SetLevel() with atomic
level support
- internal/handler/system.go: add GetLogLevel, SetLogLevel, Health
handlers
- internal/router/router.go: route /health to systemHandler
- internal/admin/handler.go: add GetLogLevel, SetLogLevel handlers
- internal/admin/router.go: add /api/v1/admin/log_level routes
### What problem does this PR solve?
_Briefly describe what this PR aims to solve. Include background context
that will help reviewers understand the purpose of the PR._
### Type of change
- [x] New Feature (non-breaking change which adds functionality)
Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
174 lines
4.2 KiB
Go
174 lines
4.2 KiB
Go
//
|
|
// 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"
|
|
"ragflow/internal/logger"
|
|
"ragflow/internal/server"
|
|
"ragflow/internal/service"
|
|
|
|
"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")
|
|
}
|
|
|
|
// Health health check
|
|
func (h *SystemHandler) Health(c *gin.Context) {
|
|
c.JSON(200, gin.H{
|
|
"status": "ok",
|
|
})
|
|
}
|
|
|
|
// 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,
|
|
})
|
|
}
|
|
|
|
// 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,
|
|
})
|
|
}
|
|
|
|
// GetLogLevel returns the current log level
|
|
func (h *SystemHandler) GetLogLevel(c *gin.Context) {
|
|
level := logger.GetLevel()
|
|
c.JSON(http.StatusOK, gin.H{
|
|
"code": 0,
|
|
"message": "success",
|
|
"data": gin.H{"level": level},
|
|
})
|
|
}
|
|
|
|
// SetLogLevelRequest set log level request
|
|
type SetLogLevelRequest struct {
|
|
Level string `json:"level" binding:"required"`
|
|
}
|
|
|
|
// SetLogLevel sets the log level at runtime
|
|
func (h *SystemHandler) SetLogLevel(c *gin.Context) {
|
|
var req SetLogLevelRequest
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
c.JSON(http.StatusBadRequest, gin.H{
|
|
"code": 400,
|
|
"message": "level is required",
|
|
})
|
|
return
|
|
}
|
|
|
|
if err := logger.SetLevel(req.Level); err != nil {
|
|
c.JSON(http.StatusBadRequest, gin.H{
|
|
"code": 400,
|
|
"message": err.Error(),
|
|
})
|
|
return
|
|
}
|
|
|
|
c.JSON(http.StatusOK, gin.H{
|
|
"code": 0,
|
|
"message": "Log level updated successfully",
|
|
"data": gin.H{"level": req.Level},
|
|
})
|
|
}
|