| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 1 | package services | 
|  | 2 |  | 
|  | 3 | import ( | 
| Jon Perritt | b0ab0d1 | 2015-01-27 12:12:51 -0700 | [diff] [blame] | 4 | "fmt" | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 5 | "strings" | 
|  | 6 |  | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 7 | "github.com/racker/perigee" | 
|  | 8 | "github.com/rackspace/gophercloud" | 
|  | 9 | "github.com/rackspace/gophercloud/pagination" | 
|  | 10 | ) | 
|  | 11 |  | 
|  | 12 | // ListOptsBuilder allows extensions to add additional parameters to the | 
|  | 13 | // List request. | 
|  | 14 | type ListOptsBuilder interface { | 
|  | 15 | ToCDNServiceListQuery() (string, error) | 
|  | 16 | } | 
|  | 17 |  | 
|  | 18 | // ListOpts allows the filtering and sorting of paginated collections through | 
|  | 19 | // the API. Marker and Limit are used for pagination. | 
|  | 20 | type ListOpts struct { | 
|  | 21 | Marker string `q:"marker"` | 
|  | 22 | Limit  int    `q:"limit"` | 
|  | 23 | } | 
|  | 24 |  | 
|  | 25 | // ToCDNServiceListQuery formats a ListOpts into a query string. | 
|  | 26 | func (opts ListOpts) ToCDNServiceListQuery() (string, error) { | 
|  | 27 | q, err := gophercloud.BuildQueryString(opts) | 
|  | 28 | if err != nil { | 
|  | 29 | return "", err | 
|  | 30 | } | 
|  | 31 | return q.String(), nil | 
|  | 32 | } | 
|  | 33 |  | 
|  | 34 | // List returns a Pager which allows you to iterate over a collection of | 
|  | 35 | // CDN services. It accepts a ListOpts struct, which allows for pagination via | 
|  | 36 | // marker and limit. | 
|  | 37 | func List(c *gophercloud.ServiceClient, opts ListOptsBuilder) pagination.Pager { | 
|  | 38 | url := listURL(c) | 
|  | 39 | if opts != nil { | 
|  | 40 | query, err := opts.ToCDNServiceListQuery() | 
|  | 41 | if err != nil { | 
|  | 42 | return pagination.Pager{Err: err} | 
|  | 43 | } | 
|  | 44 | url += query | 
|  | 45 | } | 
|  | 46 |  | 
|  | 47 | createPage := func(r pagination.PageResult) pagination.Page { | 
|  | 48 | p := ServicePage{pagination.MarkerPageBase{PageResult: r}} | 
|  | 49 | p.MarkerPageBase.Owner = p | 
|  | 50 | return p | 
|  | 51 | } | 
|  | 52 |  | 
|  | 53 | pager := pagination.NewPager(c, url, createPage) | 
|  | 54 | return pager | 
|  | 55 | } | 
|  | 56 |  | 
|  | 57 | // CreateOptsBuilder is the interface options structs have to satisfy in order | 
|  | 58 | // to be used in the main Create operation in this package. Since many | 
|  | 59 | // extensions decorate or modify the common logic, it is useful for them to | 
|  | 60 | // satisfy a basic interface in order for them to be used. | 
|  | 61 | type CreateOptsBuilder interface { | 
|  | 62 | ToCDNServiceCreateMap() (map[string]interface{}, error) | 
|  | 63 | } | 
|  | 64 |  | 
|  | 65 | // CreateOpts is the common options struct used in this package's Create | 
|  | 66 | // operation. | 
|  | 67 | type CreateOpts struct { | 
|  | 68 | // REQUIRED. Specifies the name of the service. The minimum length for name is | 
|  | 69 | // 3. The maximum length is 256. | 
|  | 70 | Name string | 
|  | 71 | // REQUIRED. Specifies a list of domains used by users to access their website. | 
|  | 72 | Domains []Domain | 
|  | 73 | // REQUIRED. Specifies a list of origin domains or IP addresses where the | 
|  | 74 | // original assets are stored. | 
|  | 75 | Origins []Origin | 
|  | 76 | // REQUIRED. Specifies the CDN provider flavor ID to use. For a list of | 
|  | 77 | // flavors, see the operation to list the available flavors. The minimum | 
|  | 78 | // length for flavor_id is 1. The maximum length is 256. | 
|  | 79 | FlavorID string | 
|  | 80 | // OPTIONAL. Specifies the TTL rules for the assets under this service. Supports wildcards for fine-grained control. | 
| Jon Perritt | 0bd2373 | 2015-01-19 20:58:57 -0700 | [diff] [blame] | 81 | Caching []CacheRule | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 82 | // OPTIONAL. Specifies the restrictions that define who can access assets (content from the CDN cache). | 
|  | 83 | Restrictions []Restriction | 
|  | 84 | } | 
|  | 85 |  | 
|  | 86 | // ToCDNServiceCreateMap casts a CreateOpts struct to a map. | 
|  | 87 | func (opts CreateOpts) ToCDNServiceCreateMap() (map[string]interface{}, error) { | 
|  | 88 | s := make(map[string]interface{}) | 
|  | 89 |  | 
|  | 90 | if opts.Name == "" { | 
|  | 91 | return nil, no("Name") | 
|  | 92 | } | 
|  | 93 | s["name"] = opts.Name | 
|  | 94 |  | 
|  | 95 | if opts.Domains == nil { | 
|  | 96 | return nil, no("Domains") | 
|  | 97 | } | 
|  | 98 | for _, domain := range opts.Domains { | 
|  | 99 | if domain.Domain == "" { | 
|  | 100 | return nil, no("Domains[].Domain") | 
|  | 101 | } | 
|  | 102 | } | 
|  | 103 | s["domains"] = opts.Domains | 
|  | 104 |  | 
|  | 105 | if opts.Origins == nil { | 
|  | 106 | return nil, no("Origins") | 
|  | 107 | } | 
|  | 108 | for _, origin := range opts.Origins { | 
|  | 109 | if origin.Origin == "" { | 
|  | 110 | return nil, no("Origins[].Origin") | 
|  | 111 | } | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 112 | if origin.Rules == nil && len(opts.Origins) > 1 { | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 113 | return nil, no("Origins[].Rules") | 
|  | 114 | } | 
|  | 115 | for _, rule := range origin.Rules { | 
|  | 116 | if rule.Name == "" { | 
|  | 117 | return nil, no("Origins[].Rules[].Name") | 
|  | 118 | } | 
|  | 119 | if rule.RequestURL == "" { | 
|  | 120 | return nil, no("Origins[].Rules[].RequestURL") | 
|  | 121 | } | 
|  | 122 | } | 
|  | 123 | } | 
|  | 124 | s["origins"] = opts.Origins | 
|  | 125 |  | 
|  | 126 | if opts.FlavorID == "" { | 
|  | 127 | return nil, no("FlavorID") | 
|  | 128 | } | 
|  | 129 | s["flavor_id"] = opts.FlavorID | 
|  | 130 |  | 
|  | 131 | if opts.Caching != nil { | 
|  | 132 | for _, cache := range opts.Caching { | 
|  | 133 | if cache.Name == "" { | 
|  | 134 | return nil, no("Caching[].Name") | 
|  | 135 | } | 
|  | 136 | if cache.Rules != nil { | 
|  | 137 | for _, rule := range cache.Rules { | 
|  | 138 | if rule.Name == "" { | 
|  | 139 | return nil, no("Caching[].Rules[].Name") | 
|  | 140 | } | 
|  | 141 | if rule.RequestURL == "" { | 
|  | 142 | return nil, no("Caching[].Rules[].RequestURL") | 
|  | 143 | } | 
|  | 144 | } | 
|  | 145 | } | 
|  | 146 | } | 
|  | 147 | s["caching"] = opts.Caching | 
|  | 148 | } | 
|  | 149 |  | 
|  | 150 | if opts.Restrictions != nil { | 
|  | 151 | for _, restriction := range opts.Restrictions { | 
|  | 152 | if restriction.Name == "" { | 
|  | 153 | return nil, no("Restrictions[].Name") | 
|  | 154 | } | 
|  | 155 | if restriction.Rules != nil { | 
|  | 156 | for _, rule := range restriction.Rules { | 
|  | 157 | if rule.Name == "" { | 
|  | 158 | return nil, no("Restrictions[].Rules[].Name") | 
|  | 159 | } | 
|  | 160 | } | 
|  | 161 | } | 
|  | 162 | } | 
|  | 163 | s["restrictions"] = opts.Restrictions | 
|  | 164 | } | 
|  | 165 |  | 
|  | 166 | return s, nil | 
|  | 167 | } | 
|  | 168 |  | 
|  | 169 | // Create accepts a CreateOpts struct and creates a new CDN service using the | 
|  | 170 | // values provided. | 
|  | 171 | func Create(c *gophercloud.ServiceClient, opts CreateOptsBuilder) CreateResult { | 
|  | 172 | var res CreateResult | 
|  | 173 |  | 
|  | 174 | reqBody, err := opts.ToCDNServiceCreateMap() | 
|  | 175 | if err != nil { | 
|  | 176 | res.Err = err | 
|  | 177 | return res | 
|  | 178 | } | 
|  | 179 |  | 
|  | 180 | // Send request to API | 
| Jon Perritt | d21966f | 2015-01-20 19:22:45 -0700 | [diff] [blame] | 181 | resp, err := perigee.Request("POST", createURL(c), perigee.Options{ | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 182 | MoreHeaders: c.AuthenticatedHeaders(), | 
|  | 183 | ReqBody:     &reqBody, | 
|  | 184 | OkCodes:     []int{202}, | 
|  | 185 | }) | 
| Jon Perritt | d21966f | 2015-01-20 19:22:45 -0700 | [diff] [blame] | 186 | res.Header = resp.HttpResponse.Header | 
|  | 187 | res.Err = err | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 188 | return res | 
|  | 189 | } | 
|  | 190 |  | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 191 | // Get retrieves a specific service based on its URL or its unique ID. For | 
|  | 192 | // example, both "96737ae3-cfc1-4c72-be88-5d0e7cc9a3f0" and | 
|  | 193 | // "https://global.cdn.api.rackspacecloud.com/v1.0/services/96737ae3-cfc1-4c72-be88-5d0e7cc9a3f0" | 
|  | 194 | // are valid options for idOrURL. | 
|  | 195 | func Get(c *gophercloud.ServiceClient, idOrURL string) GetResult { | 
|  | 196 | var url string | 
|  | 197 | if strings.Contains(idOrURL, "/") { | 
|  | 198 | url = idOrURL | 
|  | 199 | } else { | 
|  | 200 | url = getURL(c, idOrURL) | 
|  | 201 | } | 
|  | 202 |  | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 203 | var res GetResult | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 204 | _, res.Err = perigee.Request("GET", url, perigee.Options{ | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 205 | MoreHeaders: c.AuthenticatedHeaders(), | 
|  | 206 | Results:     &res.Body, | 
|  | 207 | OkCodes:     []int{200}, | 
|  | 208 | }) | 
|  | 209 | return res | 
|  | 210 | } | 
|  | 211 |  | 
| Ash Wilson | a623ff7 | 2015-01-28 15:50:37 -0500 | [diff] [blame] | 212 | // Path is a JSON pointer location that indicates which service parameter is being added, replaced, | 
|  | 213 | // or removed. | 
|  | 214 | type Path struct { | 
|  | 215 | baseElement string | 
|  | 216 | } | 
|  | 217 |  | 
|  | 218 | func (p Path) renderDash() string { | 
|  | 219 | return fmt.Sprintf("/%s/-", p.baseElement) | 
|  | 220 | } | 
|  | 221 |  | 
|  | 222 | func (p Path) renderIndex(index int64) string { | 
|  | 223 | return fmt.Sprintf("/%s/%d", p.baseElement, index) | 
|  | 224 | } | 
|  | 225 |  | 
|  | 226 | var ( | 
|  | 227 | // PathDomains indicates that an update operation is to be performed on a Domain. | 
|  | 228 | PathDomains = Path{baseElement: "domains"} | 
|  | 229 |  | 
|  | 230 | // PathOrigins indicates that an update operation is to be performed on an Origin. | 
|  | 231 | PathOrigins = Path{baseElement: "origins"} | 
|  | 232 |  | 
|  | 233 | // PathCaching indicates that an update operation is to be performed on a CacheRule. | 
|  | 234 | PathCaching = Path{baseElement: "caching"} | 
|  | 235 | ) | 
|  | 236 |  | 
| Ash Wilson | 4ee0501 | 2015-01-28 16:13:43 -0500 | [diff] [blame] | 237 | type value interface { | 
| Ash Wilson | b47ebed | 2015-01-29 11:08:41 -0500 | [diff] [blame^] | 238 | toPatchValue() interface{} | 
| Ash Wilson | 4ee0501 | 2015-01-28 16:13:43 -0500 | [diff] [blame] | 239 | appropriatePath() Path | 
|  | 240 | } | 
|  | 241 |  | 
| Ash Wilson | 7b72953 | 2015-01-28 16:15:23 -0500 | [diff] [blame] | 242 | // Patch represents a single update to an existing Service. Multiple updates to a service can be | 
|  | 243 | // submitted at the same time. | 
|  | 244 | type Patch interface { | 
|  | 245 | ToCDNServiceUpdateMap() map[string]interface{} | 
|  | 246 | } | 
|  | 247 |  | 
| Ash Wilson | 299363d | 2015-01-29 10:49:40 -0500 | [diff] [blame] | 248 | // Insertion is a Patch that requests the addition of a value (Domain, Origin, or CacheRule) to | 
|  | 249 | // a Service at a fixed index. Use an Append instead to append the new value to the end of its | 
|  | 250 | // collection. Pass it to the Update function as part of the Patch slice. | 
|  | 251 | type Insertion struct { | 
|  | 252 | Index int64 | 
| Ash Wilson | 334277c | 2015-01-29 09:08:52 -0500 | [diff] [blame] | 253 | Value value | 
|  | 254 | } | 
|  | 255 |  | 
| Ash Wilson | 299363d | 2015-01-29 10:49:40 -0500 | [diff] [blame] | 256 | // ToCDNServiceUpdateMap converts an Insertion into a request body fragment suitable for the | 
| Ash Wilson | 334277c | 2015-01-29 09:08:52 -0500 | [diff] [blame] | 257 | // Update call. | 
| Ash Wilson | 299363d | 2015-01-29 10:49:40 -0500 | [diff] [blame] | 258 | func (i Insertion) ToCDNServiceUpdateMap() map[string]interface{} { | 
|  | 259 | return map[string]interface{}{ | 
|  | 260 | "op":    "add", | 
|  | 261 | "path":  i.Value.appropriatePath().renderIndex(i.Index), | 
|  | 262 | "value": i.Value.toPatchValue(), | 
|  | 263 | } | 
|  | 264 | } | 
|  | 265 |  | 
|  | 266 | // Append is a Patch that requests the addition of a value (Domain, Origin, or CacheRule) to a | 
|  | 267 | // Service at the end of its respective collection. Use an Insertion instead to insert the value | 
|  | 268 | // at a fixed index within the collection. Pass this to the Update function as part of its | 
|  | 269 | // Patch slice. | 
|  | 270 | type Append struct { | 
|  | 271 | Value value | 
|  | 272 | } | 
|  | 273 |  | 
|  | 274 | // ToCDNServiceUpdateMap converts an Append into a request body fragment suitable for the | 
|  | 275 | // Update call. | 
|  | 276 | func (a Append) ToCDNServiceUpdateMap() map[string]interface{} { | 
| Ash Wilson | 334277c | 2015-01-29 09:08:52 -0500 | [diff] [blame] | 277 | return map[string]interface{}{ | 
|  | 278 | "op":    "add", | 
|  | 279 | "path":  a.Value.appropriatePath().renderDash(), | 
|  | 280 | "value": a.Value.toPatchValue(), | 
|  | 281 | } | 
|  | 282 | } | 
|  | 283 |  | 
|  | 284 | // Replacement is a Patch that alters a specific service parameter (Domain, Origin, or CacheRule) | 
|  | 285 | // in-place by index. Pass it to the Update function as part of the Patch slice. | 
|  | 286 | type Replacement struct { | 
|  | 287 | Value value | 
|  | 288 | Index int64 | 
|  | 289 | } | 
|  | 290 |  | 
|  | 291 | // ToCDNServiceUpdateMap converts a Replacement into a request body fragment suitable for the | 
|  | 292 | // Update call. | 
|  | 293 | func (r Replacement) ToCDNServiceUpdateMap() map[string]interface{} { | 
|  | 294 | return map[string]interface{}{ | 
|  | 295 | "op":    "replace", | 
|  | 296 | "path":  r.Value.appropriatePath().renderIndex(r.Index), | 
|  | 297 | "value": r.Value.toPatchValue(), | 
|  | 298 | } | 
|  | 299 | } | 
|  | 300 |  | 
|  | 301 | // Removal is a Patch that requests the removal of a service parameter (Domain, Origin, or | 
|  | 302 | // CacheRule) by index. Pass it to the Update function as part of the Patch slice. | 
|  | 303 | type Removal struct { | 
|  | 304 | Path  Path | 
|  | 305 | Index int64 | 
|  | 306 | } | 
|  | 307 |  | 
|  | 308 | // ToCDNServiceUpdateMap converts a Removal into a request body fragment suitable for the | 
|  | 309 | // Update call. | 
|  | 310 | func (r Removal) ToCDNServiceUpdateMap() map[string]interface{} { | 
|  | 311 | return map[string]interface{}{ | 
|  | 312 | "op":   "remove", | 
|  | 313 | "path": r.Path.renderIndex(r.Index), | 
|  | 314 | } | 
|  | 315 | } | 
|  | 316 |  | 
| Ash Wilson | 299363d | 2015-01-29 10:49:40 -0500 | [diff] [blame] | 317 | // Update accepts a slice of Patch operations (Insertion, Append, Replacement or Removal) and | 
|  | 318 | // updates an existing CDN service using the values provided. idOrURL can be either the service's | 
|  | 319 | // URL or its ID. For example, both "96737ae3-cfc1-4c72-be88-5d0e7cc9a3f0" and | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 320 | // "https://global.cdn.api.rackspacecloud.com/v1.0/services/96737ae3-cfc1-4c72-be88-5d0e7cc9a3f0" | 
|  | 321 | // are valid options for idOrURL. | 
| Ash Wilson | 09d2a28 | 2015-01-29 10:05:53 -0500 | [diff] [blame] | 322 | func Update(c *gophercloud.ServiceClient, idOrURL string, patches []Patch) UpdateResult { | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 323 | var url string | 
|  | 324 | if strings.Contains(idOrURL, "/") { | 
|  | 325 | url = idOrURL | 
|  | 326 | } else { | 
|  | 327 | url = updateURL(c, idOrURL) | 
|  | 328 | } | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 329 |  | 
| Ash Wilson | 09d2a28 | 2015-01-29 10:05:53 -0500 | [diff] [blame] | 330 | reqBody := make([]map[string]interface{}, len(patches)) | 
|  | 331 | for i, patch := range patches { | 
|  | 332 | reqBody[i] = patch.ToCDNServiceUpdateMap() | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 333 | } | 
|  | 334 |  | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 335 | resp, err := perigee.Request("PATCH", url, perigee.Options{ | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 336 | MoreHeaders: c.AuthenticatedHeaders(), | 
|  | 337 | ReqBody:     &reqBody, | 
|  | 338 | OkCodes:     []int{202}, | 
|  | 339 | }) | 
| Ash Wilson | 09d2a28 | 2015-01-29 10:05:53 -0500 | [diff] [blame] | 340 | var result UpdateResult | 
|  | 341 | result.Header = resp.HttpResponse.Header | 
|  | 342 | result.Err = err | 
|  | 343 | return result | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 344 | } | 
|  | 345 |  | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 346 | // Delete accepts a service's ID or its URL and deletes the CDN service | 
|  | 347 | // associated with it. For example, both "96737ae3-cfc1-4c72-be88-5d0e7cc9a3f0" and | 
|  | 348 | // "https://global.cdn.api.rackspacecloud.com/v1.0/services/96737ae3-cfc1-4c72-be88-5d0e7cc9a3f0" | 
|  | 349 | // are valid options for idOrURL. | 
|  | 350 | func Delete(c *gophercloud.ServiceClient, idOrURL string) DeleteResult { | 
|  | 351 | var url string | 
|  | 352 | if strings.Contains(idOrURL, "/") { | 
|  | 353 | url = idOrURL | 
|  | 354 | } else { | 
|  | 355 | url = deleteURL(c, idOrURL) | 
|  | 356 | } | 
|  | 357 |  | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 358 | var res DeleteResult | 
| Jon Perritt | b8713ad | 2015-01-21 15:02:58 -0700 | [diff] [blame] | 359 | _, res.Err = perigee.Request("DELETE", url, perigee.Options{ | 
| Jon Perritt | e7b86d1 | 2015-01-16 20:37:11 -0700 | [diff] [blame] | 360 | MoreHeaders: c.AuthenticatedHeaders(), | 
|  | 361 | OkCodes:     []int{202}, | 
|  | 362 | }) | 
|  | 363 | return res | 
|  | 364 | } |