Files
pdns-admin/internal/pdns/client_test.go
2026-06-18 22:32:42 -03:00

358 lines
9.9 KiB
Go

package pdns
import (
"context"
"encoding/json"
"net/http"
"net/http/httptest"
"strconv"
"strings"
"testing"
)
func TestListZonesSendsAPIKey(t *testing.T) {
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if r.URL.Path != "/api/v1/servers/localhost/zones" {
t.Fatalf("unexpected path: %s", r.URL.Path)
}
if got := r.Header.Get("X-API-Key"); got != "secret" {
t.Fatalf("unexpected api key: %q", got)
}
_ = json.NewEncoder(w).Encode([]Zone{{ID: "example.org.", Name: "example.org."}})
}))
defer server.Close()
client := NewClient(server.URL, "secret", "localhost", server.Client())
zones, err := client.ListZones(context.Background())
if err != nil {
t.Fatalf("ListZones returned error: %v", err)
}
if len(zones) != 1 || zones[0].ID != "example.org." {
t.Fatalf("unexpected zones: %#v", zones)
}
}
func TestCreateRRSetPatchesZone(t *testing.T) {
var patchCount int
var getCount int
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if !strings.HasPrefix(r.URL.Path, "/api/v1/servers/localhost/zones/example.org.") {
t.Fatalf("unexpected path: %s", r.URL.Path)
}
switch r.Method {
case http.MethodGet:
getCount++
serial := uint64(10)
if getCount == 2 {
serial = 11
}
writeZone(t, w, serial)
case http.MethodPatch:
patchCount++
var payload struct {
RRSets []changeRRSet `json:"rrsets"`
}
if err := json.NewDecoder(r.Body).Decode(&payload); err != nil {
t.Fatalf("decode request: %v", err)
}
if len(payload.RRSets) != 1 || payload.RRSets[0].ChangeType != "REPLACE" {
t.Fatalf("unexpected payload: %#v", payload)
}
w.WriteHeader(http.StatusNoContent)
default:
t.Fatalf("unexpected method: %s", r.Method)
}
}))
defer server.Close()
client := NewClient(server.URL, "secret", "localhost", server.Client())
err := client.CreateRRSet(context.Background(), "example.org.", RRSet{
Name: "www.example.org.",
Type: "A",
TTL: 300,
Records: []Record{{
Content: "192.0.2.10",
}},
})
if err != nil {
t.Fatalf("CreateRRSet returned error: %v", err)
}
if patchCount != 1 {
t.Fatalf("expected one patch when serial increases, got %d", patchCount)
}
}
func TestCreateRRSetBumpsSOAWhenSerialDoesNotIncrease(t *testing.T) {
var patchCount int
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if !strings.HasPrefix(r.URL.Path, "/api/v1/servers/localhost/zones/example.org.") {
t.Fatalf("unexpected path: %s", r.URL.Path)
}
switch r.Method {
case http.MethodGet:
writeZone(t, w, 10)
case http.MethodPatch:
patchCount++
var payload struct {
RRSets []changeRRSet `json:"rrsets"`
}
if err := json.NewDecoder(r.Body).Decode(&payload); err != nil {
t.Fatalf("decode request: %v", err)
}
if len(payload.RRSets) != 1 {
t.Fatalf("unexpected payload: %#v", payload)
}
if patchCount == 2 {
rrset := payload.RRSets[0]
if rrset.Type != "SOA" {
t.Fatalf("expected SOA fallback patch, got %#v", rrset)
}
if got := rrset.Records[0].Content; !strings.Contains(got, " 11 ") {
t.Fatalf("expected bumped SOA serial, got %q", got)
}
}
w.WriteHeader(http.StatusNoContent)
default:
t.Fatalf("unexpected method: %s", r.Method)
}
}))
defer server.Close()
client := NewClient(server.URL, "secret", "localhost", server.Client())
err := client.CreateRRSet(context.Background(), "example.org.", RRSet{
Name: "www.example.org.",
Type: "A",
TTL: 300,
Records: []Record{{
Content: "192.0.2.10",
}},
})
if err != nil {
t.Fatalf("CreateRRSet returned error: %v", err)
}
if patchCount != 2 {
t.Fatalf("expected requested patch plus SOA fallback patch, got %d", patchCount)
}
}
func TestCreateRRSetMergesMultiValueRecordTypes(t *testing.T) {
var patched RRSet
var getCount int
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if !strings.HasPrefix(r.URL.Path, "/api/v1/servers/localhost/zones/example.org.") {
t.Fatalf("unexpected path: %s", r.URL.Path)
}
switch r.Method {
case http.MethodGet:
getCount++
writeZoneWithRRSets(t, w, 10, []RRSet{{
Name: "example.org.",
Type: "NS",
TTL: 3600,
Records: []Record{{
Content: "ns1.example.org.",
}},
}})
case http.MethodPatch:
var payload struct {
RRSets []changeRRSet `json:"rrsets"`
}
if err := json.NewDecoder(r.Body).Decode(&payload); err != nil {
t.Fatalf("decode request: %v", err)
}
if payload.RRSets[0].Type == "NS" {
patched = RRSet{
Name: payload.RRSets[0].Name,
Type: payload.RRSets[0].Type,
TTL: payload.RRSets[0].TTL,
Records: payload.RRSets[0].Records,
}
}
w.WriteHeader(http.StatusNoContent)
default:
t.Fatalf("unexpected method: %s", r.Method)
}
}))
defer server.Close()
client := NewClient(server.URL, "secret", "localhost", server.Client())
err := client.CreateRRSet(context.Background(), "example.org.", RRSet{
Name: "example.org.",
Type: "NS",
TTL: 3600,
Records: []Record{{
Content: "ns2.example.org.",
}},
})
if err != nil {
t.Fatalf("CreateRRSet returned error: %v", err)
}
if getCount == 0 {
t.Fatal("expected zone read before merge")
}
if len(patched.Records) != 2 {
t.Fatalf("expected merged records, got %#v", patched.Records)
}
if patched.Records[0].Content != "ns1.example.org." || patched.Records[1].Content != "ns2.example.org." {
t.Fatalf("unexpected merged records: %#v", patched.Records)
}
}
func TestCreateRRSetDoesNotMergeSingleValueRecordTypes(t *testing.T) {
var patched RRSet
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if !strings.HasPrefix(r.URL.Path, "/api/v1/servers/localhost/zones/www.example.org.") {
t.Fatalf("unexpected path: %s", r.URL.Path)
}
switch r.Method {
case http.MethodGet:
writeZoneWithRRSets(t, w, 11, []RRSet{{
Name: "www.example.org.",
Type: "CNAME",
TTL: 3600,
Records: []Record{{
Content: "old.example.org.",
}},
}})
case http.MethodPatch:
var payload struct {
RRSets []changeRRSet `json:"rrsets"`
}
if err := json.NewDecoder(r.Body).Decode(&payload); err != nil {
t.Fatalf("decode request: %v", err)
}
if payload.RRSets[0].Type == "CNAME" {
patched = RRSet{
Name: payload.RRSets[0].Name,
Type: payload.RRSets[0].Type,
TTL: payload.RRSets[0].TTL,
Records: payload.RRSets[0].Records,
}
}
w.WriteHeader(http.StatusNoContent)
default:
t.Fatalf("unexpected method: %s", r.Method)
}
}))
defer server.Close()
client := NewClient(server.URL, "secret", "localhost", server.Client())
err := client.CreateRRSet(context.Background(), "www.example.org.", RRSet{
Name: "www.example.org.",
Type: "CNAME",
TTL: 3600,
Records: []Record{{
Content: "new.example.org.",
}},
})
if err != nil {
t.Fatalf("CreateRRSet returned error: %v", err)
}
if len(patched.Records) != 1 || patched.Records[0].Content != "new.example.org." {
t.Fatalf("expected replacement record only, got %#v", patched.Records)
}
}
func TestGetServerUsesConfiguredServerID(t *testing.T) {
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if r.URL.Path != "/api/v1/servers/localhost" {
t.Fatalf("unexpected path: %s", r.URL.Path)
}
_ = json.NewEncoder(w).Encode(Server{ID: "localhost", Version: "5.0.0"})
}))
defer server.Close()
client := NewClient(server.URL, "secret", "localhost", server.Client())
got, err := client.GetServer(context.Background())
if err != nil {
t.Fatalf("GetServer returned error: %v", err)
}
if got.ID != "localhost" {
t.Fatalf("unexpected server: %#v", got)
}
}
func TestCreateZonePostsZone(t *testing.T) {
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
t.Fatalf("unexpected method: %s", r.Method)
}
if r.URL.Path != "/api/v1/servers/localhost/zones" {
t.Fatalf("unexpected path: %s", r.URL.Path)
}
var payload Zone
if err := json.NewDecoder(r.Body).Decode(&payload); err != nil {
t.Fatalf("decode request: %v", err)
}
if payload.Name != "example.org." || payload.Kind != "Native" {
t.Fatalf("unexpected payload: %#v", payload)
}
w.WriteHeader(http.StatusCreated)
_ = json.NewEncoder(w).Encode(payload)
}))
defer server.Close()
client := NewClient(server.URL, "secret", "localhost", server.Client())
created, err := client.CreateZone(context.Background(), Zone{Name: "example.org.", Kind: "Native"})
if err != nil {
t.Fatalf("CreateZone returned error: %v", err)
}
if created.Name != "example.org." {
t.Fatalf("unexpected zone: %#v", created)
}
}
func TestDeleteZoneDeletesZone(t *testing.T) {
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodDelete {
t.Fatalf("unexpected method: %s", r.Method)
}
if r.URL.Path != "/api/v1/servers/localhost/zones/example.org." {
t.Fatalf("unexpected path: %s", r.URL.Path)
}
w.WriteHeader(http.StatusNoContent)
}))
defer server.Close()
client := NewClient(server.URL, "secret", "localhost", server.Client())
if err := client.DeleteZone(context.Background(), "example.org."); err != nil {
t.Fatalf("DeleteZone returned error: %v", err)
}
}
func writeZone(t *testing.T, w http.ResponseWriter, serial uint64) {
t.Helper()
writeZoneWithRRSets(t, w, serial, nil)
}
func writeZoneWithRRSets(t *testing.T, w http.ResponseWriter, serial uint64, rrsets []RRSet) {
t.Helper()
allRRSets := []RRSet{{
Name: "example.org.",
Type: "SOA",
TTL: 3600,
Records: []Record{{
Content: "ns1.example.org. hostmaster.example.org. " + strconv.FormatUint(serial, 10) + " 3600 600 604800 300",
}},
}}
allRRSets = append(allRRSets, rrsets...)
_ = json.NewEncoder(w).Encode(Zone{
ID: "example.org.",
Name: "example.org.",
Serial: serial,
RRSets: allRRSets,
})
}