blob: 15b1988ef8437a58cebacf64cad16261c15fc6e7 [file] [log] [blame]
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
}
type TypeSchema struct {
Attributes map[string]interface{} `mapstructure:"attributes"`
Properties map[string]interface{} `mapstrucutre:"properties"`
ResourceType string `mapstructure:"resource_type"`
}
type SchemaResult struct {
gophercloud.Result
}
func (r SchemaResult) Extract() (*TypeSchema, error) {
if r.Err != nil {
return nil, r.Err
}
var res TypeSchema
if err := mapstructure.Decode(r.Body, &res); err != nil {
return nil, err
}
return &res, nil
}
type TypeTemplate struct {
HeatTemplateFormatVersion string
Outputs map[string]interface{}
Parameters map[string]interface{}
Resources map[string]interface{}
}
type TemplateResult struct {
gophercloud.Result
}
func (r TemplateResult) Extract() (*TypeTemplate, error) {
if r.Err != nil {
return nil, r.Err
}
var res TypeTemplate
if err := mapstructure.Decode(r.Body, &res); err != nil {
return nil, err
}
return &res, nil
}