mirror of https://github.com/usual2970/certimate
				
				
				
			
		
			
				
	
	
		
			155 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			155 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Go
		
	
	
package deployer
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"encoding/json"
 | 
						|
	"fmt"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	xerrors "github.com/pkg/errors"
 | 
						|
	tcCdn "github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/cdn/v20180606"
 | 
						|
	"github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/common"
 | 
						|
	"github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/common/profile"
 | 
						|
	tcSsl "github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/ssl/v20191205"
 | 
						|
 | 
						|
	"github.com/usual2970/certimate/internal/domain"
 | 
						|
	"github.com/usual2970/certimate/internal/pkg/core/uploader"
 | 
						|
	uploaderTcSsl "github.com/usual2970/certimate/internal/pkg/core/uploader/providers/tencentcloud-ssl"
 | 
						|
)
 | 
						|
 | 
						|
type TencentECDNDeployer struct {
 | 
						|
	option *DeployerOption
 | 
						|
	infos  []string
 | 
						|
 | 
						|
	sdkClients  *tencentECDNDeployerSdkClients
 | 
						|
	sslUploader uploader.Uploader
 | 
						|
}
 | 
						|
 | 
						|
type tencentECDNDeployerSdkClients struct {
 | 
						|
	ssl *tcSsl.Client
 | 
						|
	cdn *tcCdn.Client
 | 
						|
}
 | 
						|
 | 
						|
func NewTencentECDNDeployer(option *DeployerOption) (Deployer, error) {
 | 
						|
	access := &domain.TencentAccess{}
 | 
						|
	if err := json.Unmarshal([]byte(option.Access), access); err != nil {
 | 
						|
		return nil, xerrors.Wrap(err, "failed to get access")
 | 
						|
	}
 | 
						|
 | 
						|
	clients, err := (&TencentECDNDeployer{}).createSdkClients(
 | 
						|
		access.SecretId,
 | 
						|
		access.SecretKey,
 | 
						|
	)
 | 
						|
	if err != nil {
 | 
						|
		return nil, xerrors.Wrap(err, "failed to create sdk clients")
 | 
						|
	}
 | 
						|
 | 
						|
	uploader, err := uploaderTcSsl.New(&uploaderTcSsl.TencentCloudSSLUploaderConfig{
 | 
						|
		SecretId:  access.SecretId,
 | 
						|
		SecretKey: access.SecretKey,
 | 
						|
	})
 | 
						|
	if err != nil {
 | 
						|
		return nil, xerrors.Wrap(err, "failed to create ssl uploader")
 | 
						|
	}
 | 
						|
 | 
						|
	return &TencentECDNDeployer{
 | 
						|
		option:      option,
 | 
						|
		infos:       make([]string, 0),
 | 
						|
		sdkClients:  clients,
 | 
						|
		sslUploader: uploader,
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
func (d *TencentECDNDeployer) GetID() string {
 | 
						|
	return fmt.Sprintf("%s-%s", d.option.AccessRecord.GetString("name"), d.option.AccessRecord.Id)
 | 
						|
}
 | 
						|
 | 
						|
func (d *TencentECDNDeployer) GetInfos() []string {
 | 
						|
	return d.infos
 | 
						|
}
 | 
						|
 | 
						|
func (d *TencentECDNDeployer) Deploy(ctx context.Context) error {
 | 
						|
	// 上传证书到 SSL
 | 
						|
	upres, err := d.sslUploader.Upload(ctx, d.option.Certificate.Certificate, d.option.Certificate.PrivateKey)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	d.infos = append(d.infos, toStr("已上传证书", upres))
 | 
						|
 | 
						|
	// 获取待部署的 ECDN 实例
 | 
						|
	// 如果是泛域名,根据证书匹配 ECDN 实例
 | 
						|
	aliInstanceIds := make([]string, 0)
 | 
						|
	domain := d.option.DeployConfig.GetConfigAsString("domain")
 | 
						|
	if strings.HasPrefix(domain, "*") {
 | 
						|
		domains, err := d.getDomainsByCertificateId(upres.CertId)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		aliInstanceIds = domains
 | 
						|
	} else {
 | 
						|
		aliInstanceIds = append(aliInstanceIds, domain)
 | 
						|
	}
 | 
						|
	if len(aliInstanceIds) == 0 {
 | 
						|
		d.infos = append(d.infos, "没有要部署的 ECDN 实例")
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	// 证书部署到 ECDN 实例
 | 
						|
	// REF: https://cloud.tencent.com/document/product/400/91667
 | 
						|
	deployCertificateInstanceReq := tcSsl.NewDeployCertificateInstanceRequest()
 | 
						|
	deployCertificateInstanceReq.CertificateId = common.StringPtr(upres.CertId)
 | 
						|
	deployCertificateInstanceReq.ResourceType = common.StringPtr("ecdn")
 | 
						|
	deployCertificateInstanceReq.Status = common.Int64Ptr(1)
 | 
						|
	deployCertificateInstanceReq.InstanceIdList = common.StringPtrs(aliInstanceIds)
 | 
						|
	deployCertificateInstanceResp, err := d.sdkClients.ssl.DeployCertificateInstance(deployCertificateInstanceReq)
 | 
						|
	if err != nil {
 | 
						|
		return xerrors.Wrap(err, "failed to execute sdk request 'ssl.DeployCertificateInstance'")
 | 
						|
	}
 | 
						|
 | 
						|
	d.infos = append(d.infos, toStr("已部署证书到云资源实例", deployCertificateInstanceResp.Response))
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (d *TencentECDNDeployer) createSdkClients(secretId, secretKey string) (*tencentECDNDeployerSdkClients, error) {
 | 
						|
	credential := common.NewCredential(secretId, secretKey)
 | 
						|
 | 
						|
	sslClient, err := tcSsl.NewClient(credential, "", profile.NewClientProfile())
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	cdnClient, err := tcCdn.NewClient(credential, "", profile.NewClientProfile())
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	return &tencentECDNDeployerSdkClients{
 | 
						|
		ssl: sslClient,
 | 
						|
		cdn: cdnClient,
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
func (d *TencentECDNDeployer) getDomainsByCertificateId(tcCertId string) ([]string, error) {
 | 
						|
	// 获取证书中的可用域名
 | 
						|
	// REF: https://cloud.tencent.com/document/product/228/42491
 | 
						|
	describeCertDomainsReq := tcCdn.NewDescribeCertDomainsRequest()
 | 
						|
	describeCertDomainsReq.CertId = common.StringPtr(tcCertId)
 | 
						|
	describeCertDomainsReq.Product = common.StringPtr("ecdn")
 | 
						|
	describeCertDomainsResp, err := d.sdkClients.cdn.DescribeCertDomains(describeCertDomainsReq)
 | 
						|
	if err != nil {
 | 
						|
		return nil, xerrors.Wrap(err, "failed to execute sdk request 'cdn.DescribeCertDomains'")
 | 
						|
	}
 | 
						|
 | 
						|
	domains := make([]string, 0)
 | 
						|
	if describeCertDomainsResp.Response.Domains == nil {
 | 
						|
		for _, domain := range describeCertDomainsResp.Response.Domains {
 | 
						|
			domains = append(domains, *domain)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return domains, nil
 | 
						|
}
 |