blob: 714c244116cf410c6bda09174deb3d70c52f09ae [file] [log] [blame]
//
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
//
package cloudstack
import (
"log"
"github.com/apache/cloudstack-go/v2/cloudstack"
"github.com/hashicorp/terraform/helper/schema"
)
// metadataSchema returns the schema to use for metadata
func metadataSchema() *schema.Schema {
return &schema.Schema{
Type: schema.TypeMap,
Optional: true,
Computed: true,
}
}
// setMetadata is a helper to set the metadata for a resource. It expects the
// metadata field to be named "metadata"
func setMetadata(cs *cloudstack.CloudStackClient, d *schema.ResourceData, resourceType string) error {
if metadata, ok := d.GetOk("metadata"); ok {
p := cs.Resourcemetadata.NewAddResourceDetailParams(
tagsFromSchema(metadata.(map[string]interface{})),
d.Id(), resourceType,
)
_, err := cs.Resourcemetadata.AddResourceDetail(p)
if err != nil {
return err
}
}
return nil
}
func getMetadata(cs *cloudstack.CloudStackClient, d *schema.ResourceData, resourceType string) (map[string]interface{}, error) {
p := cs.Resourcemetadata.NewListResourceDetailsParams(resourceType)
p.SetResourceid(d.Id())
response, err := cs.Resourcemetadata.ListResourceDetails(p)
if err != nil {
return nil, err
}
// Only return metadata values that were explicitely set
var existingFilter map[string]interface{}
if metadata, ok := d.GetOk("metadata"); ok {
existingFilter = metadata.(map[string]interface{})
}
metadata := make(map[string]interface{}, response.Count)
for _, detail := range response.ResourceDetails {
if _, ok := existingFilter[detail.Key]; ok {
metadata[detail.Key] = detail.Value
}
}
return metadata, nil
}
// updateMetadata is a helper to update only when metadata field change metadata
// field to be named "metadata"
func updateMetadata(cs *cloudstack.CloudStackClient, d *schema.ResourceData, resourceType string) error {
oraw, nraw := d.GetChange("metadata")
o := oraw.(map[string]interface{})
n := nraw.(map[string]interface{})
remove, create := diffTags(tagsFromSchema(o), tagsFromSchema(n))
log.Printf("[DEBUG] metadata to remove: %v", remove)
log.Printf("[DEBUG] metadata to create: %v", create)
// First remove any obsolete metadata
if len(remove) > 0 {
log.Printf("[DEBUG] Removing metadata: %v from %s", remove, d.Id())
p := cs.Resourcemetadata.NewRemoveResourceDetailParams(d.Id(), resourceType)
for key := range remove {
p.SetKey(key)
_, err := cs.Resourcemetadata.RemoveResourceDetail(p)
if err != nil {
return err
}
}
}
// Then add any new metadata
if len(create) > 0 {
log.Printf("[DEBUG] Creating metadata: %v for %s", create, d.Id())
p := cs.Resourcemetadata.NewAddResourceDetailParams(create, d.Id(), resourceType)
_, err := cs.Resourcemetadata.AddResourceDetail(p)
if err != nil {
return err
}
}
return nil
}