2021-03-09 08:25:55 +08:00
|
|
|
package hostingde
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2022-08-15 00:50:15 +08:00
|
|
|
"io"
|
2021-03-09 08:25:55 +08:00
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/diff"
|
|
|
|
"golang.org/x/net/idna"
|
|
|
|
)
|
|
|
|
|
|
|
|
const endpoint = "%s/api/%s/v1/json/%s"
|
|
|
|
|
|
|
|
type hostingdeProvider struct {
|
2023-02-01 00:03:29 +08:00
|
|
|
authToken string
|
|
|
|
ownerAccountID string
|
2023-02-28 09:28:17 +08:00
|
|
|
filterAccountID string
|
2023-02-01 00:03:29 +08:00
|
|
|
baseURL string
|
|
|
|
nameservers []string
|
2023-02-06 22:04:31 +08:00
|
|
|
defaultSoa soaValues
|
2021-03-09 08:25:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (hp *hostingdeProvider) getDomainConfig(domain string) (*domainConfig, error) {
|
|
|
|
params := request{
|
2023-01-31 23:22:54 +08:00
|
|
|
Filter: &filter{
|
2021-03-09 08:25:55 +08:00
|
|
|
Field: "domainName",
|
2021-11-28 02:44:28 +08:00
|
|
|
Value: domain,
|
2021-03-09 08:25:55 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := hp.get("domain", "domainsFind", params)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error getting domain info: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
domainConf := []*domainConfig{}
|
|
|
|
if err := json.Unmarshal(resp.Data, &domainConf); err != nil {
|
|
|
|
return nil, fmt.Errorf("error parsing response: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(domainConf) == 0 {
|
|
|
|
return nil, fmt.Errorf("could not get domain config: %s", domain)
|
|
|
|
}
|
|
|
|
|
|
|
|
return domainConf[0], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (hp *hostingdeProvider) createZone(domain string) error {
|
|
|
|
t, err := idna.ToASCII(domain)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
records := []*record{}
|
2022-02-22 23:54:02 +08:00
|
|
|
for _, ns := range hp.nameservers {
|
2021-03-09 08:25:55 +08:00
|
|
|
records = append(records, &record{
|
|
|
|
Name: domain,
|
|
|
|
Type: "NS",
|
|
|
|
Content: ns,
|
|
|
|
TTL: 86400,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
params := request{
|
|
|
|
ZoneConfig: &zoneConfig{
|
|
|
|
Name: t,
|
|
|
|
Type: "NATIVE",
|
|
|
|
},
|
|
|
|
Records: records,
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = hp.get("dns", "zoneCreate", params)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error creating zone: %w", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (hp *hostingdeProvider) getNameservers(domain string) ([]string, error) {
|
|
|
|
t, err := idna.ToASCII(domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
domainConf, err := hp.getDomainConfig(t)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error getting domain config: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
nss := []string{}
|
|
|
|
for _, ns := range domainConf.Nameservers {
|
|
|
|
// Currently does not support glued IP addresses
|
|
|
|
if len(ns.IPs) > 0 {
|
|
|
|
return nil, fmt.Errorf("domain %s has glued IP addresses which are not supported", domain)
|
|
|
|
}
|
|
|
|
|
|
|
|
nss = append(nss, ns.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nss, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (hp *hostingdeProvider) updateNameservers(nss []string, domain string) func() error {
|
|
|
|
return func() error {
|
|
|
|
domainConf, err := hp.getDomainConfig(domain)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
nameservers := []nameserver{}
|
|
|
|
for _, ns := range nss {
|
|
|
|
nameservers = append(nameservers, nameserver{Name: ns})
|
|
|
|
}
|
|
|
|
|
|
|
|
domainConf.Nameservers = nameservers
|
|
|
|
|
|
|
|
params := request{
|
|
|
|
Domain: domainConf,
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := hp.get("domain", "domainUpdate", params); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-28 09:28:17 +08:00
|
|
|
func (hp *hostingdeProvider) updateZone(zc *zoneConfig, options *dnsSecOptions, create, del, mod diff.Changeset) error {
|
2021-03-09 08:25:55 +08:00
|
|
|
toAdd := []*record{}
|
|
|
|
for _, c := range create {
|
|
|
|
r := recordToNative(c.Desired)
|
|
|
|
toAdd = append(toAdd, r)
|
|
|
|
}
|
|
|
|
|
|
|
|
toDelete := []*record{}
|
|
|
|
for _, d := range del {
|
|
|
|
r := recordToNative(d.Existing)
|
2023-02-23 03:26:25 +08:00
|
|
|
r.ID = d.Existing.Original.(record).ID
|
2021-03-09 08:25:55 +08:00
|
|
|
toDelete = append(toDelete, r)
|
|
|
|
}
|
|
|
|
|
|
|
|
toModify := []*record{}
|
|
|
|
for _, m := range mod {
|
|
|
|
r := recordToNative(m.Desired)
|
2023-02-23 03:26:25 +08:00
|
|
|
r.ID = m.Existing.Original.(record).ID
|
2021-03-09 08:25:55 +08:00
|
|
|
toModify = append(toModify, r)
|
|
|
|
}
|
|
|
|
|
|
|
|
params := request{
|
|
|
|
ZoneConfig: zc,
|
|
|
|
RecordsToAdd: toAdd,
|
|
|
|
RecordsToDelete: toDelete,
|
|
|
|
RecordsToModify: toModify,
|
2023-02-28 09:28:17 +08:00
|
|
|
DNSSECOptions: options,
|
2021-03-09 08:25:55 +08:00
|
|
|
}
|
|
|
|
|
2023-02-01 00:02:32 +08:00
|
|
|
_, err := hp.get("dns", "zoneUpdate", params)
|
2021-03-09 08:25:55 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-02-01 00:02:32 +08:00
|
|
|
func (hp *hostingdeProvider) getZone(domain string) (*zone, error) {
|
|
|
|
t, err := idna.ToASCII(domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
params := request{
|
|
|
|
Filter: &filter{
|
|
|
|
Field: "ZoneName",
|
|
|
|
Value: t,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := hp.get("dns", "zonesFind", params)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not get zone config: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
zones := []*zone{}
|
|
|
|
if err := json.Unmarshal(resp.Data, &zones); err != nil {
|
|
|
|
return nil, fmt.Errorf("could not parse response: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(zones) == 0 {
|
|
|
|
return nil, errZoneNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
return zones[0], nil
|
|
|
|
}
|
|
|
|
|
2021-03-09 08:25:55 +08:00
|
|
|
func (hp *hostingdeProvider) getZoneConfig(domain string) (*zoneConfig, error) {
|
|
|
|
t, err := idna.ToASCII(domain)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
params := request{
|
2023-01-31 23:22:54 +08:00
|
|
|
Filter: &filter{
|
2021-03-09 08:25:55 +08:00
|
|
|
Field: "ZoneName",
|
|
|
|
Value: t,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := hp.get("dns", "zoneConfigsFind", params)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not get zone config: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
zc := []*zoneConfig{}
|
|
|
|
if err := json.Unmarshal(resp.Data, &zc); err != nil {
|
|
|
|
return nil, fmt.Errorf("could not parse response: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(zc) == 0 {
|
|
|
|
return nil, errZoneNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
return zc[0], nil
|
|
|
|
}
|
|
|
|
|
2023-02-28 09:28:17 +08:00
|
|
|
func (hp *hostingdeProvider) getDNSSECOptions(zoneConfigID string) (*dnsSecOptions, error) {
|
2023-02-01 00:02:32 +08:00
|
|
|
params := request{
|
|
|
|
Filter: &filter{
|
|
|
|
Field: "zoneConfigId",
|
2023-02-28 09:28:17 +08:00
|
|
|
Value: zoneConfigID,
|
2023-02-01 00:02:32 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := hp.get("dns", "dnsSecOptionsFind", params)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not get dnssec options: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
dnsSecOptions := []*dnsSecOptions{}
|
|
|
|
if err := json.Unmarshal(resp.Data, &dnsSecOptions); err != nil {
|
|
|
|
return nil, fmt.Errorf("could not parse response: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(dnsSecOptions) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return dnsSecOptions[0], nil
|
|
|
|
}
|
|
|
|
|
2023-02-07 20:09:18 +08:00
|
|
|
func (hp *hostingdeProvider) dnsSecKeyModify(domain string, add []dnsSecEntry, remove []dnsSecEntry) error {
|
|
|
|
params := request{
|
|
|
|
DomainName: domain,
|
|
|
|
Add: add,
|
|
|
|
Remove: remove,
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := hp.get("domain", "dnsSecKeyModify", params)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-03-09 08:25:55 +08:00
|
|
|
func (hp *hostingdeProvider) get(service, method string, params request) (*responseData, error) {
|
|
|
|
params.AuthToken = hp.authToken
|
|
|
|
params.OwnerAccountID = hp.ownerAccountID
|
|
|
|
reqBody, err := json.Marshal(params)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not marshal request body: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
url := fmt.Sprintf(endpoint, hp.baseURL, service, method)
|
|
|
|
resp, err := http.Post(url, "application/json", bytes.NewBuffer(reqBody))
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not carry out request: %w", err)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode >= 400 {
|
|
|
|
return nil, fmt.Errorf("error occurred: %s", resp.Status)
|
|
|
|
}
|
|
|
|
|
2022-08-15 00:50:15 +08:00
|
|
|
bodyBytes, err := io.ReadAll(resp.Body)
|
2021-03-09 08:25:55 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not read response body: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
respData := &response{}
|
|
|
|
if err := json.Unmarshal(bodyBytes, &respData); err != nil {
|
|
|
|
return nil, fmt.Errorf("could not unmarshal response body: %w", err)
|
|
|
|
}
|
|
|
|
if len(respData.Errors) > 0 && respData.Status == "error" {
|
|
|
|
return nil, fmt.Errorf("%+v", respData.Errors)
|
|
|
|
}
|
|
|
|
|
|
|
|
return respData.Response, nil
|
|
|
|
}
|
2023-01-31 23:22:54 +08:00
|
|
|
|
|
|
|
func (hp *hostingdeProvider) getAllZoneConfigs() ([]*zoneConfig, error) {
|
|
|
|
params := request{
|
|
|
|
Limit: 10000,
|
|
|
|
}
|
2023-02-28 09:28:17 +08:00
|
|
|
if hp.filterAccountID != "" {
|
2023-02-01 00:03:29 +08:00
|
|
|
params.Filter = &filter{
|
|
|
|
Field: "accountId",
|
2023-02-28 09:28:17 +08:00
|
|
|
Value: hp.filterAccountID,
|
2023-02-01 00:03:29 +08:00
|
|
|
}
|
|
|
|
}
|
2023-01-31 23:22:54 +08:00
|
|
|
|
|
|
|
resp, err := hp.get("dns", "zoneConfigsFind", params)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not get zones: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
zc := []*zoneConfig{}
|
|
|
|
if err := json.Unmarshal(resp.Data, &zc); err != nil {
|
|
|
|
return nil, fmt.Errorf("could not parse response: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return zc, nil
|
|
|
|
}
|