2022-11-11 09:41:39 +00:00
|
|
|
package service
|
|
|
|
|
|
|
|
import (
|
2022-11-16 02:31:35 +00:00
|
|
|
"context"
|
|
|
|
"crypto/x509"
|
2022-11-17 09:39:33 +00:00
|
|
|
"encoding/pem"
|
2022-12-14 07:26:51 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/app/dto/request"
|
|
|
|
"github.com/1Panel-dev/1Panel/backend/app/dto/response"
|
2022-11-16 02:31:35 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/app/model"
|
2023-07-04 08:14:11 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/app/repo"
|
2022-12-05 07:50:53 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/buserr"
|
2022-11-24 06:58:29 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/constant"
|
2023-01-04 03:44:43 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/global"
|
2023-10-31 08:01:50 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/utils/files"
|
2022-11-16 02:31:35 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/utils/ssl"
|
2022-11-24 06:58:29 +00:00
|
|
|
"path"
|
2023-07-13 06:17:18 +00:00
|
|
|
"strconv"
|
2022-11-16 02:31:35 +00:00
|
|
|
"strings"
|
2022-11-11 09:41:39 +00:00
|
|
|
)
|
|
|
|
|
2022-12-13 09:20:13 +00:00
|
|
|
type WebsiteSSLService struct {
|
2022-11-11 09:41:39 +00:00
|
|
|
}
|
|
|
|
|
2023-01-04 03:44:43 +00:00
|
|
|
type IWebsiteSSLService interface {
|
|
|
|
Page(search request.WebsiteSSLSearch) (int64, []response.WebsiteSSLDTO, error)
|
|
|
|
GetSSL(id uint) (*response.WebsiteSSLDTO, error)
|
2023-07-04 08:14:11 +00:00
|
|
|
Search(req request.WebsiteSSLSearch) ([]response.WebsiteSSLDTO, error)
|
2023-01-04 03:44:43 +00:00
|
|
|
Create(create request.WebsiteSSLCreate) (request.WebsiteSSLCreate, error)
|
|
|
|
Renew(sslId uint) error
|
|
|
|
GetDNSResolve(req request.WebsiteDNSReq) ([]response.WebsiteDNSRes, error)
|
|
|
|
GetWebsiteSSL(websiteId uint) (response.WebsiteSSLDTO, error)
|
|
|
|
Delete(id uint) error
|
2023-03-21 06:42:50 +00:00
|
|
|
Update(update request.WebsiteSSLUpdate) error
|
2023-10-31 08:01:50 +00:00
|
|
|
Upload(req request.WebsiteSSLUpload) error
|
2023-01-04 03:44:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewIWebsiteSSLService() IWebsiteSSLService {
|
|
|
|
return &WebsiteSSLService{}
|
|
|
|
}
|
|
|
|
|
2022-12-14 07:26:51 +00:00
|
|
|
func (w WebsiteSSLService) Page(search request.WebsiteSSLSearch) (int64, []response.WebsiteSSLDTO, error) {
|
2023-07-04 08:14:11 +00:00
|
|
|
var (
|
|
|
|
result []response.WebsiteSSLDTO
|
|
|
|
)
|
2022-11-11 09:41:39 +00:00
|
|
|
total, sslList, err := websiteSSLRepo.Page(search.Page, search.PageSize, commonRepo.WithOrderBy("created_at desc"))
|
2022-11-20 10:32:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, nil, err
|
|
|
|
}
|
2023-07-04 08:14:11 +00:00
|
|
|
for _, sslModel := range sslList {
|
|
|
|
result = append(result, response.WebsiteSSLDTO{
|
|
|
|
WebsiteSSL: sslModel,
|
2022-11-11 09:41:39 +00:00
|
|
|
})
|
|
|
|
}
|
2023-07-04 08:14:11 +00:00
|
|
|
return total, result, err
|
2022-11-11 09:41:39 +00:00
|
|
|
}
|
|
|
|
|
2022-12-29 06:55:20 +00:00
|
|
|
func (w WebsiteSSLService) GetSSL(id uint) (*response.WebsiteSSLDTO, error) {
|
|
|
|
var res response.WebsiteSSLDTO
|
|
|
|
websiteSSL, err := websiteSSLRepo.GetFirst(commonRepo.WithByID(id))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
res.WebsiteSSL = websiteSSL
|
|
|
|
return &res, nil
|
|
|
|
}
|
|
|
|
|
2023-07-04 08:14:11 +00:00
|
|
|
func (w WebsiteSSLService) Search(search request.WebsiteSSLSearch) ([]response.WebsiteSSLDTO, error) {
|
|
|
|
var (
|
|
|
|
opts []repo.DBOption
|
|
|
|
result []response.WebsiteSSLDTO
|
|
|
|
)
|
2023-07-13 06:17:18 +00:00
|
|
|
opts = append(opts, commonRepo.WithOrderBy("created_at desc"))
|
|
|
|
if search.AcmeAccountID != "" {
|
|
|
|
acmeAccountID, err := strconv.ParseUint(search.AcmeAccountID, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
opts = append(opts, websiteSSLRepo.WithByAcmeAccountId(uint(acmeAccountID)))
|
|
|
|
}
|
2023-07-04 08:14:11 +00:00
|
|
|
sslList, err := websiteSSLRepo.List(opts...)
|
2022-11-20 10:32:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-07-04 08:14:11 +00:00
|
|
|
for _, sslModel := range sslList {
|
|
|
|
result = append(result, response.WebsiteSSLDTO{
|
|
|
|
WebsiteSSL: sslModel,
|
2022-11-20 10:32:56 +00:00
|
|
|
})
|
|
|
|
}
|
2023-07-04 08:14:11 +00:00
|
|
|
return result, err
|
2022-11-20 10:32:56 +00:00
|
|
|
}
|
|
|
|
|
2022-12-14 07:26:51 +00:00
|
|
|
func (w WebsiteSSLService) Create(create request.WebsiteSSLCreate) (request.WebsiteSSLCreate, error) {
|
|
|
|
var res request.WebsiteSSLCreate
|
2022-11-16 02:31:35 +00:00
|
|
|
acmeAccount, err := websiteAcmeRepo.GetFirst(commonRepo.WithByID(create.AcmeAccountID))
|
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
2023-11-14 10:44:09 +00:00
|
|
|
client, err := ssl.NewAcmeClient(acmeAccount)
|
2022-11-16 02:31:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
|
|
|
|
2023-04-12 15:00:30 +00:00
|
|
|
var websiteSSL model.WebsiteSSL
|
|
|
|
|
2022-11-17 09:39:33 +00:00
|
|
|
switch create.Provider {
|
2022-12-14 07:26:51 +00:00
|
|
|
case constant.DNSAccount:
|
2022-11-17 09:39:33 +00:00
|
|
|
dnsAccount, err := websiteDnsRepo.GetFirst(commonRepo.WithByID(create.DnsAccountID))
|
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
2022-11-16 02:31:35 +00:00
|
|
|
if err := client.UseDns(ssl.DnsType(dnsAccount.Type), dnsAccount.Authorization); err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
2023-04-12 15:00:30 +00:00
|
|
|
websiteSSL.AutoRenew = create.AutoRenew
|
2022-12-14 07:26:51 +00:00
|
|
|
case constant.Http:
|
2023-02-22 09:13:08 +00:00
|
|
|
appInstall, err := getAppInstallByKey(constant.AppOpenresty)
|
2022-11-24 06:58:29 +00:00
|
|
|
if err != nil {
|
2022-12-14 07:26:51 +00:00
|
|
|
return request.WebsiteSSLCreate{}, err
|
2022-11-24 06:58:29 +00:00
|
|
|
}
|
2023-08-24 10:20:15 +00:00
|
|
|
if err := client.UseHTTP(path.Join(appInstall.GetPath(), "root")); err != nil {
|
2022-11-24 06:58:29 +00:00
|
|
|
return res, err
|
|
|
|
}
|
2023-04-12 15:00:30 +00:00
|
|
|
websiteSSL.AutoRenew = create.AutoRenew
|
2023-03-16 10:44:32 +00:00
|
|
|
case constant.DnsManual:
|
|
|
|
if err := client.UseManualDns(); err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
2022-11-16 02:31:35 +00:00
|
|
|
}
|
|
|
|
|
2022-11-17 09:39:33 +00:00
|
|
|
domains := []string{create.PrimaryDomain}
|
|
|
|
otherDomainArray := strings.Split(create.OtherDomains, "\n")
|
|
|
|
if create.OtherDomains != "" {
|
|
|
|
domains = append(otherDomainArray, domains...)
|
|
|
|
}
|
|
|
|
resource, err := client.ObtainSSL(domains)
|
2022-11-16 02:31:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
2023-04-12 15:00:30 +00:00
|
|
|
|
2023-10-09 03:52:18 +00:00
|
|
|
if create.Provider == constant.DNSAccount {
|
|
|
|
websiteSSL.DnsAccountID = create.DnsAccountID
|
|
|
|
}
|
2022-11-17 09:39:33 +00:00
|
|
|
websiteSSL.AcmeAccountID = acmeAccount.ID
|
2022-12-14 07:26:51 +00:00
|
|
|
websiteSSL.Provider = create.Provider
|
2022-11-17 09:39:33 +00:00
|
|
|
websiteSSL.Domains = strings.Join(otherDomainArray, ",")
|
|
|
|
websiteSSL.PrimaryDomain = create.PrimaryDomain
|
2022-11-16 02:31:35 +00:00
|
|
|
websiteSSL.PrivateKey = string(resource.PrivateKey)
|
|
|
|
websiteSSL.Pem = string(resource.Certificate)
|
|
|
|
websiteSSL.CertURL = resource.CertURL
|
2022-11-17 09:39:33 +00:00
|
|
|
certBlock, _ := pem.Decode(resource.Certificate)
|
|
|
|
cert, err := x509.ParseCertificate(certBlock.Bytes)
|
2022-11-16 02:31:35 +00:00
|
|
|
if err != nil {
|
2022-12-14 07:26:51 +00:00
|
|
|
return request.WebsiteSSLCreate{}, err
|
2022-11-16 02:31:35 +00:00
|
|
|
}
|
|
|
|
websiteSSL.ExpireDate = cert.NotAfter
|
|
|
|
websiteSSL.StartDate = cert.NotBefore
|
|
|
|
websiteSSL.Type = cert.Issuer.CommonName
|
2022-11-17 09:39:33 +00:00
|
|
|
websiteSSL.Organization = cert.Issuer.Organization[0]
|
2022-11-16 02:31:35 +00:00
|
|
|
|
|
|
|
if err := websiteSSLRepo.Create(context.TODO(), &websiteSSL); err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return create, nil
|
|
|
|
}
|
|
|
|
|
2022-12-13 09:20:13 +00:00
|
|
|
func (w WebsiteSSLService) Renew(sslId uint) error {
|
2022-11-17 09:39:33 +00:00
|
|
|
websiteSSL, err := websiteSSLRepo.GetFirst(commonRepo.WithByID(sslId))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
acmeAccount, err := websiteAcmeRepo.GetFirst(commonRepo.WithByID(websiteSSL.AcmeAccountID))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-11-14 10:44:09 +00:00
|
|
|
client, err := ssl.NewAcmeClient(acmeAccount)
|
2022-11-17 09:39:33 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
switch websiteSSL.Provider {
|
2022-12-14 07:26:51 +00:00
|
|
|
case constant.DNSAccount:
|
2022-11-17 09:39:33 +00:00
|
|
|
dnsAccount, err := websiteDnsRepo.GetFirst(commonRepo.WithByID(websiteSSL.DnsAccountID))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := client.UseDns(ssl.DnsType(dnsAccount.Type), dnsAccount.Authorization); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-12-14 07:26:51 +00:00
|
|
|
case constant.Http:
|
2023-02-22 09:13:08 +00:00
|
|
|
appInstall, err := getAppInstallByKey(constant.AppOpenresty)
|
2022-11-24 09:50:47 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-02-22 09:13:08 +00:00
|
|
|
if err := client.UseHTTP(path.Join(constant.AppInstallDir, constant.AppOpenresty, appInstall.Name, "root")); err != nil {
|
2022-11-24 09:50:47 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-11-17 09:39:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
resource, err := client.RenewSSL(websiteSSL.CertURL)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
websiteSSL.PrivateKey = string(resource.PrivateKey)
|
|
|
|
websiteSSL.Pem = string(resource.Certificate)
|
|
|
|
websiteSSL.CertURL = resource.CertURL
|
|
|
|
certBlock, _ := pem.Decode(resource.Certificate)
|
|
|
|
cert, err := x509.ParseCertificate(certBlock.Bytes)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
websiteSSL.ExpireDate = cert.NotAfter
|
|
|
|
websiteSSL.StartDate = cert.NotBefore
|
|
|
|
websiteSSL.Type = cert.Issuer.CommonName
|
|
|
|
websiteSSL.Organization = cert.Issuer.Organization[0]
|
|
|
|
|
2023-01-04 03:44:43 +00:00
|
|
|
if err := websiteSSLRepo.Save(websiteSSL); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
websites, _ := websiteRepo.GetBy(websiteRepo.WithWebsiteSSLID(sslId))
|
|
|
|
for _, website := range websites {
|
|
|
|
if err := createPemFile(website, websiteSSL); err != nil {
|
|
|
|
global.LOG.Errorf("create website [%s] ssl file failed! err:%s", website.PrimaryDomain, err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(websites) > 0 {
|
2023-02-22 09:13:08 +00:00
|
|
|
nginxInstall, err := getAppInstallByKey(constant.AppOpenresty)
|
2023-01-04 03:44:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := opNginx(nginxInstall.ContainerName, constant.NginxReload); err != nil {
|
|
|
|
return buserr.New(constant.ErrSSLApply)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2022-11-17 09:39:33 +00:00
|
|
|
}
|
|
|
|
|
2022-12-14 07:26:51 +00:00
|
|
|
func (w WebsiteSSLService) GetDNSResolve(req request.WebsiteDNSReq) ([]response.WebsiteDNSRes, error) {
|
2022-11-16 02:31:35 +00:00
|
|
|
acmeAccount, err := websiteAcmeRepo.GetFirst(commonRepo.WithByID(req.AcmeAccountID))
|
|
|
|
if err != nil {
|
2022-11-24 16:08:44 +00:00
|
|
|
return nil, err
|
2022-11-16 02:31:35 +00:00
|
|
|
}
|
|
|
|
|
2023-11-14 10:44:09 +00:00
|
|
|
client, err := ssl.NewAcmeClient(acmeAccount)
|
2022-11-16 02:31:35 +00:00
|
|
|
if err != nil {
|
2022-11-24 16:08:44 +00:00
|
|
|
return nil, err
|
2022-11-16 02:31:35 +00:00
|
|
|
}
|
2022-11-24 16:08:44 +00:00
|
|
|
resolves, err := client.GetDNSResolve(req.Domains)
|
2022-11-16 02:31:35 +00:00
|
|
|
if err != nil {
|
2022-11-24 16:08:44 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
2022-12-14 07:26:51 +00:00
|
|
|
var res []response.WebsiteDNSRes
|
2022-11-24 16:08:44 +00:00
|
|
|
for k, v := range resolves {
|
2022-12-14 07:26:51 +00:00
|
|
|
res = append(res, response.WebsiteDNSRes{
|
2022-11-24 16:08:44 +00:00
|
|
|
Domain: k,
|
|
|
|
Key: v.Key,
|
|
|
|
Value: v.Value,
|
|
|
|
Err: v.Err,
|
|
|
|
})
|
2022-11-16 02:31:35 +00:00
|
|
|
}
|
|
|
|
return res, nil
|
|
|
|
}
|
2022-11-11 09:41:39 +00:00
|
|
|
|
2022-12-14 07:26:51 +00:00
|
|
|
func (w WebsiteSSLService) GetWebsiteSSL(websiteId uint) (response.WebsiteSSLDTO, error) {
|
|
|
|
var res response.WebsiteSSLDTO
|
2022-11-17 09:39:33 +00:00
|
|
|
website, err := websiteRepo.GetFirst(commonRepo.WithByID(websiteId))
|
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
2022-12-13 09:20:13 +00:00
|
|
|
websiteSSL, err := websiteSSLRepo.GetFirst(commonRepo.WithByID(website.WebsiteSSLID))
|
2022-11-17 09:39:33 +00:00
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
2022-12-13 09:20:13 +00:00
|
|
|
res.WebsiteSSL = websiteSSL
|
2022-11-17 09:39:33 +00:00
|
|
|
return res, nil
|
|
|
|
}
|
|
|
|
|
2022-12-13 09:20:13 +00:00
|
|
|
func (w WebsiteSSLService) Delete(id uint) error {
|
2022-12-05 07:50:53 +00:00
|
|
|
if websites, _ := websiteRepo.GetBy(websiteRepo.WithWebsiteSSLID(id)); len(websites) > 0 {
|
|
|
|
return buserr.New(constant.ErrSSLCannotDelete)
|
|
|
|
}
|
2022-11-11 09:41:39 +00:00
|
|
|
return websiteSSLRepo.DeleteBy(commonRepo.WithByID(id))
|
|
|
|
}
|
2023-03-21 06:42:50 +00:00
|
|
|
|
|
|
|
func (w WebsiteSSLService) Update(update request.WebsiteSSLUpdate) error {
|
|
|
|
websiteSSL, err := websiteSSLRepo.GetFirst(commonRepo.WithByID(update.ID))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
websiteSSL.AutoRenew = update.AutoRenew
|
|
|
|
return websiteSSLRepo.Save(websiteSSL)
|
|
|
|
}
|
2023-10-31 08:01:50 +00:00
|
|
|
|
|
|
|
func (w WebsiteSSLService) Upload(req request.WebsiteSSLUpload) error {
|
|
|
|
newSSL := &model.WebsiteSSL{
|
|
|
|
Provider: constant.Manual,
|
|
|
|
}
|
|
|
|
|
|
|
|
if req.Type == "local" {
|
|
|
|
fileOp := files.NewFileOp()
|
|
|
|
if !fileOp.Stat(req.PrivateKeyPath) {
|
|
|
|
return buserr.New("ErrSSLKeyNotFound")
|
|
|
|
}
|
|
|
|
if !fileOp.Stat(req.CertificatePath) {
|
|
|
|
return buserr.New("ErrSSLCertificateNotFound")
|
|
|
|
}
|
|
|
|
if content, err := fileOp.GetContent(req.PrivateKeyPath); err != nil {
|
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
newSSL.PrivateKey = string(content)
|
|
|
|
}
|
|
|
|
if content, err := fileOp.GetContent(req.CertificatePath); err != nil {
|
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
newSSL.Pem = string(content)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
newSSL.PrivateKey = req.PrivateKey
|
|
|
|
newSSL.Pem = req.Certificate
|
|
|
|
}
|
|
|
|
|
|
|
|
privateKeyCertBlock, _ := pem.Decode([]byte(newSSL.PrivateKey))
|
|
|
|
if privateKeyCertBlock == nil {
|
|
|
|
return buserr.New("ErrSSLKeyFormat")
|
|
|
|
}
|
|
|
|
|
|
|
|
certBlock, _ := pem.Decode([]byte(newSSL.Pem))
|
|
|
|
if certBlock == nil {
|
|
|
|
return buserr.New("ErrSSLCertificateFormat")
|
|
|
|
}
|
|
|
|
cert, err := x509.ParseCertificate(certBlock.Bytes)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
newSSL.ExpireDate = cert.NotAfter
|
|
|
|
newSSL.StartDate = cert.NotBefore
|
|
|
|
newSSL.Type = cert.Issuer.CommonName
|
|
|
|
if len(cert.Issuer.Organization) > 0 {
|
|
|
|
newSSL.Organization = cert.Issuer.Organization[0]
|
|
|
|
} else {
|
|
|
|
newSSL.Organization = cert.Issuer.CommonName
|
|
|
|
}
|
|
|
|
if len(cert.DNSNames) > 0 {
|
|
|
|
newSSL.PrimaryDomain = cert.DNSNames[0]
|
|
|
|
newSSL.Domains = strings.Join(cert.DNSNames, ",")
|
|
|
|
}
|
|
|
|
|
|
|
|
return websiteSSLRepo.Create(context.Background(), newSSL)
|
|
|
|
}
|