mirror of
https://github.com/usual2970/certimate.git
synced 2025-06-10 06:29:52 +00:00
feat: add wangsu cdnpro deployer
This commit is contained in:
parent
b0973b5ca8
commit
f970ae7529
@ -74,6 +74,7 @@ import (
|
|||||||
pVolcEngineImageX "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/volcengine-imagex"
|
pVolcEngineImageX "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/volcengine-imagex"
|
||||||
pVolcEngineLive "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/volcengine-live"
|
pVolcEngineLive "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/volcengine-live"
|
||||||
pVolcEngineTOS "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/volcengine-tos"
|
pVolcEngineTOS "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/volcengine-tos"
|
||||||
|
pWangsuCDNPro "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/wangsu-cdnpro"
|
||||||
pWebhook "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/webhook"
|
pWebhook "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/webhook"
|
||||||
"github.com/usual2970/certimate/internal/pkg/utils/maputil"
|
"github.com/usual2970/certimate/internal/pkg/utils/maputil"
|
||||||
"github.com/usual2970/certimate/internal/pkg/utils/sliceutil"
|
"github.com/usual2970/certimate/internal/pkg/utils/sliceutil"
|
||||||
@ -1003,6 +1004,30 @@ func createDeployer(options *deployerOptions) (deployer.Deployer, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case domain.DeployProviderTypeWangsuCDNPro:
|
||||||
|
{
|
||||||
|
access := domain.AccessConfigForWangsu{}
|
||||||
|
if err := maputil.Populate(options.ProviderAccessConfig, &access); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to populate provider access config: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
switch options.Provider {
|
||||||
|
case domain.DeployProviderTypeWangsuCDNPro:
|
||||||
|
deployer, err := pWangsuCDNPro.NewDeployer(&pWangsuCDNPro.DeployerConfig{
|
||||||
|
AccessKeyId: access.AccessKeyId,
|
||||||
|
AccessKeySecret: access.AccessKeySecret,
|
||||||
|
Environment: maputil.GetOrDefaultString(options.ProviderDeployConfig, "environment", "production"),
|
||||||
|
Domain: maputil.GetString(options.ProviderDeployConfig, "domain"),
|
||||||
|
CertificateId: maputil.GetString(options.ProviderDeployConfig, "certificateId"),
|
||||||
|
WebhookId: maputil.GetString(options.ProviderDeployConfig, "webhookId"),
|
||||||
|
})
|
||||||
|
return deployer, err
|
||||||
|
|
||||||
|
default:
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
case domain.DeployProviderTypeWebhook:
|
case domain.DeployProviderTypeWebhook:
|
||||||
{
|
{
|
||||||
access := domain.AccessConfigForWebhook{}
|
access := domain.AccessConfigForWebhook{}
|
||||||
|
@ -228,6 +228,11 @@ type AccessConfigForVolcEngine struct {
|
|||||||
SecretAccessKey string `json:"secretAccessKey"`
|
SecretAccessKey string `json:"secretAccessKey"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type AccessConfigForWangsu struct {
|
||||||
|
AccessKeyId string `json:"accessKeyId"`
|
||||||
|
AccessKeySecret string `json:"accessKeySecret"`
|
||||||
|
}
|
||||||
|
|
||||||
type AccessConfigForWebhook struct {
|
type AccessConfigForWebhook struct {
|
||||||
Url string `json:"url"`
|
Url string `json:"url"`
|
||||||
AllowInsecureConnections bool `json:"allowInsecureConnections,omitempty"`
|
AllowInsecureConnections bool `json:"allowInsecureConnections,omitempty"`
|
||||||
|
@ -61,6 +61,7 @@ const (
|
|||||||
AccessProviderTypeUpyun = AccessProviderType("upyun")
|
AccessProviderTypeUpyun = AccessProviderType("upyun")
|
||||||
AccessProviderTypeVercel = AccessProviderType("vercel")
|
AccessProviderTypeVercel = AccessProviderType("vercel")
|
||||||
AccessProviderTypeVolcEngine = AccessProviderType("volcengine")
|
AccessProviderTypeVolcEngine = AccessProviderType("volcengine")
|
||||||
|
AccessProviderTypeWangsu = AccessProviderType("wangsu")
|
||||||
AccessProviderTypeWebhook = AccessProviderType("webhook")
|
AccessProviderTypeWebhook = AccessProviderType("webhook")
|
||||||
AccessProviderTypeWestcn = AccessProviderType("westcn")
|
AccessProviderTypeWestcn = AccessProviderType("westcn")
|
||||||
AccessProviderTypeZeroSSL = AccessProviderType("zerossl")
|
AccessProviderTypeZeroSSL = AccessProviderType("zerossl")
|
||||||
@ -212,5 +213,6 @@ const (
|
|||||||
DeployProviderTypeVolcEngineImageX = DeployProviderType("volcengine-imagex")
|
DeployProviderTypeVolcEngineImageX = DeployProviderType("volcengine-imagex")
|
||||||
DeployProviderTypeVolcEngineLive = DeployProviderType("volcengine-live")
|
DeployProviderTypeVolcEngineLive = DeployProviderType("volcengine-live")
|
||||||
DeployProviderTypeVolcEngineTOS = DeployProviderType("volcengine-tos")
|
DeployProviderTypeVolcEngineTOS = DeployProviderType("volcengine-tos")
|
||||||
|
DeployProviderTypeWangsuCDNPro = DeployProviderType("wangsu-cdnpro")
|
||||||
DeployProviderTypeWebhook = DeployProviderType("webhook")
|
DeployProviderTypeWebhook = DeployProviderType("webhook")
|
||||||
)
|
)
|
||||||
|
@ -0,0 +1,276 @@
|
|||||||
|
package wangsucdnpro
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"context"
|
||||||
|
"crypto/aes"
|
||||||
|
"crypto/cipher"
|
||||||
|
"crypto/hmac"
|
||||||
|
"crypto/sha256"
|
||||||
|
"encoding/base64"
|
||||||
|
"encoding/hex"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"log/slog"
|
||||||
|
"regexp"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/alibabacloud-go/tea/tea"
|
||||||
|
xerrors "github.com/pkg/errors"
|
||||||
|
|
||||||
|
"github.com/usual2970/certimate/internal/pkg/core/deployer"
|
||||||
|
"github.com/usual2970/certimate/internal/pkg/utils/certutil"
|
||||||
|
wangsucdn "github.com/usual2970/certimate/internal/pkg/vendors/wangsu-sdk/cdn"
|
||||||
|
)
|
||||||
|
|
||||||
|
type DeployerConfig struct {
|
||||||
|
// 网宿云 AccessKeyId。
|
||||||
|
AccessKeyId string `json:"accessKeyId"`
|
||||||
|
// 网宿云 AccessKeySecret。
|
||||||
|
AccessKeySecret string `json:"accessKeySecret"`
|
||||||
|
// 网宿云环境。
|
||||||
|
Environment string `json:"environment"`
|
||||||
|
// 加速域名(支持泛域名)。
|
||||||
|
Domain string `json:"domain"`
|
||||||
|
// 证书 ID。
|
||||||
|
// 选填。
|
||||||
|
CertificateId string `json:"certificateId,omitempty"`
|
||||||
|
// Webhook ID。
|
||||||
|
// 选填。
|
||||||
|
WebhookId string `json:"webhookId,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type DeployerProvider struct {
|
||||||
|
config *DeployerConfig
|
||||||
|
logger *slog.Logger
|
||||||
|
sdkClient *wangsucdn.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ deployer.Deployer = (*DeployerProvider)(nil)
|
||||||
|
|
||||||
|
func NewDeployer(config *DeployerConfig) (*DeployerProvider, error) {
|
||||||
|
if config == nil {
|
||||||
|
panic("config is nil")
|
||||||
|
}
|
||||||
|
|
||||||
|
client, err := createSdkClient(config.AccessKeyId, config.AccessKeySecret)
|
||||||
|
if err != nil {
|
||||||
|
return nil, xerrors.Wrap(err, "failed to create sdk client")
|
||||||
|
}
|
||||||
|
|
||||||
|
return &DeployerProvider{
|
||||||
|
config: config,
|
||||||
|
logger: slog.Default(),
|
||||||
|
sdkClient: client,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *DeployerProvider) WithLogger(logger *slog.Logger) deployer.Deployer {
|
||||||
|
if logger == nil {
|
||||||
|
d.logger = slog.Default()
|
||||||
|
} else {
|
||||||
|
d.logger = logger
|
||||||
|
}
|
||||||
|
return d
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *DeployerProvider) Deploy(ctx context.Context, certPem string, privkeyPem string) (*deployer.DeployResult, error) {
|
||||||
|
if d.config.Domain == "" {
|
||||||
|
return nil, errors.New("config `domain` is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
// 解析证书内容
|
||||||
|
certX509, err := certutil.ParseCertificateFromPEM(certPem)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// 查询已部署加速域名的详情
|
||||||
|
getHostnameDetailResp, err := d.sdkClient.GetHostnameDetail(d.config.Domain)
|
||||||
|
d.logger.Debug("sdk request 'cdn.GetHostnameDetail'", slog.String("hostname", d.config.Domain), slog.Any("response", getHostnameDetailResp))
|
||||||
|
if err != nil {
|
||||||
|
return nil, xerrors.Wrap(err, "failed to execute sdk request 'cdn.GetHostnameDetail'")
|
||||||
|
}
|
||||||
|
|
||||||
|
// 生成网宿云证书参数
|
||||||
|
encryptedPrivateKey, err := encryptPrivateKey(privkeyPem, d.config.AccessKeySecret, time.Now().Unix())
|
||||||
|
if err != nil {
|
||||||
|
return nil, xerrors.Wrap(err, "failed to encrypt private key")
|
||||||
|
}
|
||||||
|
certificateNewVersionInfo := &wangsucdn.CertificateVersion{
|
||||||
|
PrivateKey: tea.String(encryptedPrivateKey),
|
||||||
|
Certificate: tea.String(certPem),
|
||||||
|
IdentificationInfo: &wangsucdn.CertificateVersionIdentificationInfo{
|
||||||
|
CommonName: tea.String(certX509.Subject.CommonName),
|
||||||
|
SubjectAlternativeNames: &certX509.DNSNames,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// 网宿云证书 URL 中包含证书 ID 及版本号
|
||||||
|
// 格式:
|
||||||
|
// http://open.chinanetcenter.com/cdn/certificates/5dca2205f9e9cc0001df7b33
|
||||||
|
// http://open.chinanetcenter.com/cdn/certificates/329f12c1fe6708c23c31e91f/versions/5
|
||||||
|
var wangsuCertUrl string
|
||||||
|
var wangsuCertId, wangsuCertVer string
|
||||||
|
|
||||||
|
// 如果原证书 ID 为空,则创建证书;否则更新证书。
|
||||||
|
timestamp := time.Now().Unix()
|
||||||
|
if d.config.CertificateId == "" {
|
||||||
|
// 创建证书
|
||||||
|
createCertificateReq := &wangsucdn.CreateCertificateRequest{
|
||||||
|
Timestamp: timestamp,
|
||||||
|
Name: tea.String(fmt.Sprintf("certimate_%d", time.Now().UnixMilli())),
|
||||||
|
AutoRenew: tea.String("Off"),
|
||||||
|
NewVersion: certificateNewVersionInfo,
|
||||||
|
}
|
||||||
|
createCertificateResp, err := d.sdkClient.CreateCertificate(createCertificateReq)
|
||||||
|
d.logger.Debug("sdk request 'cdn.CreateCertificate'", slog.Any("request", createCertificateReq), slog.Any("response", createCertificateResp))
|
||||||
|
if err != nil {
|
||||||
|
return nil, xerrors.Wrap(err, "failed to execute sdk request 'cdn.CreateCertificate'")
|
||||||
|
}
|
||||||
|
|
||||||
|
wangsuCertUrl = createCertificateResp.CertificateUrl
|
||||||
|
d.logger.Info("ssl certificate uploaded", slog.Any("certUrl", wangsuCertUrl))
|
||||||
|
|
||||||
|
wangsuCertIdMatches := regexp.MustCompile(`/certificates/([a-zA-Z0-9-]+)`).FindStringSubmatch(wangsuCertUrl)
|
||||||
|
if len(wangsuCertIdMatches) > 1 {
|
||||||
|
wangsuCertId = wangsuCertIdMatches[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
wangsuCertVer = "1"
|
||||||
|
} else {
|
||||||
|
// 更新证书
|
||||||
|
updateCertificateReq := &wangsucdn.UpdateCertificateRequest{
|
||||||
|
Timestamp: timestamp,
|
||||||
|
Name: tea.String(fmt.Sprintf("certimate_%d", time.Now().UnixMilli())),
|
||||||
|
AutoRenew: tea.String("Off"),
|
||||||
|
NewVersion: certificateNewVersionInfo,
|
||||||
|
}
|
||||||
|
updateCertificateResp, err := d.sdkClient.UpdateCertificate(d.config.CertificateId, updateCertificateReq)
|
||||||
|
d.logger.Debug("sdk request 'cdn.CreateCertificate'", slog.Any("certificateId", d.config.CertificateId), slog.Any("request", updateCertificateReq), slog.Any("response", updateCertificateResp))
|
||||||
|
if err != nil {
|
||||||
|
return nil, xerrors.Wrap(err, "failed to execute sdk request 'cdn.UpdateCertificate'")
|
||||||
|
}
|
||||||
|
|
||||||
|
wangsuCertUrl = updateCertificateResp.CertificateUrl
|
||||||
|
d.logger.Info("ssl certificate uploaded", slog.Any("certUrl", wangsuCertUrl))
|
||||||
|
|
||||||
|
wangsuCertIdMatches := regexp.MustCompile(`/certificates/([a-zA-Z0-9-]+)`).FindStringSubmatch(wangsuCertUrl)
|
||||||
|
if len(wangsuCertIdMatches) > 1 {
|
||||||
|
wangsuCertId = wangsuCertIdMatches[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
wangsuCertVerMatches := regexp.MustCompile(`/versions/(\d+)`).FindStringSubmatch(wangsuCertUrl)
|
||||||
|
if len(wangsuCertVerMatches) > 1 {
|
||||||
|
wangsuCertVer = wangsuCertVerMatches[1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 创建部署任务
|
||||||
|
// REF: https://www.wangsu.com/document/api-doc/27034
|
||||||
|
createDeploymentTaskReq := &wangsucdn.CreateDeploymentTaskRequest{
|
||||||
|
Name: tea.String(fmt.Sprintf("certimate_%d", time.Now().UnixMilli())),
|
||||||
|
Target: tea.String(d.config.Environment),
|
||||||
|
Actions: &[]wangsucdn.DeploymentTaskAction{
|
||||||
|
{
|
||||||
|
Action: tea.String("deploy_cert"),
|
||||||
|
CertificateId: tea.String(wangsuCertId),
|
||||||
|
Version: tea.String(wangsuCertVer),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
if d.config.WebhookId != "" {
|
||||||
|
createDeploymentTaskReq.Webhook = tea.String(d.config.WebhookId)
|
||||||
|
}
|
||||||
|
createDeploymentTaskResp, err := d.sdkClient.CreateDeploymentTask(createDeploymentTaskReq)
|
||||||
|
d.logger.Debug("sdk request 'cdn.CreateCertificate'", slog.Any("request", createDeploymentTaskReq), slog.Any("response", createDeploymentTaskResp))
|
||||||
|
if err != nil {
|
||||||
|
return nil, xerrors.Wrap(err, "failed to execute sdk request 'cdn.CreateDeploymentTask'")
|
||||||
|
}
|
||||||
|
|
||||||
|
// 循环获取部署任务详细信息,等待任务状态变更
|
||||||
|
// REF: https://www.wangsu.com/document/api-doc/27038
|
||||||
|
var wangsuTaskId string
|
||||||
|
wangsuTaskMatches := regexp.MustCompile(`/deploymentTasks/([a-zA-Z0-9-]+)`).FindStringSubmatch(wangsuCertUrl)
|
||||||
|
if len(wangsuTaskMatches) > 1 {
|
||||||
|
wangsuTaskId = wangsuTaskMatches[1]
|
||||||
|
}
|
||||||
|
for {
|
||||||
|
if ctx.Err() != nil {
|
||||||
|
return nil, ctx.Err()
|
||||||
|
}
|
||||||
|
|
||||||
|
getDeploymentTaskDetailResp, err := d.sdkClient.GetDeploymentTaskDetail(wangsuTaskId)
|
||||||
|
d.logger.Debug("sdk request 'cdn.GetDeploymentTaskDetail'", slog.Any("taskId", wangsuTaskId), slog.Any("response", getDeploymentTaskDetailResp))
|
||||||
|
if err != nil {
|
||||||
|
return nil, xerrors.Wrap(err, "failed to execute sdk request 'cdn.GetDeploymentTaskDetail'")
|
||||||
|
}
|
||||||
|
|
||||||
|
if getDeploymentTaskDetailResp.Status == "failed" {
|
||||||
|
return nil, errors.New("unexpected deployment task status")
|
||||||
|
} else if getDeploymentTaskDetailResp.Status == "succeeded" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
d.logger.Info("waiting for deployment task completion ...")
|
||||||
|
time.Sleep(time.Second * 15)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &deployer.DeployResult{}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func createSdkClient(accessKeyId, accessKeySecret string) (*wangsucdn.Client, error) {
|
||||||
|
if accessKeyId == "" {
|
||||||
|
return nil, errors.New("invalid wangsu access key id")
|
||||||
|
}
|
||||||
|
|
||||||
|
if accessKeySecret == "" {
|
||||||
|
return nil, errors.New("invalid wangsu access key secret")
|
||||||
|
}
|
||||||
|
|
||||||
|
return wangsucdn.NewClient(accessKeyId, accessKeySecret), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func encryptPrivateKey(privkeyPem string, secretKey string, timestamp int64) (string, error) {
|
||||||
|
date := time.Unix(timestamp, 0).UTC()
|
||||||
|
dateStr := date.Format("Mon, 02 Jan 2006 15:04:05 GMT")
|
||||||
|
|
||||||
|
mac := hmac.New(sha256.New, []byte(secretKey))
|
||||||
|
mac.Write([]byte(dateStr))
|
||||||
|
aesivkey := mac.Sum(nil)
|
||||||
|
aesivkeyHex := hex.EncodeToString(aesivkey)
|
||||||
|
|
||||||
|
if len(aesivkeyHex) != 64 {
|
||||||
|
return "", fmt.Errorf("invalid hmac length: %d", len(aesivkeyHex))
|
||||||
|
}
|
||||||
|
ivHex := aesivkeyHex[:32]
|
||||||
|
keyHex := aesivkeyHex[32:64]
|
||||||
|
|
||||||
|
iv, err := hex.DecodeString(ivHex)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("failed to decode iv: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
key, err := hex.DecodeString(keyHex)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("failed to decode key: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
block, err := aes.NewCipher(key)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
plainBytes := []byte(privkeyPem)
|
||||||
|
padlen := aes.BlockSize - len(plainBytes)%aes.BlockSize
|
||||||
|
if padlen > 0 {
|
||||||
|
paddata := bytes.Repeat([]byte{byte(padlen)}, padlen)
|
||||||
|
plainBytes = append(plainBytes, paddata...)
|
||||||
|
}
|
||||||
|
|
||||||
|
encBytes := make([]byte, len(plainBytes))
|
||||||
|
mode := cipher.NewCBCEncrypter(block, iv)
|
||||||
|
mode.CryptBlocks(encBytes, plainBytes)
|
||||||
|
|
||||||
|
return base64.StdEncoding.EncodeToString(encBytes), nil
|
||||||
|
}
|
@ -0,0 +1,90 @@
|
|||||||
|
package wangsucdnpro_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
provider "github.com/usual2970/certimate/internal/pkg/core/deployer/providers/wangsu-cdnpro"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
fInputCertPath string
|
||||||
|
fInputKeyPath string
|
||||||
|
fAccessKeyId string
|
||||||
|
fAccessKeySecret string
|
||||||
|
fEnvironment string
|
||||||
|
fDomain string
|
||||||
|
fCertificateId string
|
||||||
|
fWebhookId string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
argsPrefix := "CERTIMATE_DEPLOYER_WANGSUCDNPRO_"
|
||||||
|
|
||||||
|
flag.StringVar(&fInputCertPath, argsPrefix+"INPUTCERTPATH", "", "")
|
||||||
|
flag.StringVar(&fInputKeyPath, argsPrefix+"INPUTKEYPATH", "", "")
|
||||||
|
flag.StringVar(&fAccessKeyId, argsPrefix+"ACCESSKEYID", "", "")
|
||||||
|
flag.StringVar(&fAccessKeySecret, argsPrefix+"ACCESSKEYSECRET", "", "")
|
||||||
|
flag.StringVar(&fEnvironment, argsPrefix+"ENVIRONMENT", "production", "")
|
||||||
|
flag.StringVar(&fDomain, argsPrefix+"DOMAIN", "", "")
|
||||||
|
flag.StringVar(&fCertificateId, argsPrefix+"CERTIFICATEID", "", "")
|
||||||
|
flag.StringVar(&fWebhookId, argsPrefix+"WEBHOOKID", "", "")
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Shell command to run this test:
|
||||||
|
|
||||||
|
go test -v ./wangsu_cdnpro_test.go -args \
|
||||||
|
--CERTIMATE_DEPLOYER_WANGSUCDNPRO_INPUTCERTPATH="/path/to/your-input-cert.pem" \
|
||||||
|
--CERTIMATE_DEPLOYER_WANGSUCDNPRO_INPUTKEYPATH="/path/to/your-input-key.pem" \
|
||||||
|
--CERTIMATE_DEPLOYER_WANGSUCDNPRO_ACCESSKEYID="your-access-key-id" \
|
||||||
|
--CERTIMATE_DEPLOYER_WANGSUCDNPRO_ACCESSKEYSECRET="your-access-key-secret" \
|
||||||
|
--CERTIMATE_DEPLOYER_WANGSUCDNPRO_ENVIRONMENT="production" \
|
||||||
|
--CERTIMATE_DEPLOYER_WANGSUCDNPRO_DOMAIN="example.com" \
|
||||||
|
--CERTIMATE_DEPLOYER_WANGSUCDNPRO_CERTIFICATEID="your-certificate-id"\
|
||||||
|
--CERTIMATE_DEPLOYER_WANGSUCDNPRO_WEBHOOKID="your-webhook-id"
|
||||||
|
*/
|
||||||
|
func TestDeploy(t *testing.T) {
|
||||||
|
flag.Parse()
|
||||||
|
|
||||||
|
t.Run("Deploy", func(t *testing.T) {
|
||||||
|
t.Log(strings.Join([]string{
|
||||||
|
"args:",
|
||||||
|
fmt.Sprintf("INPUTCERTPATH: %v", fInputCertPath),
|
||||||
|
fmt.Sprintf("INPUTKEYPATH: %v", fInputKeyPath),
|
||||||
|
fmt.Sprintf("ACCESSKEYID: %v", fAccessKeyId),
|
||||||
|
fmt.Sprintf("ACCESSKEYSECRET: %v", fAccessKeySecret),
|
||||||
|
fmt.Sprintf("ENVIRONMENT: %v", fEnvironment),
|
||||||
|
fmt.Sprintf("DOMAIN: %v", fDomain),
|
||||||
|
fmt.Sprintf("CERTIFICATEID: %v", fCertificateId),
|
||||||
|
fmt.Sprintf("WEBHOOKID: %v", fWebhookId),
|
||||||
|
}, "\n"))
|
||||||
|
|
||||||
|
deployer, err := provider.NewDeployer(&provider.DeployerConfig{
|
||||||
|
AccessKeyId: fAccessKeyId,
|
||||||
|
AccessKeySecret: fAccessKeySecret,
|
||||||
|
Environment: fEnvironment,
|
||||||
|
Domain: fDomain,
|
||||||
|
CertificateId: fCertificateId,
|
||||||
|
WebhookId: fWebhookId,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("err: %+v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
fInputCertData, _ := os.ReadFile(fInputCertPath)
|
||||||
|
fInputKeyData, _ := os.ReadFile(fInputKeyPath)
|
||||||
|
res, err := deployer.Deploy(context.Background(), string(fInputCertData), string(fInputKeyData))
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("err: %+v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Logf("ok: %v", res)
|
||||||
|
})
|
||||||
|
}
|
58
internal/pkg/vendors/wangsu-sdk/cdn/api.go
vendored
Normal file
58
internal/pkg/vendors/wangsu-sdk/cdn/api.go
vendored
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
package cdn
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
|
||||||
|
"github.com/go-resty/resty/v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (c *Client) CreateCertificate(req *CreateCertificateRequest) (*CreateCertificateResponse, error) {
|
||||||
|
resp := &CreateCertificateResponse{}
|
||||||
|
r, err := c.client.SendRequestWithResult(http.MethodPost, "/cdn/certificates", req, resp, func(r *resty.Request) {
|
||||||
|
r.SetHeader("x-cnc-timestamp", fmt.Sprintf("%d", req.Timestamp))
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
resp.CertificateUrl = r.Header().Get("Location")
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) UpdateCertificate(certificateId string, req *UpdateCertificateRequest) (*UpdateCertificateResponse, error) {
|
||||||
|
resp := &UpdateCertificateResponse{}
|
||||||
|
r, err := c.client.SendRequestWithResult(http.MethodPatch, fmt.Sprintf("/cdn/certificates/%s", url.PathEscape(certificateId)), req, resp, func(r *resty.Request) {
|
||||||
|
r.SetHeader("x-cnc-timestamp", fmt.Sprintf("%d", req.Timestamp))
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
resp.CertificateUrl = r.Header().Get("Location")
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) GetHostnameDetail(hostname string) (*GetHostnameDetailResponse, error) {
|
||||||
|
resp := &GetHostnameDetailResponse{}
|
||||||
|
_, err := c.client.SendRequestWithResult(http.MethodGet, fmt.Sprintf("/cdn/hostnames/%s", url.PathEscape(hostname)), nil, resp)
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) CreateDeploymentTask(req *CreateDeploymentTaskRequest) (*CreateDeploymentTaskResponse, error) {
|
||||||
|
resp := &CreateDeploymentTaskResponse{}
|
||||||
|
r, err := c.client.SendRequestWithResult(http.MethodPost, "/cdn/deploymentTasks", req, resp)
|
||||||
|
if err != nil {
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
resp.DeploymentTaskUrl = r.Header().Get("Location")
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) GetDeploymentTaskDetail(deploymentTaskId string) (*GetDeploymentTaskDetailResponse, error) {
|
||||||
|
resp := &GetDeploymentTaskDetailResponse{}
|
||||||
|
_, err := c.client.SendRequestWithResult(http.MethodGet, fmt.Sprintf("/cdn/deploymentTasks/%s", url.PathEscape(hostname)), nil, resp)
|
||||||
|
return resp, err
|
||||||
|
}
|
20
internal/pkg/vendors/wangsu-sdk/cdn/client.go
vendored
Normal file
20
internal/pkg/vendors/wangsu-sdk/cdn/client.go
vendored
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
package cdn
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/usual2970/certimate/internal/pkg/vendors/wangsu-sdk/openapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Client struct {
|
||||||
|
client *openapi.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewClient(accessKey, secretKey string) *Client {
|
||||||
|
return &Client{client: openapi.NewClient(accessKey, secretKey)}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) WithTimeout(timeout time.Duration) *Client {
|
||||||
|
c.client.WithTimeout(timeout)
|
||||||
|
return c
|
||||||
|
}
|
107
internal/pkg/vendors/wangsu-sdk/cdn/models.go
vendored
Normal file
107
internal/pkg/vendors/wangsu-sdk/cdn/models.go
vendored
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
package cdn
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/usual2970/certimate/internal/pkg/vendors/wangsu-sdk/openapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
type baseResponse struct {
|
||||||
|
RequestId *string `json:"-"`
|
||||||
|
Code *string `json:"code,omitempty"`
|
||||||
|
Message *string `json:"message,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ openapi.Result = (*baseResponse)(nil)
|
||||||
|
|
||||||
|
func (r *baseResponse) SetRequestId(requestId string) {
|
||||||
|
r.RequestId = &requestId
|
||||||
|
}
|
||||||
|
|
||||||
|
type CertificateVersion struct {
|
||||||
|
Comments *string `json:"comments,omitempty"`
|
||||||
|
PrivateKey *string `json:"privateKey,omitempty"`
|
||||||
|
Certificate *string `json:"certificate,omitempty"`
|
||||||
|
ChainCert *string `json:"chainCert,omitempty"`
|
||||||
|
IdentificationInfo *CertificateVersionIdentificationInfo `json:"identificationInfo,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type CertificateVersionIdentificationInfo struct {
|
||||||
|
Country *string `json:"country,omitempty"`
|
||||||
|
State *string `json:"state,omitempty"`
|
||||||
|
City *string `json:"city,omitempty"`
|
||||||
|
Company *string `json:"company,omitempty"`
|
||||||
|
Department *string `json:"department,omitempty"`
|
||||||
|
CommonName *string `json:"commonName,omitempty" required:"true"`
|
||||||
|
Email *string `json:"email,omitempty"`
|
||||||
|
SubjectAlternativeNames *[]string `json:"subjectAlternativeNames,omitempty" required:"true"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type CreateCertificateRequest struct {
|
||||||
|
Timestamp int64 `json:"-"`
|
||||||
|
Name *string `json:"name,omitempty" required:"true"`
|
||||||
|
Description *string `json:"description,omitempty"`
|
||||||
|
AutoRenew *string `json:"autoRenew,omitempty"`
|
||||||
|
ForceRenew *bool `json:"forceRenew,omitempty"`
|
||||||
|
NewVersion *CertificateVersion `json:"newVersion,omitempty" required:"true"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type CreateCertificateResponse struct {
|
||||||
|
baseResponse
|
||||||
|
CertificateUrl string `json:"-"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UpdateCertificateRequest struct {
|
||||||
|
Timestamp int64 `json:"-"`
|
||||||
|
Name *string `json:"name,omitempty"`
|
||||||
|
Description *string `json:"description,omitempty"`
|
||||||
|
AutoRenew *string `json:"autoRenew,omitempty"`
|
||||||
|
ForceRenew *bool `json:"forceRenew,omitempty"`
|
||||||
|
NewVersion *CertificateVersion `json:"newVersion,omitempty" required:"true"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UpdateCertificateResponse struct {
|
||||||
|
baseResponse
|
||||||
|
CertificateUrl string `json:"-"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type HostnameProperty struct {
|
||||||
|
PropertyId string `json:"propertyId"`
|
||||||
|
Version int32 `json:"version"`
|
||||||
|
CertificateId *string `json:"certificateId,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type GetHostnameDetailResponse struct {
|
||||||
|
baseResponse
|
||||||
|
Hostname string `json:"hostname"`
|
||||||
|
PropertyInProduction *HostnameProperty `json:"propertyInProduction,omitempty"`
|
||||||
|
PropertyInStaging *HostnameProperty `json:"propertyInStaging,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type DeploymentTaskAction struct {
|
||||||
|
Action *string `json:"action,omitempty" required:"true"`
|
||||||
|
PropertyId *string `json:"propertyId,omitempty"`
|
||||||
|
CertificateId *string `json:"certificateId,omitempty"`
|
||||||
|
Version *string `json:"version,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type CreateDeploymentTaskRequest struct {
|
||||||
|
Name *string `json:"name,omitempty"`
|
||||||
|
Target *string `json:"target,omitempty" required:"true"`
|
||||||
|
Actions *[]DeploymentTaskAction `json:"actions,omitempty" required:"true"`
|
||||||
|
Webhook *string `json:"webhook,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type CreateDeploymentTaskResponse struct {
|
||||||
|
baseResponse
|
||||||
|
DeploymentTaskUrl string `json:"-"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type GetDeploymentTaskDetailResponse struct {
|
||||||
|
baseResponse
|
||||||
|
Target string `json:"target"`
|
||||||
|
Actions []DeploymentTaskAction `json:"actions"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
StatusDetails string `json:"statusDetails"`
|
||||||
|
SubmissionTime string `json:"submissionTime"`
|
||||||
|
FinishTime string `json:"finishTime"`
|
||||||
|
ApiRequestId string `json:"apiRequestId"`
|
||||||
|
}
|
187
internal/pkg/vendors/wangsu-sdk/openapi/client.go
vendored
Normal file
187
internal/pkg/vendors/wangsu-sdk/openapi/client.go
vendored
Normal file
@ -0,0 +1,187 @@
|
|||||||
|
package openapi
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/hmac"
|
||||||
|
"crypto/sha256"
|
||||||
|
"encoding/hex"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/go-resty/resty/v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Client struct {
|
||||||
|
accessKey string
|
||||||
|
secretKey string
|
||||||
|
|
||||||
|
client *resty.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
type Result interface {
|
||||||
|
SetRequestId(requestId string)
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewClient(accessKey, secretKey string) *Client {
|
||||||
|
client := resty.New().
|
||||||
|
SetBaseURL("https://open.chinanetcenter.com").
|
||||||
|
SetHeader("Host", "open.chinanetcenter.com").
|
||||||
|
SetHeader("Accept", "application/json").
|
||||||
|
SetHeader("Content-Type", "application/json").
|
||||||
|
SetPreRequestHook(func(c *resty.Client, req *http.Request) error {
|
||||||
|
// Step 1: Get request method
|
||||||
|
method := req.Method
|
||||||
|
method = strings.ToUpper(method)
|
||||||
|
|
||||||
|
// Step 2: Get request path
|
||||||
|
path := "/"
|
||||||
|
if req.URL != nil {
|
||||||
|
path = req.URL.Path
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step 3: Get unencoded query string
|
||||||
|
queryString := ""
|
||||||
|
if method != http.MethodPost && req.URL != nil {
|
||||||
|
queryString = req.URL.RawQuery
|
||||||
|
|
||||||
|
s, err := url.QueryUnescape(queryString)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
queryString = s
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step 4: Get canonical headers & signed headers
|
||||||
|
canonicalHeaders := "" +
|
||||||
|
"content-type:" + strings.TrimSpace(strings.ToLower(req.Header.Get("Content-Type"))) + "\n" +
|
||||||
|
"host:" + strings.TrimSpace(strings.ToLower(req.Header.Get("Host"))) + "\n"
|
||||||
|
signedHeaders := "content-type;host"
|
||||||
|
|
||||||
|
// Step 5: Get request payload
|
||||||
|
payload := ""
|
||||||
|
if method != http.MethodGet && req.Body != nil {
|
||||||
|
reader, err := req.GetBody()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
defer reader.Close()
|
||||||
|
|
||||||
|
payloadb, err := io.ReadAll(reader)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
payload = string(payloadb)
|
||||||
|
}
|
||||||
|
hashedPayload := sha256.Sum256([]byte(payload))
|
||||||
|
hashedPayloadHex := strings.ToLower(hex.EncodeToString(hashedPayload[:]))
|
||||||
|
|
||||||
|
// Step 6: Get timestamp
|
||||||
|
var reqtime time.Time
|
||||||
|
timestampString := req.Header.Get("x-cnc-timestamp")
|
||||||
|
if timestampString == "" {
|
||||||
|
reqtime = time.Now().UTC()
|
||||||
|
timestampString = fmt.Sprintf("%d", reqtime.Unix())
|
||||||
|
} else {
|
||||||
|
timestamp, err := strconv.ParseInt(timestampString, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
reqtime = time.Unix(timestamp, 0).UTC()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step 7: Get canonical request string
|
||||||
|
canonicalRequest := fmt.Sprintf("%s\n%s\n%s\n%s\n%s\n%s", method, path, queryString, canonicalHeaders, signedHeaders, hashedPayloadHex)
|
||||||
|
hashedCanonicalRequest := sha256.Sum256([]byte(canonicalRequest))
|
||||||
|
hashedCanonicalRequestHex := strings.ToLower(hex.EncodeToString(hashedCanonicalRequest[:]))
|
||||||
|
|
||||||
|
// Step 8: String to sign
|
||||||
|
const SignAlgorithmHeader = "CNC-HMAC-SHA256"
|
||||||
|
stringToSign := fmt.Sprintf("%s\n%s\n%s", SignAlgorithmHeader, timestampString, hashedCanonicalRequestHex)
|
||||||
|
hmac := hmac.New(sha256.New, []byte(secretKey))
|
||||||
|
hmac.Write([]byte(stringToSign))
|
||||||
|
sign := hmac.Sum(nil)
|
||||||
|
signHex := strings.ToLower(hex.EncodeToString(sign))
|
||||||
|
|
||||||
|
// Step 9: Add headers to request
|
||||||
|
req.Header.Set("x-cnc-accessKey", accessKey)
|
||||||
|
req.Header.Set("x-cnc-timestamp", timestampString)
|
||||||
|
req.Header.Set("x-cnc-auth-method", "AKSK")
|
||||||
|
req.Header.Set("Authorization", fmt.Sprintf("%s Credential=%s, SignedHeaders=%s, Signature=%s", SignAlgorithmHeader, accessKey, signedHeaders, signHex))
|
||||||
|
req.Header.Set("Date", reqtime.Format("Mon, 02 Jan 2006 15:04:05 GMT"))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
|
return &Client{
|
||||||
|
accessKey: accessKey,
|
||||||
|
secretKey: secretKey,
|
||||||
|
client: client,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) WithTimeout(timeout time.Duration) *Client {
|
||||||
|
c.client.SetTimeout(timeout)
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) sendRequest(method string, path string, params interface{}, configureReq ...func(req *resty.Request)) (*resty.Response, error) {
|
||||||
|
req := c.client.R()
|
||||||
|
req.Method = method
|
||||||
|
req.URL = path
|
||||||
|
if strings.EqualFold(method, http.MethodGet) {
|
||||||
|
qs := make(map[string]string)
|
||||||
|
if params != nil {
|
||||||
|
temp := make(map[string]any)
|
||||||
|
jsonb, _ := json.Marshal(params)
|
||||||
|
json.Unmarshal(jsonb, &temp)
|
||||||
|
for k, v := range temp {
|
||||||
|
if v != nil {
|
||||||
|
qs[k] = fmt.Sprintf("%v", v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
req = req.SetQueryParams(qs)
|
||||||
|
} else {
|
||||||
|
req = req.SetBody(params)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, fn := range configureReq {
|
||||||
|
fn(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := req.Send()
|
||||||
|
if err != nil {
|
||||||
|
return resp, fmt.Errorf("wangsu api error: failed to send request: %w", err)
|
||||||
|
} else if resp.IsError() {
|
||||||
|
return resp, fmt.Errorf("wangsu api error: unexpected status code: %d, resp: %s", resp.StatusCode(), resp.Body())
|
||||||
|
}
|
||||||
|
|
||||||
|
return resp, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) SendRequestWithResult(method string, path string, params interface{}, result Result, configureReq ...func(req *resty.Request)) (*resty.Response, error) {
|
||||||
|
resp, err := c.sendRequest(method, path, params, configureReq...)
|
||||||
|
if err != nil {
|
||||||
|
if resp != nil {
|
||||||
|
json.Unmarshal(resp.Body(), &result)
|
||||||
|
result.SetRequestId(resp.Header().Get("x-cnc-request-id"))
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := json.Unmarshal(resp.Body(), &result); err != nil {
|
||||||
|
return resp, fmt.Errorf("wangsu api error: failed to parse response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
result.SetRequestId(resp.Header().Get("x-cnc-request-id"))
|
||||||
|
return resp, nil
|
||||||
|
}
|
91
migrations/1744192800_upgrade.go
Normal file
91
migrations/1744192800_upgrade.go
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
package migrations
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/pocketbase/pocketbase/core"
|
||||||
|
m "github.com/pocketbase/pocketbase/migrations"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
m.Register(func(app core.App) error {
|
||||||
|
collection, err := app.FindCollectionByNameOrId("4yzbv8urny5ja1e")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// update field
|
||||||
|
if err := collection.Fields.AddMarshaledJSONAt(2, []byte(`{
|
||||||
|
"hidden": false,
|
||||||
|
"id": "hwy7m03o",
|
||||||
|
"maxSelect": 1,
|
||||||
|
"name": "provider",
|
||||||
|
"presentable": false,
|
||||||
|
"required": false,
|
||||||
|
"system": false,
|
||||||
|
"type": "select",
|
||||||
|
"values": [
|
||||||
|
"1panel",
|
||||||
|
"acmehttpreq",
|
||||||
|
"akamai",
|
||||||
|
"aliyun",
|
||||||
|
"aws",
|
||||||
|
"azure",
|
||||||
|
"baiducloud",
|
||||||
|
"baishan",
|
||||||
|
"baotapanel",
|
||||||
|
"byteplus",
|
||||||
|
"buypass",
|
||||||
|
"cachefly",
|
||||||
|
"cdnfly",
|
||||||
|
"cloudflare",
|
||||||
|
"cloudns",
|
||||||
|
"cmcccloud",
|
||||||
|
"ctcccloud",
|
||||||
|
"cucccloud",
|
||||||
|
"desec",
|
||||||
|
"dnsla",
|
||||||
|
"dogecloud",
|
||||||
|
"dynv6",
|
||||||
|
"edgio",
|
||||||
|
"fastly",
|
||||||
|
"gname",
|
||||||
|
"gcore",
|
||||||
|
"godaddy",
|
||||||
|
"goedge",
|
||||||
|
"googletrustservices",
|
||||||
|
"huaweicloud",
|
||||||
|
"jdcloud",
|
||||||
|
"k8s",
|
||||||
|
"letsencrypt",
|
||||||
|
"letsencryptstaging",
|
||||||
|
"local",
|
||||||
|
"namecheap",
|
||||||
|
"namedotcom",
|
||||||
|
"namesilo",
|
||||||
|
"ns1",
|
||||||
|
"porkbun",
|
||||||
|
"powerdns",
|
||||||
|
"qiniu",
|
||||||
|
"qingcloud",
|
||||||
|
"rainyun",
|
||||||
|
"safeline",
|
||||||
|
"ssh",
|
||||||
|
"sslcom",
|
||||||
|
"tencentcloud",
|
||||||
|
"ucloud",
|
||||||
|
"upyun",
|
||||||
|
"vercel",
|
||||||
|
"volcengine",
|
||||||
|
"wangsu",
|
||||||
|
"webhook",
|
||||||
|
"westcn",
|
||||||
|
"zerossl"
|
||||||
|
]
|
||||||
|
}`)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return app.Save(collection)
|
||||||
|
}, func(app core.App) error {
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
}
|
1
ui/public/imgs/providers/wangsu.svg
Normal file
1
ui/public/imgs/providers/wangsu.svg
Normal file
@ -0,0 +1 @@
|
|||||||
|
<svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" x="0" y="0" width="200" height="200" viewBox="0 0 340 100"><g><path d="M290,61.5c-4.1,4.6-29.3,36.7-56.9,26.7c-3.6-1.3-7.2-3.3-10.9-6.2c3.8,0.2,7.5,0,10.9-0.3 c5.3-0.5,9.8-1.5,12.4-2.9c6.1-3,4.5-13.2-12.4-16.2c-0.4-0.1-0.9-0.2-1.3-0.2c-11.3-1.7-19.5,1.7-23.7,5.3 c-0.1,0-0.3-0.1-0.4-0.2c-2.6-1.1-5.2-2.2-7.8-3.2c20-0.9,23.4-14.1,4.8-26.4c-7.5-5-16.3-9.4-25.6-13.4 c-10.7-4.5-22.3-8.5-33.7-11.7c-7.9-2.2-15.7-4.1-23.1-5.7C109.9,4.6,98.7,2.8,90,2C71.1,0.3,58.1,1.5,50.8,5.8 C47.7,7.6,46,10,46,13v0.3c0,1.6,0.5,3.3,1.4,5.2c3,6.3,11.1,13.5,24.1,21.5c11.5,7.1,28.8,15.2,43.5,20.3 c0.1,0.8,0.2,1.6,0.6,2.6c0.9,2.4,3.1,5.2,6.7,8.3c0.5,0.5,1.1,1,1.8,1.5c5.7,4.5,12.7,8.7,21.1,12.2l0.2,0.1 c10.9,4.6,22.6,8.2,33.7,10.8c16.7,3.8,32.2,5.4,42.6,4.9c0.5-0.2,0.7-0.4,0.4-0.6c-0.1-0.1-0.2-0.2-0.4-0.2 c-5.7-0.4-12.9-1.2-21.7-2.3c-7.3-1-14.3-2.3-21-3.8c-12.1-2.7-23.1-6.2-33-10.5c-0.2-0.1-0.4-0.2-0.7-0.3 c-6.7-2.9-10.5-5.7-11.4-8.2c-1.2-3.4,1.9-5.2,5-6c2,0.7,4.1,1.3,6.4,1.9c10.2,2.9,22.2,5.5,33.7,7.4c11.1,1.9,21.7,3.2,29.5,3.6 c7.6,6.7,16.1,10.2,24.6,11.3c15.2,1.8,30.4-4.1,41.1-13.8c7.3-6.6,13.2-13,16.3-17L290,61.5z M145.3,30.3 c8.5,0.3,17.4,1.3,26.7,3.1c2.5,0.5,4.8,1,7,1.5c6.7,1.6,11.5,3.3,14.5,5.2c3.4,2.1,4.1,3.7,2.1,4.9c-1.5,0.9-4.6,1.5-9.4,2 c-2.4,0.2-4.8,0.4-7.2,0.4c-2.4,0.1-4.9,0-7.4,0c-4.9-0.2-9.8-0.9-14.8-1.8c-4.9-1-8.3-2-10-3c-1.3-0.8-1.6-1.4-1-1.9 c0.4-0.2,1.1-0.4,2.3-0.5c0.9-0.1,1.5-0.4,2.1-0.7c1.1-0.7,1.4-1.6,0.8-2.9c-0.7-1.3-2-2.6-4-3.8c-0.5-0.3-1.1-0.6-1.7-1 c-0.7-0.4-1.6-0.8-2.6-1.3C143.6,30.3,144.4,30.3,145.3,30.3z M145.3,51c-6-0.5-11.4-0.5-16.2,0.1c-2.7,0.3-5,0.7-6.9,1.3 c-1.3,0.4-2.5,0.9-3.4,1.4c-0.3-0.1-0.6-0.2-1-0.4c-12.8-4.5-26.4-14.1-19.6-18.1c4.8-2.8,12.8-4.4,24-4.8c2-0.1,4.1-0.1,6.3-0.1 c-1.7,5.1,2.7,10.8,12.9,17.1c1.3,0.8,2.6,1.6,3.8,2.3c0.9,0.5,1.7,1,2.6,1.4C147,51.1,146.1,51.1,145.3,51z M206.7,79.4 c-8.5-0.8-18.3-2.9-27.7-5.2c-10.6-2.7-20.7-5.7-27.7-7.9c6.4-1.5,14.8-2.2,25.6-2c0.7,0,1.4,0,2.1,0c10.8,0.4,20,1.9,27.8,4.6 C204.6,71.5,203.8,74.9,206.7,79.4z M233.1,66.9c0.4,0,0.7,0,1.1,0.1c3.6,0.4,6.7,0.9,9.8,3.1c3.5,2.4,3.5,4.4,2.1,6.5 c-1.3,2.1-7.1,2.9-13,3.2c-4.6,0.2-9.3,0.1-12.2,0C216.1,74.7,221.9,66.4,233.1,66.9z" fill="#005BAC"/></g></svg>
|
After Width: | Height: | Size: 2.2 KiB |
@ -51,6 +51,7 @@ import AccessFormUCloudConfig from "./AccessFormUCloudConfig";
|
|||||||
import AccessFormUpyunConfig from "./AccessFormUpyunConfig";
|
import AccessFormUpyunConfig from "./AccessFormUpyunConfig";
|
||||||
import AccessFormVercelConfig from "./AccessFormVercelConfig";
|
import AccessFormVercelConfig from "./AccessFormVercelConfig";
|
||||||
import AccessFormVolcEngineConfig from "./AccessFormVolcEngineConfig";
|
import AccessFormVolcEngineConfig from "./AccessFormVolcEngineConfig";
|
||||||
|
import AccessFormWangsuConfig from "./AccessFormWangsuConfig";
|
||||||
import AccessFormWebhookConfig from "./AccessFormWebhookConfig";
|
import AccessFormWebhookConfig from "./AccessFormWebhookConfig";
|
||||||
import AccessFormWestcnConfig from "./AccessFormWestcnConfig";
|
import AccessFormWestcnConfig from "./AccessFormWestcnConfig";
|
||||||
import AccessFormZeroSSLConfig from "./AccessFormZeroSSLConfig";
|
import AccessFormZeroSSLConfig from "./AccessFormZeroSSLConfig";
|
||||||
@ -229,6 +230,8 @@ const AccessForm = forwardRef<AccessFormInstance, AccessFormProps>(({ className,
|
|||||||
return <AccessFormVercelConfig {...nestedFormProps} />;
|
return <AccessFormVercelConfig {...nestedFormProps} />;
|
||||||
case ACCESS_PROVIDERS.VOLCENGINE:
|
case ACCESS_PROVIDERS.VOLCENGINE:
|
||||||
return <AccessFormVolcEngineConfig {...nestedFormProps} />;
|
return <AccessFormVolcEngineConfig {...nestedFormProps} />;
|
||||||
|
case ACCESS_PROVIDERS.WANGSU:
|
||||||
|
return <AccessFormWangsuConfig {...nestedFormProps} />;
|
||||||
case ACCESS_PROVIDERS.WEBHOOK:
|
case ACCESS_PROVIDERS.WEBHOOK:
|
||||||
return <AccessFormWebhookConfig {...nestedFormProps} />;
|
return <AccessFormWebhookConfig {...nestedFormProps} />;
|
||||||
case ACCESS_PROVIDERS.WESTCN:
|
case ACCESS_PROVIDERS.WESTCN:
|
||||||
|
76
ui/src/components/access/AccessFormWangsuConfig.tsx
Normal file
76
ui/src/components/access/AccessFormWangsuConfig.tsx
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Form, type FormInstance, Input } from "antd";
|
||||||
|
import { createSchemaFieldRule } from "antd-zod";
|
||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
import { type AccessConfigForWangsu } from "@/domain/access";
|
||||||
|
|
||||||
|
type AccessFormWangsuConfigFieldValues = Nullish<AccessConfigForWangsu>;
|
||||||
|
|
||||||
|
export type AccessFormWangsuConfigProps = {
|
||||||
|
form: FormInstance;
|
||||||
|
formName: string;
|
||||||
|
disabled?: boolean;
|
||||||
|
initialValues?: AccessFormWangsuConfigFieldValues;
|
||||||
|
onValuesChange?: (values: AccessFormWangsuConfigFieldValues) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
const initFormModel = (): AccessFormWangsuConfigFieldValues => {
|
||||||
|
return {
|
||||||
|
accessKeyId: "",
|
||||||
|
accessKeySecret: "",
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const AccessFormWangsuConfig = ({ form: formInst, formName, disabled, initialValues, onValuesChange: onValuesChange }: AccessFormWangsuConfigProps) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
accessKeyId: z
|
||||||
|
.string()
|
||||||
|
.min(1, t("access.form.wangsu_access_key_id.placeholder"))
|
||||||
|
.max(64, t("common.errmsg.string_max", { max: 64 }))
|
||||||
|
.trim(),
|
||||||
|
accessKeySecret: z
|
||||||
|
.string()
|
||||||
|
.min(1, t("access.form.wangsu_access_key_secret.placeholder"))
|
||||||
|
.max(64, t("common.errmsg.string_max", { max: 64 }))
|
||||||
|
.trim(),
|
||||||
|
});
|
||||||
|
const formRule = createSchemaFieldRule(formSchema);
|
||||||
|
|
||||||
|
const handleFormChange = (_: unknown, values: z.infer<typeof formSchema>) => {
|
||||||
|
onValuesChange?.(values);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Form
|
||||||
|
form={formInst}
|
||||||
|
disabled={disabled}
|
||||||
|
initialValues={initialValues ?? initFormModel()}
|
||||||
|
layout="vertical"
|
||||||
|
name={formName}
|
||||||
|
onValuesChange={handleFormChange}
|
||||||
|
>
|
||||||
|
<Form.Item
|
||||||
|
name="accessKeyId"
|
||||||
|
label={t("access.form.wangsu_access_key_id.label")}
|
||||||
|
rules={[formRule]}
|
||||||
|
tooltip={<span dangerouslySetInnerHTML={{ __html: t("access.form.wangsu_access_key_id.tooltip") }}></span>}
|
||||||
|
>
|
||||||
|
<Input autoComplete="new-password" placeholder={t("access.form.wangsu_access_key_id.placeholder")} />
|
||||||
|
</Form.Item>
|
||||||
|
|
||||||
|
<Form.Item
|
||||||
|
name="accessKeySecret"
|
||||||
|
label={t("access.form.wangsu_access_key_secret.label")}
|
||||||
|
rules={[formRule]}
|
||||||
|
tooltip={<span dangerouslySetInnerHTML={{ __html: t("access.form.wangsu_access_key_secret.tooltip") }}></span>}
|
||||||
|
>
|
||||||
|
<Input.Password autoComplete="new-password" placeholder={t("access.form.wangsu_access_key_secret.placeholder")} />
|
||||||
|
</Form.Item>
|
||||||
|
</Form>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default AccessFormWangsuConfig;
|
@ -81,6 +81,7 @@ import DeployNodeConfigFormVolcEngineDCDNConfig from "./DeployNodeConfigFormVolc
|
|||||||
import DeployNodeConfigFormVolcEngineImageXConfig from "./DeployNodeConfigFormVolcEngineImageXConfig.tsx";
|
import DeployNodeConfigFormVolcEngineImageXConfig from "./DeployNodeConfigFormVolcEngineImageXConfig.tsx";
|
||||||
import DeployNodeConfigFormVolcEngineLiveConfig from "./DeployNodeConfigFormVolcEngineLiveConfig.tsx";
|
import DeployNodeConfigFormVolcEngineLiveConfig from "./DeployNodeConfigFormVolcEngineLiveConfig.tsx";
|
||||||
import DeployNodeConfigFormVolcEngineTOSConfig from "./DeployNodeConfigFormVolcEngineTOSConfig.tsx";
|
import DeployNodeConfigFormVolcEngineTOSConfig from "./DeployNodeConfigFormVolcEngineTOSConfig.tsx";
|
||||||
|
import DeployNodeConfigFormWangsuCDNProConfig from "./DeployNodeConfigFormWangsuCDNProConfig.tsx";
|
||||||
import DeployNodeConfigFormWebhookConfig from "./DeployNodeConfigFormWebhookConfig.tsx";
|
import DeployNodeConfigFormWebhookConfig from "./DeployNodeConfigFormWebhookConfig.tsx";
|
||||||
|
|
||||||
type DeployNodeConfigFormFieldValues = Partial<WorkflowNodeConfigForDeploy>;
|
type DeployNodeConfigFormFieldValues = Partial<WorkflowNodeConfigForDeploy>;
|
||||||
@ -302,6 +303,8 @@ const DeployNodeConfigForm = forwardRef<DeployNodeConfigFormInstance, DeployNode
|
|||||||
return <DeployNodeConfigFormVolcEngineLiveConfig {...nestedFormProps} />;
|
return <DeployNodeConfigFormVolcEngineLiveConfig {...nestedFormProps} />;
|
||||||
case DEPLOY_PROVIDERS.VOLCENGINE_TOS:
|
case DEPLOY_PROVIDERS.VOLCENGINE_TOS:
|
||||||
return <DeployNodeConfigFormVolcEngineTOSConfig {...nestedFormProps} />;
|
return <DeployNodeConfigFormVolcEngineTOSConfig {...nestedFormProps} />;
|
||||||
|
case DEPLOY_PROVIDERS.WANGSU_CDNPRO:
|
||||||
|
return <DeployNodeConfigFormWangsuCDNProConfig {...nestedFormProps} />;
|
||||||
case DEPLOY_PROVIDERS.WEBHOOK:
|
case DEPLOY_PROVIDERS.WEBHOOK:
|
||||||
return <DeployNodeConfigFormWebhookConfig {...nestedFormProps} />;
|
return <DeployNodeConfigFormWebhookConfig {...nestedFormProps} />;
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,107 @@
|
|||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Form, type FormInstance, Input, Select } from "antd";
|
||||||
|
import { createSchemaFieldRule } from "antd-zod";
|
||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
import { validDomainName } from "@/utils/validators";
|
||||||
|
|
||||||
|
type DeployNodeConfigFormBaishanCDNConfigFieldValues = Nullish<{
|
||||||
|
environment: string;
|
||||||
|
domain: string;
|
||||||
|
certificateId?: string;
|
||||||
|
webhookId?: string;
|
||||||
|
}>;
|
||||||
|
|
||||||
|
export type DeployNodeConfigFormBaishanCDNConfigProps = {
|
||||||
|
form: FormInstance;
|
||||||
|
formName: string;
|
||||||
|
disabled?: boolean;
|
||||||
|
initialValues?: DeployNodeConfigFormBaishanCDNConfigFieldValues;
|
||||||
|
onValuesChange?: (values: DeployNodeConfigFormBaishanCDNConfigFieldValues) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
const ENVIRONMENT_PRODUCTION = "production" as const;
|
||||||
|
const ENVIRONMENT_STAGING = "stating" as const;
|
||||||
|
|
||||||
|
const initFormModel = (): DeployNodeConfigFormBaishanCDNConfigFieldValues => {
|
||||||
|
return {
|
||||||
|
environment: ENVIRONMENT_PRODUCTION,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const DeployNodeConfigFormBaishanCDNConfig = ({
|
||||||
|
form: formInst,
|
||||||
|
formName,
|
||||||
|
disabled,
|
||||||
|
initialValues,
|
||||||
|
onValuesChange,
|
||||||
|
}: DeployNodeConfigFormBaishanCDNConfigProps) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
resourceType: z.union([z.literal(ENVIRONMENT_PRODUCTION), z.literal(ENVIRONMENT_STAGING)], {
|
||||||
|
message: t("workflow_node.deploy.form.wangsu_cdnpro_environment.placeholder"),
|
||||||
|
}),
|
||||||
|
domain: z
|
||||||
|
.string({ message: t("workflow_node.deploy.form.wangsu_cdnpro_domain.placeholder") })
|
||||||
|
.refine((v) => validDomainName(v, { allowWildcard: true }), t("common.errmsg.domain_invalid")),
|
||||||
|
certificateId: z.string().nullish(),
|
||||||
|
webhookId: z.string().nullish(),
|
||||||
|
});
|
||||||
|
const formRule = createSchemaFieldRule(formSchema);
|
||||||
|
|
||||||
|
const handleFormChange = (_: unknown, values: z.infer<typeof formSchema>) => {
|
||||||
|
onValuesChange?.(values);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Form
|
||||||
|
form={formInst}
|
||||||
|
disabled={disabled}
|
||||||
|
initialValues={initialValues ?? initFormModel()}
|
||||||
|
layout="vertical"
|
||||||
|
name={formName}
|
||||||
|
onValuesChange={handleFormChange}
|
||||||
|
>
|
||||||
|
<Form.Item name="environment" label={t("workflow_node.deploy.form.wangsu_cdnpro_environment.label")} rules={[formRule]}>
|
||||||
|
<Select placeholder={t("workflow_node.deploy.form.wangsu_cdnpro_environment.placeholder")}>
|
||||||
|
<Select.Option key={ENVIRONMENT_PRODUCTION} value={ENVIRONMENT_PRODUCTION}>
|
||||||
|
{t("workflow_node.deploy.form.wangsu_cdnpro_environment.option.production.label")}
|
||||||
|
</Select.Option>
|
||||||
|
<Select.Option key={ENVIRONMENT_STAGING} value={ENVIRONMENT_STAGING}>
|
||||||
|
{t("workflow_node.deploy.form.wangsu_cdnpro_environment.option.staging.label")}
|
||||||
|
</Select.Option>
|
||||||
|
</Select>
|
||||||
|
</Form.Item>
|
||||||
|
|
||||||
|
<Form.Item
|
||||||
|
name="domain"
|
||||||
|
label={t("workflow_node.deploy.form.wangsu_cdnpro_domain.label")}
|
||||||
|
rules={[formRule]}
|
||||||
|
tooltip={<span dangerouslySetInnerHTML={{ __html: t("workflow_node.deploy.form.wangsu_cdnpro_domain.tooltip") }}></span>}
|
||||||
|
>
|
||||||
|
<Input placeholder={t("workflow_node.deploy.form.wangsu_cdnpro_domain.placeholder")} />
|
||||||
|
</Form.Item>
|
||||||
|
|
||||||
|
<Form.Item
|
||||||
|
name="certificateId"
|
||||||
|
label={t("workflow_node.deploy.form.wangsu_cdnpro_certificate_id.label")}
|
||||||
|
rules={[formRule]}
|
||||||
|
tooltip={<span dangerouslySetInnerHTML={{ __html: t("workflow_node.deploy.form.wangsu_cdnpro_certificate_id.tooltip") }}></span>}
|
||||||
|
>
|
||||||
|
<Input placeholder={t("workflow_node.deploy.form.wangsu_cdnpro_certificate_id.placeholder")} />
|
||||||
|
</Form.Item>
|
||||||
|
|
||||||
|
<Form.Item
|
||||||
|
name="webhookId"
|
||||||
|
label={t("workflow_node.deploy.form.wangsu_cdnpro_webhook_id.label")}
|
||||||
|
rules={[formRule]}
|
||||||
|
tooltip={<span dangerouslySetInnerHTML={{ __html: t("workflow_node.deploy.form.wangsu_cdnpro_webhook_id.tooltip") }}></span>}
|
||||||
|
>
|
||||||
|
<Input placeholder={t("workflow_node.deploy.form.wangsu_cdnpro_webhook_id.placeholder")} />
|
||||||
|
</Form.Item>
|
||||||
|
</Form>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default DeployNodeConfigFormBaishanCDNConfig;
|
@ -47,6 +47,7 @@ export interface AccessModel extends BaseModel {
|
|||||||
| AccessConfigForUpyun
|
| AccessConfigForUpyun
|
||||||
| AccessConfigForVercel
|
| AccessConfigForVercel
|
||||||
| AccessConfigForVolcEngine
|
| AccessConfigForVolcEngine
|
||||||
|
| AccessConfigForWangsu
|
||||||
| AccessConfigForWebhook
|
| AccessConfigForWebhook
|
||||||
| AccessConfigForWestcn
|
| AccessConfigForWestcn
|
||||||
| AccessConfigForZeroSSL
|
| AccessConfigForZeroSSL
|
||||||
@ -268,6 +269,11 @@ export type AccessConfigForVolcEngine = {
|
|||||||
secretAccessKey: string;
|
secretAccessKey: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type AccessConfigForWangsu = {
|
||||||
|
accessKeyId: string;
|
||||||
|
accessKeySecret: string;
|
||||||
|
};
|
||||||
|
|
||||||
export type AccessConfigForWebhook = {
|
export type AccessConfigForWebhook = {
|
||||||
url: string;
|
url: string;
|
||||||
allowInsecureConnections?: boolean;
|
allowInsecureConnections?: boolean;
|
||||||
|
@ -50,6 +50,7 @@ export const ACCESS_PROVIDERS = Object.freeze({
|
|||||||
UPYUN: "upyun",
|
UPYUN: "upyun",
|
||||||
VERCEL: "vercel",
|
VERCEL: "vercel",
|
||||||
VOLCENGINE: "volcengine",
|
VOLCENGINE: "volcengine",
|
||||||
|
WANGSU: "wangsu",
|
||||||
WEBHOOK: "webhook",
|
WEBHOOK: "webhook",
|
||||||
WESTCN: "westcn",
|
WESTCN: "westcn",
|
||||||
ZEROSSL: "zerossl",
|
ZEROSSL: "zerossl",
|
||||||
@ -99,6 +100,7 @@ export const accessProvidersMap: Map<AccessProvider["type"] | string, AccessProv
|
|||||||
[ACCESS_PROVIDERS.QINIU, "provider.qiniu", "/imgs/providers/qiniu.svg", [ACCESS_USAGES.HOSTING]],
|
[ACCESS_PROVIDERS.QINIU, "provider.qiniu", "/imgs/providers/qiniu.svg", [ACCESS_USAGES.HOSTING]],
|
||||||
[ACCESS_PROVIDERS.UPYUN, "provider.upyun", "/imgs/providers/upyun.svg", [ACCESS_USAGES.HOSTING]],
|
[ACCESS_PROVIDERS.UPYUN, "provider.upyun", "/imgs/providers/upyun.svg", [ACCESS_USAGES.HOSTING]],
|
||||||
[ACCESS_PROVIDERS.BAISHAN, "provider.baishan", "/imgs/providers/baishan.png", [ACCESS_USAGES.HOSTING]],
|
[ACCESS_PROVIDERS.BAISHAN, "provider.baishan", "/imgs/providers/baishan.png", [ACCESS_USAGES.HOSTING]],
|
||||||
|
[ACCESS_PROVIDERS.WANGSU, "provider.wangsu", "/imgs/providers/wangsu.svg", [ACCESS_USAGES.HOSTING]],
|
||||||
[ACCESS_PROVIDERS.DOGECLOUD, "provider.dogecloud", "/imgs/providers/dogecloud.png", [ACCESS_USAGES.HOSTING]],
|
[ACCESS_PROVIDERS.DOGECLOUD, "provider.dogecloud", "/imgs/providers/dogecloud.png", [ACCESS_USAGES.HOSTING]],
|
||||||
[ACCESS_PROVIDERS.BYTEPLUS, "provider.byteplus", "/imgs/providers/byteplus.svg", [ACCESS_USAGES.HOSTING]],
|
[ACCESS_PROVIDERS.BYTEPLUS, "provider.byteplus", "/imgs/providers/byteplus.svg", [ACCESS_USAGES.HOSTING]],
|
||||||
[ACCESS_PROVIDERS.UCLOUD, "provider.ucloud", "/imgs/providers/ucloud.svg", [ACCESS_USAGES.HOSTING]],
|
[ACCESS_PROVIDERS.UCLOUD, "provider.ucloud", "/imgs/providers/ucloud.svg", [ACCESS_USAGES.HOSTING]],
|
||||||
@ -370,6 +372,7 @@ export const DEPLOY_PROVIDERS = Object.freeze({
|
|||||||
VOLCENGINE_IMAGEX: `${ACCESS_PROVIDERS.VOLCENGINE}-imagex`,
|
VOLCENGINE_IMAGEX: `${ACCESS_PROVIDERS.VOLCENGINE}-imagex`,
|
||||||
VOLCENGINE_LIVE: `${ACCESS_PROVIDERS.VOLCENGINE}-live`,
|
VOLCENGINE_LIVE: `${ACCESS_PROVIDERS.VOLCENGINE}-live`,
|
||||||
VOLCENGINE_TOS: `${ACCESS_PROVIDERS.VOLCENGINE}-tos`,
|
VOLCENGINE_TOS: `${ACCESS_PROVIDERS.VOLCENGINE}-tos`,
|
||||||
|
WANGSU_CDNPRO: `${ACCESS_PROVIDERS.WANGSU}-cdnpro`,
|
||||||
WEBHOOK: `${ACCESS_PROVIDERS.WEBHOOK}`,
|
WEBHOOK: `${ACCESS_PROVIDERS.WEBHOOK}`,
|
||||||
} as const);
|
} as const);
|
||||||
|
|
||||||
@ -458,6 +461,7 @@ export const deployProvidersMap: Map<DeployProvider["type"] | string, DeployProv
|
|||||||
[DEPLOY_PROVIDERS.UPYUN_FILE, "provider.upyun.file", DEPLOY_CATEGORIES.STORAGE],
|
[DEPLOY_PROVIDERS.UPYUN_FILE, "provider.upyun.file", DEPLOY_CATEGORIES.STORAGE],
|
||||||
[DEPLOY_PROVIDERS.UPYUN_CDN, "provider.upyun.cdn", DEPLOY_CATEGORIES.CDN],
|
[DEPLOY_PROVIDERS.UPYUN_CDN, "provider.upyun.cdn", DEPLOY_CATEGORIES.CDN],
|
||||||
[DEPLOY_PROVIDERS.BAISHAN_CDN, "provider.baishan.cdn", DEPLOY_CATEGORIES.CDN],
|
[DEPLOY_PROVIDERS.BAISHAN_CDN, "provider.baishan.cdn", DEPLOY_CATEGORIES.CDN],
|
||||||
|
[DEPLOY_PROVIDERS.WANGSU_CDNPRO, "provider.wangsu.cdnpro", DEPLOY_CATEGORIES.CDN],
|
||||||
[DEPLOY_PROVIDERS.DOGECLOUD_CDN, "provider.dogecloud.cdn", DEPLOY_CATEGORIES.CDN],
|
[DEPLOY_PROVIDERS.DOGECLOUD_CDN, "provider.dogecloud.cdn", DEPLOY_CATEGORIES.CDN],
|
||||||
[DEPLOY_PROVIDERS.BYTEPLUS_CDN, "provider.byteplus.cdn", DEPLOY_CATEGORIES.CDN],
|
[DEPLOY_PROVIDERS.BYTEPLUS_CDN, "provider.byteplus.cdn", DEPLOY_CATEGORIES.CDN],
|
||||||
[DEPLOY_PROVIDERS.UCLOUD_US3, "provider.ucloud.us3", DEPLOY_CATEGORIES.STORAGE],
|
[DEPLOY_PROVIDERS.UCLOUD_US3, "provider.ucloud.us3", DEPLOY_CATEGORIES.STORAGE],
|
||||||
|
@ -298,6 +298,12 @@
|
|||||||
"access.form.volcengine_secret_access_key.label": "VolcEngine SecretAccessKey",
|
"access.form.volcengine_secret_access_key.label": "VolcEngine SecretAccessKey",
|
||||||
"access.form.volcengine_secret_access_key.placeholder": "Please enter VolcEngine SecretAccessKey",
|
"access.form.volcengine_secret_access_key.placeholder": "Please enter VolcEngine SecretAccessKey",
|
||||||
"access.form.volcengine_secret_access_key.tooltip": "For more information, see <a href=\"https://www.volcengine.com/docs/6291/216571\" target=\"_blank\">https://www.volcengine.com/docs/6291/216571</a>",
|
"access.form.volcengine_secret_access_key.tooltip": "For more information, see <a href=\"https://www.volcengine.com/docs/6291/216571\" target=\"_blank\">https://www.volcengine.com/docs/6291/216571</a>",
|
||||||
|
"access.form.wangsu_access_key_id.label": "Wangsu Cloud AccessKeyId",
|
||||||
|
"access.form.wangsu_access_key_id.placeholder": "Please enter Wangsu Cloud AccessKeyId",
|
||||||
|
"access.form.wangsu_access_key_id.tooltip": "For more information, see <a href=\"https://en.wangsu.com/document/account-manage/15775\" target=\"_blank\">https://en.wangsu.com/document/account-manage/15775</a>",
|
||||||
|
"access.form.wangsu_access_key_secret.label": "Wangsu Cloud AccessKeySecret",
|
||||||
|
"access.form.wangsu_access_key_secret.placeholder": "Please enter Wangsu Cloud AccessKeySecret",
|
||||||
|
"access.form.wangsu_access_key_secret.tooltip": "For more information, see <a href=\"https://en.wangsu.com/document/account-manage/15775\" target=\"_blank\">https://en.wangsu.com/document/account-manage/15775</a>",
|
||||||
"access.form.webhook_url.label": "Webhook URL",
|
"access.form.webhook_url.label": "Webhook URL",
|
||||||
"access.form.webhook_url.placeholder": "Please enter Webhook URL",
|
"access.form.webhook_url.placeholder": "Please enter Webhook URL",
|
||||||
"access.form.webhook_allow_insecure_conns.label": "Insecure SSL/TLS connections",
|
"access.form.webhook_allow_insecure_conns.label": "Insecure SSL/TLS connections",
|
||||||
|
@ -125,6 +125,8 @@
|
|||||||
"provider.volcengine.imagex": "Volcengine - ImageX",
|
"provider.volcengine.imagex": "Volcengine - ImageX",
|
||||||
"provider.volcengine.live": "Volcengine - Live",
|
"provider.volcengine.live": "Volcengine - Live",
|
||||||
"provider.volcengine.tos": "Volcengine - TOS (Tinder Object Storage)",
|
"provider.volcengine.tos": "Volcengine - TOS (Tinder Object Storage)",
|
||||||
|
"provider.wangsu": "Wangsu Cloud",
|
||||||
|
"provider.wangsu.cdnpro": "Wangsu Cloud - CDN Pro",
|
||||||
"provider.webhook": "Webhook",
|
"provider.webhook": "Webhook",
|
||||||
"provider.westcn": "West.cn",
|
"provider.westcn": "West.cn",
|
||||||
"provider.zerossl": "ZeroSSL",
|
"provider.zerossl": "ZeroSSL",
|
||||||
|
@ -95,7 +95,7 @@
|
|||||||
"workflow_node.deploy.form.provider.placeholder": "Please select deploy target",
|
"workflow_node.deploy.form.provider.placeholder": "Please select deploy target",
|
||||||
"workflow_node.deploy.form.provider_access.label": "Host provider authorization",
|
"workflow_node.deploy.form.provider_access.label": "Host provider authorization",
|
||||||
"workflow_node.deploy.form.provider_access.placeholder": "Please select an authorization of host provider",
|
"workflow_node.deploy.form.provider_access.placeholder": "Please select an authorization of host provider",
|
||||||
"workflow_node.deploy.form.provider_access.tooltip": "Used to deploy certificates.",
|
"workflow_node.deploy.form.provider_access.tooltip": "Used to invoke API during deployment.",
|
||||||
"workflow_node.deploy.form.provider_access.button": "Create",
|
"workflow_node.deploy.form.provider_access.button": "Create",
|
||||||
"workflow_node.deploy.form.provider_access.guide_for_local": "Tips: If you are running Certimate in Docker, the \"Local\" refers to the container rather than the host.",
|
"workflow_node.deploy.form.provider_access.guide_for_local": "Tips: If you are running Certimate in Docker, the \"Local\" refers to the container rather than the host.",
|
||||||
"workflow_node.deploy.form.certificate.label": "Certificate",
|
"workflow_node.deploy.form.certificate.label": "Certificate",
|
||||||
@ -269,11 +269,11 @@
|
|||||||
"workflow_node.deploy.form.baiducloud_cdn_domain.label": "Baidu Cloud CDN domain",
|
"workflow_node.deploy.form.baiducloud_cdn_domain.label": "Baidu Cloud CDN domain",
|
||||||
"workflow_node.deploy.form.baiducloud_cdn_domain.placeholder": "Please enter Baidu Cloud CDN domain name",
|
"workflow_node.deploy.form.baiducloud_cdn_domain.placeholder": "Please enter Baidu Cloud CDN domain name",
|
||||||
"workflow_node.deploy.form.baiducloud_cdn_domain.tooltip": "For more information, see <a href=\"https://console.bce.baidu.com/cdn\" target=\"_blank\">https://console.bce.baidu.com/cdn</a>",
|
"workflow_node.deploy.form.baiducloud_cdn_domain.tooltip": "For more information, see <a href=\"https://console.bce.baidu.com/cdn\" target=\"_blank\">https://console.bce.baidu.com/cdn</a>",
|
||||||
"workflow_node.deploy.form.baishan_cdn_domain.label": "Baishan CDN domain",
|
"workflow_node.deploy.form.baishan_cdn_domain.label": "Baishan Cloud CDN domain",
|
||||||
"workflow_node.deploy.form.baishan_cdn_domain.placeholder": "Please enter Baishan CDN domain name",
|
"workflow_node.deploy.form.baishan_cdn_domain.placeholder": "Please enter Baishan Cloud CDN domain name",
|
||||||
"workflow_node.deploy.form.baishan_cdn_domain.tooltip": "For more information, see <a href=\"https://cdnx.console.baishan.com\" target=\"_blank\">https://cdnx.console.baishan.com</a>",
|
"workflow_node.deploy.form.baishan_cdn_domain.tooltip": "For more information, see <a href=\"https://cdnx.console.baishan.com\" target=\"_blank\">https://cdnx.console.baishan.com</a>",
|
||||||
"workflow_node.deploy.form.baishan_cdn_certificate_id.label": "Baishan CDN certificate ID (Optional",
|
"workflow_node.deploy.form.baishan_cdn_certificate_id.label": "Baishan Cloud CDN certificate ID (Optional)",
|
||||||
"workflow_node.deploy.form.baishan_cdn_certificate_id.placeholder": "Please enter Baishan CDN certificate ID",
|
"workflow_node.deploy.form.baishan_cdn_certificate_id.placeholder": "Please enter Baishan Cloud CDN certificate ID",
|
||||||
"workflow_node.deploy.form.baishan_cdn_certificate_id.tooltip": "For more information, see <a href=\"https://cdnx.console.baishan.com/#/cdn/cert\" target=\"_blank\">https://cdnx.console.baishan.com/#/cdn/cert</a>",
|
"workflow_node.deploy.form.baishan_cdn_certificate_id.tooltip": "For more information, see <a href=\"https://cdnx.console.baishan.com/#/cdn/cert\" target=\"_blank\">https://cdnx.console.baishan.com/#/cdn/cert</a>",
|
||||||
"workflow_node.deploy.form.baotapanel_console_auto_restart.label": "Auto restart after deployment",
|
"workflow_node.deploy.form.baotapanel_console_auto_restart.label": "Auto restart after deployment",
|
||||||
"workflow_node.deploy.form.baotapanel_site_type.label": "aaPanel site type",
|
"workflow_node.deploy.form.baotapanel_site_type.label": "aaPanel site type",
|
||||||
@ -639,6 +639,19 @@
|
|||||||
"workflow_node.deploy.form.volcengine_tos_domain.label": "VolcEngine TOS domain",
|
"workflow_node.deploy.form.volcengine_tos_domain.label": "VolcEngine TOS domain",
|
||||||
"workflow_node.deploy.form.volcengine_tos_domain.placeholder": "Please enter VolcEngine TOS domain name",
|
"workflow_node.deploy.form.volcengine_tos_domain.placeholder": "Please enter VolcEngine TOS domain name",
|
||||||
"workflow_node.deploy.form.volcengine_tos_domain.tooltip": "For more information, see <a href=\"https://console.volcengine.com/tos\" target=\"_blank\">https://console.volcengine.com/tos</a>",
|
"workflow_node.deploy.form.volcengine_tos_domain.tooltip": "For more information, see <a href=\"https://console.volcengine.com/tos\" target=\"_blank\">https://console.volcengine.com/tos</a>",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_environment.label": "Wangsu Cloud environment",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_environment.placeholder": "Please select Wangsu Cloud environment",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_environment.option.production.label": "Production environment",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_environment.option.staging.label": "Staging environment",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_domain.label": "Wangsu Cloud CDN domain",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_domain.placeholder": "Please enter Wangsu Cloud CDN domain name",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_domain.tooltip": "For more information, see <a href=\"https://cdnpro.console.wangsu.com/v2/index/#/properties\" target=\"_blank\">https://cdnpro.console.wangsu.com/v2/index/#/properties</a>",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_certificate_id.label": "Wangsu Cloud CDN certificate ID (Optional)",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_certificate_id.placeholder": "Please enter Wangsu Cloud CDN certificate ID",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_certificate_id.tooltip": "For more information, see <a href=\"https://cdnpro.console.wangsu.com/v2/index/#/certificate\" target=\"_blank\">https://cdnpro.console.wangsu.com/v2/index/#/certificate</a>",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_webhook_id.label": "Wangsu Cloud CDN Webhook ID (Optional)",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_webhook_id.placeholder": "Please enter Wangsu Cloud CDN Webhook ID",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_webhook_id.tooltip": "For more information, see <a href=\"https://cdnpro.console.wangsu.com/v2/index/#/certificate\" target=\"_blank\">https://cdnpro.console.wangsu.com/v2/index/#/certificate</a>",
|
||||||
"workflow_node.deploy.form.webhook_data.label": "Webhook data (JSON format)",
|
"workflow_node.deploy.form.webhook_data.label": "Webhook data (JSON format)",
|
||||||
"workflow_node.deploy.form.webhook_data.placeholder": "Please enter Webhook data",
|
"workflow_node.deploy.form.webhook_data.placeholder": "Please enter Webhook data",
|
||||||
"workflow_node.deploy.form.webhook_data.guide": "Tips: The Webhook data should be a key-value pair in JSON format. The values in JSON support template variables, which will be replaced by actual values when sent to the Webhook URL. <br><br>Supported variables: <br><strong>${DOMAIN}</strong>: The primary domain of the certificate (<i>CommonName</i>).<br><strong>${DOMAINS}</strong>: The domain list of the certificate (<i>SubjectAltNames</i>).<br><strong>${CERTIFICATE}</strong>: The PEM format content of the certificate file.<br><strong>${PRIVATE_KEY}</strong>: The PEM format content of the private key file.",
|
"workflow_node.deploy.form.webhook_data.guide": "Tips: The Webhook data should be a key-value pair in JSON format. The values in JSON support template variables, which will be replaced by actual values when sent to the Webhook URL. <br><br>Supported variables: <br><strong>${DOMAIN}</strong>: The primary domain of the certificate (<i>CommonName</i>).<br><strong>${DOMAINS}</strong>: The domain list of the certificate (<i>SubjectAltNames</i>).<br><strong>${CERTIFICATE}</strong>: The PEM format content of the certificate file.<br><strong>${PRIVATE_KEY}</strong>: The PEM format content of the private key file.",
|
||||||
|
@ -298,6 +298,12 @@
|
|||||||
"access.form.volcengine_secret_access_key.label": "火山引擎 SecretAccessKey",
|
"access.form.volcengine_secret_access_key.label": "火山引擎 SecretAccessKey",
|
||||||
"access.form.volcengine_secret_access_key.placeholder": "请输入火山引擎 SecretAccessKey",
|
"access.form.volcengine_secret_access_key.placeholder": "请输入火山引擎 SecretAccessKey",
|
||||||
"access.form.volcengine_secret_access_key.tooltip": "这是什么?请参阅 <a href=\"https://www.volcengine.com/docs/6291/216571\" target=\"_blank\">https://www.volcengine.com/docs/6291/216571</a>",
|
"access.form.volcengine_secret_access_key.tooltip": "这是什么?请参阅 <a href=\"https://www.volcengine.com/docs/6291/216571\" target=\"_blank\">https://www.volcengine.com/docs/6291/216571</a>",
|
||||||
|
"access.form.wangsu_access_key_id.label": "网宿云 AccessKeyId",
|
||||||
|
"access.form.wangsu_access_key_id.placeholder": "请输入网宿科技 AccessKeyId",
|
||||||
|
"access.form.wangsu_access_key_id.tooltip": "这是什么?请参阅 <a href=\"https://www.wangsu.com/document/account-manage/15775\" target=\"_blank\">https://www.wangsu.com/document/account-manage/15775</a>",
|
||||||
|
"access.form.wangsu_access_key_secret.label": "网宿科技 AccessKeySecret",
|
||||||
|
"access.form.wangsu_access_key_secret.placeholder": "请输入网宿科技 AccessKeySecret",
|
||||||
|
"access.form.wangsu_access_key_secret.tooltip": "这是什么?请参阅 <a href=\"https://www.wangsu.com/document/account-manage/15775\" target=\"_blank\">https://www.wangsu.com/document/account-manage/15775</a>",
|
||||||
"access.form.webhook_url.label": "Webhook 回调地址",
|
"access.form.webhook_url.label": "Webhook 回调地址",
|
||||||
"access.form.webhook_url.placeholder": "请输入 Webhook 回调地址",
|
"access.form.webhook_url.placeholder": "请输入 Webhook 回调地址",
|
||||||
"access.form.webhook_allow_insecure_conns.label": "忽略 SSL/TLS 证书错误",
|
"access.form.webhook_allow_insecure_conns.label": "忽略 SSL/TLS 证书错误",
|
||||||
|
@ -125,6 +125,8 @@
|
|||||||
"provider.volcengine.imagex": "火山引擎 - 图片服务 ImageX",
|
"provider.volcengine.imagex": "火山引擎 - 图片服务 ImageX",
|
||||||
"provider.volcengine.live": "火山引擎 - 视频直播 Live",
|
"provider.volcengine.live": "火山引擎 - 视频直播 Live",
|
||||||
"provider.volcengine.tos": "火山引擎 - 对象存储 TOS",
|
"provider.volcengine.tos": "火山引擎 - 对象存储 TOS",
|
||||||
|
"provider.wangsu": "网宿云",
|
||||||
|
"provider.wangsu.cdnpro": "网宿云 - CDN Pro",
|
||||||
"provider.webhook": "Webhook",
|
"provider.webhook": "Webhook",
|
||||||
"provider.westcn": "西部数码",
|
"provider.westcn": "西部数码",
|
||||||
"provider.zerossl": "ZeroSSL",
|
"provider.zerossl": "ZeroSSL",
|
||||||
|
@ -94,7 +94,7 @@
|
|||||||
"workflow_node.deploy.form.provider.placeholder": "请选择部署目标",
|
"workflow_node.deploy.form.provider.placeholder": "请选择部署目标",
|
||||||
"workflow_node.deploy.form.provider_access.label": "主机提供商授权",
|
"workflow_node.deploy.form.provider_access.label": "主机提供商授权",
|
||||||
"workflow_node.deploy.form.provider_access.placeholder": "请选择主机提供商授权",
|
"workflow_node.deploy.form.provider_access.placeholder": "请选择主机提供商授权",
|
||||||
"workflow_node.deploy.form.provider_access.tooltip": "用于部署证书,注意与申请阶段所需的 DNS 提供商相区分。",
|
"workflow_node.deploy.form.provider_access.tooltip": "用于部署证书时调用相关 API,注意与申请阶段所需的 DNS 提供商相区分。",
|
||||||
"workflow_node.deploy.form.provider_access.button": "新建",
|
"workflow_node.deploy.form.provider_access.button": "新建",
|
||||||
"workflow_node.deploy.form.provider_access.guide_for_local": "小贴士:如果你正在使用 Docker 运行 Certimate,“本地”指的是容器内而非宿主机。",
|
"workflow_node.deploy.form.provider_access.guide_for_local": "小贴士:如果你正在使用 Docker 运行 Certimate,“本地”指的是容器内而非宿主机。",
|
||||||
"workflow_node.deploy.form.certificate.label": "待部署证书",
|
"workflow_node.deploy.form.certificate.label": "待部署证书",
|
||||||
@ -638,6 +638,19 @@
|
|||||||
"workflow_node.deploy.form.volcengine_tos_domain.label": "火山引擎 TOS 自定义域名",
|
"workflow_node.deploy.form.volcengine_tos_domain.label": "火山引擎 TOS 自定义域名",
|
||||||
"workflow_node.deploy.form.volcengine_tos_domain.placeholder": "请输入火山引擎 TOS 自定义域名",
|
"workflow_node.deploy.form.volcengine_tos_domain.placeholder": "请输入火山引擎 TOS 自定义域名",
|
||||||
"workflow_node.deploy.form.volcengine_tos_domain.tooltip": "这是什么?请参阅 see <a href=\"https://console.volcengine.com/tos\" target=\"_blank\">https://console.volcengine.com/tos</a>",
|
"workflow_node.deploy.form.volcengine_tos_domain.tooltip": "这是什么?请参阅 see <a href=\"https://console.volcengine.com/tos\" target=\"_blank\">https://console.volcengine.com/tos</a>",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_environment.label": "网宿云环境",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_environment.placeholder": "请选择网宿云环境",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_environment.option.production.label": "生产环境",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_environment.option.staging.label": "演练环境",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_domain.label": "网宿云 CDN Pro 加速域名",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_domain.placeholder": "请输入网宿云 CDN Pro 加速域名(支持泛域名)",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_domain.tooltip": "这是什么?请参阅 <a href=\"https://cdnpro.console.wangsu.com/v2/index/#/properties\" target=\"_blank\">https://cdnpro.console.wangsu.com/v2/index/#/properties</a>",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_certificate_id.label": "网宿云 CDN Pro 原证书 ID(可选)",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_certificate_id.placeholder": "请输入网宿云 CDN Pro 原证书 ID",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_certificate_id.tooltip": "这是什么?请参阅 <a href=\"https://cdnpro.console.wangsu.com/v2/index/#/certificate\" target=\"_blank\">https://cdnpro.console.wangsu.com/v2/index/#/certificate</a><br><br>不填写时,将上传新证书;否则,将替换原证书。",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_webhook_id.label": "网宿云 CDN Pro 部署任务 Webhook ID(可选)",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_webhook_id.placeholder": "请输入网宿云 CDN Pro 部署任务 Webhook ID",
|
||||||
|
"workflow_node.deploy.form.wangsu_cdnpro_webhook_id.tooltip": "这是什么?请参阅 <a href=\"https://cdnpro.console.wangsu.com/v2/index/#/certificate\" target=\"_blank\">https://cdnpro.console.wangsu.com/v2/index/#/certificate</a>",
|
||||||
"workflow_node.deploy.form.webhook_data.label": "Webhook 回调数据(JSON 格式)",
|
"workflow_node.deploy.form.webhook_data.label": "Webhook 回调数据(JSON 格式)",
|
||||||
"workflow_node.deploy.form.webhook_data.placeholder": "请输入 Webhook 回调数据",
|
"workflow_node.deploy.form.webhook_data.placeholder": "请输入 Webhook 回调数据",
|
||||||
"workflow_node.deploy.form.webhook_data.guide": "小贴士:回调数据是一个 JSON 格式的键值对。其中值支持模板变量,将在被发送到指定的 Webhook URL 时被替换为实际值;其他内容将保持原样。<br><br>支持的变量:<br><strong>${DOMAIN}</strong>:证书的主域名(即 <i>CommonName</i>)<br><strong>${DOMAINS}</strong>:证书的多域名列表(即 <i>SubjectAltNames</i>)<br><strong>${CERTIFICATE}</strong>:证书文件 PEM 格式内容<br><strong>${PRIVATE_KEY}</strong>:私钥文件 PEM 格式内容",
|
"workflow_node.deploy.form.webhook_data.guide": "小贴士:回调数据是一个 JSON 格式的键值对。其中值支持模板变量,将在被发送到指定的 Webhook URL 时被替换为实际值;其他内容将保持原样。<br><br>支持的变量:<br><strong>${DOMAIN}</strong>:证书的主域名(即 <i>CommonName</i>)<br><strong>${DOMAINS}</strong>:证书的多域名列表(即 <i>SubjectAltNames</i>)<br><strong>${CERTIFICATE}</strong>:证书文件 PEM 格式内容<br><strong>${PRIVATE_KEY}</strong>:私钥文件 PEM 格式内容",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user