package godo import ( "encoding/json" "fmt" "net/http" "reflect" "testing" "time" ) func TestStorageVolumes_ListStorageVolumes(t *testing.T) { setup() defer teardown() jBlob := ` { "volumes": [ { "user_id": 42, "region": {"slug": "nyc3"}, "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "my volume", "description": "my description", "size_gigabytes": 100, "droplet_ids": [10], "created_at": "2002-10-02T15:00:00.05Z", "filesystem_type": "", "filesystem_label": "", "tags": ["tag1", "tag2"] }, { "user_id": 42, "region": {"slug": "nyc3"}, "id": "96d414c6-295e-4e3a-ac59-eb9456c1e1d1", "name": "my other volume", "description": "my other description", "size_gigabytes": 100, "created_at": "2012-10-03T15:00:01.05Z", "filesystem_type": "ext4", "filesystem_label": "my-volume", "tags": [] } ], "links": { "pages": { "last": "https://api.digitalocean.com/v2/volumes?page=2", "next": "https://api.digitalocean.com/v2/volumes?page=2" } }, "meta": { "total": 28 } }` mux.HandleFunc("/v2/volumes/", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, jBlob) }) volumes, resp, err := client.Storage.ListVolumes(ctx, nil) if err != nil { t.Errorf("Storage.ListVolumes returned error: %v", err) } expectedVolume := []Volume{ { Region: &Region{Slug: "nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my volume", Description: "my description", SizeGigaBytes: 100, DropletIDs: []int{10}, CreatedAt: time.Date(2002, 10, 02, 15, 00, 00, 50000000, time.UTC), Tags: []string{"tag1", "tag2"}, }, { Region: &Region{Slug: "nyc3"}, ID: "96d414c6-295e-4e3a-ac59-eb9456c1e1d1", Name: "my other volume", Description: "my other description", SizeGigaBytes: 100, CreatedAt: time.Date(2012, 10, 03, 15, 00, 01, 50000000, time.UTC), FilesystemType: "ext4", FilesystemLabel: "my-volume", Tags: []string{}, }, } if !reflect.DeepEqual(volumes, expectedVolume) { t.Errorf("Storage.ListVolumes returned volumes %+v, expected %+v", volumes, expectedVolume) } expectedMeta := &Meta{ Total: 28, } if !reflect.DeepEqual(resp.Meta, expectedMeta) { t.Errorf("Storage.ListVolumes returned meta %+v, expected %+v", resp.Meta, expectedMeta) } } func TestStorageVolumes_Get(t *testing.T) { setup() defer teardown() want := &Volume{ Region: &Region{Slug: "nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my volume", Description: "my description", SizeGigaBytes: 100, CreatedAt: time.Date(2002, 10, 02, 15, 00, 00, 50000000, time.UTC), FilesystemType: "xfs", FilesystemLabel: "my-vol", Tags: []string{"tag1", "tag2"}, } jBlob := `{ "volume":{ "region": {"slug":"nyc3"}, "attached_to_droplet": null, "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "my volume", "description": "my description", "size_gigabytes": 100, "created_at": "2002-10-02T15:00:00.05Z", "filesystem_type": "xfs", "filesystem_label": "my-vol", "tags": ["tag1", "tag2"] } }` mux.HandleFunc("/v2/volumes/80d414c6-295e-4e3a-ac58-eb9456c1e1d1", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, jBlob) }) got, _, err := client.Storage.GetVolume(ctx, "80d414c6-295e-4e3a-ac58-eb9456c1e1d1") if err != nil { t.Errorf("Storage.GetVolume returned error: %v", err) } if !reflect.DeepEqual(got, want) { t.Errorf("Storage.GetVolume returned %+v, want %+v", got, want) } } func TestStorageVolumes_ListVolumesByName(t *testing.T) { setup() defer teardown() jBlob := `{ "volumes": [ { "region": {"slug": "nyc3"}, "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "myvolume", "description": "my description", "size_gigabytes": 100, "droplet_ids": [10], "created_at": "2002-10-02T15:00:00.05Z", "filesystem_type": "", "filesystem_label": "", "tags": ["tag1", "tag2"] } ], "links": {}, "meta": { "total": 1 } }` expectedVolumes := []Volume{ { Region: &Region{Slug: "nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "myvolume", Description: "my description", SizeGigaBytes: 100, DropletIDs: []int{10}, CreatedAt: time.Date(2002, 10, 02, 15, 00, 00, 50000000, time.UTC), Tags: []string{"tag1", "tag2"}, }, } mux.HandleFunc("/v2/volumes", func(w http.ResponseWriter, r *http.Request) { if r.URL.Query().Get("name") != "myvolume" || r.URL.Query().Get("region") != "" { t.Errorf("Storage.ListVolumeByName did not request the correct name or region") } testMethod(t, r, http.MethodGet) fmt.Fprint(w, jBlob) }) options := &ListVolumeParams{ Name: "myvolume", } volumes, resp, err := client.Storage.ListVolumes(ctx, options) if err != nil { t.Errorf("Storage.ListVolumeByName returned error: %v", err) } if !reflect.DeepEqual(volumes, expectedVolumes) { t.Errorf("Storage.ListVolumeByName returned volumes %+v, expected %+v", volumes, expectedVolumes) } expectedMeta := &Meta{ Total: 1, } if !reflect.DeepEqual(resp.Meta, expectedMeta) { t.Errorf("Storage.ListVolumeByName returned meta %+v, expected %+v", resp.Meta, expectedMeta) } } func TestStorageVolumes_ListVolumesByRegion(t *testing.T) { setup() defer teardown() jBlob := `{ "volumes": [ { "region": {"slug": "nyc3"}, "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "myvolume", "description": "my description", "size_gigabytes": 100, "droplet_ids": [10], "created_at": "2002-10-02T15:00:00.05Z", "filesystem_type": "", "filesystem_label": "", "tags": ["tag1", "tag2"] } ], "links": {}, "meta": { "total": 1 } }` expectedVolumes := []Volume{ { Region: &Region{Slug: "nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "myvolume", Description: "my description", SizeGigaBytes: 100, DropletIDs: []int{10}, CreatedAt: time.Date(2002, 10, 02, 15, 00, 00, 50000000, time.UTC), Tags: []string{"tag1", "tag2"}, }, } mux.HandleFunc("/v2/volumes", func(w http.ResponseWriter, r *http.Request) { if r.URL.Query().Get("region") != "nyc3" || r.URL.Query().Get("name") != "" { t.Errorf("Storage.ListVolumeByName did not request the correct name or region") } testMethod(t, r, http.MethodGet) fmt.Fprint(w, jBlob) }) options := &ListVolumeParams{ Region: "nyc3", } volumes, resp, err := client.Storage.ListVolumes(ctx, options) if err != nil { t.Errorf("Storage.ListVolumeByName returned error: %v", err) } if !reflect.DeepEqual(volumes, expectedVolumes) { t.Errorf("Storage.ListVolumeByName returned volumes %+v, expected %+v", volumes, expectedVolumes) } expectedMeta := &Meta{ Total: 1, } if !reflect.DeepEqual(resp.Meta, expectedMeta) { t.Errorf("Storage.ListVolumeByName returned meta %+v, expected %+v", resp.Meta, expectedMeta) } } func TestStorageVolumes_ListVolumesByNameAndRegion(t *testing.T) { setup() defer teardown() jBlob := `{ "volumes": [ { "region": {"slug": "nyc3"}, "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "myvolume", "description": "my description", "size_gigabytes": 100, "droplet_ids": [10], "created_at": "2002-10-02T15:00:00.05Z", "filesystem_type": "", "filesystem_label": "", "tags": ["tag1", "tag2"] } ], "links": {}, "meta": { "total": 1 } }` expectedVolumes := []Volume{ { Region: &Region{Slug: "nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "myvolume", Description: "my description", SizeGigaBytes: 100, DropletIDs: []int{10}, CreatedAt: time.Date(2002, 10, 02, 15, 00, 00, 50000000, time.UTC), Tags: []string{"tag1", "tag2"}, }, } mux.HandleFunc("/v2/volumes", func(w http.ResponseWriter, r *http.Request) { if r.URL.Query().Get("region") != "nyc3" || r.URL.Query().Get("name") != "myvolume" { t.Errorf("Storage.ListVolumeByName did not request the correct name or region") } testMethod(t, r, http.MethodGet) fmt.Fprint(w, jBlob) }) options := &ListVolumeParams{ Region: "nyc3", Name: "myvolume", } volumes, resp, err := client.Storage.ListVolumes(ctx, options) if err != nil { t.Errorf("Storage.ListVolumeByName returned error: %v", err) } if !reflect.DeepEqual(volumes, expectedVolumes) { t.Errorf("Storage.ListVolumeByName returned volumes %+v, expected %+v", volumes, expectedVolumes) } expectedMeta := &Meta{ Total: 1, } if !reflect.DeepEqual(resp.Meta, expectedMeta) { t.Errorf("Storage.ListVolumeByName returned meta %+v, expected %+v", resp.Meta, expectedMeta) } } func TestStorageVolumes_Create(t *testing.T) { setup() defer teardown() createRequest := &VolumeCreateRequest{ Region: "nyc3", Name: "my volume", Description: "my description", SizeGigaBytes: 100, Tags: []string{"tag1", "tag2"}, } want := &Volume{ Region: &Region{Slug: "nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my volume", Description: "my description", SizeGigaBytes: 100, CreatedAt: time.Date(2002, 10, 02, 15, 00, 00, 50000000, time.UTC), Tags: []string{"tag1", "tag2"}, } jBlob := `{ "volume":{ "region": {"slug":"nyc3"}, "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "my volume", "description": "my description", "size_gigabytes": 100, "created_at": "2002-10-02T15:00:00.05Z", "tags": ["tag1", "tag2"] }, "links": {} }` mux.HandleFunc("/v2/volumes", func(w http.ResponseWriter, r *http.Request) { v := new(VolumeCreateRequest) 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, jBlob) }) got, _, err := client.Storage.CreateVolume(ctx, createRequest) if err != nil { t.Errorf("Storage.CreateVolume returned error: %v", err) } if !reflect.DeepEqual(got, want) { t.Errorf("Storage.CreateVolume returned %+v, want %+v", got, want) } } func TestStorageVolumes_CreateFormatted(t *testing.T) { setup() defer teardown() createRequest := &VolumeCreateRequest{ Region: "nyc3", Name: "my volume", Description: "my description", SizeGigaBytes: 100, FilesystemType: "xfs", Tags: []string{"tag1", "tag2"}, } want := &Volume{ Region: &Region{Slug: "nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my volume", Description: "my description", SizeGigaBytes: 100, CreatedAt: time.Date(2002, 10, 02, 15, 00, 00, 50000000, time.UTC), FilesystemType: "xfs", Tags: []string{"tag1", "tag2"}, } jBlob := `{ "volume":{ "region": {"slug":"nyc3"}, "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "my volume", "description": "my description", "size_gigabytes": 100, "created_at": "2002-10-02T15:00:00.05Z", "filesystem_type": "xfs", "filesystem_label": "", "tags": ["tag1", "tag2"] }, "links": {} }` mux.HandleFunc("/v2/volumes", func(w http.ResponseWriter, r *http.Request) { v := new(VolumeCreateRequest) 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, jBlob) }) got, _, err := client.Storage.CreateVolume(ctx, createRequest) if err != nil { t.Errorf("Storage.CreateVolume returned error: %v", err) } if !reflect.DeepEqual(got, want) { t.Errorf("Storage.CreateVolume returned %+v, want %+v", got, want) } } func TestStorageVolumes_CreateFromSnapshot(t *testing.T) { setup() defer teardown() createRequest := &VolumeCreateRequest{ Name: "my-volume-from-a-snapshot", Description: "my description", SizeGigaBytes: 100, SnapshotID: "0d165eff-0b4c-11e7-9093-0242ac110207", Tags: []string{"tag1", "tag2"}, } want := &Volume{ Region: &Region{Slug: "nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my-volume-from-a-snapshot", Description: "my description", SizeGigaBytes: 100, CreatedAt: time.Date(2002, 10, 02, 15, 00, 00, 50000000, time.UTC), Tags: []string{"tag1", "tag2"}, } jBlob := `{ "volume":{ "region": {"slug":"nyc3"}, "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "my-volume-from-a-snapshot", "description": "my description", "size_gigabytes": 100, "created_at": "2002-10-02T15:00:00.05Z", "tags": ["tag1", "tag2"] }, "links": {} }` mux.HandleFunc("/v2/volumes", func(w http.ResponseWriter, r *http.Request) { v := new(VolumeCreateRequest) 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, jBlob) }) got, _, err := client.Storage.CreateVolume(ctx, createRequest) if err != nil { t.Errorf("Storage.CreateVolume returned error: %v", err) } if !reflect.DeepEqual(got, want) { t.Errorf("Storage.CreateVolume returned %+v, want %+v", got, want) } } func TestStorageVolumes_Destroy(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/volumes/80d414c6-295e-4e3a-ac58-eb9456c1e1d1", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodDelete) }) _, err := client.Storage.DeleteVolume(ctx, "80d414c6-295e-4e3a-ac58-eb9456c1e1d1") if err != nil { t.Errorf("Storage.DeleteVolume returned error: %v", err) } } func TestStorageSnapshots_ListStorageSnapshots(t *testing.T) { setup() defer teardown() jBlob := ` { "snapshots": [ { "regions": ["nyc3"], "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "my snapshot", "size_gigabytes": 100, "created_at": "2002-10-02T15:00:00.05Z" }, { "regions": ["nyc3"], "id": "96d414c6-295e-4e3a-ac59-eb9456c1e1d1", "name": "my other snapshot", "size_gigabytes": 100, "created_at": "2012-10-03T15:00:01.05Z" } ], "links": { "pages": { "last": "https://api.digitalocean.com/v2/volumes?page=2", "next": "https://api.digitalocean.com/v2/volumes?page=2" } }, "meta": { "total": 28 } }` mux.HandleFunc("/v2/volumes/98d414c6-295e-4e3a-ac58-eb9456c1e1d1/snapshots", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, jBlob) }) volumes, resp, err := client.Storage.ListSnapshots(ctx, "98d414c6-295e-4e3a-ac58-eb9456c1e1d1", nil) if err != nil { t.Errorf("Storage.ListSnapshots returned error: %v", err) } expectedSnapshots := []Snapshot{ { Regions: []string{"nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my snapshot", SizeGigaBytes: 100, Created: "2002-10-02T15:00:00.05Z", }, { Regions: []string{"nyc3"}, ID: "96d414c6-295e-4e3a-ac59-eb9456c1e1d1", Name: "my other snapshot", SizeGigaBytes: 100, Created: "2012-10-03T15:00:01.05Z", }, } if !reflect.DeepEqual(volumes, expectedSnapshots) { t.Errorf("Storage.ListSnapshots returned snapshots %+v, expected %+v", volumes, expectedSnapshots) } expectedMeta := &Meta{ Total: 28, } if !reflect.DeepEqual(resp.Meta, expectedMeta) { t.Errorf("Storage.ListSnapshots returned meta %+v, expected %+v", resp.Meta, expectedMeta) } } func TestStorageSnapshots_Get(t *testing.T) { setup() defer teardown() want := &Snapshot{ Regions: []string{"nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my snapshot", SizeGigaBytes: 100, Created: "2002-10-02T15:00:00.05Z", } jBlob := `{ "snapshot":{ "regions": ["nyc3"], "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "my snapshot", "size_gigabytes": 100, "created_at": "2002-10-02T15:00:00.05Z" } }` mux.HandleFunc("/v2/snapshots/80d414c6-295e-4e3a-ac58-eb9456c1e1d1", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodGet) fmt.Fprint(w, jBlob) }) got, _, err := client.Storage.GetSnapshot(ctx, "80d414c6-295e-4e3a-ac58-eb9456c1e1d1") if err != nil { t.Errorf("Storage.GetSnapshot returned error: %v", err) } if !reflect.DeepEqual(got, want) { t.Errorf("Storage.GetSnapshot returned %+v, want %+v", got, want) } } func TestStorageSnapshots_Create(t *testing.T) { setup() defer teardown() createRequest := &SnapshotCreateRequest{ VolumeID: "98d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my snapshot", Description: "my description", Tags: []string{"one", "two"}, } want := &Snapshot{ Regions: []string{"nyc3"}, ID: "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", Name: "my snapshot", SizeGigaBytes: 100, Created: "2002-10-02T15:00:00.05Z", Tags: []string{"one", "two"}, } jBlob := `{ "snapshot":{ "regions": ["nyc3"], "id": "80d414c6-295e-4e3a-ac58-eb9456c1e1d1", "name": "my snapshot", "description": "my description", "size_gigabytes": 100, "created_at": "2002-10-02T15:00:00.05Z", "tags": ["one", "two"] }, "links": { "pages": { "last": "https://api.digitalocean.com/v2/volumes/98d414c6-295e-4e3a-ac58-eb9456c1e1d1/snapshots?page=2", "next": "https://api.digitalocean.com/v2/volumes/98d414c6-295e-4e3a-ac58-eb9456c1e1d1/snapshots?page=2" } }, "meta": { "total": 28 } }` mux.HandleFunc("/v2/volumes/98d414c6-295e-4e3a-ac58-eb9456c1e1d1/snapshots", func(w http.ResponseWriter, r *http.Request) { v := new(SnapshotCreateRequest) 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, jBlob) }) got, _, err := client.Storage.CreateSnapshot(ctx, createRequest) if err != nil { t.Errorf("Storage.CreateSnapshot returned error: %v", err) } if !reflect.DeepEqual(got, want) { t.Errorf("Storage.CreateSnapshot returned %+v, want %+v", got, want) } } func TestStorageSnapshots_Destroy(t *testing.T) { setup() defer teardown() mux.HandleFunc("/v2/snapshots/80d414c6-295e-4e3a-ac58-eb9456c1e1d1", func(w http.ResponseWriter, r *http.Request) { testMethod(t, r, http.MethodDelete) }) _, err := client.Storage.DeleteSnapshot(ctx, "80d414c6-295e-4e3a-ac58-eb9456c1e1d1") if err != nil { t.Errorf("Storage.DeleteSnapshot returned error: %v", err) } }