blob: c2c25a8620b26e8f0afb3133d85e7130e6c59226 [file] [log] [blame]
Jon Perritt8c93a302014-09-28 22:35:57 -05001package containers
2
3import (
4 "fmt"
Jon Perritt8c93a302014-09-28 22:35:57 -05005 "strings"
Jon Perritt8314f4e2014-12-01 10:58:40 -07006 "time"
Jon Perritt5db08922014-09-30 21:32:48 -05007
Ash Wilsonaf262872014-10-20 09:32:29 -04008 "github.com/rackspace/gophercloud"
Jon Perritt5db08922014-09-30 21:32:48 -05009 "github.com/rackspace/gophercloud/pagination"
Jon Perrittb3461402014-10-09 21:36:17 -050010
11 "github.com/mitchellh/mapstructure"
Jon Perritt8c93a302014-09-28 22:35:57 -050012)
13
Jamie Hannaford4ff79962014-10-06 16:28:23 +020014// Container represents a container resource.
Jon Perritt8aa40262014-09-29 15:41:32 -050015type Container struct {
Jamie Hannaford4ff79962014-10-06 16:28:23 +020016 // The total number of bytes stored in the container.
17 Bytes int `json:"bytes" mapstructure:"bytes"`
18
19 // The total number of objects stored in the container.
20 Count int `json:"count" mapstructure:"count"`
21
22 // The name of the container.
23 Name string `json:"name" mapstructure:"name"`
Jon Perritt8aa40262014-09-29 15:41:32 -050024}
Jon Perritt8c93a302014-09-28 22:35:57 -050025
Jamie Hannaford4ff79962014-10-06 16:28:23 +020026// ContainerPage is the page returned by a pager when traversing over a
27// collection of containers.
Jon Perritt8c93a302014-09-28 22:35:57 -050028type ContainerPage struct {
29 pagination.MarkerPageBase
30}
31
32// IsEmpty returns true if a ListResult contains no container names.
33func (r ContainerPage) IsEmpty() (bool, error) {
34 names, err := ExtractNames(r)
35 if err != nil {
36 return true, err
37 }
38 return len(names) == 0, nil
39}
40
41// LastMarker returns the last container name in a ListResult.
42func (r ContainerPage) LastMarker() (string, error) {
43 names, err := ExtractNames(r)
44 if err != nil {
45 return "", err
46 }
47 if len(names) == 0 {
48 return "", nil
49 }
50 return names[len(names)-1], nil
51}
52
53// ExtractInfo is a function that takes a ListResult and returns the containers' information.
54func ExtractInfo(page pagination.Page) ([]Container, error) {
55 untyped := page.(ContainerPage).Body.([]interface{})
56 results := make([]Container, len(untyped))
57 for index, each := range untyped {
Jon Perritt8aa40262014-09-29 15:41:32 -050058 container := each.(map[string]interface{})
Jon Perrittfdac6e52014-09-29 19:43:45 -050059 err := mapstructure.Decode(container, &results[index])
Jon Perritt8aa40262014-09-29 15:41:32 -050060 if err != nil {
61 return results, err
62 }
Jon Perritt8c93a302014-09-28 22:35:57 -050063 }
64 return results, nil
65}
66
67// ExtractNames is a function that takes a ListResult and returns the containers' names.
68func ExtractNames(page pagination.Page) ([]string, error) {
69 casted := page.(ContainerPage)
Ash Wilson72e4d2c2014-10-20 10:27:30 -040070 ct := casted.Header.Get("Content-Type")
Jon Perritt8c93a302014-09-28 22:35:57 -050071
72 switch {
73 case strings.HasPrefix(ct, "application/json"):
74 parsed, err := ExtractInfo(page)
75 if err != nil {
76 return nil, err
77 }
78
79 names := make([]string, 0, len(parsed))
80 for _, container := range parsed {
Jon Perritt8aa40262014-09-29 15:41:32 -050081 names = append(names, container.Name)
Jon Perritt8c93a302014-09-28 22:35:57 -050082 }
83 return names, nil
84 case strings.HasPrefix(ct, "text/plain"):
85 names := make([]string, 0, 50)
86
87 body := string(page.(ContainerPage).Body.([]uint8))
88 for _, name := range strings.Split(body, "\n") {
89 if len(name) > 0 {
90 names = append(names, name)
91 }
92 }
93
94 return names, nil
95 default:
96 return nil, fmt.Errorf("Cannot extract names from response with content-type: [%s]", ct)
97 }
98}
99
Jon Perritt8314f4e2014-12-01 10:58:40 -0700100// GetHeader represents the headers returned in the response from a Get request.
101type GetHeader struct {
102 BytesUsed int64 `json:"X-Account-Bytes-Used"`
103 ContentLength int64 `json:"Content-Length"`
104 ContentType string `json:"Content-Type"`
105 Date time.Time `mapstructure:"-" json:"-"`
106 ObjectCount int64 `json:"X-Account-Object-Count"`
107 TransID string `json:"X-Trans-Id"`
108}
109
Jon Perritt5db08922014-09-30 21:32:48 -0500110// GetResult represents the result of a get operation.
111type GetResult struct {
Jon Perrittd50f93e2014-10-27 14:19:27 -0500112 gophercloud.HeaderResult
Jon Perritt5db08922014-09-30 21:32:48 -0500113}
114
Jon Perritt8314f4e2014-12-01 10:58:40 -0700115// Extract will return a struct of headers returned from a call to Get. To obtain
116// a map of headers, call the ExtractHeader method on the GetResult.
117func (gr GetResult) Extract() (GetHeader, error) {
118 var gh GetHeader
119
120 if err := mapstructure.Decode(gr.Header, &gh); err != nil {
121 return gh, err
122 }
123
124 if date, ok := gr.Header["Date"]; ok && len(date) > 0 {
125 t, err := time.Parse(time.RFC1123, gr.Header["Date"][0])
126 if err != nil {
127 return gh, err
128 }
129 gh.Date = t
130 }
131
132 return gh, nil
133}
134
Jon Perritt8c93a302014-09-28 22:35:57 -0500135// ExtractMetadata is a function that takes a GetResult (of type *http.Response)
136// and returns the custom metadata associated with the container.
137func (gr GetResult) ExtractMetadata() (map[string]string, error) {
138 if gr.Err != nil {
139 return nil, gr.Err
140 }
141 metadata := make(map[string]string)
Ash Wilson72e4d2c2014-10-20 10:27:30 -0400142 for k, v := range gr.Header {
Jon Perritt8c93a302014-09-28 22:35:57 -0500143 if strings.HasPrefix(k, "X-Container-Meta-") {
144 key := strings.TrimPrefix(k, "X-Container-Meta-")
145 metadata[key] = v[0]
146 }
147 }
148 return metadata, nil
149}
Jon Perritt5db08922014-09-30 21:32:48 -0500150
Jon Perritt8314f4e2014-12-01 10:58:40 -0700151// CreateHeader represents the headers returned in the response from a Create request.
152type CreateHeader struct {
153 ContentLength int64 `json:"Content-Length"`
154 ContentType string `json:"Content-Type"`
155 Date time.Time `mapstructure:"-" json:"-"`
156 TransID string `json:"X-Trans-Id"`
157}
158
Jamie Hannaford22ec4792014-10-07 10:07:41 +0200159// CreateResult represents the result of a create operation. To extract the
Jon Perritt9856a342014-10-27 13:44:06 -0500160// the headers from the HTTP response, you can invoke the 'ExtractHeader'
Jamie Hannaford22ec4792014-10-07 10:07:41 +0200161// method on the result struct.
Jon Perritt5db08922014-09-30 21:32:48 -0500162type CreateResult struct {
Jon Perrittd50f93e2014-10-27 14:19:27 -0500163 gophercloud.HeaderResult
Jon Perritt5db08922014-09-30 21:32:48 -0500164}
165
Jon Perritt8314f4e2014-12-01 10:58:40 -0700166// Extract will return a struct of headers returned from a call to Create. To obtain
167// a map of headers, call the ExtractHeader method on the CreateResult.
168func (cr CreateResult) Extract() (CreateHeader, error) {
169 var ch CreateHeader
170
171 if err := mapstructure.Decode(cr.Header, &ch); err != nil {
172 return ch, err
173 }
174
175 if date, ok := cr.Header["Date"]; ok && len(date) > 0 {
176 t, err := time.Parse(time.RFC1123, cr.Header["Date"][0])
177 if err != nil {
178 return ch, err
179 }
180 ch.Date = t
181 }
182
183 return ch, nil
184}
185
186// UpdateHeader represents the headers returned in the response from a Update request.
187type UpdateHeader struct {
188 ContentLength int64 `json:"Content-Length"`
189 ContentType string `json:"Content-Type"`
190 Date time.Time `mapstructure:"-" json:"-"`
191 TransID string `json:"X-Trans-Id"`
192}
193
Jamie Hannaford22ec4792014-10-07 10:07:41 +0200194// UpdateResult represents the result of an update operation. To extract the
Jon Perritt9856a342014-10-27 13:44:06 -0500195// the headers from the HTTP response, you can invoke the 'ExtractHeader'
Jamie Hannaford22ec4792014-10-07 10:07:41 +0200196// method on the result struct.
Jon Perritt5db08922014-09-30 21:32:48 -0500197type UpdateResult struct {
Jon Perrittd50f93e2014-10-27 14:19:27 -0500198 gophercloud.HeaderResult
Jon Perritt5db08922014-09-30 21:32:48 -0500199}
200
Jon Perritt8314f4e2014-12-01 10:58:40 -0700201// Extract will return a struct of headers returned from a call to Update. To obtain
202// a map of headers, call the ExtractHeader method on the UpdateResult.
203func (ur UpdateResult) Extract() (UpdateHeader, error) {
204 var uh UpdateHeader
205
206 if err := mapstructure.Decode(ur.Header, &uh); err != nil {
207 return uh, err
208 }
209
210 if date, ok := ur.Header["Date"]; ok && len(date) > 0 {
211 t, err := time.Parse(time.RFC1123, ur.Header["Date"][0])
212 if err != nil {
213 return uh, err
214 }
215 uh.Date = t
216 }
217
218 return uh, nil
219}
220
221// DeleteHeader represents the headers returned in the response from a Delete request.
222type DeleteHeader struct {
223 ContentLength int64 `json:"Content-Length"`
224 ContentType string `json:"Content-Type"`
225 Date time.Time `mapstructure:"-" json:"-"`
226 TransID string `json:"X-Trans-Id"`
227}
228
Jamie Hannaford22ec4792014-10-07 10:07:41 +0200229// DeleteResult represents the result of a delete operation. To extract the
Jon Perritt9856a342014-10-27 13:44:06 -0500230// the headers from the HTTP response, you can invoke the 'ExtractHeader'
Jamie Hannaford22ec4792014-10-07 10:07:41 +0200231// method on the result struct.
Jon Perritt5db08922014-09-30 21:32:48 -0500232type DeleteResult struct {
Jon Perrittd50f93e2014-10-27 14:19:27 -0500233 gophercloud.HeaderResult
Jon Perritt5db08922014-09-30 21:32:48 -0500234}
Jon Perritt8314f4e2014-12-01 10:58:40 -0700235
236// Extract will return a struct of headers returned from a call to Delete. To obtain
237// a map of headers, call the ExtractHeader method on the DeleteResult.
238func (dr DeleteResult) Extract() (DeleteHeader, error) {
239 var dh DeleteHeader
240
241 if err := mapstructure.Decode(dr.Header, &dh); err != nil {
242 return dh, err
243 }
244
245 if date, ok := dr.Header["Date"]; ok && len(date) > 0 {
246 t, err := time.Parse(time.RFC1123, dr.Header["Date"][0])
247 if err != nil {
248 return dh, err
249 }
250 dh.Date = t
251 }
252
253 return dh, nil
254}