2014-08-18 17:49:54 -05:00
|
|
|
package heroku
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
2014-08-27 23:05:09 -05:00
|
|
|
"strings"
|
2014-08-30 19:02:11 -05:00
|
|
|
"sync"
|
2014-08-18 17:49:54 -05:00
|
|
|
|
2014-08-27 09:50:37 -05:00
|
|
|
"github.com/cyberdelia/heroku-go/v3"
|
2014-08-18 17:49:54 -05:00
|
|
|
"github.com/hashicorp/terraform/helper/schema"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Global lock to prevent parallelism for heroku_addon since
|
|
|
|
// the Heroku API cannot handle a single application requesting
|
|
|
|
// multiple addons simultaneously.
|
|
|
|
var addonLock sync.Mutex
|
|
|
|
|
|
|
|
func resourceHerokuAddon() *schema.Resource {
|
|
|
|
return &schema.Resource{
|
|
|
|
Create: resourceHerokuAddonCreate,
|
|
|
|
Read: resourceHerokuAddonRead,
|
|
|
|
Update: resourceHerokuAddonUpdate,
|
|
|
|
Delete: resourceHerokuAddonDelete,
|
|
|
|
|
|
|
|
Schema: map[string]*schema.Schema{
|
|
|
|
"app": &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
Required: true,
|
|
|
|
ForceNew: true,
|
|
|
|
},
|
|
|
|
|
|
|
|
"plan": &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
Required: true,
|
|
|
|
},
|
|
|
|
|
|
|
|
"config": &schema.Schema{
|
|
|
|
Type: schema.TypeList,
|
|
|
|
Optional: true,
|
|
|
|
ForceNew: true,
|
|
|
|
Elem: &schema.Schema{
|
|
|
|
Type: schema.TypeMap,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
"provider_id": &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
Computed: true,
|
|
|
|
},
|
|
|
|
|
|
|
|
"config_vars": &schema.Schema{
|
|
|
|
Type: schema.TypeList,
|
|
|
|
Computed: true,
|
2015-05-07 09:11:31 -05:00
|
|
|
Elem: &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
},
|
2014-08-18 17:49:54 -05:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func resourceHerokuAddonCreate(d *schema.ResourceData, meta interface{}) error {
|
|
|
|
addonLock.Lock()
|
|
|
|
defer addonLock.Unlock()
|
|
|
|
|
2014-08-27 09:50:37 -05:00
|
|
|
client := meta.(*heroku.Service)
|
2014-08-18 17:49:54 -05:00
|
|
|
|
|
|
|
app := d.Get("app").(string)
|
2014-08-27 09:50:37 -05:00
|
|
|
opts := heroku.AddonCreateOpts{Plan: d.Get("plan").(string)}
|
2014-08-18 17:49:54 -05:00
|
|
|
|
|
|
|
if v := d.Get("config"); v != nil {
|
|
|
|
config := make(map[string]string)
|
|
|
|
for _, v := range v.([]interface{}) {
|
|
|
|
for k, v := range v.(map[string]interface{}) {
|
|
|
|
config[k] = v.(string)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
opts.Config = &config
|
|
|
|
}
|
|
|
|
|
2014-08-27 09:50:37 -05:00
|
|
|
log.Printf("[DEBUG] Addon create configuration: %#v, %#v", app, opts)
|
|
|
|
a, err := client.AddonCreate(app, opts)
|
2014-08-18 17:49:54 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-08-27 09:50:37 -05:00
|
|
|
d.SetId(a.ID)
|
2014-08-18 17:49:54 -05:00
|
|
|
log.Printf("[INFO] Addon ID: %s", d.Id())
|
|
|
|
|
|
|
|
return resourceHerokuAddonRead(d, meta)
|
|
|
|
}
|
|
|
|
|
|
|
|
func resourceHerokuAddonRead(d *schema.ResourceData, meta interface{}) error {
|
2014-08-27 09:50:37 -05:00
|
|
|
client := meta.(*heroku.Service)
|
2014-08-18 17:49:54 -05:00
|
|
|
|
2014-12-15 08:26:17 -06:00
|
|
|
addon, err := resourceHerokuAddonRetrieve(
|
2014-08-18 17:49:54 -05:00
|
|
|
d.Get("app").(string), d.Id(), client)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-08-27 23:05:09 -05:00
|
|
|
// Determine the plan. If we were configured without a specific plan,
|
|
|
|
// then just avoid the plan altogether (accepting anything that
|
|
|
|
// Heroku sends down).
|
|
|
|
plan := addon.Plan.Name
|
|
|
|
if v := d.Get("plan").(string); v != "" {
|
|
|
|
if idx := strings.IndexRune(v, ':'); idx == -1 {
|
|
|
|
idx = strings.IndexRune(plan, ':')
|
|
|
|
if idx > -1 {
|
|
|
|
plan = plan[:idx]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-18 17:49:54 -05:00
|
|
|
d.Set("name", addon.Name)
|
2014-08-27 23:05:09 -05:00
|
|
|
d.Set("plan", plan)
|
2014-08-27 09:50:37 -05:00
|
|
|
d.Set("provider_id", addon.ProviderID)
|
2015-05-07 09:11:31 -05:00
|
|
|
if err := d.Set("config_vars", addon.ConfigVars); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2014-08-18 17:49:54 -05:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func resourceHerokuAddonUpdate(d *schema.ResourceData, meta interface{}) error {
|
2014-08-27 09:50:37 -05:00
|
|
|
client := meta.(*heroku.Service)
|
2014-08-18 17:49:54 -05:00
|
|
|
|
|
|
|
app := d.Get("app").(string)
|
|
|
|
|
|
|
|
if d.HasChange("plan") {
|
|
|
|
ad, err := client.AddonUpdate(
|
2014-08-27 09:50:37 -05:00
|
|
|
app, d.Id(), heroku.AddonUpdateOpts{Plan: d.Get("plan").(string)})
|
2014-08-18 17:49:54 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store the new ID
|
2014-08-27 09:50:37 -05:00
|
|
|
d.SetId(ad.ID)
|
2014-08-18 17:49:54 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return resourceHerokuAddonRead(d, meta)
|
|
|
|
}
|
|
|
|
|
|
|
|
func resourceHerokuAddonDelete(d *schema.ResourceData, meta interface{}) error {
|
2014-08-27 09:50:37 -05:00
|
|
|
client := meta.(*heroku.Service)
|
2014-08-18 17:49:54 -05:00
|
|
|
|
|
|
|
log.Printf("[INFO] Deleting Addon: %s", d.Id())
|
|
|
|
|
|
|
|
// Destroy the app
|
|
|
|
err := client.AddonDelete(d.Get("app").(string), d.Id())
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Error deleting addon: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
d.SetId("")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-12-15 08:26:17 -06:00
|
|
|
func resourceHerokuAddonRetrieve(app string, id string, client *heroku.Service) (*heroku.Addon, error) {
|
2014-08-18 17:49:54 -05:00
|
|
|
addon, err := client.AddonInfo(app, id)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("Error retrieving addon: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return addon, nil
|
|
|
|
}
|