blob: 6525d8b4d2b92b4faf0078f4a06270e15a87350d [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 Perritt27249f42016-02-18 10:35:59 -06006 "github.com/gophercloud/gophercloud"
7 "github.com/gophercloud/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
Jamie Hannaford6181fdb2015-03-24 14:55:50 +010086 _, res.Err = client.Post(createURL(client), reqBody, &res.Body, &gophercloud.RequestOpts{
87 OkCodes: []int{200, 201},
Jon Perritt94963ad2014-05-05 12:14:39 -050088 })
Jon Perritt6d5561b2014-10-01 21:42:15 -050089 return res
Jon Perrittd1d6a742014-09-17 01:10:59 -050090}
91
Jon Perritt42b3a2a2014-10-02 23:06:07 -050092// Delete will delete the existing Volume with the provided ID.
Jamie Hannafordce9f9082014-10-27 11:27:12 +010093func Delete(client *gophercloud.ServiceClient, id string) DeleteResult {
94 var res DeleteResult
Jamie Hannaford6181fdb2015-03-24 14:55:50 +010095 _, res.Err = client.Delete(deleteURL(client, id), nil)
Jamie Hannafordce9f9082014-10-27 11:27:12 +010096 return res
Jon Perritt6d5561b2014-10-01 21:42:15 -050097}
Jon Perrittd1d6a742014-09-17 01:10:59 -050098
Jon Perritt1c2356b2014-10-13 19:56:43 -050099// Get retrieves the Volume with the provided ID. To extract the Volume object
100// from the response, call the Extract method on the GetResult.
Jon Perritt03cb46d2014-09-22 20:46:20 -0500101func Get(client *gophercloud.ServiceClient, id string) GetResult {
Jon Perritt6d5561b2014-10-01 21:42:15 -0500102 var res GetResult
Jamie Hannaford6181fdb2015-03-24 14:55:50 +0100103 _, res.Err = client.Get(getURL(client, id), &res.Body, nil)
Jon Perritt6d5561b2014-10-01 21:42:15 -0500104 return res
Jon Perritt70dd47d2014-05-01 13:51:53 -0500105}
106
Jon Perritt1c2356b2014-10-13 19:56:43 -0500107// ListOptsBuilder allows extensions to add additional parameters to the List
108// request.
109type ListOptsBuilder interface {
Jon Perritt26780d52014-10-14 11:35:58 -0500110 ToVolumeListQuery() (string, error)
Jon Perritt1c2356b2014-10-13 19:56:43 -0500111}
112
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500113// ListOpts holds options for listing Volumes. It is passed to the volumes.List
114// function.
115type ListOpts struct {
Jon Perritt1c2356b2014-10-13 19:56:43 -0500116 // admin-only option. Set it to true to see all tenant volumes.
117 AllTenants bool `q:"all_tenants"`
118 // List only volumes that contain Metadata.
119 Metadata map[string]string `q:"metadata"`
120 // List only volumes that have Name as the display name.
121 Name string `q:"name"`
122 // List only volumes that have a status of Status.
123 Status string `q:"status"`
124}
125
Jon Perritt26780d52014-10-14 11:35:58 -0500126// ToVolumeListQuery formats a ListOpts into a query string.
127func (opts ListOpts) ToVolumeListQuery() (string, error) {
Jon Perritt1c2356b2014-10-13 19:56:43 -0500128 q, err := gophercloud.BuildQueryString(opts)
129 if err != nil {
130 return "", err
131 }
132 return q.String(), nil
Jon Perritt97347a02014-09-21 13:34:48 -0500133}
134
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500135// List returns Volumes optionally limited by the conditions provided in ListOpts.
Jon Perritt1c2356b2014-10-13 19:56:43 -0500136func List(client *gophercloud.ServiceClient, opts ListOptsBuilder) pagination.Pager {
Jon Perritt0871a812014-10-03 11:02:35 -0500137 url := listURL(client)
138 if opts != nil {
Jon Perritt26780d52014-10-14 11:35:58 -0500139 query, err := opts.ToVolumeListQuery()
Jon Perritt0871a812014-10-03 11:02:35 -0500140 if err != nil {
141 return pagination.Pager{Err: err}
142 }
Jon Perritt1c2356b2014-10-13 19:56:43 -0500143 url += query
Jon Perritt0871a812014-10-03 11:02:35 -0500144 }
Ash Wilsonb8b16f82014-10-20 10:19:49 -0400145 createPage := func(r pagination.PageResult) pagination.Page {
Jon Perritt12395212016-02-24 10:41:17 -0600146 return VolumePage{pagination.SinglePageBase(r)}
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500147 }
Jon Perritt3d0a1852015-02-19 08:51:39 -0700148
149 return pagination.NewPager(client, url, createPage)
Jon Perritt1c2356b2014-10-13 19:56:43 -0500150}
151
152// UpdateOptsBuilder allows extensions to add additional parameters to the
153// Update request.
154type UpdateOptsBuilder interface {
155 ToVolumeUpdateMap() (map[string]interface{}, error)
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500156}
157
158// UpdateOpts contain options for updating an existing Volume. This object is passed
159// to the volumes.Update function. For more information about the parameters, see
160// the Volume object.
161type UpdateOpts struct {
Jon Perritt1c2356b2014-10-13 19:56:43 -0500162 // OPTIONAL
163 Name string
164 // OPTIONAL
165 Description string
166 // OPTIONAL
167 Metadata map[string]string
168}
169
170// ToVolumeUpdateMap assembles a request body based on the contents of an
171// UpdateOpts.
172func (opts UpdateOpts) ToVolumeUpdateMap() (map[string]interface{}, error) {
173 v := make(map[string]interface{})
174
175 if opts.Description != "" {
176 v["display_description"] = opts.Description
177 }
178 if opts.Metadata != nil {
179 v["metadata"] = opts.Metadata
180 }
181 if opts.Name != "" {
182 v["display_name"] = opts.Name
183 }
184
185 return map[string]interface{}{"volume": v}, nil
Jon Perritt42b3a2a2014-10-02 23:06:07 -0500186}
187
188// Update will update the Volume with provided information. To extract the updated
189// Volume from the response, call the Extract method on the UpdateResult.
Jon Perritt04851d32014-10-14 02:07:13 -0500190func Update(client *gophercloud.ServiceClient, id string, opts UpdateOptsBuilder) UpdateResult {
Jon Perritt6d5561b2014-10-01 21:42:15 -0500191 var res UpdateResult
Jon Perritt97347a02014-09-21 13:34:48 -0500192
Jon Perritt1c2356b2014-10-13 19:56:43 -0500193 reqBody, err := opts.ToVolumeUpdateMap()
194 if err != nil {
195 res.Err = err
196 return res
197 }
198
Jamie Hannaford6181fdb2015-03-24 14:55:50 +0100199 _, res.Err = client.Put(updateURL(client, id), reqBody, &res.Body, &gophercloud.RequestOpts{
200 OkCodes: []int{200},
Jon Perritt97347a02014-09-21 13:34:48 -0500201 })
Jon Perritt6d5561b2014-10-01 21:42:15 -0500202 return res
Jon Perritte77b9b22014-05-01 13:11:12 -0500203}
Jon Perritt24c20832015-06-30 09:57:00 -0600204
205// IDFromName is a convienience function that returns a server's ID given its name.
206func IDFromName(client *gophercloud.ServiceClient, name string) (string, error) {
207 volumeCount := 0
208 volumeID := ""
209 if name == "" {
210 return "", fmt.Errorf("A volume name must be provided.")
211 }
212 pager := List(client, nil)
213 pager.EachPage(func(page pagination.Page) (bool, error) {
214 volumeList, err := ExtractVolumes(page)
215 if err != nil {
216 return false, err
217 }
218
219 for _, s := range volumeList {
220 if s.Name == name {
221 volumeCount++
222 volumeID = s.ID
223 }
224 }
225 return true, nil
226 })
227
228 switch volumeCount {
229 case 0:
230 return "", fmt.Errorf("Unable to find volume: %s", name)
231 case 1:
232 return volumeID, nil
233 default:
234 return "", fmt.Errorf("Found %d volumes matching %s", volumeCount, name)
235 }
236}