Skip to content

Commit

Permalink
Merge pull request digitalocean#340 from viola/master
Browse files Browse the repository at this point in the history
Add RecordsByType, RecordsByName and RecordsByTypeAndName to the DomainsService
  • Loading branch information
Verolop authored Jun 9, 2020
2 parents 6c3f52d + 9b58e0f commit c107de7
Show file tree
Hide file tree
Showing 2 changed files with 243 additions and 10 deletions.
85 changes: 77 additions & 8 deletions domains.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,9 @@ type DomainsService interface {
Delete(context.Context, string) (*Response, error)

Records(context.Context, string, *ListOptions) ([]DomainRecord, *Response, error)
RecordsByType(context.Context, string, string, *ListOptions) ([]DomainRecord, *Response, error)
RecordsByName(context.Context, string, string, *ListOptions) ([]DomainRecord, *Response, error)
RecordsByTypeAndName(context.Context, string, string, string, *ListOptions) ([]DomainRecord, *Response, error)
Record(context.Context, string, int) (*DomainRecord, *Response, error)
DeleteRecord(context.Context, string, int) (*Response, error)
EditRecord(context.Context, string, int, *DomainRecordEditRequest) (*DomainRecord, *Response, error)
Expand Down Expand Up @@ -201,7 +204,7 @@ func (d DomainRecordEditRequest) String() string {
return Stringify(d)
}

// Records returns a slice of DomainRecords for a domain
// Records returns a slice of DomainRecord for a domain.
func (s *DomainsServiceOp) Records(ctx context.Context, domain string, opt *ListOptions) ([]DomainRecord, *Response, error) {
if len(domain) < 1 {
return nil, nil, NewArgError("domain", "cannot be an empty string")
Expand All @@ -213,21 +216,68 @@ func (s *DomainsServiceOp) Records(ctx context.Context, domain string, opt *List
return nil, nil, err
}

req, err := s.client.NewRequest(ctx, http.MethodGet, path, nil)
return s.records(ctx, path)
}

// RecordsByType returns a slice of DomainRecord for a domain matched by record type.
func (s *DomainsServiceOp) RecordsByType(ctx context.Context, domain, ofType string, opt *ListOptions) ([]DomainRecord, *Response, error) {
if len(domain) < 1 {
return nil, nil, NewArgError("domain", "cannot be an empty string")
}

if len(ofType) < 1 {
return nil, nil, NewArgError("type", "cannot be an empty string")
}

path := fmt.Sprintf("%s/%s/records?type=%s", domainsBasePath, domain, ofType)
path, err := addOptions(path, opt)
if err != nil {
return nil, nil, err
}

root := new(domainRecordsRoot)
resp, err := s.client.Do(ctx, req, root)
return s.records(ctx, path)
}

// RecordsByName returns a slice of DomainRecord for a domain matched by record name.
func (s *DomainsServiceOp) RecordsByName(ctx context.Context, domain, name string, opt *ListOptions) ([]DomainRecord, *Response, error) {
if len(domain) < 1 {
return nil, nil, NewArgError("domain", "cannot be an empty string")
}

if len(name) < 1 {
return nil, nil, NewArgError("name", "cannot be an empty string")
}

path := fmt.Sprintf("%s/%s/records?name=%s", domainsBasePath, domain, name)
path, err := addOptions(path, opt)
if err != nil {
return nil, resp, err
return nil, nil, err
}
if l := root.Links; l != nil {
resp.Links = l

return s.records(ctx, path)
}

// RecordsByTypeAndName returns a slice of DomainRecord for a domain matched by record type and name.
func (s *DomainsServiceOp) RecordsByTypeAndName(ctx context.Context, domain, ofType, name string, opt *ListOptions) ([]DomainRecord, *Response, error) {
if len(domain) < 1 {
return nil, nil, NewArgError("domain", "cannot be an empty string")
}

return root.DomainRecords, resp, err
if len(ofType) < 1 {
return nil, nil, NewArgError("type", "cannot be an empty string")
}

if len(name) < 1 {
return nil, nil, NewArgError("name", "cannot be an empty string")
}

path := fmt.Sprintf("%s/%s/records?type=%s&name=%s", domainsBasePath, domain, ofType, name)
path, err := addOptions(path, opt)
if err != nil {
return nil, nil, err
}

return s.records(ctx, path)
}

// Record returns the record id from a domain
Expand Down Expand Up @@ -339,3 +389,22 @@ func (s *DomainsServiceOp) CreateRecord(ctx context.Context,

return d.DomainRecord, resp, err
}

// Performs a domain records request given a path.
func (s *DomainsServiceOp) records(ctx context.Context, path string) ([]DomainRecord, *Response, error) {
req, err := s.client.NewRequest(ctx, http.MethodGet, path, nil)
if err != nil {
return nil, nil, err
}

root := new(domainRecordsRoot)
resp, err := s.client.Do(ctx, req, root)
if err != nil {
return nil, resp, err
}
if l := root.Links; l != nil {
resp.Links = l
}

return root.DomainRecords, resp, err
}
168 changes: 166 additions & 2 deletions domains_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,11 @@ import (
"fmt"
"net/http"
"reflect"
"strconv"
"testing"

"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)

func TestDomains_ListDomains(t *testing.T) {
Expand Down Expand Up @@ -181,7 +185,7 @@ func TestDomains_AllRecordsForDomainName(t *testing.T) {

expected := []DomainRecord{{ID: 1}, {ID: 2}}
if !reflect.DeepEqual(records, expected) {
t.Errorf("Domains.List returned %+v, expected %+v", records, expected)
t.Errorf("Domains.Records returned %+v, expected %+v", records, expected)
}
}

Expand All @@ -206,7 +210,167 @@ func TestDomains_AllRecordsForDomainName_PerPage(t *testing.T) {

expected := []DomainRecord{{ID: 1}, {ID: 2}}
if !reflect.DeepEqual(records, expected) {
t.Errorf("Domains.List returned %+v, expected %+v", 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)
}
}
})
}
}

Expand Down

0 comments on commit c107de7

Please sign in to comment.