This commit is contained in:
@@ -1,224 +0,0 @@
|
||||
package ionosAPI
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"io"
|
||||
"net"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"realdnydns/model/common"
|
||||
)
|
||||
|
||||
/**
|
||||
* Docs: https://developer.hosting.ionos.com/docs/dns
|
||||
*/
|
||||
type IonosAPI interface {
|
||||
GetARecord(tld string, subdomain string) (*common.ARecord, error)
|
||||
SetARecord(tld string, subdomain string, ip net.IP, ttl int, prio int, disabled bool) (*common.ARecord, error)
|
||||
GetZoneId(tld string) (string, error)
|
||||
GetRecordId(zoneId string, tld string, subdomain string, recordType string) (string, error)
|
||||
HttpCall(method string, url string, body io.Reader, queryParams map[string]string) (*http.Response, error)
|
||||
}
|
||||
|
||||
type IonosAPIImpl struct {
|
||||
APIKey string
|
||||
BaseURL string
|
||||
HTTPClient *http.Client
|
||||
}
|
||||
|
||||
type ZonesResponse []struct {
|
||||
Name string `json:"name"`
|
||||
Id string `json:"id"`
|
||||
Type string `json:"type"`
|
||||
}
|
||||
|
||||
type ZoneResponse struct {
|
||||
Id string `json:"id"`
|
||||
Name string `json:"name"`
|
||||
Type string `json:"type"`
|
||||
Records []RecordResponse `json:"records"`
|
||||
}
|
||||
|
||||
type RecordResponse struct {
|
||||
Id string `json:"id"`
|
||||
Name string `json:"name"`
|
||||
RootName string `json:"rootName"`
|
||||
Type string `json:"type"`
|
||||
Content string `json:"content"`
|
||||
ChangeDate string `json:"changeDate"`
|
||||
TTL int `json:"ttl"`
|
||||
Prio int `json:"prio"`
|
||||
Disabled bool `json:"disabled"`
|
||||
}
|
||||
|
||||
type ChangeRecord struct {
|
||||
Name string `json:"name"`
|
||||
Type string `json:"type"`
|
||||
Content string `json:"content"`
|
||||
TTL int `json:"ttl"`
|
||||
Prio int `json:"prio"`
|
||||
Disabled bool `json:"disabled"`
|
||||
}
|
||||
|
||||
type ChangeRecordRequest struct {
|
||||
Content string `json:"content"`
|
||||
TTL int `json:"ttl"`
|
||||
Prio int `json:"prio"`
|
||||
Disabled bool `json:"disabled"`
|
||||
}
|
||||
|
||||
func New(APIKey string, BaseURL string) IonosAPI {
|
||||
return &IonosAPIImpl{
|
||||
APIKey: APIKey,
|
||||
BaseURL: BaseURL,
|
||||
HTTPClient: &http.Client{},
|
||||
}
|
||||
}
|
||||
|
||||
func (i *IonosAPIImpl) HttpCall(method string, path string, body io.Reader, queryParams map[string]string) (*http.Response, error) {
|
||||
requestUrl, _ := url.Parse(i.BaseURL + path)
|
||||
|
||||
query := requestUrl.Query()
|
||||
for key, value := range queryParams {
|
||||
query.Add(key, value)
|
||||
}
|
||||
requestUrl.RawQuery = query.Encode()
|
||||
|
||||
req, err := http.NewRequest(method, requestUrl.String(), body)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
req.Header.Add("X-API-Key", i.APIKey)
|
||||
req.Header.Add("Content-Type", "application/json")
|
||||
req.Header.Add("Accept", "application/json")
|
||||
|
||||
return i.HTTPClient.Do(req)
|
||||
}
|
||||
|
||||
func (i *IonosAPIImpl) GetZoneId(tld string) (string, error) {
|
||||
res, err := i.HttpCall("GET", "/v1/zones", nil, nil)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
responseBody := make([]byte, res.ContentLength)
|
||||
res.Body.Read(responseBody)
|
||||
|
||||
zones := []ZoneResponse{}
|
||||
json.Unmarshal(responseBody, &zones)
|
||||
|
||||
for _, z := range zones {
|
||||
if z.Name == tld {
|
||||
return z.Id, nil
|
||||
}
|
||||
}
|
||||
|
||||
return "", errors.New("zone not found")
|
||||
}
|
||||
|
||||
func (i *IonosAPIImpl) GetRecordId(zoneId string, tld string, subdomain string, recordType string) (string, error) {
|
||||
var domain string
|
||||
if subdomain == "@" || subdomain == "" {
|
||||
domain = tld
|
||||
} else {
|
||||
domain = subdomain + "." + tld
|
||||
}
|
||||
|
||||
res, err := i.HttpCall("GET", "/v1/zones/"+zoneId, nil, map[string]string{"recordName": domain, "recordType": recordType})
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
responseBody := make([]byte, res.ContentLength)
|
||||
res.Body.Read(responseBody)
|
||||
|
||||
zone := ZoneResponse{}
|
||||
json.Unmarshal(responseBody, &zone)
|
||||
|
||||
for _, record := range zone.Records {
|
||||
if record.Type == recordType && record.Name == domain {
|
||||
return record.Id, nil
|
||||
}
|
||||
}
|
||||
|
||||
return "", errors.New("record not found")
|
||||
}
|
||||
|
||||
func (i *IonosAPIImpl) SetARecord(tld string, subdomain string, ip net.IP, ttl int, prio int, disabled bool) (*common.ARecord, error) {
|
||||
zoneId, err := i.GetZoneId(tld)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
recordId, err := i.GetRecordId(zoneId, tld, subdomain, "A")
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
changeRecordRequest, err := json.Marshal(ChangeRecordRequest{
|
||||
Content: ip.String(),
|
||||
TTL: ttl,
|
||||
Prio: prio,
|
||||
Disabled: false,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
res, err := i.HttpCall("PUT", "/v1/zones/"+zoneId+"/records/"+recordId, bytes.NewReader(changeRecordRequest), nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
responseBody := make([]byte, res.ContentLength)
|
||||
res.Body.Read(responseBody)
|
||||
|
||||
if res.StatusCode != 200 {
|
||||
return nil, errors.New("error updating record")
|
||||
}
|
||||
|
||||
changeRecord := ChangeRecord{}
|
||||
json.Unmarshal(responseBody, &changeRecord)
|
||||
|
||||
return &common.ARecord{
|
||||
Domain: changeRecord.Name,
|
||||
IP: changeRecord.Content,
|
||||
TTL: changeRecord.TTL,
|
||||
Prio: changeRecord.Prio,
|
||||
Disabled: changeRecord.Disabled,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func (ionos *IonosAPIImpl) GetARecord(tld string, subdomain string) (*common.ARecord, error) {
|
||||
zoneId, err := ionos.GetZoneId(tld)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
recordId, err := ionos.GetRecordId(zoneId, tld, subdomain, "A")
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
res, err := ionos.HttpCall("GET", "/v1/zones/"+zoneId+"/records/"+recordId, nil, nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
responseBody := make([]byte, res.ContentLength)
|
||||
res.Body.Read(responseBody)
|
||||
|
||||
record := RecordResponse{}
|
||||
json.Unmarshal(responseBody, &record)
|
||||
|
||||
return &common.ARecord{
|
||||
Domain: record.Name,
|
||||
IP: record.Content,
|
||||
TTL: record.TTL,
|
||||
Prio: record.Prio,
|
||||
Disabled: record.Disabled,
|
||||
}, nil
|
||||
}
|
||||
@@ -1,285 +0,0 @@
|
||||
package ionosAPI
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"net"
|
||||
"net/http"
|
||||
"net/http/httptest"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func utilMockServerImpl() func(w http.ResponseWriter, r *http.Request) {
|
||||
zonesResponse := []ZoneResponse{
|
||||
{
|
||||
Name: "example.com",
|
||||
Id: "1234567890",
|
||||
Type: "NATIVE",
|
||||
},
|
||||
{
|
||||
Name: "notTheExample.org",
|
||||
Id: "0987654321",
|
||||
Type: "SLAVE",
|
||||
},
|
||||
}
|
||||
zonesResponseJson, _ := json.Marshal(zonesResponse)
|
||||
|
||||
recordResponseSub := RecordResponse{
|
||||
Id: "abcdefghij",
|
||||
Name: "example.com",
|
||||
RootName: "example.com",
|
||||
Type: "A",
|
||||
Content: "127.0.0.1",
|
||||
ChangeDate: "2019-12-09T13:04:25.772Z",
|
||||
TTL: 300,
|
||||
Prio: 0,
|
||||
Disabled: false,
|
||||
}
|
||||
recordResponseSubJson, _ := json.Marshal(recordResponseSub)
|
||||
|
||||
recordResponseTLD := RecordResponse{
|
||||
Id: "jihgfedcba",
|
||||
Name: "sub.example.com",
|
||||
RootName: "example.com",
|
||||
Type: "A",
|
||||
Content: "127.0.0.2",
|
||||
ChangeDate: "2019-12-09T13:04:25.772Z",
|
||||
TTL: 300,
|
||||
Prio: 0,
|
||||
Disabled: false,
|
||||
}
|
||||
recordResponseTLDJson, _ := json.Marshal(recordResponseTLD)
|
||||
|
||||
zoneResponse := ZoneResponse{
|
||||
Id: "1234567890",
|
||||
Name: "example.com",
|
||||
Type: "NATIVE",
|
||||
Records: []RecordResponse{
|
||||
recordResponseSub,
|
||||
recordResponseTLD,
|
||||
},
|
||||
}
|
||||
zoneResponseJson, _ := json.Marshal(zoneResponse)
|
||||
|
||||
changeRecord := ChangeRecord{
|
||||
Name: "sub.example.com",
|
||||
Type: "A",
|
||||
Content: "127.0.0.1",
|
||||
TTL: 300,
|
||||
Prio: 0,
|
||||
Disabled: false,
|
||||
}
|
||||
changeRecordJson, _ := json.Marshal(changeRecord)
|
||||
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
var response []byte
|
||||
|
||||
if r.Method == "GET" {
|
||||
if r.RequestURI == "/v1/zones" {
|
||||
response = zonesResponseJson
|
||||
} else if r.RequestURI == "/v1/zones/1234567890?recordName=example.com&recordType=A" ||
|
||||
r.RequestURI == "/v1/zones/1234567890?recordName=sub.example.com&recordType=A" {
|
||||
response = zoneResponseJson
|
||||
} else if r.RequestURI == "/v1/zones/1234567890/records/abcdefghij" {
|
||||
response = recordResponseSubJson
|
||||
} else if r.RequestURI == "/v1/zones/1234567890/records/jihgfedcba" {
|
||||
response = recordResponseTLDJson
|
||||
}
|
||||
} else if r.Method == "PUT" {
|
||||
response = changeRecordJson
|
||||
} else {
|
||||
response = []byte("404")
|
||||
}
|
||||
|
||||
w.Write(response)
|
||||
}
|
||||
}
|
||||
|
||||
func TestHttpCall(t *testing.T) {
|
||||
mockServer := httptest.NewServer(http.HandlerFunc(utilMockServerImpl()))
|
||||
defer mockServer.Close()
|
||||
|
||||
ionosAPI := New("dummyKey", mockServer.URL)
|
||||
|
||||
t.Run("returns response for GET request", testHttpCallGet(ionosAPI))
|
||||
t.Run("returns response for PUT request", testHttpCallPut(ionosAPI))
|
||||
t.Run("returns error for non existing endpoint", testHttpCallNonExistingEndpoint())
|
||||
}
|
||||
|
||||
func testHttpCallGet(api IonosAPI) func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
res, err := api.HttpCall("GET", "/v1/zones", nil, nil)
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("HttpCall() returned unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if res.StatusCode != 200 {
|
||||
t.Fatalf("HttpCall() returned unexpected status code: %v instead of 200", res.StatusCode)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func testHttpCallPut(api IonosAPI) func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
res, err := api.HttpCall("PUT", "/v1/zones/1234567890/records/abcdefghij", nil, nil)
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("HttpCall() returned unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if res.StatusCode != 200 {
|
||||
t.Fatalf("HttpCall() returned unexpected status code: %v instead of 200", res.StatusCode)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func testHttpCallNonExistingEndpoint() func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
mockServer := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||
w.WriteHeader(404)
|
||||
}))
|
||||
defer mockServer.Close()
|
||||
|
||||
api := New("dummyKey", mockServer.URL)
|
||||
|
||||
res, err := api.HttpCall("GET", "/v1/nonExistingEndpoint", nil, nil)
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("HttpCall() returned unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if res.StatusCode != 404 {
|
||||
t.Fatalf("HttpCall() returned unexpected status code: %v instead of 404", res.StatusCode)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestGetZoneId(t *testing.T) {
|
||||
mockServer := httptest.NewServer(http.HandlerFunc(utilMockServerImpl()))
|
||||
|
||||
ionosAPI := New("dummyKey", mockServer.URL)
|
||||
|
||||
t.Run("returns zoneId for tracked domain", testGetZoneIdSuccess(ionosAPI))
|
||||
t.Run("returns error for non tracked domain", testGetZoneIdNoMatch(ionosAPI))
|
||||
}
|
||||
|
||||
func testGetZoneIdSuccess(api IonosAPI) func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
zoneId, err := api.GetZoneId("example.com")
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("GetZoneId() returned unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if zoneId != "1234567890" {
|
||||
t.Fatalf("GetZoneId() returned unexpected zoneId: %v instead of 1234567890", zoneId)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func testGetZoneIdNoMatch(api IonosAPI) func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
zoneId, err := api.GetZoneId("nonTrackedDomain.com")
|
||||
|
||||
if err == nil || zoneId != "" {
|
||||
t.Fatalf("GetZoneId() did not return an error for a non tracked domain")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestGetRecordId(t *testing.T) {
|
||||
mockServer := httptest.NewServer(http.HandlerFunc(utilMockServerImpl()))
|
||||
defer mockServer.Close()
|
||||
|
||||
ionosAPI := New("dummyKey", mockServer.URL)
|
||||
|
||||
t.Run("returns zoneId for top level domain", testGetRecordIdForTopLevelDomain(ionosAPI))
|
||||
t.Run("returns zoneId for subdomain", testGetRecordIdForSubdomain(ionosAPI))
|
||||
t.Run("returns error for untracked subdomain", testGetRecordIdWithUntrackedSubdomain(ionosAPI))
|
||||
}
|
||||
|
||||
func testGetRecordIdForTopLevelDomain(api IonosAPI) func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
recordId, err := api.GetRecordId("1234567890", "example.com", "", "A")
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("GetZoneId() returned unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if recordId != "abcdefghij" {
|
||||
t.Fatalf("GetZoneId() returned unexpected zoneId: %v instead of abcdefghij", recordId)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func testGetRecordIdForSubdomain(api IonosAPI) func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
recordId, err := api.GetRecordId("1234567890", "example.com", "sub", "A")
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("GetZoneId() returned unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if recordId != "jihgfedcba" {
|
||||
t.Fatalf("GetZoneId() returned unexpected zoneId: %v instead of jihgfedcba", recordId)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func testGetRecordIdWithUntrackedSubdomain(api IonosAPI) func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
recordId, err := api.GetRecordId("1234567890", "example.com", "untrackedSub", "A")
|
||||
|
||||
if err == nil && recordId == "" {
|
||||
t.Fatalf("GetZoneId() did not return an error for a non tracked domain")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestGetARecord(t *testing.T) {
|
||||
mockServer := httptest.NewServer(http.HandlerFunc(utilMockServerImpl()))
|
||||
defer mockServer.Close()
|
||||
|
||||
ionosAPI := New("dummyKey", mockServer.URL)
|
||||
|
||||
t.Run("returns A record for top level domain", testGetARecordFor(ionosAPI, "example.com", ""))
|
||||
t.Run("returns A record for subdomain", testGetARecordFor(ionosAPI, "example.com", "sub"))
|
||||
}
|
||||
|
||||
func testGetARecordFor(api IonosAPI, tld string, subdomain string) func(t *testing.T) {
|
||||
return func(t *testing.T) {
|
||||
var domain string
|
||||
if subdomain == "" {
|
||||
domain = tld
|
||||
} else {
|
||||
domain = subdomain + "." + tld
|
||||
}
|
||||
|
||||
record, err := api.GetARecord(tld, subdomain)
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("GetARecord() returned unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if record.Domain != domain {
|
||||
t.Fatalf("GetARecord() returned unexpected record for: %v instead of %v", record.Domain, domain)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetARecord(t *testing.T) {
|
||||
mockServer := httptest.NewServer(http.HandlerFunc(utilMockServerImpl()))
|
||||
defer mockServer.Close()
|
||||
|
||||
ionosAPI := New("dummyKey", mockServer.URL)
|
||||
|
||||
changedARecord, err := ionosAPI.SetARecord("example.com", "sub", net.ParseIP("127.0.0.1"), 300, 0, false)
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("SetARecord() returned unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if changedARecord.Domain != "sub.example.com" {
|
||||
t.Fatalf("SetARecord() returned unexpected record for: %v instead of sub.example.com", changedARecord.Domain)
|
||||
}
|
||||
}
|
||||
@@ -1,20 +1,30 @@
|
||||
package ionosDnsProvider
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"net"
|
||||
"net/http"
|
||||
"realdnydns/model/common"
|
||||
"realdnydns/pkg/dnsProvider"
|
||||
ionosAPI "realdnydns/pkg/dnsProvider/ionos/api"
|
||||
"sync"
|
||||
|
||||
ionosDnsClient "gitea.t000-n.de/t.behrendt/ionosDnsClient"
|
||||
)
|
||||
|
||||
type IONOS struct {
|
||||
API ionosAPI.IonosAPI
|
||||
client *ionosDnsClient.ClientWithResponses
|
||||
zoneIdMap map[string]string
|
||||
defaultTtl int
|
||||
defaultPrio int
|
||||
zoneIdMapMutex sync.Mutex
|
||||
}
|
||||
|
||||
type IONOSConfig struct {
|
||||
APIKey string `yaml:"api_key"`
|
||||
BaseURL string `yaml:"base_url"`
|
||||
APIKey string `yaml:"api_key"`
|
||||
BaseURL string `yaml:"base_url"`
|
||||
DefaultTTL *int `yaml:"default_ttl,omitempty"`
|
||||
DefaultPrio *int `yaml:"default_prio,omitempty"`
|
||||
}
|
||||
|
||||
func NewIonos(config *IONOSConfig) (dnsProvider.DNSProvider, error) {
|
||||
@@ -26,15 +36,231 @@ func NewIonos(config *IONOSConfig) (dnsProvider.DNSProvider, error) {
|
||||
return nil, errors.New("base_url is required")
|
||||
}
|
||||
|
||||
options := []ionosDnsClient.ClientOption{
|
||||
ionosDnsClient.WithRequestEditorFn(func(ctx context.Context, req *http.Request) error {
|
||||
req.Header.Set("X-API-Key", config.APIKey)
|
||||
return nil
|
||||
}),
|
||||
}
|
||||
|
||||
client, err := ionosDnsClient.NewClientWithResponses(config.BaseURL, options...)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
// Set default values for TTL and Prio if not provided
|
||||
defaultTtl := 300
|
||||
if config.DefaultTTL != nil {
|
||||
defaultTtl = *config.DefaultTTL
|
||||
}
|
||||
|
||||
defaultPrio := 0
|
||||
if config.DefaultPrio != nil {
|
||||
defaultPrio = *config.DefaultPrio
|
||||
}
|
||||
|
||||
return &IONOS{
|
||||
ionosAPI.New(config.APIKey, config.BaseURL),
|
||||
client: client,
|
||||
zoneIdMap: make(map[string]string),
|
||||
defaultTtl: defaultTtl,
|
||||
defaultPrio: defaultPrio,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func (ionos *IONOS) UpdateRecord(tld string, subdomain string, ip net.IP, ttl int, prio int, disabled bool) (*common.ARecord, error) {
|
||||
return ionos.API.SetARecord(tld, subdomain, ip, ttl, prio, disabled)
|
||||
func (i *IONOS) UpdateRecord(tld string, subdomain string, ip net.IP, ttl int, prio int, disabled bool) (*common.ARecord, error) {
|
||||
|
||||
zoneId, err := i.getZoneIdForTld(tld)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
domain := assembleTldSubdomainToDomain(tld, subdomain)
|
||||
recordId, err := i.getRecordIdForDomain(zoneId, domain)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
ipString := ip.String()
|
||||
res, err := i.client.UpdateRecordWithResponse(context.Background(), zoneId, recordId, ionosDnsClient.RecordUpdate{
|
||||
Content: &ipString,
|
||||
Ttl: &ttl,
|
||||
Prio: &prio,
|
||||
Disabled: &disabled,
|
||||
})
|
||||
if err != nil || res.StatusCode() != 200 {
|
||||
return nil, errors.New("failed to update record")
|
||||
}
|
||||
|
||||
if res.JSON200 == nil {
|
||||
return nil, errors.New("record response is empty")
|
||||
}
|
||||
|
||||
record := *res.JSON200
|
||||
if record.Name == nil || record.Content == nil {
|
||||
return nil, errors.New("record is empty")
|
||||
}
|
||||
|
||||
// Handle optional fields with defaults
|
||||
recordTtl := 0
|
||||
if record.Ttl != nil {
|
||||
recordTtl = *record.Ttl
|
||||
}
|
||||
|
||||
recordPrio := 0
|
||||
if record.Prio != nil {
|
||||
recordPrio = *record.Prio
|
||||
}
|
||||
|
||||
recordDisabled := false
|
||||
if record.Disabled != nil {
|
||||
recordDisabled = *record.Disabled
|
||||
}
|
||||
|
||||
return &common.ARecord{
|
||||
Domain: *record.Name,
|
||||
IP: *record.Content,
|
||||
TTL: recordTtl,
|
||||
Prio: recordPrio,
|
||||
Disabled: recordDisabled,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func (ionos *IONOS) GetRecord(tld string, subdomain string) (*common.ARecord, error) {
|
||||
return ionos.API.GetARecord(tld, subdomain)
|
||||
func (i *IONOS) getZoneIdForTld(tld string) (string, error) {
|
||||
i.zoneIdMapMutex.Lock()
|
||||
defer i.zoneIdMapMutex.Unlock()
|
||||
if zoneId, ok := i.zoneIdMap[tld]; ok {
|
||||
return zoneId, nil
|
||||
}
|
||||
|
||||
res, err := i.client.GetZonesWithResponse(context.Background())
|
||||
if err != nil {
|
||||
return "", err
|
||||
} else if res.StatusCode() != 200 {
|
||||
return "", errors.New("failed to get zones")
|
||||
}
|
||||
|
||||
if res.JSON200 == nil {
|
||||
return "", errors.New("zones response is empty")
|
||||
}
|
||||
|
||||
zones := *res.JSON200
|
||||
|
||||
for _, zone := range zones {
|
||||
if *zone.Name == tld {
|
||||
if zone.Id == nil || *zone.Id == "" {
|
||||
return "", errors.New("zone id is empty")
|
||||
}
|
||||
|
||||
i.zoneIdMap[tld] = *zone.Id
|
||||
return *zone.Id, nil
|
||||
}
|
||||
}
|
||||
|
||||
return "", errors.New("no zone found")
|
||||
}
|
||||
|
||||
var recordTypeA = "A"
|
||||
|
||||
func assembleTldSubdomainToDomain(tld string, subdomain string) string {
|
||||
if subdomain == "@" || subdomain == "" {
|
||||
return tld
|
||||
}
|
||||
return subdomain + "." + tld
|
||||
}
|
||||
|
||||
func (i *IONOS) getRecordIdForDomain(zoneId string, domain string) (string, error) {
|
||||
res, err := i.client.GetZoneWithResponse(context.Background(), zoneId, &ionosDnsClient.GetZoneParams{
|
||||
RecordName: &domain,
|
||||
RecordType: &recordTypeA,
|
||||
})
|
||||
if err != nil || res.StatusCode() != 200 {
|
||||
return "", errors.New("failed to get zone")
|
||||
}
|
||||
|
||||
if res.JSON200 == nil {
|
||||
return "", errors.New("zone response is empty")
|
||||
}
|
||||
|
||||
zone := *res.JSON200
|
||||
|
||||
for _, record := range *zone.Records {
|
||||
if *record.Name == domain && *record.Type == ionosDnsClient.RecordTypes("A") {
|
||||
return *record.Id, nil
|
||||
}
|
||||
}
|
||||
|
||||
return "", errors.New("no record found")
|
||||
}
|
||||
|
||||
func (i *IONOS) GetRecord(tld string, subdomain string) (*common.ARecord, error) {
|
||||
zoneId, err := i.getZoneIdForTld(tld)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
domain := assembleTldSubdomainToDomain(tld, subdomain)
|
||||
recordId, err := i.getRecordIdForDomain(zoneId, domain)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
res, err := i.client.GetRecordWithResponse(context.Background(), zoneId, recordId)
|
||||
if err != nil || res.StatusCode() != 200 || res.JSON200 == nil {
|
||||
return nil, errors.New("failed to get record")
|
||||
}
|
||||
|
||||
record := *res.JSON200
|
||||
|
||||
if record.Name == nil || *record.Name == "" || *record.Name != domain {
|
||||
return nil, errors.New("record name does not match or is empty")
|
||||
}
|
||||
|
||||
if record.Content == nil || *record.Content == "" {
|
||||
return nil, errors.New("record content is empty")
|
||||
}
|
||||
|
||||
needsUpdate := false
|
||||
|
||||
ttl := i.defaultTtl
|
||||
if record.Ttl != nil {
|
||||
ttl = *record.Ttl
|
||||
} else {
|
||||
needsUpdate = true
|
||||
}
|
||||
|
||||
prio := i.defaultPrio
|
||||
if record.Prio != nil {
|
||||
prio = *record.Prio
|
||||
} else {
|
||||
needsUpdate = true
|
||||
}
|
||||
|
||||
disabled := false
|
||||
if record.Disabled != nil {
|
||||
disabled = *record.Disabled
|
||||
} else {
|
||||
needsUpdate = true
|
||||
}
|
||||
|
||||
// realDynDns requires every field to be set, so we need to update the record if any of the fields are missing
|
||||
if needsUpdate {
|
||||
ip := net.ParseIP(*record.Content)
|
||||
if ip == nil {
|
||||
return nil, errors.New("invalid IP address in record content")
|
||||
}
|
||||
|
||||
updatedRecord, err := i.UpdateRecord(tld, subdomain, ip, ttl, prio, disabled)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return updatedRecord, nil
|
||||
}
|
||||
|
||||
return &common.ARecord{
|
||||
Domain: *record.Name,
|
||||
IP: *record.Content,
|
||||
TTL: ttl,
|
||||
Prio: prio,
|
||||
Disabled: disabled,
|
||||
}, nil
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user