blob: e620ca61400eaf19651f782cdbfdc95a8c18fef2 [file] [log] [blame]
Jon Perrittee6074f2014-04-30 18:42:32 -05001package volumes
2
3import (
Jon Perritt1c2356b2014-10-13 19:56:43 -05004 "fmt"
5
Jon Perrittd1d6a742014-09-17 01:10:59 -05006 "github.com/rackspace/gophercloud"
7 "github.com/rackspace/gophercloud/pagination"
Jon Perrittee6074f2014-04-30 18:42:32 -05008)
9
Jon Perritt1c2356b2014-10-13 19:56:43 -050010// CreateOptsBuilder allows extensions to add additional parameters to the
11// Create request.
12type CreateOptsBuilder interface {
13 ToVolumeCreateMap() (map[string]interface{}, error)
14}
15
Jon Perritt42b3a2a2014-10-02 23:06:07 -050016// CreateOpts contains options for creating a Volume. This object is passed to
17// the volumes.Create function. For more information about these parameters,
18// see the Volume object.
Jon Perritt97347a02014-09-21 13:34:48 -050019type CreateOpts struct {
Jon Perritt1c2356b2014-10-13 19:56:43 -050020 // OPTIONAL
21 Availability string
22 // OPTIONAL
23 Description string
24 // OPTIONAL
25 Metadata map[string]string
26 // OPTIONAL
27 Name string
28 // REQUIRED
29 Size int
30 // OPTIONAL
31 SnapshotID, SourceVolID, ImageID string
32 // OPTIONAL
33 VolumeType string
34}
35
36// ToVolumeCreateMap assembles a request body based on the contents of a
37// CreateOpts.
38func (opts CreateOpts) ToVolumeCreateMap() (map[string]interface{}, error) {
39 v := make(map[string]interface{})
40
41 if opts.Size == 0 {
42 return nil, fmt.Errorf("Required CreateOpts field 'Size' not set.")
43 }
44 v["size"] = opts.Size
45
46 if opts.Availability != "" {
47 v["availability_zone"] = opts.Availability
48 }
49 if opts.Description != "" {
50 v["display_description"] = opts.Description
51 }
52 if opts.ImageID != "" {
53 v["imageRef"] = opts.ImageID
54 }
55 if opts.Metadata != nil {
56 v["metadata"] = opts.Metadata
57 }
58 if opts.Name != "" {
59 v["display_name"] = opts.Name
60 }
61 if opts.SourceVolID != "" {
62 v["source_volid"] = opts.SourceVolID
63 }
64 if opts.SnapshotID != "" {
65 v["snapshot_id"] = opts.SnapshotID
66 }
67 if opts.VolumeType != "" {
68 v["volume_type"] = opts.VolumeType
69 }
70
71 return map[string]interface{}{"volume": v}, nil
Jon Perrittee6074f2014-04-30 18:42:32 -050072}
Jon Perritte77b9b22014-05-01 13:11:12 -050073
Jon Perritt42b3a2a2014-10-02 23:06:07 -050074// Create will create a new Volume based on the values in CreateOpts. To extract
Jon Perritt1c2356b2014-10-13 19:56:43 -050075// the Volume object from the response, call the Extract method on the
76// CreateResult.
77func Create(client *gophercloud.ServiceClient, opts CreateOptsBuilder) CreateResult {
Jon Perritt6d5561b2014-10-01 21:42:15 -050078 var res CreateResult
Jon Perritt1c2356b2014-10-13 19:56:43 -050079
80 reqBody, err := opts.ToVolumeCreateMap()
81 if err != nil {
82 res.Err = err
83 return res
84 }
85
Ash Wilsonacb1b902015-02-12 14:29:31 -050086 _, res.Err = client.Request("POST", createURL(client), gophercloud.RequestOpts{
87 OkCodes: []int{200, 201},
88 JSONBody: &reqBody,
89 JSONResponse: &res.Body,
Jon Perritt94963ad2014-05-05 12:14:39 -050090 })
Jon Perritt6d5561b2014-10-01 21:42:15 -050091 return res
Jon Perrittd1d6a742014-09-17 01:10:59 -050092}
93
Jon Perritt42b3a2a2014-10-02 23:06:07 -050094// Delete will delete the existing Volume with the provided ID.
Jamie Hannafordce9f9082014-10-27 11:27:12 +010095func Delete(client *gophercloud.ServiceClient, id string) DeleteResult {
96 var res DeleteResult
Ash Wilsonacb1b902015-02-12 14:29:31 -050097 _, res.Err = client.Request("DELETE", deleteURL(client, id), gophercloud.RequestOpts{
98 OkCodes: []int{202, 204},
Jon Perritt42b3a2a2014-10-02 23:06:07 -050099 })
Jamie Hannafordce9f9082014-10-27 11:27:12 +0100100 return res
Jon Perritt6d5561b2014-10-01 21:42:15 -0500101}
Jon Perrittd1d6a742014-09-17 01:10:59 -0500102
Jon Perritt1c2356b2014-10-13 19:56:43 -0500103// Get retrieves the Volume with the provided ID. To extract the Volume object
104// from the response, call the Extract method on the GetResult.
Jon Perritt03cb46d2014-09-22 20:46:20 -0500105func Get(client *gophercloud.ServiceClient, id string) GetResult {
Jon Perritt6d5561b2014-10-01 21:42:15 -0500106 var res GetResult
Ash Wilsonacb1b902015-02-12 14:29:31 -0500107 _, res.Err = client.Request("GET", getURL(client, id), gophercloud.RequestOpts{
108 JSONResponse: &res.Body,
109 OkCodes: []int{200},
Jon Perritt70dd47d2014-05-01 13:51:53 -0500110 })
Jon Perritt6d5561b2014-10-01 21:42:15 -0500111 return res
Jon Perritt70dd47d2014-05-01 13:51:53 -0500112}
113
Jon Perritt1c2356b2014-10-13 19:56:43 -0500114// ListOptsBuilder allows extensions to add additional parameters to the List
115// request.
116type ListOptsBuilder interface {
Jon Perritt26780d52014-10-14 11:35:58 -0500117 ToVolumeListQuery() (string, error)
Jon Perritt1c2356b2014-10-13 19:56:43 -0500118}
119
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500120// ListOpts holds options for listing Volumes. It is passed to the volumes.List
121// function.
122type ListOpts struct {
Jon Perritt1c2356b2014-10-13 19:56:43 -0500123 // admin-only option. Set it to true to see all tenant volumes.
124 AllTenants bool `q:"all_tenants"`
125 // List only volumes that contain Metadata.
126 Metadata map[string]string `q:"metadata"`
127 // List only volumes that have Name as the display name.
128 Name string `q:"name"`
129 // List only volumes that have a status of Status.
130 Status string `q:"status"`
131}
132
Jon Perritt26780d52014-10-14 11:35:58 -0500133// ToVolumeListQuery formats a ListOpts into a query string.
134func (opts ListOpts) ToVolumeListQuery() (string, error) {
Jon Perritt1c2356b2014-10-13 19:56:43 -0500135 q, err := gophercloud.BuildQueryString(opts)
136 if err != nil {
137 return "", err
138 }
139 return q.String(), nil
Jon Perritt97347a02014-09-21 13:34:48 -0500140}
141
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500142// List returns Volumes optionally limited by the conditions provided in ListOpts.
Jon Perritt1c2356b2014-10-13 19:56:43 -0500143func List(client *gophercloud.ServiceClient, opts ListOptsBuilder) pagination.Pager {
Jon Perritt0871a812014-10-03 11:02:35 -0500144 url := listURL(client)
145 if opts != nil {
Jon Perritt26780d52014-10-14 11:35:58 -0500146 query, err := opts.ToVolumeListQuery()
Jon Perritt0871a812014-10-03 11:02:35 -0500147 if err != nil {
148 return pagination.Pager{Err: err}
149 }
Jon Perritt1c2356b2014-10-13 19:56:43 -0500150 url += query
Jon Perritt0871a812014-10-03 11:02:35 -0500151 }
Ash Wilsonb8b16f82014-10-20 10:19:49 -0400152 createPage := func(r pagination.PageResult) pagination.Page {
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500153 return ListResult{pagination.SinglePageBase(r)}
154 }
Jon Perritt1c2356b2014-10-13 19:56:43 -0500155 return pagination.NewPager(client, url, createPage)
156}
157
158// UpdateOptsBuilder allows extensions to add additional parameters to the
159// Update request.
160type UpdateOptsBuilder interface {
161 ToVolumeUpdateMap() (map[string]interface{}, error)
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500162}
163
164// UpdateOpts contain options for updating an existing Volume. This object is passed
165// to the volumes.Update function. For more information about the parameters, see
166// the Volume object.
167type UpdateOpts struct {
Jon Perritt1c2356b2014-10-13 19:56:43 -0500168 // OPTIONAL
169 Name string
170 // OPTIONAL
171 Description string
172 // OPTIONAL
173 Metadata map[string]string
174}
175
176// ToVolumeUpdateMap assembles a request body based on the contents of an
177// UpdateOpts.
178func (opts UpdateOpts) ToVolumeUpdateMap() (map[string]interface{}, error) {
179 v := make(map[string]interface{})
180
181 if opts.Description != "" {
182 v["display_description"] = opts.Description
183 }
184 if opts.Metadata != nil {
185 v["metadata"] = opts.Metadata
186 }
187 if opts.Name != "" {
188 v["display_name"] = opts.Name
189 }
190
191 return map[string]interface{}{"volume": v}, nil
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500192}
193
194// Update will update the Volume with provided information. To extract the updated
195// Volume from the response, call the Extract method on the UpdateResult.
Jon Perritt04851d32014-10-14 02:07:13 -0500196func Update(client *gophercloud.ServiceClient, id string, opts UpdateOptsBuilder) UpdateResult {
Jon Perritt6d5561b2014-10-01 21:42:15 -0500197 var res UpdateResult
Jon Perritt97347a02014-09-21 13:34:48 -0500198
Jon Perritt1c2356b2014-10-13 19:56:43 -0500199 reqBody, err := opts.ToVolumeUpdateMap()
200 if err != nil {
201 res.Err = err
202 return res
203 }
204
Ash Wilsonacb1b902015-02-12 14:29:31 -0500205 _, res.Err = client.Request("PUT", updateURL(client, id), gophercloud.RequestOpts{
206 OkCodes: []int{200},
207 JSONBody: &reqBody,
208 JSONResponse: &res.Body,
Jon Perritt97347a02014-09-21 13:34:48 -0500209 })
Jon Perritt6d5561b2014-10-01 21:42:15 -0500210 return res
Jon Perritte77b9b22014-05-01 13:11:12 -0500211}