2016-01-15 12:33:17 -05:00
|
|
|
package docker
|
|
|
|
|
|
|
|
|
|
import (
|
2018-07-03 11:30:53 -04:00
|
|
|
"context"
|
2016-01-15 12:33:17 -05:00
|
|
|
"fmt"
|
2018-07-03 11:30:53 -04:00
|
|
|
"github.com/docker/docker/api/types"
|
|
|
|
|
"github.com/docker/docker/api/types/volume"
|
|
|
|
|
"github.com/hashicorp/terraform/helper/resource"
|
|
|
|
|
"github.com/hashicorp/terraform/helper/schema"
|
2018-01-29 06:01:46 -05:00
|
|
|
"log"
|
2018-05-16 12:00:04 -04:00
|
|
|
"strings"
|
2018-01-29 06:01:46 -05:00
|
|
|
"time"
|
2016-01-15 12:33:17 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func resourceDockerVolume() *schema.Resource {
|
|
|
|
|
return &schema.Resource{
|
|
|
|
|
Create: resourceDockerVolumeCreate,
|
|
|
|
|
Read: resourceDockerVolumeRead,
|
|
|
|
|
Delete: resourceDockerVolumeDelete,
|
|
|
|
|
|
|
|
|
|
Schema: map[string]*schema.Schema{
|
|
|
|
|
"name": &schema.Schema{
|
|
|
|
|
Type: schema.TypeString,
|
|
|
|
|
Optional: true,
|
|
|
|
|
Computed: true,
|
|
|
|
|
ForceNew: true,
|
|
|
|
|
},
|
|
|
|
|
"driver": &schema.Schema{
|
|
|
|
|
Type: schema.TypeString,
|
|
|
|
|
Optional: true,
|
|
|
|
|
Computed: true,
|
|
|
|
|
ForceNew: true,
|
|
|
|
|
},
|
|
|
|
|
"driver_opts": &schema.Schema{
|
|
|
|
|
Type: schema.TypeMap,
|
|
|
|
|
Optional: true,
|
|
|
|
|
ForceNew: true,
|
|
|
|
|
},
|
|
|
|
|
"mountpoint": &schema.Schema{
|
|
|
|
|
Type: schema.TypeString,
|
|
|
|
|
Computed: true,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func resourceDockerVolumeCreate(d *schema.ResourceData, meta interface{}) error {
|
2017-11-21 04:14:07 -05:00
|
|
|
client := meta.(*ProviderConfig).DockerClient
|
2018-07-03 11:30:53 -04:00
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
|
|
createOpts := volume.VolumeCreateBody{}
|
2016-01-15 12:33:17 -05:00
|
|
|
|
|
|
|
|
if v, ok := d.GetOk("name"); ok {
|
|
|
|
|
createOpts.Name = v.(string)
|
|
|
|
|
}
|
|
|
|
|
if v, ok := d.GetOk("driver"); ok {
|
|
|
|
|
createOpts.Driver = v.(string)
|
|
|
|
|
}
|
|
|
|
|
if v, ok := d.GetOk("driver_opts"); ok {
|
|
|
|
|
createOpts.DriverOpts = mapTypeMapValsToString(v.(map[string]interface{}))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var err error
|
2018-07-03 11:30:53 -04:00
|
|
|
var retVolume types.Volume
|
|
|
|
|
retVolume, err = client.VolumeCreate(ctx, createOpts)
|
|
|
|
|
|
|
|
|
|
if err != nil {
|
2016-01-15 12:33:17 -05:00
|
|
|
return fmt.Errorf("Unable to create volume: %s", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
d.SetId(retVolume.Name)
|
|
|
|
|
d.Set("name", retVolume.Name)
|
|
|
|
|
d.Set("driver", retVolume.Driver)
|
|
|
|
|
d.Set("mountpoint", retVolume.Mountpoint)
|
|
|
|
|
|
2018-07-03 11:30:53 -04:00
|
|
|
return resourceDockerVolumeRead(d, meta)
|
2016-01-15 12:33:17 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func resourceDockerVolumeRead(d *schema.ResourceData, meta interface{}) error {
|
2017-11-21 04:14:07 -05:00
|
|
|
client := meta.(*ProviderConfig).DockerClient
|
2018-07-03 11:30:53 -04:00
|
|
|
ctx := context.Background()
|
2016-01-15 12:33:17 -05:00
|
|
|
|
|
|
|
|
var err error
|
2018-07-03 11:30:53 -04:00
|
|
|
var retVolume types.Volume
|
|
|
|
|
retVolume, err = client.VolumeInspect(ctx, d.Id())
|
|
|
|
|
|
|
|
|
|
if err != nil {
|
2016-01-15 12:33:17 -05:00
|
|
|
return fmt.Errorf("Unable to inspect volume: %s", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
d.Set("name", retVolume.Name)
|
|
|
|
|
d.Set("driver", retVolume.Driver)
|
|
|
|
|
d.Set("mountpoint", retVolume.Mountpoint)
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func resourceDockerVolumeDelete(d *schema.ResourceData, meta interface{}) error {
|
2018-07-03 11:30:53 -04:00
|
|
|
log.Printf("[INFO] Waiting for volume: '%s' to get removed: max '%v seconds'", d.Id(), 30)
|
|
|
|
|
|
|
|
|
|
stateConf := &resource.StateChangeConf{
|
|
|
|
|
Pending: []string{"in_use"},
|
|
|
|
|
Target: []string{"removed"},
|
|
|
|
|
Refresh: resourceDockerVolumeRemoveRefreshFunc(d.Id(), meta),
|
|
|
|
|
Timeout: 30 * time.Second,
|
|
|
|
|
MinTimeout: 5 * time.Second,
|
|
|
|
|
Delay: 2 * time.Second,
|
|
|
|
|
}
|
2016-01-15 12:33:17 -05:00
|
|
|
|
2018-07-03 11:30:53 -04:00
|
|
|
// Wait, catching any errors
|
|
|
|
|
_, err := stateConf.WaitForState()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
2016-01-15 12:33:17 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
d.SetId("")
|
|
|
|
|
return nil
|
|
|
|
|
}
|
2018-07-03 11:30:53 -04:00
|
|
|
|
|
|
|
|
func resourceDockerVolumeRemoveRefreshFunc(
|
|
|
|
|
volumeID string, meta interface{}) resource.StateRefreshFunc {
|
|
|
|
|
return func() (interface{}, string, error) {
|
|
|
|
|
client := meta.(*ProviderConfig).DockerClient
|
|
|
|
|
forceDelete := true
|
|
|
|
|
|
|
|
|
|
if err := client.VolumeRemove(context.Background(), volumeID, forceDelete); err != nil {
|
|
|
|
|
if strings.Contains(err.Error(), "volume is in use") { // store.IsInUse(err)
|
|
|
|
|
log.Printf("[INFO] Volume with id '%v' is still in use", volumeID)
|
|
|
|
|
return volumeID, "in_use", nil
|
|
|
|
|
}
|
|
|
|
|
log.Printf("[INFO] Removing volume with id '%v' caused an error: %v", volumeID, err)
|
|
|
|
|
return nil, "", err
|
|
|
|
|
}
|
|
|
|
|
log.Printf("[INFO] Removing volume with id '%v' got removed", volumeID)
|
|
|
|
|
return volumeID, "removed", nil
|
|
|
|
|
}
|
|
|
|
|
}
|