package godo import ( "encoding/json" "fmt" "net/http" "reflect" "strconv" "testing" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) func TestDomains_ListDomains(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, `{ "domains": [ { "name":"foo.com" }, { "name":"bar.com" } ], "meta": { "total": 2 } }`) }) domains, resp, err := client.Domains.List(ctx, nil) if err != nil { t.Errorf("Domains.List returned error: %v", err) } expectedDomains := []Domain{{Name: "foo.com"}, {Name: "bar.com"}} if !reflect.DeepEqual(domains, expectedDomains) { t.Errorf("Domains.List returned domains %+v, expected %+v", domains, expectedDomains) } expectedMeta := &Meta{Total: 2} if !reflect.DeepEqual(resp.Meta, expectedMeta) { t.Errorf("Domains.List returned meta %+v, expected %+v", resp.Meta, expectedMeta) } } func TestDomains_ListDomainsMultiplePages(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, `{"domains": [{"id":1},{"id":2}], "links":{"pages":{"next":"http://example.com/v2/domains/?page=2"}}}`) }) _, resp, err := client.Domains.List(ctx, nil) if err != nil { t.Fatal(err) } checkCurrentPage(t, resp, 1) } func TestDomains_RetrievePageByNumber(t *testing.T) { setup() defer teardown() jBlob := ` { "domains": [{"id":1},{"id":2}], "links":{ "pages":{ "next":"http://example.com/v2/domains/?page=3", "prev":"http://example.com/v2/domains/?page=1", "last":"http://example.com/v2/domains/?page=3", "first":"http://example.com/v2/domains/?page=1" } }, "meta":{ "total":2 } }` mux.HandleFunc("/v2/domains", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, jBlob) }) opt := &ListOptions{Page: 2} _, resp, err := client.Domains.List(ctx, opt) if err != nil { t.Fatal(err) } checkCurrentPage(t, resp, 2) } func TestDomains_GetDomain(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, `{"domain":{"name":"example.com"}}`) }) domains, _, err := client.Domains.Get(ctx, "example.com") if err != nil { t.Errorf("domain.Get returned error: %v", err) } expected := &Domain{Name: "example.com"} if !reflect.DeepEqual(domains, expected) { t.Errorf("domains.Get returned %+v, expected %+v", domains, expected) } } func TestDomains_Create(t *testing.T) { setup() defer teardown() createRequest := &DomainCreateRequest{ Name: "example.com", IPAddress: "127.0.0.1", } mux.HandleFunc("/v2/domains", func(w http.ResponseWriter, r *http.Request) { v := new(DomainCreateRequest) err := json.NewDecoder(r.Body).Decode(v) if err != nil { t.Fatal(err) } testMethod(t, r, http.MethodPost) if !reflect.DeepEqual(v, createRequest) { t.Errorf("Request body = %+v, expected %+v", v, createRequest) } fmt.Fprint(w, `{"domain":{"name":"example.com"}}`) }) domain, _, err := client.Domains.Create(ctx, createRequest) if err != nil { t.Errorf("Domains.Create returned error: %v", err) } expected := &Domain{Name: "example.com"} if !reflect.DeepEqual(domain, expected) { t.Errorf("Domains.Create returned %+v, expected %+v", domain, expected) } } func TestDomains_Destroy(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodDelete) }) _, err := client.Domains.Delete(ctx, "example.com") if err != nil { t.Errorf("Domains.Delete returned error: %v", err) } } func TestDomains_AllRecordsForDomainName(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com/records", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, `{"domain_records":[{"id":1},{"id":2}]}`) }) records, _, err := client.Domains.Records(ctx, "example.com", nil) if err != nil { t.Errorf("Domains.List returned error: %v", err) } expected := []DomainRecord{{ID: 1}, {ID: 2}} if !reflect.DeepEqual(records, expected) { t.Errorf("Domains.Records returned %+v, expected %+v", records, expected) } } func TestDomains_AllRecordsForDomainName_PerPage(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com/records", func(w http.ResponseWriter, r *http.Request) { perPage := r.URL.Query().Get("per_page") if perPage != "2" { t.Fatalf("expected '2', got '%s'", perPage) } fmt.Fprint(w, `{"domain_records":[{"id":1},{"id":2}]}`) }) dro := &ListOptions{PerPage: 2} records, _, err := client.Domains.Records(ctx, "example.com", dro) if err != nil { t.Errorf("Domains.List returned error: %v", err) } expected := []DomainRecord{{ID: 1}, {ID: 2}} if !reflect.DeepEqual(records, expected) { t.Errorf("Domains.Records returned %+v, expected %+v", records, expected) } } func TestDomains_RecordsByType(t *testing.T) { tests := []struct { name string recordType string pagination *ListOptions expectedErr *ArgError }{ { name: "success", recordType: "CNAME", }, { name: "when record type is empty it returns argument error", expectedErr: &ArgError{arg: "type", reason: "cannot be an empty string"}, }, { name: "with pagination", recordType: "CNAME", pagination: &ListOptions{Page: 1, PerPage: 10}, }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com/records", func(w http.ResponseWriter, r *http.Request) { require.Equal(t, tt.recordType, r.URL.Query().Get("type")) if tt.pagination != nil { require.Equal(t, strconv.Itoa(tt.pagination.Page), r.URL.Query().Get("page")) require.Equal(t, strconv.Itoa(tt.pagination.PerPage), r.URL.Query().Get("per_page")) } testMethod(t, r, http.MethodGet) fmt.Fprint(w, `{"domain_records":[{"id":1},{"id":2}]}`) }) records, _, err := client.Domains.RecordsByType(ctx, "example.com", tt.recordType, tt.pagination) if tt.expectedErr != nil { assert.Equal(t, tt.expectedErr, err) } else { expected := []DomainRecord{{ID: 1}, {ID: 2}} if !reflect.DeepEqual(records, expected) { t.Errorf("Domains.RecordsByType returned %+v, expected %+v", records, expected) } } }) } } func TestDomains_RecordsByName(t *testing.T) { tests := []struct { name string recordName string pagination *ListOptions expectedErr *ArgError }{ { name: "success", recordName: "foo.com", }, { name: "when record name is empty it returns argument error", expectedErr: &ArgError{arg: "name", reason: "cannot be an empty string"}, }, { name: "with pagination", recordName: "foo.com", pagination: &ListOptions{Page: 2, PerPage: 1}, }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com/records", func(w http.ResponseWriter, r *http.Request) { require.Equal(t, tt.recordName, r.URL.Query().Get("name")) if tt.pagination != nil { require.Equal(t, strconv.Itoa(tt.pagination.Page), r.URL.Query().Get("page")) require.Equal(t, strconv.Itoa(tt.pagination.PerPage), r.URL.Query().Get("per_page")) } testMethod(t, r, http.MethodGet) fmt.Fprint(w, `{"domain_records":[{"id":1},{"id":2}]}`) }) records, _, err := client.Domains.RecordsByName(ctx, "example.com", tt.recordName, tt.pagination) if tt.expectedErr != nil { assert.Equal(t, tt.expectedErr, err) } else { expected := []DomainRecord{{ID: 1}, {ID: 2}} if !reflect.DeepEqual(records, expected) { t.Errorf("Domains.RecordsByName returned %+v, expected %+v", records, expected) } } }) } } func TestDomains_RecordsByTypeAndName(t *testing.T) { tests := []struct { name string recordType string recordName string pagination *ListOptions expectedErr *ArgError }{ { name: "success", recordType: "NS", recordName: "foo.com", }, { name: "when record type is empty it returns argument error", recordName: "foo.com", expectedErr: &ArgError{arg: "type", reason: "cannot be an empty string"}, }, { name: "when record name is empty it returns argument error", recordType: "NS", expectedErr: &ArgError{arg: "name", reason: "cannot be an empty string"}, }, { name: "with pagination", recordType: "CNAME", recordName: "foo.com", pagination: &ListOptions{Page: 1, PerPage: 1}, }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com/records", func(w http.ResponseWriter, r *http.Request) { require.Equal(t, tt.recordType, r.URL.Query().Get("type")) require.Equal(t, tt.recordName, r.URL.Query().Get("name")) if tt.pagination != nil { require.Equal(t, strconv.Itoa(tt.pagination.Page), r.URL.Query().Get("page")) require.Equal(t, strconv.Itoa(tt.pagination.PerPage), r.URL.Query().Get("per_page")) } testMethod(t, r, http.MethodGet) fmt.Fprint(w, `{"domain_records":[{"id":1},{"id":2}]}`) }) records, _, err := client.Domains.RecordsByTypeAndName(ctx, "example.com", tt.recordType, tt.recordName, tt.pagination) if tt.expectedErr != nil { assert.Equal(t, tt.expectedErr, err) } else { expected := []DomainRecord{{ID: 1}, {ID: 2}} if !reflect.DeepEqual(records, expected) { t.Errorf("Domains.RecordsByTypeAndName returned %+v, expected %+v", records, expected) } } }) } } func TestDomains_GetRecordforDomainName(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com/records/1", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, `{"domain_record":{"id":1}}`) }) record, _, err := client.Domains.Record(ctx, "example.com", 1) if err != nil { t.Errorf("Domains.GetRecord returned error: %v", err) } expected := &DomainRecord{ID: 1} if !reflect.DeepEqual(record, expected) { t.Errorf("Domains.GetRecord returned %+v, expected %+v", record, expected) } } func TestDomains_DeleteRecordForDomainName(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/domains/example.com/records/1", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodDelete) }) _, err := client.Domains.DeleteRecord(ctx, "example.com", 1) if err != nil { t.Errorf("Domains.RecordDelete returned error: %v", err) } } func TestDomains_CreateRecordForDomainName(t *testing.T) { setup() defer teardown() createRequest := &DomainRecordEditRequest{ Type: "CNAME", Name: "example", Data: "@", Priority: 10, Port: 10, TTL: 1800, Weight: 10, Flags: 1, Tag: "test", } mux.HandleFunc("/v2/domains/example.com/records", func(w http.ResponseWriter, r *http.Request) { v := new(DomainRecordEditRequest) err := json.NewDecoder(r.Body).Decode(v) if err != nil { t.Fatalf("decode json: %v", err) } testMethod(t, r, http.MethodPost) if !reflect.DeepEqual(v, createRequest) { t.Errorf("Request body = %+v, expected %+v", v, createRequest) } fmt.Fprintf(w, `{"domain_record": {"id":1}}`) }) record, _, err := client.Domains.CreateRecord(ctx, "example.com", createRequest) if err != nil { t.Errorf("Domains.CreateRecord returned error: %v", err) } expected := &DomainRecord{ID: 1} if !reflect.DeepEqual(record, expected) { t.Errorf("Domains.CreateRecord returned %+v, expected %+v", record, expected) } } func TestDomains_EditRecordForDomainName(t *testing.T) { setup() defer teardown() editRequest := &DomainRecordEditRequest{ Type: "CNAME", Name: "example", Data: "@", Priority: 10, Port: 10, TTL: 1800, Weight: 10, Flags: 1, Tag: "test", } mux.HandleFunc("/v2/domains/example.com/records/1", func(w http.ResponseWriter, r *http.Request) { v := new(DomainRecordEditRequest) err := json.NewDecoder(r.Body).Decode(v) if err != nil { t.Fatalf("decode json: %v", err) } testMethod(t, r, http.MethodPut) if !reflect.DeepEqual(v, editRequest) { t.Errorf("Request body = %+v, expected %+v", v, editRequest) } fmt.Fprintf(w, `{"domain_record": {"id":1, "type": "CNAME", "name": "example"}}`) }) record, _, err := client.Domains.EditRecord(ctx, "example.com", 1, editRequest) if err != nil { t.Errorf("Domains.EditRecord returned error: %v", err) } expected := &DomainRecord{ID: 1, Type: "CNAME", Name: "example"} if !reflect.DeepEqual(record, expected) { t.Errorf("Domains.EditRecord returned %+v, expected %+v", record, expected) } } func TestDomainRecord_String(t *testing.T) { record := &DomainRecord{ ID: 1, Type: "CNAME", Name: "example", Data: "@", Priority: 10, Port: 10, TTL: 1800, Weight: 10, Flags: 1, Tag: "test", } stringified := record.String() expected := `godo.DomainRecord{ID:1, Type:"CNAME", Name:"example", Data:"@", Priority:10, Port:10, TTL:1800, Weight:10, Flags:1, Tag:"test"}` if expected != stringified { t.Errorf("DomainRecord.String returned %+v, expected %+v", stringified, expected) } } func TestDomainRecordEditRequest_String(t *testing.T) { record := &DomainRecordEditRequest{ Type: "CNAME", Name: "example", Data: "@", Priority: 10, Port: 10, TTL: 1800, Weight: 10, Flags: 1, Tag: "test", } stringified := record.String() expected := `godo.DomainRecordEditRequest{Type:"CNAME", Name:"example", Data:"@", Priority:10, Port:10, TTL:1800, Weight:10, Flags:1, Tag:"test"}` if expected != stringified { t.Errorf("DomainRecordEditRequest.String returned %+v, expected %+v", stringified, expected) } }