nezha/cmd/dashboard/controller/ddns.go

209 lines
5.1 KiB
Go
Raw Normal View History

package controller
import (
"errors"
"fmt"
"strconv"
"strings"
"github.com/gin-gonic/gin"
2024-10-23 22:21:59 -04:00
"github.com/jinzhu/copier"
2024-10-21 04:22:30 -04:00
"golang.org/x/net/idna"
"github.com/naiba/nezha/model"
"github.com/naiba/nezha/service/singleton"
)
2024-10-23 22:21:59 -04:00
// List DDNS Profiles
// @Summary List DDNS profiles
// @Schemes
// @Description List DDNS profiles
// @Security BearerAuth
// @Tags auth required
// @Produce json
2024-10-24 02:11:06 -04:00
// @Success 200 {object} model.CommonResponse[[]model.DDNSProfile]
2024-10-23 22:21:59 -04:00
// @Router /ddns [get]
func listDDNS(c *gin.Context) ([]*model.DDNSProfile, error) {
var ddnsProfiles []*model.DDNSProfile
singleton.DDNSCacheLock.RLock()
defer singleton.DDNSCacheLock.RUnlock()
if err := copier.Copy(&ddnsProfiles, &singleton.DDNSList); err != nil {
return nil, err
}
return ddnsProfiles, nil
}
2024-10-21 04:22:30 -04:00
// Add DDNS profile
// @Summary Add DDNS profile
// @Security BearerAuth
// @Schemes
2024-10-21 04:22:30 -04:00
// @Description Add DDNS profile
// @Tags auth required
// @Accept json
// @param request body model.DDNSForm true "DDNS Request"
// @Produce json
2024-10-23 05:56:51 -04:00
// @Success 200 {object} model.CommonResponse[uint64]
// @Router /ddns [post]
2024-10-23 05:56:51 -04:00
func createDDNS(c *gin.Context) (uint64, error) {
var df model.DDNSForm
var p model.DDNSProfile
if err := c.ShouldBindJSON(&df); err != nil {
2024-10-23 05:56:51 -04:00
return 0, err
}
if df.MaxRetries < 1 || df.MaxRetries > 10 {
2024-10-23 05:56:51 -04:00
return 0, errors.New("重试次数必须为大于 1 且不超过 10 的整数")
}
p.Name = df.Name
enableIPv4 := df.EnableIPv4
enableIPv6 := df.EnableIPv6
p.EnableIPv4 = &enableIPv4
p.EnableIPv6 = &enableIPv6
p.MaxRetries = df.MaxRetries
p.Provider = df.Provider
p.DomainsRaw = df.DomainsRaw
p.Domains = strings.Split(p.DomainsRaw, ",")
p.AccessID = df.AccessID
p.AccessSecret = df.AccessSecret
p.WebhookURL = df.WebhookURL
p.WebhookMethod = df.WebhookMethod
p.WebhookRequestType = df.WebhookRequestType
p.WebhookRequestBody = df.WebhookRequestBody
p.WebhookHeaders = df.WebhookHeaders
for n, domain := range p.Domains {
// IDN to ASCII
domainValid, domainErr := idna.Lookup.ToASCII(domain)
if domainErr != nil {
2024-10-23 05:56:51 -04:00
return 0, fmt.Errorf("域名 %s 解析错误: %v", domain, domainErr)
}
p.Domains[n] = domainValid
}
if err := singleton.DB.Create(&p).Error; err != nil {
2024-10-23 05:56:51 -04:00
return 0, newGormError("%v", err)
}
2024-10-23 22:21:59 -04:00
singleton.OnDDNSUpdate(&p)
singleton.UpdateDDNSList()
2024-10-23 05:56:51 -04:00
return p.ID, nil
}
2024-10-21 04:22:30 -04:00
// Edit DDNS profile
// @Summary Edit DDNS profile
// @Security BearerAuth
// @Schemes
2024-10-21 04:22:30 -04:00
// @Description Edit DDNS profile
// @Tags auth required
// @Accept json
2024-10-24 21:09:08 -04:00
// @param id path uint true "Profile ID"
// @param request body model.DDNSForm true "DDNS Request"
// @Produce json
// @Success 200 {object} model.CommonResponse[any]
// @Router /ddns/{id} [patch]
2024-10-23 05:56:51 -04:00
func updateDDNS(c *gin.Context) (any, error) {
idStr := c.Param("id")
2024-10-21 11:00:51 -04:00
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
2024-10-23 05:56:51 -04:00
return nil, err
}
2024-10-21 11:00:51 -04:00
var df model.DDNSForm
if err := c.ShouldBindJSON(&df); err != nil {
2024-10-23 05:56:51 -04:00
return nil, err
}
if df.MaxRetries < 1 || df.MaxRetries > 10 {
2024-10-23 05:56:51 -04:00
return nil, errors.New("重试次数必须为大于 1 且不超过 10 的整数")
}
2024-10-21 11:00:51 -04:00
var p model.DDNSProfile
if err = singleton.DB.First(&p, id).Error; err != nil {
2024-10-23 05:56:51 -04:00
return nil, fmt.Errorf("profile id %d does not exist", id)
2024-10-21 11:00:51 -04:00
}
p.Name = df.Name
p.ID = id
enableIPv4 := df.EnableIPv4
enableIPv6 := df.EnableIPv6
p.EnableIPv4 = &enableIPv4
p.EnableIPv6 = &enableIPv6
p.MaxRetries = df.MaxRetries
p.Provider = df.Provider
p.DomainsRaw = df.DomainsRaw
p.Domains = strings.Split(p.DomainsRaw, ",")
p.AccessID = df.AccessID
p.AccessSecret = df.AccessSecret
p.WebhookURL = df.WebhookURL
p.WebhookMethod = df.WebhookMethod
p.WebhookRequestType = df.WebhookRequestType
p.WebhookRequestBody = df.WebhookRequestBody
p.WebhookHeaders = df.WebhookHeaders
for n, domain := range p.Domains {
// IDN to ASCII
domainValid, domainErr := idna.Lookup.ToASCII(domain)
if domainErr != nil {
2024-10-23 05:56:51 -04:00
return nil, fmt.Errorf("域名 %s 解析错误: %v", domain, domainErr)
}
p.Domains[n] = domainValid
}
if err = singleton.DB.Save(&p).Error; err != nil {
2024-10-23 05:56:51 -04:00
return nil, newGormError("%v", err)
}
2024-10-23 22:21:59 -04:00
singleton.OnDDNSUpdate(&p)
singleton.UpdateDDNSList()
2024-10-23 05:56:51 -04:00
return nil, nil
}
// Batch delete DDNS configurations
// @Summary Batch delete DDNS configurations
// @Security BearerAuth
// @Schemes
// @Description Batch delete DDNS configurations
// @Tags auth required
// @Accept json
// @param request body []uint64 true "id list"
// @Produce json
// @Success 200 {object} model.CommonResponse[any]
// @Router /batch-delete/ddns [post]
2024-10-23 05:56:51 -04:00
func batchDeleteDDNS(c *gin.Context) (any, error) {
var ddnsConfigs []uint64
if err := c.ShouldBindJSON(&ddnsConfigs); err != nil {
2024-10-23 05:56:51 -04:00
return nil, err
}
if err := singleton.DB.Unscoped().Delete(&model.DDNSProfile{}, "id in (?)", ddnsConfigs).Error; err != nil {
2024-10-23 05:56:51 -04:00
return nil, newGormError("%v", err)
}
2024-10-23 22:21:59 -04:00
singleton.OnDDNSDelete(ddnsConfigs)
singleton.UpdateDDNSList()
2024-10-23 05:56:51 -04:00
return nil, nil
}
// List DDNS Providers
// @Summary List DDNS providers
// @Schemes
// @Description List DDNS providers
// @Security BearerAuth
// @Tags auth required
// @Produce json
// @Success 200 {object} model.CommonResponse[[]string]
// @Router /ddns/providers [get]
2024-10-23 05:56:51 -04:00
func listProviders(c *gin.Context) ([]string, error) {
return model.ProviderList, nil
}