273 lines
7.1 KiB
Go
273 lines
7.1 KiB
Go
package controllers
|
||
|
||
import (
|
||
"encoding/json"
|
||
"fmt"
|
||
"io"
|
||
"strconv"
|
||
"strings"
|
||
"time"
|
||
|
||
"server/models"
|
||
"server/pkg/jwtutil"
|
||
|
||
beego "github.com/beego/beego/v2/server/web"
|
||
)
|
||
|
||
// SiteSettingsController 租户站点设置(站点基本信息)
|
||
// 对应前端 normalSettings.vue 的:
|
||
// - GET /backend/normalInfos
|
||
// - POST /backend/saveNormalInfos
|
||
// - GET /platform/normalInfos
|
||
// - POST /platform/saveNormalInfos
|
||
type SiteSettingsController struct {
|
||
beego.Controller
|
||
}
|
||
|
||
func (c *SiteSettingsController) jsonErr(httpStatus, bizCode int, msg string) {
|
||
c.Ctx.Output.SetStatus(httpStatus)
|
||
c.Data["json"] = map[string]interface{}{"code": bizCode, "msg": msg}
|
||
_ = c.ServeJSON()
|
||
}
|
||
|
||
func (c *SiteSettingsController) claimsByPath() (*jwtutil.Claims, error) {
|
||
auth := c.Ctx.Request.Header.Get("Authorization")
|
||
if auth == "" {
|
||
return nil, fmt.Errorf("未登录")
|
||
}
|
||
parts := strings.SplitN(auth, " ", 2)
|
||
if len(parts) != 2 || parts[0] != "Bearer" {
|
||
return nil, fmt.Errorf("认证信息格式错误")
|
||
}
|
||
claims, err := jwtutil.ParseToken(parts[1])
|
||
if err != nil {
|
||
return nil, fmt.Errorf("无效的token")
|
||
}
|
||
|
||
path := strings.ToLower(c.Ctx.Request.URL.Path)
|
||
if strings.HasPrefix(path, "/platform/") {
|
||
if claims.UserType != "platform" {
|
||
return nil, fmt.Errorf("无权访问")
|
||
}
|
||
} else if strings.HasPrefix(path, "/backend/") {
|
||
if claims.UserType != "backend" {
|
||
return nil, fmt.Errorf("无权访问")
|
||
}
|
||
}
|
||
|
||
return claims, nil
|
||
}
|
||
|
||
func parseUint64Flexible(v interface{}) uint64 {
|
||
if v == nil {
|
||
return 0
|
||
}
|
||
switch x := v.(type) {
|
||
case float64:
|
||
if x <= 0 {
|
||
return 0
|
||
}
|
||
return uint64(x)
|
||
case string:
|
||
s := strings.TrimSpace(x)
|
||
if s == "" {
|
||
return 0
|
||
}
|
||
n, err := strconv.ParseUint(s, 10, 64)
|
||
if err != nil || n == 0 {
|
||
return 0
|
||
}
|
||
return n
|
||
default:
|
||
return 0
|
||
}
|
||
}
|
||
|
||
type normalInfosOutput struct {
|
||
Sitename string `json:"sitename"`
|
||
Companyintroduction string `json:"companyintroduction"`
|
||
Description string `json:"description"`
|
||
Copyright string `json:"copyright"`
|
||
Companyname string `json:"companyname"`
|
||
Icp string `json:"icp"`
|
||
Logo string `json:"logo"`
|
||
Logow string `json:"logow"`
|
||
Ico string `json:"ico"`
|
||
}
|
||
|
||
// GetNormalInfos GET /backend/normalInfos 或 /platform/normalInfos
|
||
func (c *SiteSettingsController) GetNormalInfos() {
|
||
claims, err := c.claimsByPath()
|
||
if err != nil {
|
||
c.jsonErr(401, 401, err.Error())
|
||
return
|
||
}
|
||
|
||
// 优先使用 token 中的租户 id;若为 0,则允许前端通过查询参数传入(兼容历史/平台端)。
|
||
tid := uint64(claims.TenantId)
|
||
if tid == 0 {
|
||
tidStr := strings.TrimSpace(c.GetString("tid"))
|
||
if tidStr != "" {
|
||
if n, err := strconv.ParseUint(tidStr, 10, 64); err == nil {
|
||
tid = n
|
||
}
|
||
}
|
||
}
|
||
|
||
out := normalInfosOutput{
|
||
Sitename: "",
|
||
Companyintroduction: "",
|
||
Description: "",
|
||
Copyright: "",
|
||
Companyname: "",
|
||
Icp: "",
|
||
Logo: "",
|
||
Logow: "",
|
||
Ico: "",
|
||
}
|
||
|
||
// tid 缺失时不报错,直接返回空对象给前端渲染(避免 UI 直接崩)。
|
||
if tid == 0 {
|
||
c.Data["json"] = map[string]interface{}{"code": 200, "msg": "success", "data": out}
|
||
_ = c.ServeJSON()
|
||
return
|
||
}
|
||
|
||
var rows []models.TenantSiteSetting
|
||
_, err = models.Orm.QueryTable(new(models.TenantSiteSetting)).
|
||
Filter("tid", tid).
|
||
Filter("delete_time__isnull", true).
|
||
Limit(1).
|
||
All(&rows)
|
||
if err != nil {
|
||
c.jsonErr(500, 500, "获取失败: "+err.Error())
|
||
return
|
||
}
|
||
if len(rows) > 0 {
|
||
r := rows[0]
|
||
out.Sitename = r.Sitename
|
||
out.Companyintroduction = r.Companyintroduction
|
||
out.Logo = r.Logo
|
||
out.Logow = r.Logow
|
||
out.Ico = r.Ico
|
||
out.Description = r.Description
|
||
out.Copyright = r.Copyright
|
||
out.Companyname = r.Companyname
|
||
out.Icp = r.Icp
|
||
}
|
||
|
||
c.Data["json"] = map[string]interface{}{"code": 200, "msg": "success", "data": out}
|
||
_ = c.ServeJSON()
|
||
}
|
||
|
||
type normalInfosPayload struct {
|
||
// 前端会传 tid(但我们仍优先使用 token 的 tenant_id)
|
||
Tid interface{} `json:"tid"`
|
||
|
||
Sitename string `json:"sitename"`
|
||
Companyintroduction string `json:"companyintroduction"`
|
||
Logo string `json:"logo"`
|
||
Logow string `json:"logow"`
|
||
Ico string `json:"ico"`
|
||
Description string `json:"description"`
|
||
Copyright string `json:"copyright"`
|
||
Companyname string `json:"companyname"`
|
||
Icp string `json:"icp"`
|
||
}
|
||
|
||
// SaveNormalInfos POST /backend/saveNormalInfos 或 /platform/saveNormalInfos
|
||
func (c *SiteSettingsController) SaveNormalInfos() {
|
||
claims, err := c.claimsByPath()
|
||
if err != nil {
|
||
c.jsonErr(401, 401, err.Error())
|
||
return
|
||
}
|
||
|
||
raw, err := io.ReadAll(c.Ctx.Request.Body)
|
||
if err != nil {
|
||
c.jsonErr(400, 400, "参数错误")
|
||
return
|
||
}
|
||
|
||
var p normalInfosPayload
|
||
if uerr := json.Unmarshal(raw, &p); uerr != nil {
|
||
c.jsonErr(400, 400, "参数错误")
|
||
return
|
||
}
|
||
|
||
tid := uint64(claims.TenantId)
|
||
if tid == 0 {
|
||
tid = parseUint64Flexible(p.Tid)
|
||
}
|
||
if tid == 0 {
|
||
c.jsonErr(400, 400, "tid不能为空")
|
||
return
|
||
}
|
||
|
||
sitename := strings.TrimSpace(p.Sitename)
|
||
if sitename == "" {
|
||
c.jsonErr(400, 400, "站点名称不能为空")
|
||
return
|
||
}
|
||
|
||
now := time.Now()
|
||
|
||
up := map[string]interface{}{
|
||
"tid": tid,
|
||
"sitename": sitename,
|
||
"companyintroduction": strings.TrimSpace(p.Companyintroduction),
|
||
"logo": strings.TrimSpace(p.Logo),
|
||
"logow": strings.TrimSpace(p.Logow),
|
||
"ico": strings.TrimSpace(p.Ico),
|
||
"description": strings.TrimSpace(p.Description),
|
||
"copyright": strings.TrimSpace(p.Copyright),
|
||
"companyname": strings.TrimSpace(p.Companyname),
|
||
"icp": strings.TrimSpace(p.Icp),
|
||
"update_time": now,
|
||
}
|
||
|
||
cnt, err := models.Orm.QueryTable(new(models.TenantSiteSetting)).
|
||
Filter("tid", tid).
|
||
Filter("delete_time__isnull", true).
|
||
Count()
|
||
if err != nil {
|
||
c.jsonErr(500, 500, "保存失败: "+err.Error())
|
||
return
|
||
}
|
||
|
||
if cnt == 0 {
|
||
row := &models.TenantSiteSetting{
|
||
Tid: tid,
|
||
Sitename: sitename,
|
||
Companyintroduction: strings.TrimSpace(p.Companyintroduction),
|
||
Logo: strings.TrimSpace(p.Logo),
|
||
Logow: strings.TrimSpace(p.Logow),
|
||
Ico: strings.TrimSpace(p.Ico),
|
||
Description: strings.TrimSpace(p.Description),
|
||
Copyright: strings.TrimSpace(p.Copyright),
|
||
Companyname: strings.TrimSpace(p.Companyname),
|
||
Icp: strings.TrimSpace(p.Icp),
|
||
CreateTime: now,
|
||
UpdateTime: &now,
|
||
}
|
||
_, err = models.Orm.Insert(row)
|
||
if err != nil {
|
||
c.jsonErr(500, 500, "保存失败: "+err.Error())
|
||
return
|
||
}
|
||
} else {
|
||
_, err = models.Orm.QueryTable(new(models.TenantSiteSetting)).
|
||
Filter("tid", tid).
|
||
Filter("delete_time__isnull", true).
|
||
Update(up)
|
||
if err != nil {
|
||
c.jsonErr(500, 500, "保存失败: "+err.Error())
|
||
return
|
||
}
|
||
}
|
||
|
||
c.Data["json"] = map[string]interface{}{"code": 200, "msg": "保存成功"}
|
||
_ = c.ServeJSON()
|
||
}
|
||
|