增加bark配置

This commit is contained in:
扫地僧 2026-06-17 23:49:19 +08:00
parent 22e0e75c35
commit 6ee1f26124
19 changed files with 784 additions and 260 deletions

View File

@ -66,6 +66,15 @@ func (c *ApiGetCardController) GetCard() {
return return
} }
// 读取机器码/MAC
machineCode := strings.TrimSpace(c.GetString("machine_code"))
if machineCode == "" {
machineCode = strings.TrimSpace(c.GetString("machineCode"))
}
if machineCode == "" {
machineCode = strings.TrimSpace(c.GetString("mac"))
}
// 参数校验 // 参数校验
if platform == "" { if platform == "" {
c.cardErr(400, 400, "缺少参数 type来源平台") c.cardErr(400, 400, "缺少参数 type来源平台")
@ -92,7 +101,7 @@ func (c *ApiGetCardController) GetCard() {
switch module { switch module {
case "cursor": case "cursor":
c.extractCursor(platform, dataType, startID, now) c.extractCursor(platform, dataType, startID, now, machineCode)
case "windsurf": case "windsurf":
c.extractWindsurf(platform, dataType, startID, now) c.extractWindsurf(platform, dataType, startID, now)
case "krio": case "krio":
@ -121,8 +130,25 @@ func (c *ApiGetCardController) readOptionalStartID() (uint64, error) {
return id, nil return id, nil
} }
func (c *ApiGetCardController) extractCursor(platform, dataType string, startID uint64, now time.Time) { func (c *ApiGetCardController) extractCursor(platform, dataType string, startID uint64, now time.Time, machineCode string) {
c.extractWithProbe("cursor", platform, dataType, now, func() (uint64, *string, *string, string, string, *int8, error) { // 优先查询该机器码是否已经绑定过未删除的卡密
if machineCode != "" {
var existing models.PlatformAccountPoolCursor
err := models.Orm.QueryTable(new(models.PlatformAccountPoolCursor)).
Filter("machine_code", machineCode).
Filter("delete_time__isnull", true).
Exclude("is_used", 0).
OrderBy("-id").
Limit(1).
One(&existing)
if err == nil {
// 直接返回已绑定的卡密信息
c.cardOK(buildCardResult(&existing.Account, &existing.Password, existing.Token, existing.DataType))
return
}
}
c.extractWithProbe("cursor", platform, dataType, now, machineCode, func() (uint64, *string, *string, string, string, *int8, error) {
var row models.PlatformAccountPoolCursor var row models.PlatformAccountPoolCursor
qs := models.Orm.QueryTable(new(models.PlatformAccountPoolCursor)). qs := models.Orm.QueryTable(new(models.PlatformAccountPoolCursor)).
Filter("is_extracted", 0). Filter("is_extracted", 0).
@ -141,7 +167,7 @@ func (c *ApiGetCardController) extractCursor(platform, dataType string, startID
} }
func (c *ApiGetCardController) extractWindsurf(platform, dataType string, startID uint64, now time.Time) { func (c *ApiGetCardController) extractWindsurf(platform, dataType string, startID uint64, now time.Time) {
c.extractWithProbe("windsurf", platform, dataType, now, func() (uint64, *string, *string, string, string, *int8, error) { c.extractWithProbe("windsurf", platform, dataType, now, "", func() (uint64, *string, *string, string, string, *int8, error) {
var row models.PlatformAccountPoolWindsurf var row models.PlatformAccountPoolWindsurf
qs := models.Orm.QueryTable(new(models.PlatformAccountPoolWindsurf)). qs := models.Orm.QueryTable(new(models.PlatformAccountPoolWindsurf)).
Filter("is_extracted", 0). Filter("is_extracted", 0).
@ -160,7 +186,7 @@ func (c *ApiGetCardController) extractWindsurf(platform, dataType string, startI
} }
func (c *ApiGetCardController) extractKrio(platform, dataType string, startID uint64, now time.Time) { func (c *ApiGetCardController) extractKrio(platform, dataType string, startID uint64, now time.Time) {
c.extractWithProbe("krio", platform, dataType, now, func() (uint64, *string, *string, string, string, *int8, error) { c.extractWithProbe("krio", platform, dataType, now, "", func() (uint64, *string, *string, string, string, *int8, error) {
var row models.PlatformAccountPoolKiro var row models.PlatformAccountPoolKiro
qs := models.Orm.QueryTable(new(models.PlatformAccountPoolKiro)). qs := models.Orm.QueryTable(new(models.PlatformAccountPoolKiro)).
Filter("is_extracted", 0). Filter("is_extracted", 0).
@ -179,7 +205,7 @@ func (c *ApiGetCardController) extractKrio(platform, dataType string, startID ui
} }
func (c *ApiGetCardController) extractCodex(platform, dataType string, startID uint64, now time.Time) { func (c *ApiGetCardController) extractCodex(platform, dataType string, startID uint64, now time.Time) {
c.extractWithProbe("codex", platform, dataType, now, func() (uint64, *string, *string, string, string, *int8, error) { c.extractWithProbe("codex", platform, dataType, now, "", func() (uint64, *string, *string, string, string, *int8, error) {
var row models.PlatformAccountPoolCodex var row models.PlatformAccountPoolCodex
qs := models.Orm.QueryTable(new(models.PlatformAccountPoolCodex)). qs := models.Orm.QueryTable(new(models.PlatformAccountPoolCodex)).
Filter("is_extracted", 0). Filter("is_extracted", 0).
@ -203,6 +229,7 @@ type poolRowFetcher func() (id uint64, account, password *string, token, rowData
func (c *ApiGetCardController) extractWithProbe( func (c *ApiGetCardController) extractWithProbe(
module, platform, dataType string, module, platform, dataType string,
now time.Time, now time.Time,
machineCode string,
fetch poolRowFetcher, fetch poolRowFetcher,
) { ) {
for { for {
@ -221,14 +248,20 @@ func (c *ApiGetCardController) extractWithProbe(
c.cardErr(500, 500, "无效模块") c.cardErr(500, 500, "无效模块")
return return
} }
_, err = models.Orm.QueryTable(tableName).
Filter("id", id). params := map[string]interface{}{
Update(map[string]interface{}{
"is_extracted": 1, "is_extracted": 1,
"extracted_time": now, "extracted_time": now,
"extracted_platform": platform, "extracted_platform": platform,
"update_time": now, "update_time": now,
}) }
if module == "cursor" && machineCode != "" {
params["machine_code"] = machineCode
}
_, err = models.Orm.QueryTable(tableName).
Filter("id", id).
Update(params)
if err != nil { if err != nil {
c.cardErr(500, 500, "提取失败") c.cardErr(500, 500, "提取失败")
return return
@ -240,10 +273,16 @@ func (c *ApiGetCardController) extractWithProbe(
c.cardOK(buildCardResult(account, password, token, rowDataType)) c.cardOK(buildCardResult(account, password, token, rowDataType))
return return
} }
if module == "cursor" && machineCode != "" {
_, _ = models.Orm.QueryTable(tableName).Filter("id", id).Update(map[string]interface{}{"machine_code": "", "update_time": time.Now()})
}
continue continue
} }
if !poolProbeToken(module, rowDataType, token, id) { if !poolProbeToken(module, rowDataType, token, id) {
if module == "cursor" && machineCode != "" {
_, _ = models.Orm.QueryTable(tableName).Filter("id", id).Update(map[string]interface{}{"machine_code": "", "update_time": time.Now()})
}
continue continue
} }

View File

@ -4,7 +4,6 @@ import (
"encoding/json" "encoding/json"
"io" "io"
"strings" "strings"
"time"
"server/models" "server/models"
"server/pkg/jwtutil" "server/pkg/jwtutil"
@ -209,40 +208,18 @@ func (c *BackendLoginVerifyController) SaveLoginVerifyInfos() {
} }
} }
now := time.Now() err = models.SavePlatformLoginVerify(&models.PlatformLoginVerify{
var existed models.PlatformLoginVerify
err = models.Orm.QueryTable(new(models.PlatformLoginVerify)).OrderBy("-id").One(&existed)
if err == nil {
_, err = models.Orm.QueryTable(new(models.PlatformLoginVerify)).
Filter("id", existed.ID).
Update(map[string]interface{}{
"open_verify_enabled": openVerifyEnabled,
"verify_type": verifyType,
"geetest3_id": geetest3ID,
"geetest3_key": geetest3Key,
"geetest4_id": geetest4ID,
"geetest4_key": geetest4Key,
"update_time": now,
})
if err != nil {
c.jsonErr(500, 500, "保存失败")
return
}
} else {
row := &models.PlatformLoginVerify{
OpenVerifyEnabled: openVerifyEnabled, OpenVerifyEnabled: openVerifyEnabled,
VerifyType: verifyType, VerifyType: verifyType,
Geetest3ID: geetest3ID, Geetest3ID: geetest3ID,
Geetest3Key: geetest3Key, Geetest3Key: geetest3Key,
Geetest4ID: geetest4ID, Geetest4ID: geetest4ID,
Geetest4Key: geetest4Key, Geetest4Key: geetest4Key,
UpdateTime: &now, })
} if err != nil {
if _, err := models.Orm.Insert(row); err != nil {
c.jsonErr(500, 500, "保存失败") c.jsonErr(500, 500, "保存失败")
return return
} }
}
c.Data["json"] = map[string]interface{}{"code": 200, "msg": "保存成功"} c.Data["json"] = map[string]interface{}{"code": 200, "msg": "保存成功"}
_ = c.ServeJSON() _ = c.ServeJSON()

View File

@ -0,0 +1,215 @@
package controllers
import (
"encoding/json"
"fmt"
"io"
"net/http"
"strings"
"time"
"server/models"
"server/pkg/jwtutil"
beego "github.com/beego/beego/v2/server/web"
)
type PlatformBarkController struct {
beego.Controller
}
func (c *PlatformBarkController) platformClaims() (*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")
}
if claims.UserType != "platform" {
return nil, fmt.Errorf("无权访问")
}
return claims, nil
}
func (c *PlatformBarkController) jsonErr(httpStatus, bizCode int, msg string) {
c.Ctx.Output.SetStatus(httpStatus)
c.Data["json"] = map[string]interface{}{"code": bizCode, "msg": msg}
_ = c.ServeJSON()
}
// GetBarkInfo GET /platform/bark/info
func (c *PlatformBarkController) GetBarkInfo() {
if _, err := c.platformClaims(); err != nil {
c.jsonErr(401, 401, err.Error())
return
}
enabledStr := models.GetPlatformSettingValue("bark_enabled", "0")
serverURL := models.GetPlatformSettingValue("bark_server_url", "https://api.day.app")
deviceKey := models.GetPlatformSettingValue("bark_device_key", "")
enabled := false
if enabledStr == "1" {
enabled = true
}
c.Data["json"] = map[string]interface{}{
"code": 200,
"msg": "success",
"data": map[string]interface{}{
"enabled": enabled,
"server_url": serverURL,
"device_key": deviceKey,
},
}
_ = c.ServeJSON()
}
type barkEditPayload struct {
Enabled bool `json:"enabled"`
ServerUrl string `json:"server_url"`
DeviceKey string `json:"device_key"`
}
// EditBarkInfo POST /platform/bark/editinfo
func (c *PlatformBarkController) EditBarkInfo() {
if _, err := c.platformClaims(); 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 barkEditPayload
if err := json.Unmarshal(raw, &p); err != nil {
c.jsonErr(400, 400, "参数错误")
return
}
enabledStr := "0"
if p.Enabled {
enabledStr = "1"
}
serverURL := strings.TrimSpace(p.ServerUrl)
if serverURL == "" {
serverURL = "https://api.day.app"
}
deviceKey := strings.TrimSpace(p.DeviceKey)
settings := []struct {
code string
name string
value string
remark string
}{
{"bark_enabled", "Bark推送启用状态", enabledStr, "0为关闭1为开启"},
{"bark_server_url", "Bark推送服务器地址", serverURL, ""},
{"bark_device_key", "Bark设备Key", deviceKey, ""},
}
for _, item := range settings {
var setting models.PlatformNormalSetting
err := models.Orm.QueryTable(new(models.PlatformNormalSetting)).
Filter("code", item.code).
Filter("delete_time__isnull", true).
One(&setting)
if err == nil {
setting.Value = item.value
setting.Name = item.name
setting.Remark = item.remark
now := time.Now()
setting.UpdateTime = &now
_, err = models.Orm.Update(&setting, "Value", "Name", "Remark", "UpdateTime")
if err != nil {
c.jsonErr(500, 500, "保存失败: "+err.Error())
return
}
} else {
newSetting := models.PlatformNormalSetting{
Name: item.name,
Code: item.code,
Value: item.value,
Remark: item.remark,
CreateTime: time.Now(),
}
_, err = models.Orm.Insert(&newSetting)
if err != nil {
c.jsonErr(500, 500, "保存失败: "+err.Error())
return
}
}
}
c.Data["json"] = map[string]interface{}{"code": 200, "msg": "保存成功"}
_ = c.ServeJSON()
}
type barkTestPayload struct {
ServerUrl string `json:"server_url"`
DeviceKey string `json:"device_key"`
}
// SendTestBark POST /platform/bark/sendtest
func (c *PlatformBarkController) SendTestBark() {
if _, err := c.platformClaims(); 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 barkTestPayload
if err := json.Unmarshal(raw, &p); err != nil {
c.jsonErr(400, 400, "参数错误")
return
}
serverURL := strings.TrimSpace(p.ServerUrl)
if serverURL == "" {
serverURL = models.GetPlatformSettingValue("bark_server_url", "https://api.day.app")
}
deviceKey := strings.TrimSpace(p.DeviceKey)
if deviceKey == "" {
deviceKey = models.GetPlatformSettingValue("bark_device_key", "")
}
if deviceKey == "" {
c.jsonErr(400, 400, "设备 Key 不能为空")
return
}
// 拼接发送 URL注意去除多余斜杠
baseURL := strings.TrimRight(serverURL, "/")
// Bark 的格式是: base_url/device_key/title/body
testURL := fmt.Sprintf("%s/%s/测试通知/您配置的 Bark 推送服务已连接成功!", baseURL, deviceKey)
client := &http.Client{Timeout: 10 * time.Second}
resp, err := client.Get(testURL)
if err != nil {
c.jsonErr(500, 500, "发送失败: "+err.Error())
return
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
bodyBytes, _ := io.ReadAll(resp.Body)
c.jsonErr(500, 500, fmt.Sprintf("发送失败HTTP 状态码: %d, 返回内容: %s", resp.StatusCode, string(bodyBytes)))
return
}
c.Data["json"] = map[string]interface{}{"code": 200, "msg": "测试推送已发出,请注意查收"}
_ = c.ServeJSON()
}

View File

@ -105,10 +105,11 @@ func (c *PlatformCursorEquipmentController) cursorActivationSummary(row *models.
return count, &latest return count, &latest
} }
func (c *PlatformCursorEquipmentController) cursorExtractSummary() (int64, *models.PlatformAccountPoolCursor) { func (c *PlatformCursorEquipmentController) cursorExtractSummary(machineCode string) (int64, *models.PlatformAccountPoolCursor) {
qs := models.Orm.QueryTable(new(models.PlatformAccountPoolCursor)). qs := models.Orm.QueryTable(new(models.PlatformAccountPoolCursor)).
Filter("delete_time__isnull", true). Filter("delete_time__isnull", true).
Filter("is_extracted__gt", 0) Filter("is_extracted__gt", 0).
Filter("machine_code", machineCode)
count, _ := qs.Count() count, _ := qs.Count()
@ -122,7 +123,7 @@ func (c *PlatformCursorEquipmentController) cursorExtractSummary() (int64, *mode
func (c *PlatformCursorEquipmentController) rowToMap(row *models.PlatformCursorEquipment) map[string]interface{} { func (c *PlatformCursorEquipmentController) rowToMap(row *models.PlatformCursorEquipment) map[string]interface{} {
activationCount, latestActivation := c.cursorActivationSummary(row) activationCount, latestActivation := c.cursorActivationSummary(row)
extractCount, latestExtract := c.cursorExtractSummary() extractCount, latestExtract := c.cursorExtractSummary(row.MachineCode)
var bindActivationCode interface{} var bindActivationCode interface{}
var activationCodeId interface{} var activationCodeId interface{}
@ -640,7 +641,8 @@ func (c *PlatformCursorEquipmentController) ExtractRecords() {
qs := models.Orm.QueryTable(new(models.PlatformAccountPoolCursor)). qs := models.Orm.QueryTable(new(models.PlatformAccountPoolCursor)).
Filter("delete_time__isnull", true). Filter("delete_time__isnull", true).
Filter("is_extracted__gt", 0) Filter("is_extracted__gt", 0).
Filter("machine_code", equipment.MachineCode)
total, _ := qs.Count() total, _ := qs.Count()

View File

@ -85,38 +85,19 @@ func (c *PlatformLoginVerifyController) SaveLoginVerifyInfos() {
} }
} }
var existed models.PlatformLoginVerify err := models.SavePlatformLoginVerify(&models.PlatformLoginVerify{
err := models.Orm.QueryTable(new(models.PlatformLoginVerify)).OrderBy("-id").One(&existed)
if err == nil {
update := map[string]interface{}{
"open_verify_enabled": openVerifyEnabled,
"verify_type": verifyType,
"geetest3_id": p.Geetest3ID,
"geetest3_key": p.Geetest3Key,
"geetest4_id": p.Geetest4ID,
"geetest4_key": p.Geetest4Key,
}
_, err = models.Orm.QueryTable(new(models.PlatformLoginVerify)).Filter("id", existed.ID).Update(update)
if err != nil {
c.Data["json"] = map[string]interface{}{"code": 500, "msg": "保存失败"}
_ = c.ServeJSON()
return
}
} else {
row := &models.PlatformLoginVerify{
OpenVerifyEnabled: openVerifyEnabled, OpenVerifyEnabled: openVerifyEnabled,
VerifyType: verifyType, VerifyType: verifyType,
Geetest3ID: p.Geetest3ID, Geetest3ID: p.Geetest3ID,
Geetest3Key: p.Geetest3Key, Geetest3Key: p.Geetest3Key,
Geetest4ID: p.Geetest4ID, Geetest4ID: p.Geetest4ID,
Geetest4Key: p.Geetest4Key, Geetest4Key: p.Geetest4Key,
} })
if _, err := models.Orm.Insert(row); err != nil { if err != nil {
c.Data["json"] = map[string]interface{}{"code": 500, "msg": "保存失败"} c.Data["json"] = map[string]interface{}{"code": 500, "msg": "保存失败"}
_ = c.ServeJSON() _ = c.ServeJSON()
return return
} }
}
c.Data["json"] = map[string]interface{}{"code": 200, "msg": "保存成功"} c.Data["json"] = map[string]interface{}{"code": 200, "msg": "保存成功"}
_ = c.ServeJSON() _ = c.ServeJSON()

View File

@ -55,24 +55,8 @@ func (c *PlatformSMSController) GetSmsInfo() {
return return
} }
var row models.SystemSMS backendURL := models.GetPlatformSettingValue("sms_custom_url", "")
// 优先默认通道,其次 custom apiKey := models.GetPlatformSettingValue("sms_custom_key", "")
err := models.Orm.QueryTable(new(models.SystemSMS)).
Filter("is_default", 1).
Filter("status", 1).
OrderBy("-weight", "-id").
Limit(1).
One(&row)
if err != nil {
_ = models.Orm.QueryTable(new(models.SystemSMS)).
Filter("config_code", "custom").
OrderBy("-id").
Limit(1).
One(&row)
}
backendURL := strings.TrimSpace(row.ApiURL)
apiKey := strings.TrimSpace(row.ApiKey)
data := []map[string]interface{}{{ data := []map[string]interface{}{{
"backend_url": backendURL, "backend_url": backendURL,
@ -93,7 +77,7 @@ type smsEditPayload struct {
} }
// EditSmsInfo POST /platform/sms/editinfo // EditSmsInfo POST /platform/sms/editinfo
// 将旧前端的 backendUrl/apiKey 落到 yz_system_sms 的 api_url/api_key写入 config_code=custom // 将旧前端的 backendUrl/apiKey 落到 yz_platform_normal_setting 表中
func (c *PlatformSMSController) EditSmsInfo() { func (c *PlatformSMSController) EditSmsInfo() {
if _, err := c.platformClaims(); err != nil { if _, err := c.platformClaims(); err != nil {
c.jsonErr(401, 401, err.Error()) c.jsonErr(401, 401, err.Error())
@ -128,46 +112,48 @@ func (c *PlatformSMSController) EditSmsInfo() {
return return
} }
// 确保只有一个默认:先清空默认,再 upsert custom 为默认 settings := []struct {
_, _ = models.Orm.QueryTable(new(models.SystemSMS)).Update(map[string]interface{}{"is_default": 0}) code string
name string
value string
remark string
}{
{"sms_custom_url", "自定义短信网关地址", backendURL, ""},
{"sms_custom_key", "自定义短信API KEY", apiKey, ""},
}
var existed models.SystemSMS for _, item := range settings {
e := models.Orm.QueryTable(new(models.SystemSMS)).Filter("config_code", "custom").Limit(1).One(&existed) var setting models.PlatformNormalSetting
if e == nil && existed.ID > 0 { err := models.Orm.QueryTable(new(models.PlatformNormalSetting)).
_, err = models.Orm.QueryTable(new(models.SystemSMS)).Filter("id", existed.ID).Update(map[string]interface{}{ Filter("code", item.code).
"config_name": "自定义网关", Filter("delete_time__isnull", true).
"channel_type": 2, One(&setting)
"api_url": backendURL, if err == nil {
"api_key": apiKey, setting.Value = item.value
"weight": 10, setting.Name = item.name
"is_default": 1, setting.Remark = item.remark
"status": 1, now := time.Now()
}) setting.UpdateTime = &now
_, err = models.Orm.Update(&setting, "Value", "Name", "Remark", "UpdateTime")
if err != nil { if err != nil {
c.jsonErr(500, 500, "更新失败: "+err.Error()) c.jsonErr(500, 500, "保存失败: "+err.Error())
return return
} }
} else { } else {
row := &models.SystemSMS{ newSetting := models.PlatformNormalSetting{
ConfigCode: "custom", Name: item.name,
ConfigName: "自定义网关", Code: item.code,
ChannelType: 2, Value: item.value,
ApiURL: backendURL, Remark: item.remark,
ApiKey: apiKey, CreateTime: time.Now(),
ApiSecret: "",
SignName: "",
TemplateID: "",
TestPhone: "",
Weight: 10,
IsDefault: 1,
Status: 1,
Remark: "",
} }
if _, err := models.Orm.Insert(row); err != nil { _, err = models.Orm.Insert(&newSetting)
c.jsonErr(500, 500, "更新失败: "+err.Error()) if err != nil {
c.jsonErr(500, 500, "保存失败: "+err.Error())
return return
} }
} }
}
updated := []map[string]interface{}{{ updated := []map[string]interface{}{{
"backend_url": backendURL, "backend_url": backendURL,
@ -232,18 +218,11 @@ func (c *PlatformSMSController) SendTestSms() {
// 兜底body 未带时从默认配置取 // 兜底body 未带时从默认配置取
if backendURL == "" || apiKey == "" { if backendURL == "" || apiKey == "" {
var row models.SystemSMS
_ = models.Orm.QueryTable(new(models.SystemSMS)).
Filter("is_default", 1).
Filter("status", 1).
OrderBy("-weight", "-id").
Limit(1).
One(&row)
if backendURL == "" { if backendURL == "" {
backendURL = strings.TrimSpace(row.ApiURL) backendURL = models.GetPlatformSettingValue("sms_custom_url", "")
} }
if apiKey == "" { if apiKey == "" {
apiKey = strings.TrimSpace(row.ApiKey) apiKey = models.GetPlatformSettingValue("sms_custom_key", "")
} }
} }
if backendURL == "" { if backendURL == "" {

View File

@ -43,14 +43,11 @@ func Init(_ string) {
new(AdminRole), new(AdminRole),
new(SystemFile), new(SystemFile),
new(SystemFilesCategory), new(SystemFilesCategory),
new(SystemEmail),
new(SystemSMS),
new(SystemSMSTask), new(SystemSMSTask),
new(SystemOperationLog), new(SystemOperationLog),
new(SystemDomainPool), new(SystemDomainPool),
new(SystemTenantDomain), new(SystemTenantDomain),
new(SystemModules), new(SystemModules),
new(PlatformLoginVerify),
new(StorageConfig), new(StorageConfig),
new(TenantSiteSetting), new(TenantSiteSetting),
new(ComplaintCategory), new(ComplaintCategory),
@ -67,8 +64,11 @@ func Init(_ string) {
new(CmsArticleCategory), new(CmsArticleCategory),
new(CmsArticle), new(CmsArticle),
new(SystemSiteReminder),
new(SystemReminderList), new(SystemReminderList),
new(SystemNormalSetting),
new(PlatformNormalSetting),
new(BackendNormalSetting),
) )
// 创建全局 Ormer // 创建全局 Ormer

View File

@ -75,6 +75,7 @@ type PlatformAccountPoolCursor struct {
IsUsed *int8 `orm:"column(is_used);null" json:"is_used"` // 0=用完/不可用 1=可用 NULL=未探测 IsUsed *int8 `orm:"column(is_used);null" json:"is_used"` // 0=用完/不可用 1=可用 NULL=未探测
ExtractedTime *time.Time `orm:"column(extracted_time);type(datetime);null" json:"extracted_time"` ExtractedTime *time.Time `orm:"column(extracted_time);type(datetime);null" json:"extracted_time"`
ExtractedPlatform *string `orm:"column(extracted_platform);size(32);null" json:"extracted_platform"` ExtractedPlatform *string `orm:"column(extracted_platform);size(32);null" json:"extracted_platform"`
MachineCode string `orm:"column(machine_code);size(128);default('')" json:"machine_code"`
CreateTime time.Time `orm:"column(create_time);auto_now_add;type(datetime)" json:"create_time"` CreateTime time.Time `orm:"column(create_time);auto_now_add;type(datetime)" json:"create_time"`
UpdateTime *time.Time `orm:"column(update_time);type(datetime);null" json:"update_time"` UpdateTime *time.Time `orm:"column(update_time);type(datetime);null" json:"update_time"`
DeleteTime *time.Time `orm:"column(delete_time);type(datetime);null" json:"delete_time"` DeleteTime *time.Time `orm:"column(delete_time);type(datetime);null" json:"delete_time"`

View File

@ -15,20 +15,118 @@ type PlatformLoginVerify struct {
UpdateTime *time.Time `orm:"column(update_time);type(datetime);auto_now;null" json:"update_time"` UpdateTime *time.Time `orm:"column(update_time);type(datetime);auto_now;null" json:"update_time"`
} }
func (m *PlatformLoginVerify) TableName() string {
return "yz_system_login_verify"
}
func GetPlatformLoginVerify() (*PlatformLoginVerify, error) { func GetPlatformLoginVerify() (*PlatformLoginVerify, error) {
var cfg PlatformLoginVerify // 从 yz_platform_normal_setting 表中按 code 获取各个配置
err := Orm.QueryTable(new(PlatformLoginVerify)).OrderBy("-id").One(&cfg) enabledStr := GetPlatformSettingValue("login_verify_enabled", "1")
if err != nil { verifyType := GetPlatformSettingValue("login_verify_type", "captcha")
// 默认配置:验证码 geetest3ID := GetPlatformSettingValue("login_verify_geetest3_id", "")
return &PlatformLoginVerify{OpenVerifyEnabled: 1, VerifyType: "captcha"}, nil geetest3Key := GetPlatformSettingValue("login_verify_geetest3_key", "")
geetest4ID := GetPlatformSettingValue("login_verify_geetest4_id", "")
geetest4Key := GetPlatformSettingValue("login_verify_geetest4_key", "")
openVerifyEnabled := int8(1)
if enabledStr == "0" {
openVerifyEnabled = 0
} }
if cfg.VerifyType == "" {
cfg.VerifyType = "captcha" cfg := &PlatformLoginVerify{
OpenVerifyEnabled: openVerifyEnabled,
VerifyType: verifyType,
} }
return &cfg, nil if geetest3ID != "" {
cfg.Geetest3ID = &geetest3ID
}
if geetest3Key != "" {
cfg.Geetest3Key = &geetest3Key
}
if geetest4ID != "" {
cfg.Geetest4ID = &geetest4ID
}
if geetest4Key != "" {
cfg.Geetest4Key = &geetest4Key
}
return cfg, nil
}
func GetPlatformSettingValue(code string, defaultVal string) string {
var setting PlatformNormalSetting
err := Orm.QueryTable(new(PlatformNormalSetting)).
Filter("code", code).
Filter("delete_time__isnull", true).
One(&setting)
if err != nil {
return defaultVal
}
return setting.Value
}
func SavePlatformLoginVerify(cfg *PlatformLoginVerify) error {
openVerifyEnabledStr := "1"
if cfg.OpenVerifyEnabled == 0 {
openVerifyEnabledStr = "0"
}
geetest3ID := ""
if cfg.Geetest3ID != nil {
geetest3ID = *cfg.Geetest3ID
}
geetest3Key := ""
if cfg.Geetest3Key != nil {
geetest3Key = *cfg.Geetest3Key
}
geetest4ID := ""
if cfg.Geetest4ID != nil {
geetest4ID = *cfg.Geetest4ID
}
geetest4Key := ""
if cfg.Geetest4Key != nil {
geetest4Key = *cfg.Geetest4Key
}
settings := []struct {
code string
name string
value string
remark string
}{
{"login_verify_enabled", "登录验证开启状态", openVerifyEnabledStr, "0为关闭1为开启"},
{"login_verify_type", "登录验证类型", cfg.VerifyType, "支持 captcha/sms/geetest/email"},
{"login_verify_geetest3_id", "极验3 ID", geetest3ID, ""},
{"login_verify_geetest3_key", "极验3 Key", geetest3Key, ""},
{"login_verify_geetest4_id", "极验4 ID", geetest4ID, ""},
{"login_verify_geetest4_key", "极验4 Key", geetest4Key, ""},
}
for _, item := range settings {
var setting PlatformNormalSetting
err := Orm.QueryTable(new(PlatformNormalSetting)).
Filter("code", item.code).
Filter("delete_time__isnull", true).
One(&setting)
if err == nil {
setting.Value = item.value
setting.Name = item.name
setting.Remark = item.remark
now := time.Now()
setting.UpdateTime = &now
_, err = Orm.Update(&setting, "Value", "Name", "Remark", "UpdateTime")
if err != nil {
return err
}
} else {
newSetting := PlatformNormalSetting{
Name: item.name,
Code: item.code,
Value: item.value,
Remark: item.remark,
CreateTime: time.Now(),
}
_, err = Orm.Insert(&newSetting)
if err != nil {
return err
}
}
}
return nil
} }

View File

@ -18,6 +18,4 @@ type SystemEmail struct {
UpdateTime time.Time `orm:"column(update_time);type(datetime);auto_now;null" json:"update_time"` UpdateTime time.Time `orm:"column(update_time);type(datetime);auto_now;null" json:"update_time"`
} }
func (m *SystemEmail) TableName() string {
return "yz_system_email"
}

View File

@ -0,0 +1,51 @@
package models
import "time"
// SystemNormalSetting 系统通用配置表: yz_system_normal_setting
type SystemNormalSetting struct {
ID uint64 `orm:"column(id);pk;auto" json:"id"`
Name string `orm:"column(name);size(128);default('')" json:"name"`
Value string `orm:"column(value);type(text);null" json:"value"`
Code string `orm:"column(code);size(64);default('')" json:"code"`
Remark string `orm:"column(remark);size(255);default('')" json:"remark"`
CreateTime time.Time `orm:"column(create_time);auto_now_add;type(datetime)" json:"create_time"`
UpdateTime *time.Time `orm:"column(update_time);type(datetime);null" json:"update_time"`
DeleteTime *time.Time `orm:"column(delete_time);type(datetime);null" json:"delete_time"`
}
func (m *SystemNormalSetting) TableName() string {
return "yz_system_normal_setting"
}
// PlatformNormalSetting 平台通用配置表: yz_platform_normal_setting
type PlatformNormalSetting struct {
ID uint64 `orm:"column(id);pk;auto" json:"id"`
Name string `orm:"column(name);size(128);default('')" json:"name"`
Value string `orm:"column(value);type(text);null" json:"value"`
Code string `orm:"column(code);size(64);default('')" json:"code"`
Remark string `orm:"column(remark);size(255);default('')" json:"remark"`
CreateTime time.Time `orm:"column(create_time);auto_now_add;type(datetime)" json:"create_time"`
UpdateTime *time.Time `orm:"column(update_time);type(datetime);null" json:"update_time"`
DeleteTime *time.Time `orm:"column(delete_time);type(datetime);null" json:"delete_time"`
}
func (m *PlatformNormalSetting) TableName() string {
return "yz_platform_normal_setting"
}
// BackendNormalSetting 管理端通用配置表: yz_backend_normal_setting
type BackendNormalSetting struct {
ID uint64 `orm:"column(id);pk;auto" json:"id"`
Name string `orm:"column(name);size(128);default('')" json:"name"`
Value string `orm:"column(value);type(text);null" json:"value"`
Code string `orm:"column(code);size(64);default('')" json:"code"`
Remark string `orm:"column(remark);size(255);default('')" json:"remark"`
CreateTime time.Time `orm:"column(create_time);auto_now_add;type(datetime)" json:"create_time"`
UpdateTime *time.Time `orm:"column(update_time);type(datetime);null" json:"update_time"`
DeleteTime *time.Time `orm:"column(delete_time);type(datetime);null" json:"delete_time"`
}
func (m *BackendNormalSetting) TableName() string {
return "yz_backend_normal_setting"
}

View File

@ -11,6 +11,4 @@ type SystemSiteReminder struct {
UpdateTime *time.Time `orm:"column(update_time);type(datetime);null" json:"update_time"` UpdateTime *time.Time `orm:"column(update_time);type(datetime);null" json:"update_time"`
} }
func (m *SystemSiteReminder) TableName() string {
return "yz_system_sitereminder"
}

View File

@ -27,6 +27,4 @@ type SystemSMS struct {
UpdateTime time.Time `orm:"column(update_time);type(datetime);auto_now" json:"update_time"` UpdateTime time.Time `orm:"column(update_time);type(datetime);auto_now" json:"update_time"`
} }
func (m *SystemSMS) TableName() string {
return "yz_system_sms"
}

View File

@ -144,6 +144,11 @@ func Register() {
beego.Router("/platform/sms/taskList", &controllers.PlatformSMSController{}, "get:GetSmsTaskList") beego.Router("/platform/sms/taskList", &controllers.PlatformSMSController{}, "get:GetSmsTaskList")
beego.Router("/platform/sms/taskEdit/:id", &controllers.PlatformSMSController{}, "post:EditSmsTask") beego.Router("/platform/sms/taskEdit/:id", &controllers.PlatformSMSController{}, "post:EditSmsTask")
// Bark 推送配置
beego.Router("/platform/bark/info", &controllers.PlatformBarkController{}, "get:GetBarkInfo")
beego.Router("/platform/bark/editinfo", &controllers.PlatformBarkController{}, "post:EditBarkInfo")
beego.Router("/platform/bark/sendtest", &controllers.PlatformBarkController{}, "post:SendTestBark")
// 文件管理yz_system_files / yz_system_files_category // 文件管理yz_system_files / yz_system_files_category
beego.Router("/platform/usercate", &controllers.PlatformFileController{}, "get:GetUserCate") beego.Router("/platform/usercate", &controllers.PlatformFileController{}, "get:GetUserCate")
beego.Router("/platform/allfiles", &controllers.PlatformFileController{}, "get:GetAllFiles") beego.Router("/platform/allfiles", &controllers.PlatformFileController{}, "get:GetAllFiles")

View File

@ -157,25 +157,11 @@ func VerifyBackendLoginCode(tenantName, account, channel, code string) error {
} }
func getDefaultSystemSMSConfig() (backendURL string, apiKey string, err error) { func getDefaultSystemSMSConfig() (backendURL string, apiKey string, err error) {
var row models.SystemSMS backendURL = models.GetPlatformSettingValue("sms_custom_url", "")
err = models.Orm.QueryTable(new(models.SystemSMS)). apiKey = models.GetPlatformSettingValue("sms_custom_key", "")
Filter("is_default", 1). if backendURL == "" || apiKey == "" {
Filter("status", 1). return "", "", fmt.Errorf("短信网关未配置")
OrderBy("-weight", "-id").
Limit(1).
One(&row)
if err != nil {
err2 := models.Orm.QueryTable(new(models.SystemSMS)).
Filter("config_code", "custom").
OrderBy("-id").
Limit(1).
One(&row)
if err2 != nil {
return "", "", err2
} }
}
backendURL = strings.TrimSpace(row.ApiURL)
apiKey = strings.TrimSpace(row.ApiKey)
return backendURL, apiKey, nil return backendURL, apiKey, nil
} }

View File

@ -2,19 +2,51 @@ package services
import ( import (
"fmt" "fmt"
"strconv"
"strings" "strings"
"time"
"server/models" "server/models"
) )
// ListSystemEmails 返回全部邮箱配置(按 id 升序,通常仅一条) // ListSystemEmails 返回从 yz_platform_normal_setting 组装的邮箱配置(切片,通常仅一条)
func ListSystemEmails() ([]models.SystemEmail, error) { func ListSystemEmails() ([]models.SystemEmail, error) {
var rows []models.SystemEmail enabledStr := models.GetPlatformSettingValue("email_enabled", "0")
_, err := models.Orm.QueryTable(new(models.SystemEmail)).OrderBy("id").All(&rows) fromAddress := models.GetPlatformSettingValue("email_from_address", "")
return rows, err fromName := models.GetPlatformSettingValue("email_from_name", "")
host := models.GetPlatformSettingValue("email_host", "")
portStr := models.GetPlatformSettingValue("email_port", "465")
password := models.GetPlatformSettingValue("email_password", "")
encryption := models.GetPlatformSettingValue("email_encryption", "ssl")
timeoutStr := models.GetPlatformSettingValue("email_timeout", "30")
status := int8(0)
if enabledStr == "1" {
status = 1
}
portVal, _ := strconv.ParseUint(portStr, 10, 32)
timeoutVal, _ := strconv.ParseUint(timeoutStr, 10, 32)
row := models.SystemEmail{
ID: 1,
FromAddress: fromAddress,
Host: host,
Port: uint(portVal),
Password: password,
Encryption: encryption,
Timeout: uint(timeoutVal),
Status: status,
CreateTime: time.Now(),
UpdateTime: time.Now(),
}
if fromName != "" {
row.FromName = &fromName
}
return []models.SystemEmail{row}, nil
} }
// UpsertFirstSystemEmail 若已有记录则更新第一条,否则插入 // UpsertFirstSystemEmail 将邮箱配置保存到 yz_platform_normal_setting 表中
func UpsertFirstSystemEmail(fromAddress string, fromName *string, host string, port uint, password string, encryption string, timeout uint, status int8, remark *string) error { func UpsertFirstSystemEmail(fromAddress string, fromName *string, host string, port uint, password string, encryption string, timeout uint, status int8, remark *string) error {
if encryption == "" { if encryption == "" {
encryption = "ssl" encryption = "ssl"
@ -28,47 +60,76 @@ func UpsertFirstSystemEmail(fromAddress string, fromName *string, host string, p
fromAddress = strings.TrimSpace(fromAddress) fromAddress = strings.TrimSpace(fromAddress)
host = strings.TrimSpace(host) host = strings.TrimSpace(host)
cnt, err := models.Orm.QueryTable(new(models.SystemEmail)).Count() fn := ""
if fromName != nil {
fn = *fromName
}
statusStr := "0"
if status == 1 {
statusStr = "1"
}
settings := []struct {
code string
name string
value string
remark string
}{
{"email_enabled", "邮件服务启用状态", statusStr, "0为关闭1为开启"},
{"email_from_address", "发件人邮箱", fromAddress, ""},
{"email_from_name", "发件人名称", fn, ""},
{"email_host", "SMTP 服务器地址", host, ""},
{"email_port", "SMTP 端口", strconv.FormatUint(uint64(port), 10), ""},
{"email_encryption", "邮件加密方式", encryption, "支持 ssl/tls/none"},
{"email_timeout", "邮件发送超时时间", strconv.FormatUint(uint64(timeout), 10), ""},
}
// 如果传入了新密码,或者目前还没有保存过密码,才更新密码
if strings.TrimSpace(password) != "" {
settings = append(settings, struct {
code string
name string
value string
remark string
}{"email_password", "邮件授权码/密码", strings.TrimSpace(password), ""})
} else {
// 校验:如果完全没有配置过密码,必须填写密码
existingPass := models.GetPlatformSettingValue("email_password", "")
if existingPass == "" {
return fmt.Errorf("首次保存必须填写授权码/密码")
}
}
for _, item := range settings {
var setting models.PlatformNormalSetting
err := models.Orm.QueryTable(new(models.PlatformNormalSetting)).
Filter("code", item.code).
Filter("delete_time__isnull", true).
One(&setting)
if err == nil {
setting.Value = item.value
setting.Name = item.name
setting.Remark = item.remark
now := time.Now()
setting.UpdateTime = &now
_, err = models.Orm.Update(&setting, "Value", "Name", "Remark", "UpdateTime")
if err != nil { if err != nil {
return err return err
} }
if cnt == 0 { } else {
if strings.TrimSpace(password) == "" { newSetting := models.PlatformNormalSetting{
return fmt.Errorf("首次保存必须填写授权码/密码") Name: item.name,
Code: item.code,
Value: item.value,
Remark: item.remark,
CreateTime: time.Now(),
} }
row := &models.SystemEmail{ _, err = models.Orm.Insert(&newSetting)
FromAddress: fromAddress, if err != nil {
FromName: fromName,
Host: host,
Port: port,
Password: strings.TrimSpace(password),
Encryption: encryption,
Timeout: timeout,
Status: status,
Remark: remark,
}
_, err = models.Orm.Insert(row)
return err return err
} }
var first models.SystemEmail
if err := models.Orm.QueryTable(new(models.SystemEmail)).OrderBy("id").Limit(1).One(&first); err != nil {
return err
} }
up := map[string]interface{}{
"from_address": fromAddress,
"from_name": fromName,
"host": host,
"port": port,
"encryption": encryption,
"timeout": timeout,
"status": status,
"remark": remark,
} }
if strings.TrimSpace(password) != "" { return nil
up["password"] = strings.TrimSpace(password)
}
_, err = models.Orm.QueryTable(new(models.SystemEmail)).Filter("id", first.ID).Update(up)
return err
} }

View File

@ -3,32 +3,36 @@ package services
import ( import (
"context" "context"
"fmt" "fmt"
"strconv"
"time" "time"
"github.com/beego/beego/v2/client/orm" "github.com/beego/beego/v2/client/orm"
"server/models" "server/models"
) )
// GetSiteReminderConfig 获取站内信配置(只读首条记录,不存在则初始化默认值 // GetSiteReminderConfig 获取站内信配置(从 yz_platform_normal_setting 读取
func GetSiteReminderConfig() (models.SystemSiteReminder, error) { func GetSiteReminderConfig() (models.SystemSiteReminder, error) {
var row models.SystemSiteReminder retentionDaysStr := models.GetPlatformSettingValue("sitemsg_retention_days", "30")
err := models.Orm.QueryTable(new(models.SystemSiteReminder)).OrderBy("id").Limit(1).One(&row) autoReadStr := models.GetPlatformSettingValue("sitemsg_auto_read", "0")
if err == orm.ErrNoRows {
// 默认配置 retentionDays, _ := strconv.Atoi(retentionDaysStr)
if retentionDays <= 0 {
retentionDays = 30
}
autoRead := int8(0)
if autoReadStr == "1" {
autoRead = 1
}
now := time.Now() now := time.Now()
row = models.SystemSiteReminder{ row := models.SystemSiteReminder{
RetentionDays: 30, ID: 1,
AutoRead: 0, RetentionDays: retentionDays,
AutoRead: autoRead,
CreateTime: &now, CreateTime: &now,
UpdateTime: &now, UpdateTime: &now,
} }
_, err = models.Orm.Insert(&row)
if err != nil {
return row, err
}
return row, nil return row, nil
}
return row, err
} }
// SaveSiteReminderConfig 保存/更新配置 // SaveSiteReminderConfig 保存/更新配置
@ -36,17 +40,53 @@ func SaveSiteReminderConfig(retentionDays int, autoRead int8) error {
if retentionDays <= 0 { if retentionDays <= 0 {
retentionDays = 30 retentionDays = 30
} }
cfg, err := GetSiteReminderConfig()
autoReadStr := "0"
if autoRead == 1 {
autoReadStr = "1"
}
settings := []struct {
code string
name string
value string
remark string
}{
{"sitemsg_retention_days", "站内信消息保留天数", strconv.Itoa(retentionDays), ""},
{"sitemsg_auto_read", "自动标记已读状态", autoReadStr, "0为关闭1为开启"},
}
for _, item := range settings {
var setting models.PlatformNormalSetting
err := models.Orm.QueryTable(new(models.PlatformNormalSetting)).
Filter("code", item.code).
Filter("delete_time__isnull", true).
One(&setting)
if err == nil {
setting.Value = item.value
setting.Name = item.name
setting.Remark = item.remark
now := time.Now()
setting.UpdateTime = &now
_, err = models.Orm.Update(&setting, "Value", "Name", "Remark", "UpdateTime")
if err != nil { if err != nil {
return err return err
} }
now := time.Now() } else {
cfg.RetentionDays = retentionDays newSetting := models.PlatformNormalSetting{
cfg.AutoRead = autoRead Name: item.name,
cfg.UpdateTime = &now Code: item.code,
Value: item.value,
_, err = models.Orm.Update(&cfg, "RetentionDays", "AutoRead", "UpdateTime") Remark: item.remark,
CreateTime: time.Now(),
}
_, err = models.Orm.Insert(&newSetting)
if err != nil {
return err return err
}
}
}
return nil
} }
// SendSiteReminder 发送站内信 // SendSiteReminder 发送站内信

View File

@ -151,3 +151,40 @@ export function saveStorageConfig(data) {
data: data, data: data,
}); });
} }
/**
* 获取 Bark 配置
* @returns {Promise}
*/
export function getBarkConfig() {
return request({
url: "/platform/bark/info",
method: "get",
});
}
/**
* 保存 Bark 配置
* @param {Object} data 要保存的数据
* @returns {Promise}
*/
export function saveBarkConfig(data) {
return request({
url: "/platform/bark/editinfo",
method: "post",
data: data,
});
}
/**
* 发送测试 Bark 推送
* @param {Object} data 测试数据
* @returns {Promise}
*/
export function sendTestBark(data) {
return request({
url: "/platform/bark/sendtest",
method: "post",
data: data,
});
}

View File

@ -155,10 +155,12 @@
</div> </div>
</template> </template>
<el-form :model="formData.sitemsg" label-width="140px" label-position="right"> <el-form :model="formData.sitemsg" label-width="140px" label-position="right">
<!--
<el-form-item label="启用状态"> <el-form-item label="启用状态">
<el-switch v-model="formData.sitemsg.enabled" disabled /> <el-switch v-model="formData.sitemsg.enabled" disabled />
<span class="form-tip-inline" style="margin-left: 10px;">站内信功能为系统核心功能必须开启</span> <span class="form-tip-inline" style="margin-left: 10px;">站内信功能为系统核心功能必须开启</span>
</el-form-item> </el-form-item>
-->
<el-form-item label="保留天数"> <el-form-item label="保留天数">
<el-input-number v-model="formData.sitemsg.retention_days" :min="1" :max="365" controls-position="right" /> <el-input-number v-model="formData.sitemsg.retention_days" :min="1" :max="365" controls-position="right" />
<span class="form-tip-inline">过期消息将自动清理</span> <span class="form-tip-inline">过期消息将自动清理</span>
@ -296,6 +298,11 @@
<el-form-item label="设备 Key"> <el-form-item label="设备 Key">
<el-input v-model="formData.bark.device_key" placeholder="请输入您的 Bark Device Key" /> <el-input v-model="formData.bark.device_key" placeholder="请输入您的 Bark Device Key" />
</el-form-item> </el-form-item>
<el-form-item label="测试推送">
<el-button type="success" plain size="small" :loading="barkTestLoading" @click="handleSendBarkTest">
发送测试推送
</el-button>
</el-form-item>
</el-form> </el-form>
</el-card> </el-card>
</div> </div>
@ -395,10 +402,12 @@ import {
import { getSmsInfo, editSmsInfo, sendTestSms } from "@/api/sms"; import { getSmsInfo, editSmsInfo, sendTestSms } from "@/api/sms";
import { getEmailInfo, editEmailInfo, sendTestEmail } from "@/api/email"; import { getEmailInfo, editEmailInfo, sendTestEmail } from "@/api/email";
import { getSiteReminderConfig, saveSiteReminderConfig } from "@/api/sitereminder"; import { getSiteReminderConfig, saveSiteReminderConfig } from "@/api/sitereminder";
import { getBarkConfig, saveBarkConfig, sendTestBark } from "@/api/sitesettings";
const STORAGE_KEY = "notification_settings_draft"; const STORAGE_KEY = "notification_settings_draft";
const activeSubTab = ref("email"); const activeSubTab = ref("email");
const submitting = ref(false); const submitting = ref(false);
const barkTestLoading = ref(false);
const formData = reactive({ const formData = reactive({
email: { email: {
@ -659,6 +668,17 @@ const handleSubmit = async () => {
} else { } else {
throw new Error(res.msg || "保存站内信配置失败"); throw new Error(res.msg || "保存站内信配置失败");
} }
} else if (activeSubTab.value === 'bark') {
const res = await saveBarkConfig({
enabled: formData.bark.enabled,
server_url: formData.bark.server_url,
device_key: formData.bark.device_key
});
if (res.code === 200) {
ElMessage.success("Bark 配置保存成功");
} else {
throw new Error(res.msg || "保存 Bark 配置失败");
}
} }
saveDraft(); saveDraft();
@ -670,11 +690,49 @@ const handleSubmit = async () => {
} }
}; };
const loadBarkConfig = async () => {
try {
const res = await getBarkConfig();
if (res.code === 200 && res.data) {
formData.bark.enabled = res.data.enabled;
formData.bark.server_url = res.data.server_url || "https://api.day.app";
formData.bark.device_key = res.data.device_key || "";
}
} catch (error) {
console.error("加载 Bark 配置失败:", error);
}
};
const handleSendBarkTest = async () => {
const key = (formData.bark.device_key || "").trim();
if (!key) {
ElMessage.warning("请先输入 Bark 设备 Key");
return;
}
try {
barkTestLoading.value = true;
const res = await sendTestBark({
server_url: formData.bark.server_url,
device_key: key
});
if (res.code === 200) {
ElMessage.success(res.msg || "测试推送发送成功");
} else {
ElMessage.error(res.msg || "测试推送失败,请稍后重试");
}
} catch (error) {
ElMessage.error(error?.message || "测试推送失败,请稍后重试");
} finally {
barkTestLoading.value = false;
}
};
onMounted(() => { onMounted(() => {
loadDraft(); loadDraft();
loadSmsConfig(); loadSmsConfig();
loadEmailConfig(); loadEmailConfig();
loadSiteReminderConfig(); loadSiteReminderConfig();
loadBarkConfig();
}); });
</script> </script>