Files
NetworkAuth/controllers/admin/variable.go

416 lines
11 KiB
Go
Raw Normal View History

2025-10-26 00:08:55 +08:00
package admin
import (
2026-03-18 21:51:17 +08:00
"NetworkAuth/controllers"
"NetworkAuth/models"
2026-03-19 05:11:44 +08:00
"NetworkAuth/services"
2025-10-26 00:08:55 +08:00
"net/http"
"regexp"
"strconv"
"strings"
2025-10-26 14:48:02 +08:00
"github.com/gin-gonic/gin"
2025-10-26 00:08:55 +08:00
"github.com/sirupsen/logrus"
)
2025-10-27 23:12:15 +08:00
// ============================================================================
// 全局变量
// ============================================================================
2025-10-26 14:48:02 +08:00
// 创建基础控制器实例
var variableBaseController = controllers.NewBaseController()
2025-10-27 23:12:15 +08:00
// ============================================================================
// 页面处理器
// ============================================================================
// VariableFragmentHandler 公共变量列表页面片段处理器
2025-10-26 14:48:02 +08:00
func VariableFragmentHandler(c *gin.Context) {
c.HTML(http.StatusOK, "variables.html", gin.H{
"Title": "公共变量",
2025-10-26 00:08:55 +08:00
})
}
2025-10-27 23:12:15 +08:00
// ============================================================================
// API处理器
// ============================================================================
2025-10-26 00:08:55 +08:00
// VariableListHandler 变量列表API处理器
2025-10-26 14:48:02 +08:00
func VariableListHandler(c *gin.Context) {
2025-10-26 00:08:55 +08:00
// 获取分页参数
2025-10-26 14:48:02 +08:00
page, _ := strconv.Atoi(c.Query("page"))
2025-10-26 00:08:55 +08:00
if page <= 0 {
page = 1
}
2025-10-26 14:48:02 +08:00
limit, _ := strconv.Atoi(c.Query("limit"))
// 兼容前端使用的page_size参数
if limit <= 0 {
limit, _ = strconv.Atoi(c.Query("page_size"))
}
2025-10-26 00:08:55 +08:00
if limit <= 0 {
limit = 10
}
// 获取搜索关键词参数(支持编号、别名、数据的综合搜索)
2025-10-26 14:48:02 +08:00
search := strings.TrimSpace(c.Query("search"))
// 兼容旧的别名搜索参数
if search == "" {
2025-10-26 14:48:02 +08:00
search = strings.TrimSpace(c.Query("alias"))
}
2025-10-26 00:08:55 +08:00
// 获取应用筛选参数
appUUID := strings.TrimSpace(c.Query("app_uuid"))
2025-10-26 00:08:55 +08:00
// 构建查询
2025-10-26 14:48:02 +08:00
db, ok := variableBaseController.GetDB(c)
if !ok {
2025-10-26 00:08:55 +08:00
return
}
// 构建基础查询
query := db.Model(&models.Variable{})
2026-03-18 01:08:15 +08:00
// 如果指定了搜索关键词,则在编号、别名、数据、备注中进行精确搜索
if search != "" {
2026-03-18 01:08:15 +08:00
// 优化:使用精确匹配提升查询性能
query = query.Where("number = ? OR alias = ? OR data = ? OR remark = ?",
search, search, search, search)
2025-10-26 00:08:55 +08:00
}
// 如果指定了应用筛选则按应用UUID筛选
if appUUID != "" {
query = query.Where("app_uuid = ?", appUUID)
}
2025-10-26 00:08:55 +08:00
// 获取总数
var total int64
if err := query.Count(&total).Error; err != nil {
logrus.WithError(err).Error("Failed to count variables")
2025-10-26 14:48:02 +08:00
variableBaseController.HandleInternalError(c, "查询变量总数失败", err)
2025-10-26 00:08:55 +08:00
return
}
// 获取分页数据
var variables []models.Variable
offset := (page - 1) * limit
if err := query.Offset(offset).Limit(limit).Order("created_at DESC").Find(&variables).Error; err != nil {
logrus.WithError(err).Error("Failed to fetch variables")
2025-10-26 14:48:02 +08:00
variableBaseController.HandleInternalError(c, "查询变量列表失败", err)
2025-10-26 00:08:55 +08:00
return
}
// 构建响应数据
type VariableResponse struct {
ID uint `json:"id"`
UUID string `json:"uuid"`
Number string `json:"number"`
AppUUID string `json:"app_uuid"`
2025-10-26 00:08:55 +08:00
Alias string `json:"alias"`
Data string `json:"data"`
Remark string `json:"remark"`
CreatedAt string `json:"created_at"`
UpdatedAt string `json:"updated_at"`
}
var responseData []VariableResponse
for _, variable := range variables {
responseData = append(responseData, VariableResponse{
ID: variable.ID,
UUID: variable.UUID,
Number: variable.Number,
AppUUID: variable.AppUUID,
2025-10-26 00:08:55 +08:00
Alias: variable.Alias,
Data: variable.Data,
Remark: variable.Remark,
CreatedAt: variable.CreatedAt.Format("2006-01-02 15:04:05"),
UpdatedAt: variable.UpdatedAt.Format("2006-01-02 15:04:05"),
})
}
2025-10-26 14:48:02 +08:00
response := gin.H{
2025-10-26 00:08:55 +08:00
"code": 0,
"msg": "success",
"count": total,
"data": responseData,
}
2025-10-26 14:48:02 +08:00
c.JSON(http.StatusOK, response)
2025-10-26 00:08:55 +08:00
}
// VariableCreateHandler 新增变量API处理器
2025-10-26 14:48:02 +08:00
func VariableCreateHandler(c *gin.Context) {
2025-10-26 00:08:55 +08:00
var req struct {
Alias string `json:"alias"`
AppUUID string `json:"app_uuid"`
Data string `json:"data"`
Remark string `json:"remark"`
2025-10-26 00:08:55 +08:00
}
2025-10-26 14:48:02 +08:00
if !variableBaseController.BindJSON(c, &req) {
2025-10-26 00:08:55 +08:00
return
}
// 验证必填字段
2025-10-26 14:48:02 +08:00
if !variableBaseController.ValidateRequired(c, map[string]interface{}{
"变量别名": req.Alias,
2025-10-26 14:48:02 +08:00
}) {
2025-10-26 00:08:55 +08:00
return
}
// 验证别名格式:必须以英文字母开头,只能包含数字和英文字母
aliasPattern := regexp.MustCompile(`^[a-zA-Z][a-zA-Z0-9]*$`)
if !aliasPattern.MatchString(req.Alias) {
2025-10-26 14:48:02 +08:00
variableBaseController.HandleValidationError(c, "别名必须以英文字母开头,只能包含数字和英文字母")
2025-10-26 00:08:55 +08:00
return
}
2025-10-26 14:48:02 +08:00
db, ok := variableBaseController.GetDB(c)
if !ok {
2025-10-26 00:08:55 +08:00
return
}
// 处理应用UUID如果为空或"0",设置为"0"(全局变量)
updateAppUUID := strings.TrimSpace(req.AppUUID)
if updateAppUUID == "" {
updateAppUUID = "0"
}
// 如果指定了应用UUID且不是"0",验证应用是否存在
if updateAppUUID != "0" {
var appCount int64
if err := db.Model(&models.App{}).Where("uuid = ?", updateAppUUID).Count(&appCount).Error; err != nil {
logrus.WithError(err).Error("Failed to check app existence")
variableBaseController.HandleInternalError(c, "验证应用失败", err)
return
}
if appCount == 0 {
variableBaseController.HandleValidationError(c, "指定的应用不存在")
return
}
}
// 处理应用UUID如果为空或"0",设置为"0"(全局变量)
appUUID := strings.TrimSpace(req.AppUUID)
if appUUID == "" {
appUUID = "0"
}
// 如果指定了应用UUID且不是"0",验证应用是否存在
if appUUID != "0" {
var appCount int64
if err := db.Model(&models.App{}).Where("uuid = ?", appUUID).Count(&appCount).Error; err != nil {
logrus.WithError(err).Error("Failed to check app existence")
variableBaseController.HandleInternalError(c, "验证应用失败", err)
return
}
if appCount == 0 {
variableBaseController.HandleValidationError(c, "指定的应用不存在")
return
}
}
2025-10-26 00:08:55 +08:00
// 创建变量
variable := models.Variable{
Alias: strings.TrimSpace(req.Alias),
AppUUID: appUUID,
Data: req.Data,
Remark: strings.TrimSpace(req.Remark),
2025-10-26 00:08:55 +08:00
}
if err := db.Create(&variable).Error; err != nil {
logrus.WithError(err).Error("Failed to create variable")
2025-10-26 14:48:02 +08:00
variableBaseController.HandleInternalError(c, "创建变量失败", err)
2025-10-26 00:08:55 +08:00
return
}
2026-03-19 05:11:44 +08:00
// 记录操作日志
operator := c.GetString("admin_username")
if operator == "" {
operator = "unknown"
}
operatorUUID := c.GetString("admin_uuid")
services.RecordOperationLog(
"创建变量",
operator,
operatorUUID,
"创建了变量: "+variable.Alias,
)
2025-10-26 14:48:02 +08:00
variableBaseController.HandleSuccess(c, "创建成功", variable)
2025-10-26 00:08:55 +08:00
}
// VariableUpdateHandler 更新变量API处理器
2025-10-26 14:48:02 +08:00
func VariableUpdateHandler(c *gin.Context) {
2025-10-26 00:08:55 +08:00
var req struct {
UUID string `json:"uuid"`
AppUUID string `json:"app_uuid"`
Data string `json:"data"`
Remark string `json:"remark"`
2025-10-26 00:08:55 +08:00
}
2025-10-26 14:48:02 +08:00
if !variableBaseController.BindJSON(c, &req) {
2025-10-26 00:08:55 +08:00
return
}
// 验证必填字段移除对alias的验证因为编辑时不允许修改别名
2025-10-26 14:48:02 +08:00
if !variableBaseController.ValidateRequired(c, map[string]interface{}{
"变量UUID": req.UUID,
}) {
2025-10-26 00:08:55 +08:00
return
}
2025-10-26 14:48:02 +08:00
db, ok := variableBaseController.GetDB(c)
if !ok {
2025-10-26 00:08:55 +08:00
return
}
// 处理应用UUID如果为空或"0",设置为"0"(全局变量)
updateAppUUID := strings.TrimSpace(req.AppUUID)
if updateAppUUID == "" {
updateAppUUID = "0"
}
// 如果指定了应用UUID且不是"0",验证应用是否存在
if updateAppUUID != "0" {
var appCount int64
if err := db.Model(&models.App{}).Where("uuid = ?", updateAppUUID).Count(&appCount).Error; err != nil {
logrus.WithError(err).Error("Failed to check app existence")
variableBaseController.HandleInternalError(c, "验证应用失败", err)
return
}
if appCount == 0 {
variableBaseController.HandleValidationError(c, "指定的应用不存在")
return
}
}
2025-10-26 00:08:55 +08:00
// 通过uuid字段查找变量
var variable models.Variable
if err := db.Where("uuid = ?", strings.TrimSpace(req.UUID)).First(&variable).Error; err != nil {
2025-10-26 14:48:02 +08:00
variableBaseController.HandleValidationError(c, "变量不存在")
2025-10-26 00:08:55 +08:00
return
}
// 更新字段不更新alias保持原有别名不变
variable.AppUUID = updateAppUUID
2025-10-26 00:08:55 +08:00
variable.Data = req.Data
variable.Remark = strings.TrimSpace(req.Remark)
if err := db.Save(&variable).Error; err != nil {
logrus.WithError(err).Error("Failed to update variable")
2025-10-26 14:48:02 +08:00
variableBaseController.HandleInternalError(c, "更新变量失败", err)
2025-10-26 00:08:55 +08:00
return
}
2025-10-26 14:48:02 +08:00
variableBaseController.HandleSuccess(c, "更新成功", variable)
2025-10-26 00:08:55 +08:00
}
// VariableDeleteHandler 删除变量API处理器
2025-10-26 14:48:02 +08:00
func VariableDeleteHandler(c *gin.Context) {
2025-10-26 00:08:55 +08:00
var req struct {
ID uint `json:"id"`
}
2025-10-26 14:48:02 +08:00
if !variableBaseController.BindJSON(c, &req) {
2025-10-26 00:08:55 +08:00
return
}
if req.ID == 0 {
2025-10-26 14:48:02 +08:00
variableBaseController.HandleValidationError(c, "变量ID不能为空")
2025-10-26 00:08:55 +08:00
return
}
2025-10-26 14:48:02 +08:00
db, ok := variableBaseController.GetDB(c)
if !ok {
2025-10-26 00:08:55 +08:00
return
}
2026-03-19 05:11:44 +08:00
// 查找变量以记录日志
var variable models.Variable
db.First(&variable, req.ID)
2025-10-26 00:08:55 +08:00
// 删除变量
if err := db.Delete(&models.Variable{}, req.ID).Error; err != nil {
logrus.WithError(err).Error("Failed to delete variable")
2025-10-26 14:48:02 +08:00
variableBaseController.HandleInternalError(c, "删除变量失败", err)
2025-10-26 00:08:55 +08:00
return
}
2026-03-19 05:11:44 +08:00
// 记录操作日志
operator := c.GetString("admin_username")
if operator == "" {
operator = "unknown"
}
operatorUUID := c.GetString("admin_uuid")
details := "删除了变量ID: " + strconv.Itoa(int(req.ID))
if variable.ID != 0 {
details = "删除了变量: " + variable.Alias
}
services.RecordOperationLog(
"删除变量",
operator,
operatorUUID,
details,
)
2026-03-18 21:51:17 +08:00
logrus.WithField("variable_id", req.ID).Debug("Successfully deleted variable")
2025-10-26 00:08:55 +08:00
2025-10-26 14:48:02 +08:00
variableBaseController.HandleSuccess(c, "删除成功", nil)
2025-10-26 00:08:55 +08:00
}
// VariablesBatchDeleteHandler 批量删除变量API处理器
2025-10-26 14:48:02 +08:00
func VariablesBatchDeleteHandler(c *gin.Context) {
2025-10-26 00:08:55 +08:00
var req struct {
IDs []uint `json:"ids"`
}
2025-10-26 14:48:02 +08:00
if !variableBaseController.BindJSON(c, &req) {
2025-10-26 00:08:55 +08:00
return
}
if len(req.IDs) == 0 {
2025-10-26 14:48:02 +08:00
variableBaseController.HandleValidationError(c, "请选择要删除的变量")
2025-10-26 00:08:55 +08:00
return
}
2025-10-26 14:48:02 +08:00
db, ok := variableBaseController.GetDB(c)
if !ok {
2025-10-26 00:08:55 +08:00
return
}
// 批量删除变量
if err := db.Delete(&models.Variable{}, req.IDs).Error; err != nil {
logrus.WithError(err).Error("Failed to batch delete variables")
2025-10-26 14:48:02 +08:00
variableBaseController.HandleInternalError(c, "批量删除失败", err)
2025-10-26 00:08:55 +08:00
return
}
2026-03-19 05:11:44 +08:00
// 记录操作日志
operator := c.GetString("admin_username")
if operator == "" {
operator = "unknown"
}
operatorUUID := c.GetString("admin_uuid")
var idStrs []string
for _, id := range req.IDs {
idStrs = append(idStrs, strconv.Itoa(int(id)))
}
details := "批量删除了变量ID: " + strings.Join(idStrs, ",")
services.RecordOperationLog(
"删除变量",
operator,
operatorUUID,
details,
)
2026-03-18 21:51:17 +08:00
logrus.WithField("variable_ids", req.IDs).Debug("Successfully batch deleted variables")
2025-10-26 00:08:55 +08:00
2025-10-26 14:48:02 +08:00
variableBaseController.HandleSuccess(c, "批量删除成功", nil)
2025-10-26 00:08:55 +08:00
}