2025-10-24 00:09:45 +08:00
|
|
|
|
package admin
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"net/http"
|
|
|
|
|
|
"networkDev/database"
|
2025-10-26 11:57:31 +08:00
|
|
|
|
"networkDev/models"
|
2025-10-24 00:09:45 +08:00
|
|
|
|
"networkDev/services"
|
|
|
|
|
|
"networkDev/utils"
|
|
|
|
|
|
"networkDev/utils/timeutil"
|
|
|
|
|
|
|
|
|
|
|
|
"github.com/spf13/viper"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2025-10-26 11:57:31 +08:00
|
|
|
|
// formatDBType 格式化数据库类型显示
|
|
|
|
|
|
// 将配置文件中的小写类型转换为友好的显示格式
|
|
|
|
|
|
func formatDBType(dbType string) string {
|
|
|
|
|
|
switch dbType {
|
|
|
|
|
|
case "mysql":
|
|
|
|
|
|
return "MySQL"
|
|
|
|
|
|
case "sqlite":
|
|
|
|
|
|
return "SQLite"
|
|
|
|
|
|
case "postgresql", "postgres":
|
|
|
|
|
|
return "PostgreSQL"
|
|
|
|
|
|
case "sqlserver":
|
|
|
|
|
|
return "SQL Server"
|
|
|
|
|
|
default:
|
|
|
|
|
|
return "SQLite" // 默认显示
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// AdminIndexHandler 后台首页处理器/admin 与 /admin/ 根路径入口
|
2025-10-24 00:09:45 +08:00
|
|
|
|
// - 未登录:重定向到 /admin/login
|
|
|
|
|
|
// - 已登录:渲染后台布局页(或重定向到 /admin/layout)
|
2025-10-26 01:51:25 +08:00
|
|
|
|
// - 自动清理失效的JWT Cookie
|
2025-10-24 00:09:45 +08:00
|
|
|
|
func AdminIndexHandler(w http.ResponseWriter, r *http.Request) {
|
2025-10-26 01:51:25 +08:00
|
|
|
|
if IsAdminAuthenticatedWithCleanup(w, r) {
|
2025-10-24 00:09:45 +08:00
|
|
|
|
// 直接渲染布局页,保持URL为 /admin
|
|
|
|
|
|
AdminLayoutHandler(w, r)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
http.Redirect(w, r, "/admin/login", http.StatusFound)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// AdminLayoutHandler 后台布局页渲染
|
|
|
|
|
|
// - 渲染 layout.html,包含顶部导航、侧边栏与动态内容容器
|
|
|
|
|
|
func AdminLayoutHandler(w http.ResponseWriter, r *http.Request) {
|
2025-10-26 03:05:27 +08:00
|
|
|
|
// 获取或生成CSRF令牌
|
|
|
|
|
|
var token string
|
|
|
|
|
|
if existingToken := utils.GetCSRFTokenFromCookie(r); existingToken != "" {
|
|
|
|
|
|
// 重用现有的Cookie令牌
|
|
|
|
|
|
token = existingToken
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 生成新的CSRF令牌并设置到Cookie
|
|
|
|
|
|
newToken, err := utils.GenerateCSRFToken()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
http.Error(w, "生成CSRF令牌失败", http.StatusInternalServerError)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
token = newToken
|
|
|
|
|
|
utils.SetCSRFToken(w, token)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 准备额外的模板数据
|
|
|
|
|
|
extraData := make(map[string]interface{})
|
2025-10-24 00:09:45 +08:00
|
|
|
|
|
|
|
|
|
|
// 从数据库读取站点标题
|
2025-10-26 03:05:27 +08:00
|
|
|
|
db, dbErr := database.GetDB()
|
|
|
|
|
|
if dbErr != nil {
|
|
|
|
|
|
extraData["Title"] = "凌动技术"
|
2025-10-24 00:09:45 +08:00
|
|
|
|
} else {
|
2025-10-26 03:05:27 +08:00
|
|
|
|
siteTitle, settingErr := services.FindSettingByName("site_title", db)
|
|
|
|
|
|
if settingErr != nil || siteTitle == nil {
|
|
|
|
|
|
extraData["Title"] = "凌动技术"
|
2025-10-24 00:09:45 +08:00
|
|
|
|
} else {
|
2025-10-26 03:05:27 +08:00
|
|
|
|
extraData["Title"] = siteTitle.Value
|
2025-10-24 00:09:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-26 03:05:27 +08:00
|
|
|
|
// 准备模板数据
|
|
|
|
|
|
data := utils.GetDefaultTemplateData()
|
|
|
|
|
|
data["CSRFToken"] = token
|
|
|
|
|
|
|
|
|
|
|
|
// 合并额外数据
|
|
|
|
|
|
for key, value := range extraData {
|
|
|
|
|
|
data[key] = value
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-24 00:09:45 +08:00
|
|
|
|
utils.RenderTemplate(w, "layout.html", data)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// DashboardFragmentHandler 仪表盘片段渲染
|
2025-10-26 11:57:31 +08:00
|
|
|
|
// - 展示系统信息:版本、开发模式、数据库类型、启动时长
|
2025-10-24 00:09:45 +08:00
|
|
|
|
func DashboardFragmentHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
|
version := "1.0.0"
|
2025-10-26 11:57:31 +08:00
|
|
|
|
mode := viper.GetBool("server.dev_mode")
|
2025-10-24 00:09:45 +08:00
|
|
|
|
dbType := viper.GetString("database.type")
|
|
|
|
|
|
if dbType == "" {
|
|
|
|
|
|
dbType = "sqlite"
|
|
|
|
|
|
}
|
|
|
|
|
|
uptime := timeutil.GetServerUptimeString()
|
|
|
|
|
|
|
|
|
|
|
|
data := map[string]interface{}{
|
|
|
|
|
|
"Version": version,
|
|
|
|
|
|
"Mode": mode,
|
2025-10-26 11:57:31 +08:00
|
|
|
|
"DBType": formatDBType(dbType),
|
2025-10-24 00:09:45 +08:00
|
|
|
|
"Uptime": uptime,
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
utils.RenderTemplate(w, "dashboard.html", data)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// SystemInfoHandler 系统信息API接口
|
|
|
|
|
|
// - 返回系统运行状态的JSON数据,用于前端定时刷新
|
|
|
|
|
|
func SystemInfoHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
|
if r.Method != http.MethodGet {
|
|
|
|
|
|
http.Error(w, "Method Not Allowed", http.StatusMethodNotAllowed)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
version := "1.0.0"
|
2025-10-26 11:57:31 +08:00
|
|
|
|
mode := viper.GetBool("server.dev_mode")
|
2025-10-24 00:09:45 +08:00
|
|
|
|
dbType := viper.GetString("database.type")
|
|
|
|
|
|
if dbType == "" {
|
|
|
|
|
|
dbType = "sqlite"
|
|
|
|
|
|
}
|
|
|
|
|
|
uptime := timeutil.GetServerUptimeString()
|
|
|
|
|
|
|
|
|
|
|
|
data := map[string]interface{}{
|
|
|
|
|
|
"version": version,
|
|
|
|
|
|
"mode": mode,
|
2025-10-26 11:57:31 +08:00
|
|
|
|
"db_type": formatDBType(dbType),
|
2025-10-24 00:09:45 +08:00
|
|
|
|
"uptime": uptime,
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
utils.JsonResponse(w, http.StatusOK, true, "ok", data)
|
|
|
|
|
|
}
|
2025-10-26 11:57:31 +08:00
|
|
|
|
|
|
|
|
|
|
// DashboardStatsHandler 仪表盘统计数据API接口
|
|
|
|
|
|
// - 返回应用统计数据的JSON数据,包括全部/启用/禁用/变量数量
|
|
|
|
|
|
func DashboardStatsHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
|
if r.Method != http.MethodGet {
|
|
|
|
|
|
http.Error(w, "Method Not Allowed", http.StatusMethodNotAllowed)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取数据库连接
|
|
|
|
|
|
db, err := database.GetDB()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
utils.JsonResponse(w, http.StatusInternalServerError, false, "数据库连接失败", nil)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 统计应用数据
|
|
|
|
|
|
var totalApps int64
|
|
|
|
|
|
var enabledApps int64
|
|
|
|
|
|
var disabledApps int64
|
|
|
|
|
|
var totalVariables int64
|
|
|
|
|
|
|
|
|
|
|
|
// 统计全部应用数量
|
|
|
|
|
|
if err := db.Model(&models.App{}).Count(&totalApps).Error; err != nil {
|
|
|
|
|
|
utils.JsonResponse(w, http.StatusInternalServerError, false, "统计应用数量失败", nil)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 统计启用应用数量
|
|
|
|
|
|
if err := db.Model(&models.App{}).Where("status = ?", 1).Count(&enabledApps).Error; err != nil {
|
|
|
|
|
|
utils.JsonResponse(w, http.StatusInternalServerError, false, "统计启用应用数量失败", nil)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 统计禁用应用数量
|
|
|
|
|
|
if err := db.Model(&models.App{}).Where("status = ?", 0).Count(&disabledApps).Error; err != nil {
|
|
|
|
|
|
utils.JsonResponse(w, http.StatusInternalServerError, false, "统计禁用应用数量失败", nil)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 统计变量数量
|
|
|
|
|
|
if err := db.Model(&models.Variable{}).Count(&totalVariables).Error; err != nil {
|
|
|
|
|
|
utils.JsonResponse(w, http.StatusInternalServerError, false, "统计变量数量失败", nil)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
data := map[string]interface{}{
|
|
|
|
|
|
"total_apps": totalApps,
|
|
|
|
|
|
"enabled_apps": enabledApps,
|
|
|
|
|
|
"disabled_apps": disabledApps,
|
|
|
|
|
|
"total_variables": totalVariables,
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
utils.JsonResponse(w, http.StatusOK, true, "ok", data)
|
|
|
|
|
|
}
|