2013-06-13 10:03:10 -04:00
|
|
|
// All of the methods used to communicate with the digital_ocean API
|
|
|
|
// are here. Their API is on a path to V2, so just plain JSON is used
|
|
|
|
// in place of a proper client library for now.
|
|
|
|
|
|
|
|
package digitalocean
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2013-06-19 00:54:15 -04:00
|
|
|
"github.com/mitchellh/mapstructure"
|
2013-06-13 10:03:10 -04:00
|
|
|
"io/ioutil"
|
2013-06-14 09:26:03 -04:00
|
|
|
"log"
|
2013-06-13 10:03:10 -04:00
|
|
|
"net/http"
|
2013-06-14 09:26:03 -04:00
|
|
|
"net/url"
|
2013-06-19 16:22:48 -04:00
|
|
|
"strings"
|
2013-06-13 10:03:10 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
const DIGITALOCEAN_API_URL = "https://api.digitalocean.com"
|
|
|
|
|
2013-06-19 00:54:15 -04:00
|
|
|
type Image struct {
|
|
|
|
Id uint
|
|
|
|
Name string
|
|
|
|
Distribution string
|
|
|
|
}
|
|
|
|
|
|
|
|
type ImagesResp struct {
|
|
|
|
Images []Image
|
|
|
|
}
|
|
|
|
|
2013-06-13 10:03:10 -04:00
|
|
|
type DigitalOceanClient struct {
|
|
|
|
// The http client for communicating
|
2013-06-13 11:58:06 -04:00
|
|
|
client *http.Client
|
2013-06-13 10:03:10 -04:00
|
|
|
|
|
|
|
// The base URL of the API
|
|
|
|
BaseURL string
|
|
|
|
|
|
|
|
// Credentials
|
|
|
|
ClientID string
|
|
|
|
APIKey string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Creates a new client for communicating with DO
|
2013-06-13 11:58:06 -04:00
|
|
|
func (d DigitalOceanClient) New(client string, key string) *DigitalOceanClient {
|
2013-06-13 10:03:10 -04:00
|
|
|
c := &DigitalOceanClient{
|
|
|
|
client: http.DefaultClient,
|
|
|
|
BaseURL: DIGITALOCEAN_API_URL,
|
|
|
|
ClientID: client,
|
|
|
|
APIKey: key,
|
|
|
|
}
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
// Creates an SSH Key and returns it's id
|
|
|
|
func (d DigitalOceanClient) CreateKey(name string, pub string) (uint, error) {
|
2013-06-14 09:26:03 -04:00
|
|
|
// Escape the public key
|
|
|
|
pub = url.QueryEscape(pub)
|
|
|
|
|
|
|
|
params := fmt.Sprintf("name=%v&ssh_pub_key=%v", name, pub)
|
2013-06-13 10:03:10 -04:00
|
|
|
|
|
|
|
body, err := NewRequest(d, "ssh_keys/new", params)
|
|
|
|
if err != nil {
|
2013-06-13 11:58:06 -04:00
|
|
|
return 0, err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read the SSH key's ID we just created
|
|
|
|
key := body["ssh_key"].(map[string]interface{})
|
|
|
|
keyId := key["id"].(float64)
|
|
|
|
return uint(keyId), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Destroys an SSH key
|
|
|
|
func (d DigitalOceanClient) DestroyKey(id uint) error {
|
2013-06-14 09:26:03 -04:00
|
|
|
path := fmt.Sprintf("ssh_keys/%v/destroy", id)
|
2013-06-13 10:03:10 -04:00
|
|
|
_, err := NewRequest(d, path, "")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Creates a droplet and returns it's id
|
|
|
|
func (d DigitalOceanClient) CreateDroplet(name string, size uint, image uint, region uint, keyId uint) (uint, error) {
|
|
|
|
params := fmt.Sprintf(
|
2013-06-14 09:26:03 -04:00
|
|
|
"name=%v&image_id=%v&size_id=%v®ion_id=%v&ssh_key_ids=%v",
|
|
|
|
name, image, size, region, keyId)
|
2013-06-13 10:03:10 -04:00
|
|
|
|
|
|
|
body, err := NewRequest(d, "droplets/new", params)
|
|
|
|
if err != nil {
|
2013-06-13 11:58:06 -04:00
|
|
|
return 0, err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read the Droplets ID
|
|
|
|
droplet := body["droplet"].(map[string]interface{})
|
|
|
|
dropletId := droplet["id"].(float64)
|
2013-06-13 11:58:06 -04:00
|
|
|
return uint(dropletId), err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Destroys a droplet
|
|
|
|
func (d DigitalOceanClient) DestroyDroplet(id uint) error {
|
2013-06-14 09:26:03 -04:00
|
|
|
path := fmt.Sprintf("droplets/%v/destroy", id)
|
2013-06-13 10:03:10 -04:00
|
|
|
_, err := NewRequest(d, path, "")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Powers off a droplet
|
2013-06-13 11:58:06 -04:00
|
|
|
func (d DigitalOceanClient) PowerOffDroplet(id uint) error {
|
2013-06-14 09:26:03 -04:00
|
|
|
path := fmt.Sprintf("droplets/%v/power_off", id)
|
2013-06-13 10:03:10 -04:00
|
|
|
|
|
|
|
_, err := NewRequest(d, path, "")
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Creates a snaphot of a droplet by it's ID
|
|
|
|
func (d DigitalOceanClient) CreateSnapshot(id uint, name string) error {
|
2013-06-14 09:26:03 -04:00
|
|
|
path := fmt.Sprintf("droplets/%v/snapshot", id)
|
|
|
|
params := fmt.Sprintf("name=%v", name)
|
2013-06-13 10:03:10 -04:00
|
|
|
|
|
|
|
_, err := NewRequest(d, path, params)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2013-06-19 00:54:15 -04:00
|
|
|
// Returns all available images.
|
|
|
|
func (d DigitalOceanClient) Images() ([]Image, error) {
|
|
|
|
resp, err := NewRequest(d, "images", "")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var result ImagesResp
|
|
|
|
if err := mapstructure.Decode(resp, &result); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return result.Images, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Destroys an image by its ID.
|
|
|
|
func (d DigitalOceanClient) DestroyImage(id uint) error {
|
|
|
|
path := fmt.Sprintf("images/%d/destroy", id)
|
|
|
|
_, err := NewRequest(d, path, "")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2013-06-13 10:03:10 -04:00
|
|
|
// Returns DO's string representation of status "off" "new" "active" etc.
|
2013-06-13 12:48:19 -04:00
|
|
|
func (d DigitalOceanClient) DropletStatus(id uint) (string, string, error) {
|
2013-06-14 09:26:03 -04:00
|
|
|
path := fmt.Sprintf("droplets/%v", id)
|
2013-06-13 10:03:10 -04:00
|
|
|
|
|
|
|
body, err := NewRequest(d, path, "")
|
|
|
|
if err != nil {
|
2013-06-13 12:48:19 -04:00
|
|
|
return "", "", err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
2013-06-14 09:26:03 -04:00
|
|
|
var ip string
|
|
|
|
|
2013-06-13 10:03:10 -04:00
|
|
|
// Read the droplet's "status"
|
|
|
|
droplet := body["droplet"].(map[string]interface{})
|
|
|
|
status := droplet["status"].(string)
|
2013-06-14 09:26:03 -04:00
|
|
|
|
|
|
|
if droplet["ip_address"] != nil {
|
|
|
|
ip = droplet["ip_address"].(string)
|
|
|
|
}
|
2013-06-13 10:03:10 -04:00
|
|
|
|
2013-06-13 12:48:19 -04:00
|
|
|
return ip, status, err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sends an api request and returns a generic map[string]interface of
|
|
|
|
// the response.
|
|
|
|
func NewRequest(d DigitalOceanClient, path string, params string) (map[string]interface{}, error) {
|
|
|
|
client := d.client
|
2013-06-19 16:18:53 -04:00
|
|
|
url := fmt.Sprintf("%s/%s?%s&client_id=%s&api_key=%s",
|
2013-06-13 10:03:10 -04:00
|
|
|
DIGITALOCEAN_API_URL, path, params, d.ClientID, d.APIKey)
|
|
|
|
|
2013-06-13 11:58:06 -04:00
|
|
|
var decodedResponse map[string]interface{}
|
|
|
|
|
2013-06-19 16:18:53 -04:00
|
|
|
// Do some basic scrubbing so sensitive information doesn't appear in logs
|
2013-06-19 16:22:48 -04:00
|
|
|
scrubbedUrl := strings.Replace(url, d.ClientID, "CLIENT_ID", -1)
|
|
|
|
scrubbedUrl = strings.Replace(scrubbedUrl, d.APIKey, "API_KEY", -1)
|
2013-06-19 16:18:53 -04:00
|
|
|
log.Printf("sending new request to digitalocean: %s", scrubbedUrl)
|
2013-06-14 09:26:03 -04:00
|
|
|
|
2013-06-13 10:03:10 -04:00
|
|
|
resp, err := client.Get(url)
|
|
|
|
if err != nil {
|
2013-06-13 11:58:06 -04:00
|
|
|
return decodedResponse, err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
2013-06-13 11:58:06 -04:00
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
2013-06-13 10:03:10 -04:00
|
|
|
|
|
|
|
resp.Body.Close()
|
|
|
|
if err != nil {
|
2013-06-13 11:58:06 -04:00
|
|
|
return decodedResponse, err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
2013-06-14 09:26:03 -04:00
|
|
|
err = json.Unmarshal(body, &decodedResponse)
|
|
|
|
|
2013-06-17 08:31:47 -04:00
|
|
|
log.Printf("response from digitalocean: %v", decodedResponse)
|
|
|
|
|
2013-06-13 10:03:10 -04:00
|
|
|
// Catch all non-200 status and return an error
|
|
|
|
if resp.StatusCode != 200 {
|
2013-06-17 08:31:47 -04:00
|
|
|
err = errors.New(fmt.Sprintf("Received non-200 HTTP status from DigitalOcean: %v", resp.StatusCode))
|
2013-06-13 11:58:06 -04:00
|
|
|
return decodedResponse, err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2013-06-13 11:58:06 -04:00
|
|
|
return decodedResponse, err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Catch all non-OK statuses from DO and return an error
|
|
|
|
status := decodedResponse["status"]
|
|
|
|
if status != "OK" {
|
2013-06-17 08:22:29 -04:00
|
|
|
// Get the actual error message if there is one
|
|
|
|
if status == "ERROR" {
|
|
|
|
status = decodedResponse["error_message"]
|
|
|
|
}
|
2013-06-17 08:31:47 -04:00
|
|
|
err = errors.New(fmt.Sprintf("Received bad status from DigitalOcean: %v", status))
|
2013-06-13 11:58:06 -04:00
|
|
|
return decodedResponse, err
|
2013-06-13 10:03:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return decodedResponse, nil
|
|
|
|
}
|