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, }) }