blob: 4d4779c0cf650af4ff4251c263cc9934f1ed7601 [file] [log] [blame]
package stacks
import (
"errors"
"fmt"
"strings"
)
// an interface to represent stack environments
type Environment struct {
TE
}
// allowed sections in a stack environment file
var EnvironmentSections = map[string]bool{
"parameters": true,
"parameter_defaults": true,
"resource_registry": true,
}
func (e *Environment) Validate() error {
if e.Parsed == nil {
if err := e.Parse(); err != nil {
return err
}
}
for key, _ := range e.Parsed {
if _, ok := EnvironmentSections[key]; !ok {
return errors.New(fmt.Sprintf("Environment has wrong section: %s", key))
}
}
return nil
}
// Parse environment file to resolve the URL's of the resources. This is done by
// reading from the `Resource Registry` section, which is why the function is
// named GetRRFileContents.
func (e *Environment) getRRFileContents(ignoreIf igFunc) error {
// initialize environment if empty
if e.Files == nil {
e.Files = make(map[string]string)
}
if e.fileMaps == nil {
e.fileMaps = make(map[string]string)
}
// get the resource registry
rr := e.Parsed["resource_registry"]
// search the resource registry for URLs
switch rr.(type) {
// process further only if the resource registry is a map
case map[string]interface{}, map[interface{}]interface{}:
rr_map, err := toStringKeys(rr)
if err != nil {
return err
}
// the resource registry might contain a base URL for the resource. If
// such a field is present, use it. Otherwise, use the default base URL.
var baseURL string
if val, ok := rr_map["base_url"]; ok {
baseURL = val.(string)
} else {
baseURL = e.baseURL
}
// The contents of the resource may be located in a remote file, which
// will be a template. Instantiate a temporary template to manage the
// contents.
tempTemplate := new(Template)
tempTemplate.baseURL = baseURL
tempTemplate.client = e.client
// Fetch the contents of remote resource URL's
if err = tempTemplate.getFileContents(rr, ignoreIf, false); err != nil {
return err
}
// check the `resources` section (if it exists) for more URL's. Note that
// the previous call to GetFileContents was (deliberately) not recursive
// as we want more control over where to look for URL's
if val, ok := rr_map["resources"]; ok {
switch val.(type) {
// process further only if the contents are a map
case map[string]interface{}, map[interface{}]interface{}:
resources_map, err := toStringKeys(val)
if err != nil {
return err
}
for _, v := range resources_map {
switch v.(type) {
case map[string]interface{}, map[interface{}]interface{}:
resource_map, err := toStringKeys(v)
if err != nil {
return err
}
var resourceBaseURL string
// if base_url for the resource type is defined, use it
if val, ok := resource_map["base_url"]; ok {
resourceBaseURL = val.(string)
} else {
resourceBaseURL = baseURL
}
tempTemplate.baseURL = resourceBaseURL
if err := tempTemplate.getFileContents(v, ignoreIf, false); err != nil {
return err
}
}
}
}
}
// if the resource registry contained any URL's, store them. This can
// then be passed as parameter to api calls to Heat api.
e.Files = tempTemplate.Files
return nil
default:
return nil
}
}
// function to choose keys whose values are other environment files
func ignoreIfEnvironment(key string, value interface{}) bool {
// base_url and hooks refer to components which cannot have urls
if key == "base_url" || key == "hooks" {
return true
}
// if value is not string, it cannot be a URL
valueString, ok := value.(string)
if !ok {
return true
}
// if value contains `::`, it must be a reference to another resource type
// e.g. OS::Nova::Server : Rackspace::Cloud::Server
if strings.Contains(valueString, "::") {
return true
}
return false
}