mirror of
https://github.com/acepanel/panel.git
synced 2026-02-04 13:47:15 +08:00
* refactor: 重构部分完成 * fix: 添加.gitkeep * fix: build * fix: lint * fix: lint * chore(deps): Update module github.com/go-playground/validator/v10 to v10.22.1 (#162) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore(deps): Update module gorm.io/gorm to v1.25.12 (#161) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore(deps): Update module golang.org/x/net to v0.29.0 (#159) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * workflow: 更新工作流 * workflow: test new download * feat: merge frontend project * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: fix frontend build * workflow: update to ubuntu-24.04 * workflow: rename build-* * workflow: 修改fetch-depth * chore(deps): Update dependency eslint to v9 (#164) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore(frontend): update dependences * chore(frontend): fix lint * chore(frontend): fix lint * workflow: add govulncheck * workflow: disable nilaway * feat: 使用新的压缩解压库 * fix: 测试 * fix: 测试 * fix: 测试 * feat: 添加ntp包 * chore(deps): Lock file maintenance (#168) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore(deps): Update module github.com/go-resty/resty/v2 to v2.15.0 (#167) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore(deps): Update dependency @iconify/json to v2.2.249 (#169) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * feat: 添加限流器 * feat: 调整登录限流 * feat: 证书 * fix: lint * feat: 证书dns * feat: 证书acme账号 * fix: 修改UserID导致的一系列问题 * feat: 低配版任务队列 * feat: 队列完成 * fix: lint * fix: lint * fix: swagger和前端路由 * fix: 去掉ntp测试 * feat: 完成插件接口 * feat: 完成cron * feat: 完成safe * chore(deps): Update dependency vue to v3.5.6 (#170) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore(deps): Update dependency @vueuse/core to v11.1.0 (#171) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore(deps): Update dependency vite to v5.4.6 (#173) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore(deps): Update unocss monorepo to v0.62.4 (#172) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> * chore: update renovate config * feat: 新的firewall客户端 * fix: lint * feat: firewall完成 * feat: ssh完成 * feat: 容器完成1/2 * feat: 容器完成 * feat: 文件完成 * feat: systemctl及设置 * fix: windows编译 * fix: session not work * fix: migrate not work * feat: 前端路由 * feat: 初步支持cli --------- Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
342 lines
7.0 KiB
Go
342 lines
7.0 KiB
Go
package service
|
|
|
|
import (
|
|
"net/http"
|
|
|
|
"github.com/go-rat/chix"
|
|
|
|
"github.com/TheTNB/panel/internal/biz"
|
|
"github.com/TheTNB/panel/internal/data"
|
|
"github.com/TheTNB/panel/internal/http/request"
|
|
"github.com/TheTNB/panel/pkg/acme"
|
|
)
|
|
|
|
type CertService struct {
|
|
certRepo biz.CertRepo
|
|
}
|
|
|
|
func NewCertService() *CertService {
|
|
return &CertService{
|
|
certRepo: data.NewCertRepo(),
|
|
}
|
|
}
|
|
|
|
// CAProviders
|
|
//
|
|
// @Summary 获取 CA 提供商
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/caProviders [get]
|
|
func (s *CertService) CAProviders(w http.ResponseWriter, r *http.Request) {
|
|
Success(w, []map[string]string{
|
|
{
|
|
"name": "Let's Encrypt",
|
|
"ca": "letsencrypt",
|
|
},
|
|
{
|
|
"name": "ZeroSSL",
|
|
"ca": "zerossl",
|
|
},
|
|
{
|
|
"name": "SSL.com",
|
|
"ca": "sslcom",
|
|
},
|
|
{
|
|
"name": "Google",
|
|
"ca": "google",
|
|
},
|
|
{
|
|
"name": "Buypass",
|
|
"ca": "buypass",
|
|
},
|
|
})
|
|
|
|
}
|
|
|
|
// DNSProviders
|
|
//
|
|
// @Summary 获取 DNS 提供商
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/dnsProviders [get]
|
|
func (s *CertService) DNSProviders(w http.ResponseWriter, r *http.Request) {
|
|
Success(w, []map[string]any{
|
|
{
|
|
"name": "DNSPod",
|
|
"dns": acme.DnsPod,
|
|
},
|
|
{
|
|
"name": "腾讯云",
|
|
"dns": acme.Tencent,
|
|
},
|
|
{
|
|
"name": "阿里云",
|
|
"dns": acme.AliYun,
|
|
},
|
|
{
|
|
"name": "CloudFlare",
|
|
"dns": acme.CloudFlare,
|
|
},
|
|
})
|
|
|
|
}
|
|
|
|
// Algorithms
|
|
//
|
|
// @Summary 获取算法列表
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/algorithms [get]
|
|
func (s *CertService) Algorithms(w http.ResponseWriter, r *http.Request) {
|
|
Success(w, []map[string]any{
|
|
{
|
|
"name": "EC256",
|
|
"key": acme.KeyEC256,
|
|
},
|
|
{
|
|
"name": "EC384",
|
|
"key": acme.KeyEC384,
|
|
},
|
|
{
|
|
"name": "RSA2048",
|
|
"key": acme.KeyRSA2048,
|
|
},
|
|
{
|
|
"name": "RSA4096",
|
|
"key": acme.KeyRSA4096,
|
|
},
|
|
})
|
|
|
|
}
|
|
|
|
// List
|
|
//
|
|
// @Summary 证书列表
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/cert [get]
|
|
func (s *CertService) List(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.Paginate](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
certs, total, err := s.certRepo.List(req.Page, req.Limit)
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, chix.M{
|
|
"total": total,
|
|
"items": certs,
|
|
})
|
|
}
|
|
|
|
// Create
|
|
//
|
|
// @Summary 创建证书
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/cert [post]
|
|
func (s *CertService) Create(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.CertCreate](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
cert, err := s.certRepo.Create(req)
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, cert)
|
|
}
|
|
|
|
// Update
|
|
//
|
|
// @Summary 更新证书
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Param id path int true "证书 ID"
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/cert/{id} [post]
|
|
func (s *CertService) Update(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.CertUpdate](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
if err = s.certRepo.Update(req); err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, nil)
|
|
}
|
|
|
|
// Get
|
|
//
|
|
// @Summary 获取证书
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Param id path int true "证书 ID"
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/cert/{id} [get]
|
|
func (s *CertService) Get(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.ID](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
cert, err := s.certRepo.Get(req.ID)
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, cert)
|
|
}
|
|
|
|
// Delete
|
|
//
|
|
// @Summary 删除证书
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Param id path int true "证书 ID"
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/cert/{id} [delete]
|
|
func (s *CertService) Delete(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.ID](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
err = s.certRepo.Delete(req.ID)
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, nil)
|
|
}
|
|
|
|
// Obtain
|
|
//
|
|
// @Summary 签发证书
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Param id path int true "证书 ID"
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/{id}/obtain [post]
|
|
func (s *CertService) Obtain(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.ID](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
cert, err := s.certRepo.Get(req.ID)
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
if cert.DNS != nil || cert.Website != nil {
|
|
_, err = s.certRepo.ObtainAuto(req.ID)
|
|
} else {
|
|
_, err = s.certRepo.ObtainManual(req.ID)
|
|
}
|
|
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, nil)
|
|
}
|
|
|
|
// Renew
|
|
//
|
|
// @Summary 续签证书
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Param id path int true "证书 ID"
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/{id}/renew [post]
|
|
func (s *CertService) Renew(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.ID](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
_, err = s.certRepo.Renew(req.ID)
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, nil)
|
|
}
|
|
|
|
// ManualDNS
|
|
//
|
|
// @Summary 手动 DNS
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Param id path int true "证书 ID"
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/{id}/manualDNS [post]
|
|
func (s *CertService) ManualDNS(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.ID](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
dns, err := s.certRepo.ManualDNS(req.ID)
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, dns)
|
|
}
|
|
|
|
// Deploy
|
|
//
|
|
// @Summary 部署证书
|
|
// @Tags 证书服务
|
|
// @Produce json
|
|
// @Param id path int true "证书 ID"
|
|
// @Param websiteID query int true "网站 ID"
|
|
// @Success 200 {object} SuccessResponse
|
|
// @Router /cert/{id}/deploy [post]
|
|
func (s *CertService) Deploy(w http.ResponseWriter, r *http.Request) {
|
|
req, err := Bind[request.CertDeploy](r)
|
|
if err != nil {
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
return
|
|
}
|
|
|
|
err = s.certRepo.Deploy(req.ID, req.WebsiteID)
|
|
if err != nil {
|
|
Error(w, http.StatusInternalServerError, err.Error())
|
|
return
|
|
}
|
|
|
|
Success(w, nil)
|
|
}
|