270 lines
6.4 KiB
Go
270 lines
6.4 KiB
Go
package consul
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"strconv"
|
|
|
|
consulapi "github.com/hashicorp/consul/api"
|
|
"github.com/hashicorp/terraform/helper/schema"
|
|
)
|
|
|
|
func resourceConsulKeys() *schema.Resource {
|
|
return &schema.Resource{
|
|
Create: resourceConsulKeysCreate,
|
|
Update: resourceConsulKeysCreate,
|
|
Read: resourceConsulKeysRead,
|
|
Delete: resourceConsulKeysDelete,
|
|
|
|
SchemaVersion: 1,
|
|
MigrateState: resourceConsulKeysMigrateState,
|
|
|
|
Schema: map[string]*schema.Schema{
|
|
"datacenter": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Optional: true,
|
|
Computed: true,
|
|
ForceNew: true,
|
|
},
|
|
|
|
"token": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Optional: true,
|
|
},
|
|
|
|
"key": &schema.Schema{
|
|
Type: schema.TypeSet,
|
|
Optional: true,
|
|
Elem: &schema.Resource{
|
|
Schema: map[string]*schema.Schema{
|
|
"name": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Required: true,
|
|
},
|
|
|
|
"path": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Required: true,
|
|
},
|
|
|
|
"value": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Optional: true,
|
|
Computed: true,
|
|
},
|
|
|
|
"default": &schema.Schema{
|
|
Type: schema.TypeString,
|
|
Optional: true,
|
|
},
|
|
|
|
"delete": &schema.Schema{
|
|
Type: schema.TypeBool,
|
|
Optional: true,
|
|
Default: false,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
|
|
"var": &schema.Schema{
|
|
Type: schema.TypeMap,
|
|
Computed: true,
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
func resourceConsulKeysCreate(d *schema.ResourceData, meta interface{}) error {
|
|
client := meta.(*consulapi.Client)
|
|
kv := client.KV()
|
|
token := d.Get("token").(string)
|
|
dc, err := getDC(d, client)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Setup the operations using the datacenter
|
|
qOpts := consulapi.QueryOptions{Datacenter: dc, Token: token}
|
|
wOpts := consulapi.WriteOptions{Datacenter: dc, Token: token}
|
|
|
|
// Store the computed vars
|
|
vars := make(map[string]string)
|
|
|
|
// Extract the keys
|
|
keys := d.Get("key").(*schema.Set).List()
|
|
for _, raw := range keys {
|
|
key, path, sub, err := parseKey(raw)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
value := sub["value"].(string)
|
|
if value != "" {
|
|
log.Printf("[DEBUG] Setting key '%s' to '%v' in %s", path, value, dc)
|
|
pair := consulapi.KVPair{Key: path, Value: []byte(value)}
|
|
if _, err := kv.Put(&pair, &wOpts); err != nil {
|
|
return fmt.Errorf("Failed to set Consul key '%s': %v", path, err)
|
|
}
|
|
vars[key] = value
|
|
} else {
|
|
log.Printf("[DEBUG] Getting key '%s' in %s", path, dc)
|
|
pair, _, err := kv.Get(path, &qOpts)
|
|
if err != nil {
|
|
return fmt.Errorf("Failed to get Consul key '%s': %v", path, err)
|
|
}
|
|
value := attributeValue(sub, key, pair)
|
|
vars[key] = value
|
|
}
|
|
}
|
|
|
|
// The ID doesn't matter, since we use provider config, datacenter,
|
|
// and key paths to address consul properly. So we just need to fill it in
|
|
// with some value to indicate the resource has been created.
|
|
d.SetId("consul")
|
|
|
|
// Set the vars we collected above
|
|
if err := d.Set("var", vars); err != nil {
|
|
return err
|
|
}
|
|
// Store the datacenter on this resource, which can be helpful for reference
|
|
// in case it was read from the provider
|
|
d.Set("datacenter", dc)
|
|
|
|
return nil
|
|
}
|
|
|
|
func resourceConsulKeysRead(d *schema.ResourceData, meta interface{}) error {
|
|
client := meta.(*consulapi.Client)
|
|
kv := client.KV()
|
|
token := d.Get("token").(string)
|
|
dc, err := getDC(d, client)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Setup the operations using the datacenter
|
|
qOpts := consulapi.QueryOptions{Datacenter: dc, Token: token}
|
|
|
|
// Store the computed vars
|
|
vars := make(map[string]string)
|
|
|
|
// Extract the keys
|
|
keys := d.Get("key").(*schema.Set).List()
|
|
for _, raw := range keys {
|
|
key, path, sub, err := parseKey(raw)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
log.Printf("[DEBUG] Refreshing value of key '%s' in %s", path, dc)
|
|
pair, _, err := kv.Get(path, &qOpts)
|
|
if err != nil {
|
|
return fmt.Errorf("Failed to get value for path '%s' from Consul: %v", path, err)
|
|
}
|
|
|
|
value := attributeValue(sub, key, pair)
|
|
vars[key] = value
|
|
}
|
|
|
|
// Update the resource
|
|
if err := d.Set("var", vars); err != nil {
|
|
return err
|
|
}
|
|
// Store the datacenter on this resource, which can be helpful for reference
|
|
// in case it was read from the provider
|
|
d.Set("datacenter", dc)
|
|
|
|
return nil
|
|
}
|
|
|
|
func resourceConsulKeysDelete(d *schema.ResourceData, meta interface{}) error {
|
|
client := meta.(*consulapi.Client)
|
|
kv := client.KV()
|
|
token := d.Get("token").(string)
|
|
dc, err := getDC(d, client)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Setup the operations using the datacenter
|
|
wOpts := consulapi.WriteOptions{Datacenter: dc, Token: token}
|
|
|
|
// Extract the keys
|
|
keys := d.Get("key").(*schema.Set).List()
|
|
for _, raw := range keys {
|
|
_, path, sub, err := parseKey(raw)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Ignore if the key is non-managed
|
|
shouldDelete, ok := sub["delete"].(bool)
|
|
if !ok || !shouldDelete {
|
|
continue
|
|
}
|
|
|
|
log.Printf("[DEBUG] Deleting key '%s' in %s", path, dc)
|
|
if _, err := kv.Delete(path, &wOpts); err != nil {
|
|
return fmt.Errorf("Failed to delete Consul key '%s': %v", path, err)
|
|
}
|
|
}
|
|
|
|
// Clear the ID
|
|
d.SetId("")
|
|
return nil
|
|
}
|
|
|
|
// parseKey is used to parse a key into a name, path, config or error
|
|
func parseKey(raw interface{}) (string, string, map[string]interface{}, error) {
|
|
sub, ok := raw.(map[string]interface{})
|
|
if !ok {
|
|
return "", "", nil, fmt.Errorf("Failed to unroll: %#v", raw)
|
|
}
|
|
|
|
key, ok := sub["name"].(string)
|
|
if !ok {
|
|
return "", "", nil, fmt.Errorf("Failed to expand key '%#v'", sub)
|
|
}
|
|
|
|
path, ok := sub["path"].(string)
|
|
if !ok {
|
|
return "", "", nil, fmt.Errorf("Failed to get path for key '%s'", key)
|
|
}
|
|
return key, path, sub, nil
|
|
}
|
|
|
|
// attributeValue determines the value for a key, potentially
|
|
// using a default value if provided.
|
|
func attributeValue(sub map[string]interface{}, key string, pair *consulapi.KVPair) string {
|
|
// Use the value if given
|
|
if pair != nil {
|
|
return string(pair.Value)
|
|
}
|
|
|
|
// Use a default if given
|
|
if raw, ok := sub["default"]; ok {
|
|
switch def := raw.(type) {
|
|
case string:
|
|
return def
|
|
case bool:
|
|
return strconv.FormatBool(def)
|
|
}
|
|
}
|
|
|
|
// No value
|
|
return ""
|
|
}
|
|
|
|
// getDC is used to get the datacenter of the local agent
|
|
func getDC(d *schema.ResourceData, client *consulapi.Client) (string, error) {
|
|
if v, ok := d.GetOk("datacenter"); ok {
|
|
return v.(string), nil
|
|
}
|
|
info, err := client.Agent().Self()
|
|
if err != nil {
|
|
return "", fmt.Errorf("Failed to get datacenter from Consul agent: %v", err)
|
|
}
|
|
return info["Config"]["Datacenter"].(string), nil
|
|
}
|