mirror of
https://github.com/woodchen-ink/certimate.git
synced 2025-07-18 09:21:56 +08:00
feat: new acme dns-01 provider: spaceship
This commit is contained in:
parent
d272b64329
commit
928a0443cc
@ -38,6 +38,7 @@ import (
|
|||||||
pPorkbun "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/porkbun"
|
pPorkbun "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/porkbun"
|
||||||
pPowerDNS "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/powerdns"
|
pPowerDNS "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/powerdns"
|
||||||
pRainYun "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/rainyun"
|
pRainYun "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/rainyun"
|
||||||
|
pSpaceship "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/spaceship"
|
||||||
pTencentCloud "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/tencentcloud"
|
pTencentCloud "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/tencentcloud"
|
||||||
pTencentCloudEO "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/tencentcloud-eo"
|
pTencentCloudEO "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/tencentcloud-eo"
|
||||||
pUCloudUDNR "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/ucloud-udnr"
|
pUCloudUDNR "github.com/certimate-go/certimate/pkg/core/ssl-applicator/acme-dns01/providers/ucloud-udnr"
|
||||||
@ -582,6 +583,22 @@ func createApplicantProvider(options *applicantProviderOptions) (challenge.Provi
|
|||||||
return applicant, err
|
return applicant, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case domain.ACMEDns01ProviderTypeSpaceship:
|
||||||
|
{
|
||||||
|
access := domain.AccessConfigForSpaceship{}
|
||||||
|
if err := xmaps.Populate(options.ProviderAccessConfig, &access); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to populate provider access config: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
applicant, err := pSpaceship.NewChallengeProvider(&pSpaceship.ChallengeProviderConfig{
|
||||||
|
ApiKey: access.ApiKey,
|
||||||
|
ApiSecret: access.ApiSecret,
|
||||||
|
DnsPropagationTimeout: options.DnsPropagationTimeout,
|
||||||
|
DnsTTL: options.DnsTTL,
|
||||||
|
})
|
||||||
|
return applicant, err
|
||||||
|
}
|
||||||
|
|
||||||
case domain.ACMEDns01ProviderTypeTencentCloud, domain.ACMEDns01ProviderTypeTencentCloudDNS, domain.ACMEDns01ProviderTypeTencentCloudEO:
|
case domain.ACMEDns01ProviderTypeTencentCloud, domain.ACMEDns01ProviderTypeTencentCloudDNS, domain.ACMEDns01ProviderTypeTencentCloudEO:
|
||||||
{
|
{
|
||||||
access := domain.AccessConfigForTencentCloud{}
|
access := domain.AccessConfigForTencentCloud{}
|
||||||
|
@ -324,6 +324,11 @@ type AccessConfigForSlackBot struct {
|
|||||||
DefaultChannelId string `json:"defaultChannelId,omitempty"`
|
DefaultChannelId string `json:"defaultChannelId,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type AccessConfigForSpaceship struct {
|
||||||
|
ApiKey string `json:"apiKey"`
|
||||||
|
ApiSecret string `json:"apiSecret"`
|
||||||
|
}
|
||||||
|
|
||||||
type AccessConfigForSSH struct {
|
type AccessConfigForSSH struct {
|
||||||
Host string `json:"host"`
|
Host string `json:"host"`
|
||||||
Port int32 `json:"port"`
|
Port int32 `json:"port"`
|
||||||
|
@ -74,6 +74,7 @@ const (
|
|||||||
AccessProviderTypeRatPanel = AccessProviderType("ratpanel")
|
AccessProviderTypeRatPanel = AccessProviderType("ratpanel")
|
||||||
AccessProviderTypeSafeLine = AccessProviderType("safeline")
|
AccessProviderTypeSafeLine = AccessProviderType("safeline")
|
||||||
AccessProviderTypeSlackBot = AccessProviderType("slackbot")
|
AccessProviderTypeSlackBot = AccessProviderType("slackbot")
|
||||||
|
AccessProviderTypeSpaceship = AccessProviderType("spaceship")
|
||||||
AccessProviderTypeSSH = AccessProviderType("ssh")
|
AccessProviderTypeSSH = AccessProviderType("ssh")
|
||||||
AccessProviderTypeSSLCOM = AccessProviderType("sslcom")
|
AccessProviderTypeSSLCOM = AccessProviderType("sslcom")
|
||||||
AccessProviderTypeTelegramBot = AccessProviderType("telegrambot")
|
AccessProviderTypeTelegramBot = AccessProviderType("telegrambot")
|
||||||
@ -159,6 +160,7 @@ const (
|
|||||||
ACMEDns01ProviderTypePorkbun = ACMEDns01ProviderType(AccessProviderTypePorkbun)
|
ACMEDns01ProviderTypePorkbun = ACMEDns01ProviderType(AccessProviderTypePorkbun)
|
||||||
ACMEDns01ProviderTypePowerDNS = ACMEDns01ProviderType(AccessProviderTypePowerDNS)
|
ACMEDns01ProviderTypePowerDNS = ACMEDns01ProviderType(AccessProviderTypePowerDNS)
|
||||||
ACMEDns01ProviderTypeRainYun = ACMEDns01ProviderType(AccessProviderTypeRainYun)
|
ACMEDns01ProviderTypeRainYun = ACMEDns01ProviderType(AccessProviderTypeRainYun)
|
||||||
|
ACMEDns01ProviderTypeSpaceship = ACMEDns01ProviderType(AccessProviderTypeSpaceship)
|
||||||
ACMEDns01ProviderTypeTencentCloud = ACMEDns01ProviderType(AccessProviderTypeTencentCloud) // 兼容旧值,等同于 [ACMEDns01ProviderTypeTencentCloudDNS]
|
ACMEDns01ProviderTypeTencentCloud = ACMEDns01ProviderType(AccessProviderTypeTencentCloud) // 兼容旧值,等同于 [ACMEDns01ProviderTypeTencentCloudDNS]
|
||||||
ACMEDns01ProviderTypeTencentCloudDNS = ACMEDns01ProviderType(AccessProviderTypeTencentCloud + "-dns")
|
ACMEDns01ProviderTypeTencentCloudDNS = ACMEDns01ProviderType(AccessProviderTypeTencentCloud + "-dns")
|
||||||
ACMEDns01ProviderTypeTencentCloudEO = ACMEDns01ProviderType(AccessProviderTypeTencentCloud + "-eo")
|
ACMEDns01ProviderTypeTencentCloudEO = ACMEDns01ProviderType(AccessProviderTypeTencentCloud + "-eo")
|
||||||
|
@ -0,0 +1,40 @@
|
|||||||
|
package spaceship
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/go-acme/lego/v4/providers/dns/spaceship"
|
||||||
|
|
||||||
|
"github.com/certimate-go/certimate/pkg/core"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ChallengeProviderConfig struct {
|
||||||
|
ApiKey string `json:"apiKey"`
|
||||||
|
ApiSecret string `json:"apiSecret"`
|
||||||
|
DnsPropagationTimeout int32 `json:"dnsPropagationTimeout,omitempty"`
|
||||||
|
DnsTTL int32 `json:"dnsTTL,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewChallengeProvider(config *ChallengeProviderConfig) (core.ACMEChallenger, error) {
|
||||||
|
if config == nil {
|
||||||
|
return nil, errors.New("the configuration of the acme challenge provider is nil")
|
||||||
|
}
|
||||||
|
|
||||||
|
providerConfig := spaceship.NewDefaultConfig()
|
||||||
|
providerConfig.APIKey = config.ApiKey
|
||||||
|
providerConfig.APISecret = config.ApiSecret
|
||||||
|
if config.DnsPropagationTimeout != 0 {
|
||||||
|
providerConfig.PropagationTimeout = time.Duration(config.DnsPropagationTimeout) * time.Second
|
||||||
|
}
|
||||||
|
if config.DnsTTL != 0 {
|
||||||
|
providerConfig.TTL = int(config.DnsTTL)
|
||||||
|
}
|
||||||
|
|
||||||
|
provider, err := spaceship.NewDNSProviderConfig(providerConfig)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return provider, nil
|
||||||
|
}
|
BIN
ui/public/imgs/providers/spaceship.png
Normal file
BIN
ui/public/imgs/providers/spaceship.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 6.5 KiB |
@ -68,6 +68,7 @@ import AccessFormRainYunConfig from "./AccessFormRainYunConfig";
|
|||||||
import AccessFormRatPanelConfig from "./AccessFormRatPanelConfig";
|
import AccessFormRatPanelConfig from "./AccessFormRatPanelConfig";
|
||||||
import AccessFormSafeLineConfig from "./AccessFormSafeLineConfig";
|
import AccessFormSafeLineConfig from "./AccessFormSafeLineConfig";
|
||||||
import AccessFormSlackBotConfig from "./AccessFormSlackBotConfig";
|
import AccessFormSlackBotConfig from "./AccessFormSlackBotConfig";
|
||||||
|
import AccessFormSpaceshipConfig from "./AccessFormSpaceshipConfig";
|
||||||
import AccessFormSSHConfig from "./AccessFormSSHConfig";
|
import AccessFormSSHConfig from "./AccessFormSSHConfig";
|
||||||
import AccessFormSSLComConfig from "./AccessFormSSLComConfig";
|
import AccessFormSSLComConfig from "./AccessFormSSLComConfig";
|
||||||
import AccessFormTelegramBotConfig from "./AccessFormTelegramBotConfig";
|
import AccessFormTelegramBotConfig from "./AccessFormTelegramBotConfig";
|
||||||
@ -301,6 +302,8 @@ const AccessForm = forwardRef<AccessFormInstance, AccessFormProps>(({ className,
|
|||||||
return <AccessFormSafeLineConfig {...nestedFormProps} />;
|
return <AccessFormSafeLineConfig {...nestedFormProps} />;
|
||||||
case ACCESS_PROVIDERS.SLACKBOT:
|
case ACCESS_PROVIDERS.SLACKBOT:
|
||||||
return <AccessFormSlackBotConfig {...nestedFormProps} />;
|
return <AccessFormSlackBotConfig {...nestedFormProps} />;
|
||||||
|
case ACCESS_PROVIDERS.SPACESHIP:
|
||||||
|
return <AccessFormSpaceshipConfig {...nestedFormProps} />;
|
||||||
case ACCESS_PROVIDERS.SSH:
|
case ACCESS_PROVIDERS.SSH:
|
||||||
return <AccessFormSSHConfig {...nestedFormProps} />;
|
return <AccessFormSSHConfig {...nestedFormProps} />;
|
||||||
case ACCESS_PROVIDERS.TELEGRAMBOT:
|
case ACCESS_PROVIDERS.TELEGRAMBOT:
|
||||||
|
68
ui/src/components/access/AccessFormSpaceshipConfig.tsx
Normal file
68
ui/src/components/access/AccessFormSpaceshipConfig.tsx
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Form, type FormInstance, Input } from "antd";
|
||||||
|
import { createSchemaFieldRule } from "antd-zod";
|
||||||
|
import { z } from "zod/v4";
|
||||||
|
|
||||||
|
import { type AccessConfigForSpaceship } from "@/domain/access";
|
||||||
|
|
||||||
|
type AccessFormSpaceshipConfigFieldValues = Nullish<AccessConfigForSpaceship>;
|
||||||
|
|
||||||
|
export type AccessFormSpaceshipConfigProps = {
|
||||||
|
form: FormInstance;
|
||||||
|
formName: string;
|
||||||
|
disabled?: boolean;
|
||||||
|
initialValues?: AccessFormSpaceshipConfigFieldValues;
|
||||||
|
onValuesChange?: (values: AccessFormSpaceshipConfigFieldValues) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
const initFormModel = (): AccessFormSpaceshipConfigFieldValues => {
|
||||||
|
return {
|
||||||
|
apiKey: "",
|
||||||
|
apiSecret: "",
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const AccessFormSpaceshipConfig = ({ form: formInst, formName, disabled, initialValues, onValuesChange }: AccessFormSpaceshipConfigProps) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
apiKey: z.string().nonempty(t("access.form.spaceship_api_key.placeholder")),
|
||||||
|
apiSecret: z.string().nonempty(t("access.form.spaceship_api_secret.placeholder")),
|
||||||
|
});
|
||||||
|
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="apiKey"
|
||||||
|
label={t("access.form.spaceship_api_key.label")}
|
||||||
|
rules={[formRule]}
|
||||||
|
tooltip={<span dangerouslySetInnerHTML={{ __html: t("access.form.spaceship_api_key.tooltip") }}></span>}
|
||||||
|
>
|
||||||
|
<Input autoComplete="new-password" placeholder={t("access.form.spaceship_api_key.placeholder")} />
|
||||||
|
</Form.Item>
|
||||||
|
|
||||||
|
<Form.Item
|
||||||
|
name="apiSecret"
|
||||||
|
label={t("access.form.spaceship_api_secret.label")}
|
||||||
|
rules={[formRule]}
|
||||||
|
tooltip={<span dangerouslySetInnerHTML={{ __html: t("access.form.spaceship_api_secret.tooltip") }}></span>}
|
||||||
|
>
|
||||||
|
<Input.Password autoComplete="new-password" placeholder={t("access.form.spaceship_api_secret.placeholder")} />
|
||||||
|
</Form.Item>
|
||||||
|
</Form>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default AccessFormSpaceshipConfig;
|
@ -62,6 +62,7 @@ export interface AccessModel extends BaseModel {
|
|||||||
| AccessConfigForRatPanel
|
| AccessConfigForRatPanel
|
||||||
| AccessConfigForSafeLine
|
| AccessConfigForSafeLine
|
||||||
| AccessConfigForSlackBot
|
| AccessConfigForSlackBot
|
||||||
|
| AccessConfigForSpaceship
|
||||||
| AccessConfigForSSH
|
| AccessConfigForSSH
|
||||||
| AccessConfigForSSLCom
|
| AccessConfigForSSLCom
|
||||||
| AccessConfigForTelegramBot
|
| AccessConfigForTelegramBot
|
||||||
@ -389,6 +390,11 @@ export type AccessConfigForSlackBot = {
|
|||||||
defaultChannelId?: string;
|
defaultChannelId?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type AccessConfigForSpaceship = {
|
||||||
|
apiKey: string;
|
||||||
|
apiSecret: string;
|
||||||
|
};
|
||||||
|
|
||||||
export type AccessConfigForSSH = {
|
export type AccessConfigForSSH = {
|
||||||
host: string;
|
host: string;
|
||||||
port: number;
|
port: number;
|
||||||
|
@ -65,6 +65,7 @@ export const ACCESS_PROVIDERS = Object.freeze({
|
|||||||
RATPANEL: "ratpanel",
|
RATPANEL: "ratpanel",
|
||||||
SAFELINE: "safeline",
|
SAFELINE: "safeline",
|
||||||
SLACKBOT: "slackbot",
|
SLACKBOT: "slackbot",
|
||||||
|
SPACESHIP: "spaceship",
|
||||||
SSH: "ssh",
|
SSH: "ssh",
|
||||||
SSLCOM: "sslcom",
|
SSLCOM: "sslcom",
|
||||||
TELEGRAMBOT: "telegrambot",
|
TELEGRAMBOT: "telegrambot",
|
||||||
@ -164,6 +165,7 @@ export const accessProvidersMap: Map<AccessProvider["type"] | string, AccessProv
|
|||||||
[ACCESS_PROVIDERS.NETCUP, "provider.netcup", "/imgs/providers/netcup.png", [ACCESS_USAGES.DNS]],
|
[ACCESS_PROVIDERS.NETCUP, "provider.netcup", "/imgs/providers/netcup.png", [ACCESS_USAGES.DNS]],
|
||||||
[ACCESS_PROVIDERS.NS1, "provider.ns1", "/imgs/providers/ns1.svg", [ACCESS_USAGES.DNS]],
|
[ACCESS_PROVIDERS.NS1, "provider.ns1", "/imgs/providers/ns1.svg", [ACCESS_USAGES.DNS]],
|
||||||
[ACCESS_PROVIDERS.PORKBUN, "provider.porkbun", "/imgs/providers/porkbun.svg", [ACCESS_USAGES.DNS]],
|
[ACCESS_PROVIDERS.PORKBUN, "provider.porkbun", "/imgs/providers/porkbun.svg", [ACCESS_USAGES.DNS]],
|
||||||
|
[ACCESS_PROVIDERS.SPACESHIP, "provider.spaceship", "/imgs/providers/spaceship.png", [ACCESS_USAGES.DNS]],
|
||||||
[ACCESS_PROVIDERS.VERCEL, "provider.vercel", "/imgs/providers/vercel.svg", [ACCESS_USAGES.DNS]],
|
[ACCESS_PROVIDERS.VERCEL, "provider.vercel", "/imgs/providers/vercel.svg", [ACCESS_USAGES.DNS]],
|
||||||
[ACCESS_PROVIDERS.CMCCCLOUD, "provider.cmcccloud", "/imgs/providers/cmcccloud.svg", [ACCESS_USAGES.DNS]],
|
[ACCESS_PROVIDERS.CMCCCLOUD, "provider.cmcccloud", "/imgs/providers/cmcccloud.svg", [ACCESS_USAGES.DNS]],
|
||||||
[ACCESS_PROVIDERS.WESTCN, "provider.westcn", "/imgs/providers/westcn.svg", [ACCESS_USAGES.DNS]],
|
[ACCESS_PROVIDERS.WESTCN, "provider.westcn", "/imgs/providers/westcn.svg", [ACCESS_USAGES.DNS]],
|
||||||
@ -296,6 +298,7 @@ export const ACME_DNS01_PROVIDERS = Object.freeze({
|
|||||||
PORKBUN: `${ACCESS_PROVIDERS.PORKBUN}`,
|
PORKBUN: `${ACCESS_PROVIDERS.PORKBUN}`,
|
||||||
POWERDNS: `${ACCESS_PROVIDERS.POWERDNS}`,
|
POWERDNS: `${ACCESS_PROVIDERS.POWERDNS}`,
|
||||||
RAINYUN: `${ACCESS_PROVIDERS.RAINYUN}`,
|
RAINYUN: `${ACCESS_PROVIDERS.RAINYUN}`,
|
||||||
|
SPACESHIP: `${ACCESS_PROVIDERS.SPACESHIP}`,
|
||||||
UCLOUD_UDNR: `${ACCESS_PROVIDERS.UCLOUD}-udnr`,
|
UCLOUD_UDNR: `${ACCESS_PROVIDERS.UCLOUD}-udnr`,
|
||||||
TENCENTCLOUD: `${ACCESS_PROVIDERS.TENCENTCLOUD}`, // 兼容旧值,等同于 `TENCENTCLOUD_DNS`
|
TENCENTCLOUD: `${ACCESS_PROVIDERS.TENCENTCLOUD}`, // 兼容旧值,等同于 `TENCENTCLOUD_DNS`
|
||||||
TENCENTCLOUD_DNS: `${ACCESS_PROVIDERS.TENCENTCLOUD}-dns`,
|
TENCENTCLOUD_DNS: `${ACCESS_PROVIDERS.TENCENTCLOUD}-dns`,
|
||||||
@ -351,6 +354,7 @@ export const acmeDns01ProvidersMap: Map<ACMEDns01Provider["type"] | string, ACME
|
|||||||
[ACME_DNS01_PROVIDERS.NETLIFY, "provider.netlify"],
|
[ACME_DNS01_PROVIDERS.NETLIFY, "provider.netlify"],
|
||||||
[ACME_DNS01_PROVIDERS.NS1, "provider.ns1"],
|
[ACME_DNS01_PROVIDERS.NS1, "provider.ns1"],
|
||||||
[ACME_DNS01_PROVIDERS.PORKBUN, "provider.porkbun"],
|
[ACME_DNS01_PROVIDERS.PORKBUN, "provider.porkbun"],
|
||||||
|
[ACME_DNS01_PROVIDERS.SPACESHIP, "provider.spaceship"],
|
||||||
[ACME_DNS01_PROVIDERS.VERCEL, "provider.vercel"],
|
[ACME_DNS01_PROVIDERS.VERCEL, "provider.vercel"],
|
||||||
[ACME_DNS01_PROVIDERS.CMCCCLOUD_DNS, "provider.cmcccloud.dns"],
|
[ACME_DNS01_PROVIDERS.CMCCCLOUD_DNS, "provider.cmcccloud.dns"],
|
||||||
[ACME_DNS01_PROVIDERS.CTCCCLOUD_SMARTDNS, "provider.ctcccloud.smartdns"],
|
[ACME_DNS01_PROVIDERS.CTCCCLOUD_SMARTDNS, "provider.ctcccloud.smartdns"],
|
||||||
|
@ -387,6 +387,12 @@
|
|||||||
"access.form.slackbot_default_channel_id.label": "Default Slack channel ID (Optional)",
|
"access.form.slackbot_default_channel_id.label": "Default Slack channel ID (Optional)",
|
||||||
"access.form.slackbot_default_channel_id.placeholder": "Please enter default Slack channel ID",
|
"access.form.slackbot_default_channel_id.placeholder": "Please enter default Slack channel ID",
|
||||||
"access.form.slackbot_default_channel_id.tooltip": "How to get it? Please refer to <a href=\"https://www.youtube.com/watch?v=Uz5Yi5C2pwQ\" target=\"_blank\">https://www.youtube.com/watch?v=Uz5Yi5C2pwQ</a>",
|
"access.form.slackbot_default_channel_id.tooltip": "How to get it? Please refer to <a href=\"https://www.youtube.com/watch?v=Uz5Yi5C2pwQ\" target=\"_blank\">https://www.youtube.com/watch?v=Uz5Yi5C2pwQ</a>",
|
||||||
|
"access.form.spaceship_api_key.label": "Spaceship API key",
|
||||||
|
"access.form.spaceship_api_key.placeholder": "Please enter Spaceship API key",
|
||||||
|
"access.form.spaceship_api_key.tooltip": "For more information, see <a href=\"https://www.spaceship.com/application/api-manager/\" target=\"_blank\">https://www.spaceship.com/application/api-manager/</a>",
|
||||||
|
"access.form.spaceship_api_secret.label": "Spaceship API secret",
|
||||||
|
"access.form.spaceship_api_secret.placeholder": "Please enter Spaceship API secret",
|
||||||
|
"access.form.spaceship_api_secret.tooltip": "For more information, see <a href=\"https://www.spaceship.com/application/api-manager/\" target=\"_blank\">https://www.spaceship.com/application/api-manager/</a>",
|
||||||
"access.form.ssh_host.label": "Server host",
|
"access.form.ssh_host.label": "Server host",
|
||||||
"access.form.ssh_host.placeholder": "Please enter server host",
|
"access.form.ssh_host.placeholder": "Please enter server host",
|
||||||
"access.form.ssh_port.label": "Server port",
|
"access.form.ssh_port.label": "Server port",
|
||||||
|
@ -130,6 +130,7 @@
|
|||||||
"provider.ratpanel.site": "RatPanel - Website",
|
"provider.ratpanel.site": "RatPanel - Website",
|
||||||
"provider.safeline": "SafeLine",
|
"provider.safeline": "SafeLine",
|
||||||
"provider.slackbot": "Slack Bot",
|
"provider.slackbot": "Slack Bot",
|
||||||
|
"provider.spaceship": "Spaceship",
|
||||||
"provider.ssh": "Remote host (SSH)",
|
"provider.ssh": "Remote host (SSH)",
|
||||||
"provider.sslcom": "SSL.com",
|
"provider.sslcom": "SSL.com",
|
||||||
"provider.telegrambot": "Telegram Bot",
|
"provider.telegrambot": "Telegram Bot",
|
||||||
|
@ -387,6 +387,12 @@
|
|||||||
"access.form.slackbot_default_channel_id.label": "默认的 Slack 频道 ID(可选)",
|
"access.form.slackbot_default_channel_id.label": "默认的 Slack 频道 ID(可选)",
|
||||||
"access.form.slackbot_default_channel_id.placeholder": "请输入默认的 Slack 频道 ID",
|
"access.form.slackbot_default_channel_id.placeholder": "请输入默认的 Slack 频道 ID",
|
||||||
"access.form.slackbot_default_channel_id.tooltip": "如何获取此参数?请参阅 <a href=\"https://www.youtube.com/watch?v=Uz5Yi5C2pwQ\" target=\"_blank\">https://www.youtube.com/watch?v=Uz5Yi5C2pwQ</a>",
|
"access.form.slackbot_default_channel_id.tooltip": "如何获取此参数?请参阅 <a href=\"https://www.youtube.com/watch?v=Uz5Yi5C2pwQ\" target=\"_blank\">https://www.youtube.com/watch?v=Uz5Yi5C2pwQ</a>",
|
||||||
|
"access.form.spaceship_api_key.label": "Spaceship API Key",
|
||||||
|
"access.form.spaceship_api_key.placeholder": "请输入 Spaceship API Key",
|
||||||
|
"access.form.spaceship_api_key.tooltip": "这是什么?请参阅 <a href=\"https://www.spaceship.com/application/api-manager/\" target=\"_blank\">https://www.spaceship.com/application/api-manager/</a>",
|
||||||
|
"access.form.spaceship_api_secret.label": "Spaceship API Secret",
|
||||||
|
"access.form.spaceship_api_secret.placeholder": "请输入 Spaceship API Secret",
|
||||||
|
"access.form.spaceship_api_secret.tooltip": "这是什么?请参阅 <a href=\"https://www.spaceship.com/application/api-manager/\" target=\"_blank\">https://www.spaceship.com/application/api-manager/</a>",
|
||||||
"access.form.ssh_host.label": "服务器地址",
|
"access.form.ssh_host.label": "服务器地址",
|
||||||
"access.form.ssh_host.placeholder": "请输入服务器地址",
|
"access.form.ssh_host.placeholder": "请输入服务器地址",
|
||||||
"access.form.ssh_port.label": "服务器端口",
|
"access.form.ssh_port.label": "服务器端口",
|
||||||
|
@ -130,6 +130,7 @@
|
|||||||
"provider.ratpanel.site": "耗子面板 - 网站",
|
"provider.ratpanel.site": "耗子面板 - 网站",
|
||||||
"provider.safeline": "雷池",
|
"provider.safeline": "雷池",
|
||||||
"provider.slackbot": "Slack 机器人",
|
"provider.slackbot": "Slack 机器人",
|
||||||
|
"provider.spaceship": "Spaceship",
|
||||||
"provider.ssh": "远程主机(SSH)",
|
"provider.ssh": "远程主机(SSH)",
|
||||||
"provider.sslcom": "SSL.com",
|
"provider.sslcom": "SSL.com",
|
||||||
"provider.telegrambot": "Telegram 机器人",
|
"provider.telegrambot": "Telegram 机器人",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user