| package gophercloud |
| |
| import ( |
| "fmt" |
| "net/http" |
| "reflect" |
| "strings" |
| "testing" |
| |
| "github.com/mitchellh/mapstructure" |
| "github.com/rackspace/gophercloud/testhelper" |
| ) |
| |
| func createClient() *ServiceClient { |
| return &ServiceClient{ |
| Provider: &ProviderClient{TokenID: "abc123"}, |
| Endpoint: testhelper.Endpoint(), |
| } |
| } |
| |
| // SinglePage sample and test cases. |
| |
| func ExtractSingleInts(page Page) ([]int, error) { |
| var response struct { |
| Ints []int `mapstructure:"ints"` |
| } |
| |
| err := mapstructure.Decode(page.(SinglePage).Body, &response) |
| if err != nil { |
| return nil, err |
| } |
| |
| return response.Ints, nil |
| } |
| |
| func setupSinglePaged() Pager { |
| testhelper.SetupHTTP() |
| client := createClient() |
| |
| testhelper.Mux.HandleFunc("/only", func(w http.ResponseWriter, r *http.Request) { |
| w.Header().Add("Content-Type", "application/json") |
| fmt.Fprintf(w, `{ "ints": [1, 2, 3] }`) |
| }) |
| |
| countPage := func(p Page) (int, error) { |
| is, err := ExtractSingleInts(p) |
| if err != nil { |
| return 0, err |
| } |
| return len(is), nil |
| } |
| |
| return NewSinglePager(client, testhelper.Server.URL+"/only", countPage) |
| } |
| |
| func TestEnumerateSinglePaged(t *testing.T) { |
| callCount := 0 |
| pager := setupSinglePaged() |
| defer testhelper.TeardownHTTP() |
| |
| err := pager.EachPage(func(page Page) (bool, error) { |
| callCount++ |
| |
| expected := []int{1, 2, 3} |
| actual, err := ExtractSingleInts(page) |
| if err != nil { |
| return false, err |
| } |
| if !reflect.DeepEqual(expected, actual) { |
| t.Errorf("Expected %v, but was %v", expected, actual) |
| } |
| return true, nil |
| }) |
| if err != nil { |
| t.Fatalf("Unexpected error calling EachPage: %v", err) |
| } |
| |
| if callCount != 1 { |
| t.Errorf("Callback was invoked %d times", callCount) |
| } |
| } |
| |
| // LinkedPager sample and test cases. |
| |
| func ExtractLinkedInts(page Page) ([]int, error) { |
| var response struct { |
| Ints []int `mapstructure:"ints"` |
| } |
| |
| err := mapstructure.Decode(page.(LinkedPage).Body, &response) |
| if err != nil { |
| return nil, err |
| } |
| |
| return response.Ints, nil |
| } |
| |
| func createLinked(t *testing.T) Pager { |
| testhelper.SetupHTTP() |
| |
| testhelper.Mux.HandleFunc("/page1", func(w http.ResponseWriter, r *http.Request) { |
| w.Header().Add("Content-Type", "application/json") |
| fmt.Fprintf(w, `{ "ints": [1, 2, 3], "links": { "next": "%s/page2" } }`, testhelper.Server.URL) |
| }) |
| |
| testhelper.Mux.HandleFunc("/page2", func(w http.ResponseWriter, r *http.Request) { |
| w.Header().Add("Content-Type", "application/json") |
| fmt.Fprintf(w, `{ "ints": [4, 5, 6], "links": { "next": "%s/page3" } }`, testhelper.Server.URL) |
| }) |
| |
| testhelper.Mux.HandleFunc("/page3", func(w http.ResponseWriter, r *http.Request) { |
| w.Header().Add("Content-Type", "application/json") |
| fmt.Fprintf(w, `{ "ints": [7, 8, 9], "links": { "next": null } }`) |
| }) |
| |
| client := createClient() |
| |
| countPage := func(p Page) (int, error) { |
| is, err := ExtractLinkedInts(p) |
| if err != nil { |
| return 0, err |
| } |
| return len(is), nil |
| } |
| |
| return NewLinkedPager(client, testhelper.Server.URL+"/page1", countPage) |
| } |
| |
| func TestEnumerateLinked(t *testing.T) { |
| pager := createLinked(t) |
| defer testhelper.TeardownHTTP() |
| |
| callCount := 0 |
| err := pager.EachPage(func(page Page) (bool, error) { |
| actual, err := ExtractLinkedInts(page) |
| if err != nil { |
| return false, err |
| } |
| |
| t.Logf("Handler invoked with %v", actual) |
| |
| var expected []int |
| switch callCount { |
| case 0: |
| expected = []int{1, 2, 3} |
| case 1: |
| expected = []int{4, 5, 6} |
| case 2: |
| expected = []int{7, 8, 9} |
| default: |
| t.Fatalf("Unexpected call count: %d", callCount) |
| return false, nil |
| } |
| |
| if !reflect.DeepEqual(expected, actual) { |
| t.Errorf("Call %d: Expected %#v, but was %#v", callCount, expected, actual) |
| } |
| |
| callCount++ |
| return true, nil |
| }) |
| if err != nil { |
| t.Errorf("Unexpected error for page iteration: %v", err) |
| } |
| |
| if callCount != 3 { |
| t.Errorf("Expected 3 calls, but was %d", callCount) |
| } |
| } |
| |
| func createMarkerPaged(t *testing.T) Pager { |
| testhelper.SetupHTTP() |
| |
| testhelper.Mux.HandleFunc("/page", func(w http.ResponseWriter, r *http.Request) { |
| r.ParseForm() |
| ms := r.Form["marker"] |
| switch { |
| case len(ms) == 0: |
| fmt.Fprintf(w, "aaa\nbbb\nccc") |
| case len(ms) == 1 && ms[0] == "ccc": |
| fmt.Fprintf(w, "ddd\neee\nfff") |
| case len(ms) == 1 && ms[0] == "fff": |
| fmt.Fprintf(w, "ggg\nhhh\niii") |
| case len(ms) == 1 && ms[0] == "iii": |
| w.WriteHeader(http.StatusNoContent) |
| default: |
| t.Errorf("Request with unexpected marker: [%v]", ms) |
| } |
| }) |
| |
| client := createClient() |
| |
| lastMark := func(p Page) (string, error) { |
| items, err := ExtractMarkerStrings(p) |
| if err != nil { |
| return "", err |
| } |
| return items[len(items)-1], nil |
| } |
| |
| countPage := func(p Page) (int, error) { |
| items, err := ExtractMarkerStrings(p) |
| if err != nil { |
| return 0, err |
| } |
| fmt.Printf("Counting items [%#v] = [%d]\n", items, len(items)) |
| return len(items), nil |
| } |
| |
| return NewMarkerPager(client, testhelper.Server.URL+"/page", lastMark, countPage) |
| } |
| |
| func ExtractMarkerStrings(page Page) ([]string, error) { |
| content := page.(MarkerPage).Body.([]uint8) |
| parts := strings.Split(string(content), "\n") |
| results := make([]string, 0, len(parts)) |
| for _, part := range parts { |
| if len(part) > 0 { |
| results = append(results, part) |
| } |
| } |
| return results, nil |
| } |
| |
| func TestEnumerateMarker(t *testing.T) { |
| pager := createMarkerPaged(t) |
| defer testhelper.TeardownHTTP() |
| |
| callCount := 0 |
| err := pager.EachPage(func(page Page) (bool, error) { |
| actual, err := ExtractMarkerStrings(page) |
| if err != nil { |
| return false, err |
| } |
| |
| t.Logf("Handler invoked with %v", actual) |
| |
| var expected []string |
| switch callCount { |
| case 0: |
| expected = []string{"aaa", "bbb", "ccc"} |
| case 1: |
| expected = []string{"ddd", "eee", "fff"} |
| case 2: |
| expected = []string{"ggg", "hhh", "iii"} |
| default: |
| t.Fatalf("Unexpected call count: %d", callCount) |
| return false, nil |
| } |
| |
| testhelper.CheckDeepEquals(t, expected, actual) |
| |
| callCount++ |
| return true, nil |
| }) |
| testhelper.AssertNoErr(t, err) |
| testhelper.AssertEquals(t, callCount, 3) |
| } |