From d57738d2b42f4c70fd5338a080bc1e8846084ccd Mon Sep 17 00:00:00 2001 From: Kevin Zander Date: Thu, 29 Aug 2024 10:15:48 -0500 Subject: [PATCH] Feature: Add function to zones API to add multiple ResourceRecordSets at once (#23) * tests: use docker compose v2, update pdns-auth version * apis/zones: add function to add multiple ResourceRecordSets * apis/zones: deprecate AddRecordSetToZone --- apis/zones/interface.go | 6 +++ apis/zones/zones_addrecordset.go | 15 +++++--- client_test.go | 64 ++++++++++++++++++++++++++------ docker-compose.yml | 5 +-- 4 files changed, 71 insertions(+), 19 deletions(-) diff --git a/apis/zones/interface.go b/apis/zones/interface.go index 1ea785b..8636459 100644 --- a/apis/zones/interface.go +++ b/apis/zones/interface.go @@ -22,8 +22,14 @@ type Client interface { // AddRecordSetToZone will add a new set of records to a zone. Existing record sets for // the exact name/type combination will be replaced. + // + // Deprecated: Superceded by AddRecordSetsToZone AddRecordSetToZone(ctx context.Context, serverID string, zoneID string, set ResourceRecordSet) error + // AddRecordSetsToZone will add new sets of records to a zone. Existing record sets for + // the exact name/type combination will be replaced. + AddRecordSetsToZone(ctx context.Context, serverID string, zoneID string, sets []ResourceRecordSet) error + // RemoveRecordSetFromZone removes a record set from a zone. The record set is matched // by name and type. RemoveRecordSetFromZone(ctx context.Context, serverID string, zoneID string, name string, recordType string) error diff --git a/apis/zones/zones_addrecordset.go b/apis/zones/zones_addrecordset.go index 0982f0c..f12d47c 100644 --- a/apis/zones/zones_addrecordset.go +++ b/apis/zones/zones_addrecordset.go @@ -3,18 +3,23 @@ package zones import ( "context" "fmt" - "github.com/mittwald/go-powerdns/pdnshttp" "net/url" + + "github.com/mittwald/go-powerdns/pdnshttp" ) func (c *client) AddRecordSetToZone(ctx context.Context, serverID string, zoneID string, set ResourceRecordSet) error { + return c.AddRecordSetsToZone(ctx, serverID, zoneID, []ResourceRecordSet{set}) +} + +func (c *client) AddRecordSetsToZone(ctx context.Context, serverID string, zoneID string, sets []ResourceRecordSet) error { path := fmt.Sprintf("/servers/%s/zones/%s", url.PathEscape(serverID), url.PathEscape(zoneID)) - set.ChangeType = ChangeTypeReplace + for idx := range sets { + sets[idx].ChangeType = ChangeTypeReplace + } patch := Zone{ - ResourceRecordSets: []ResourceRecordSet{ - set, - }, + ResourceRecordSets: sets, } return c.httpClient.Patch(ctx, path, nil, pdnshttp.WithJSONRequestBody(&patch)) diff --git a/client_test.go b/client_test.go index 7bd6ffe..87a27b2 100644 --- a/client_test.go +++ b/client_test.go @@ -4,16 +4,17 @@ import ( "context" "flag" "fmt" - "github.com/mittwald/go-powerdns/apis/search" - "github.com/mittwald/go-powerdns/apis/zones" - "github.com/mittwald/go-powerdns/pdnshttp" - "github.com/stretchr/testify/assert" - "github.com/stretchr/testify/require" "io" "os" "os/exec" "testing" "time" + + "github.com/mittwald/go-powerdns/apis/search" + "github.com/mittwald/go-powerdns/apis/zones" + "github.com/mittwald/go-powerdns/pdnshttp" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" ) func TestMain(m *testing.M) { @@ -24,12 +25,12 @@ func TestMain(m *testing.M) { os.Exit(0) } - runOrPanic("docker-compose", "rm", "-sfv") - runOrPanic("docker-compose", "down", "-v") - runOrPanic("docker-compose", "up", "-d") + runOrPanic("docker", "compose", "rm", "-sfv") + runOrPanic("docker", "compose", "down", "-v") + runOrPanic("docker", "compose", "up", "-d") defer func() { - runOrPanic("docker-compose", "down", "-v") + runOrPanic("docker", "compose", "down", "-v") }() c, err := New( @@ -53,7 +54,7 @@ func TestMain(m *testing.M) { fmt.Println("Leaving containers running for further inspection") fmt.Println("") } else { - runOrPanic("docker-compose", "down", "-v") + runOrPanic("docker", "compose", "down", "-v") } os.Exit(e) @@ -245,6 +246,47 @@ func TestAddRecordToZone(t *testing.T) { require.NotNil(t, rs) } +func TestAddRecordSetsToZone(t *testing.T) { + c := buildClient(t) + + zone := zones.Zone{ + Name: "example6.de.", + Type: zones.ZoneTypeZone, + Kind: zones.ZoneKindNative, + Nameservers: []string{ + "ns1.example.com.", + "ns2.example.com.", + }, + ResourceRecordSets: []zones.ResourceRecordSet{ + {Name: "foo.example6.de.", Type: "A", TTL: 60, Records: []zones.Record{{Content: "127.0.0.1"}}}, + }, + } + + ctx, cancel := context.WithTimeout(context.Background(), 15*time.Second) + defer cancel() + created, err := c.Zones().CreateZone(ctx, "localhost", zone) + + require.Nil(t, err, "CreateZone returned error") + + err = c.Zones().AddRecordSetsToZone(ctx, "localhost", created.ID, + []zones.ResourceRecordSet{ + {Name: "bar.example6.de.", Type: "A", TTL: 60, Records: []zones.Record{{Content: "127.0.0.2"}}}, + {Name: "baz.example6.de.", Type: "A", TTL: 60, Records: []zones.Record{{Content: "127.0.0.3"}}}, + }, + ) + + require.Nil(t, err, "AddRecordSetsToZone returned error") + + updated, err := c.Zones().GetZone(ctx, "localhost", created.ID) + + require.Nil(t, err) + + rs := updated.GetRecordSet("bar.example6.de.", "A") + require.NotNil(t, rs) + rs = updated.GetRecordSet("baz.example6.de.", "A") + require.NotNil(t, rs) +} + func TestSelectZoneWithoutRRSets(t *testing.T) { c := buildClient(t) @@ -410,7 +452,7 @@ func TestExportZone(t *testing.T) { export, sErr := c.Zones().ExportZone(ctx, "localhost", created.ID) - date := time.Now().Format("20060102") + "01" + date := time.Now().UTC().Format("20060102") + "01" require.Nil(t, sErr) require.Equal(t, "example-export.de.\t60\tIN\tA\t127.0.0.1\nexample-export.de.\t3600\tIN\tNS\tns1.example.com.\nexample-export.de.\t3600\tIN\tNS\tns2.example.com.\nexample-export.de.\t3600\tIN\tSOA\ta.misconfigured.dns.server.invalid. hostmaster.example-export.de. "+date+" 10800 3600 604800 3600\n", string(export)) diff --git a/docker-compose.yml b/docker-compose.yml index 65c8002..de9d6ef 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,10 +1,9 @@ -version: "3" services: powerdns: - image: powerdns/pdns-auth-46:4.6.3 + image: powerdns/pdns-auth-49:4.9.1 environment: PDNS_AUTH_API_KEY: secret ports: - 8081:8081 - 8053:53/udp - - 8053:53/tcp \ No newline at end of file + - 8053:53/tcp