| package stackresources |
| |
| import ( |
| "time" |
| |
| "github.com/mitchellh/mapstructure" |
| "github.com/rackspace/gophercloud" |
| "github.com/rackspace/gophercloud/pagination" |
| ) |
| |
| type Resource struct { |
| Links []gophercloud.Link `mapstructure:"links"` |
| LogicalID string `mapstructure:"logical_resource_id"` |
| Name string `mapstructure:"resource_name"` |
| PhysicalID string `mapstructure:"physical_resource_id"` |
| RequiredBy []interface{} `mapstructure:"required_by"` |
| Status string `mapstructure:"resource_status"` |
| StatusReason string `mapstructure:"resource_status_reason"` |
| Type string `mapstructure:"resource_type"` |
| UpdatedTime time.Time `mapstructure:"-"` |
| } |
| |
| type FindResult struct { |
| gophercloud.Result |
| } |
| |
| func (r FindResult) Extract() ([]Resource, error) { |
| if r.Err != nil { |
| return nil, r.Err |
| } |
| |
| var res struct { |
| Res []Resource `mapstructure:"resources"` |
| } |
| |
| if err := mapstructure.Decode(r.Body, &res); err != nil { |
| return nil, err |
| } |
| |
| resources := r.Body.(map[string]interface{})["resources"].([]interface{}) |
| |
| for i, resourceRaw := range resources { |
| resource := resourceRaw.(map[string]interface{}) |
| if date, ok := resource["updated_time"]; ok && date != nil { |
| t, err := time.Parse(time.RFC3339, date.(string)) |
| if err != nil { |
| return nil, err |
| } |
| res.Res[i].UpdatedTime = t |
| } |
| } |
| |
| return res.Res, nil |
| } |
| |
| // ResourcePage abstracts the raw results of making a List() request against the API. |
| // As OpenStack extensions may freely alter the response bodies of structures returned to the client, you may only safely access the |
| // data provided through the ExtractResources call. |
| type ResourcePage struct { |
| pagination.LinkedPageBase |
| } |
| |
| // IsEmpty returns true if a page contains no Server results. |
| func (page ResourcePage) IsEmpty() (bool, error) { |
| resources, err := ExtractResources(page) |
| if err != nil { |
| return true, err |
| } |
| return len(resources) == 0, nil |
| } |
| |
| // NextPageURL uses the response's embedded link reference to navigate to the next page of results. |
| func (page ResourcePage) NextPageURL() (string, error) { |
| type resp struct { |
| Links []gophercloud.Link `mapstructure:"servers_links"` |
| } |
| |
| var r resp |
| err := mapstructure.Decode(page.Body, &r) |
| if err != nil { |
| return "", err |
| } |
| |
| return gophercloud.ExtractNextURL(r.Links) |
| } |
| |
| // ExtractResources interprets the results of a single page from a List() call, producing a slice of Resource entities. |
| func ExtractResources(page pagination.Page) ([]Resource, error) { |
| casted := page.(ResourcePage).Body |
| |
| var response struct { |
| Resources []Resource `mapstructure:"resources"` |
| } |
| err := mapstructure.Decode(casted, &response) |
| |
| resources := casted.(map[string]interface{})["resources"].([]interface{}) |
| |
| for i, resourceRaw := range resources { |
| resource := resourceRaw.(map[string]interface{}) |
| if date, ok := resource["updated_time"]; ok && date != nil { |
| t, err := time.Parse(time.RFC3339, date.(string)) |
| if err != nil { |
| return nil, err |
| } |
| response.Resources[i].UpdatedTime = t |
| } |
| } |
| |
| return response.Resources, err |
| } |
| |
| type GetResult struct { |
| gophercloud.Result |
| } |
| |
| func (r GetResult) Extract() (*Resource, error) { |
| if r.Err != nil { |
| return nil, r.Err |
| } |
| |
| var res struct { |
| Res *Resource `mapstructure:"resource"` |
| } |
| |
| if err := mapstructure.Decode(r.Body, &res); err != nil { |
| return nil, err |
| } |
| |
| resource := r.Body.(map[string]interface{})["resource"].(map[string]interface{}) |
| |
| if date, ok := resource["updated_time"]; ok && date != nil { |
| t, err := time.Parse(time.RFC3339, date.(string)) |
| if err != nil { |
| return nil, err |
| } |
| res.Res.UpdatedTime = t |
| } |
| |
| return res.Res, nil |
| } |
| |
| type MetadataResult struct { |
| gophercloud.Result |
| } |
| |
| func (r MetadataResult) Extract() (map[string]string, error) { |
| if r.Err != nil { |
| return nil, r.Err |
| } |
| |
| var res struct { |
| Meta map[string]string `mapstructure:"metadata"` |
| } |
| |
| if err := mapstructure.Decode(r.Body, &res); err != nil { |
| return nil, err |
| } |
| |
| return res.Meta, nil |
| } |
| |
| // ResourceTypePage abstracts the raw results of making a ListTypes() request against the API. |
| // As OpenStack extensions may freely alter the response bodies of structures returned to the client, you may only safely access the |
| // data provided through the ExtractResourceTypes call. |
| type ResourceTypePage struct { |
| pagination.SinglePageBase |
| } |
| |
| // IsEmpty returns true if a ResourceTypePage contains no resource types. |
| func (r ResourceTypePage) IsEmpty() (bool, error) { |
| rts, err := ExtractResourceTypes(r) |
| if err != nil { |
| return true, err |
| } |
| return len(rts) == 0, nil |
| } |
| |
| // ExtractResourceTypes extracts and returns resource types. |
| func ExtractResourceTypes(page pagination.Page) ([]string, error) { |
| var response struct { |
| ResourceTypes []string `mapstructure:"resource_types"` |
| } |
| |
| err := mapstructure.Decode(page.(ResourceTypePage).Body, &response) |
| return response.ResourceTypes, err |
| } |