/* Copyright 2016-2017 gtalent2@gmail.com This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0. If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/. */ package main import ( "errors" "github.com/digitalocean/godo" "golang.org/x/oauth2" "log" "time" ) const DROPLET_NS = "dospin-" type tokenSource struct { AccessToken string } func (t *tokenSource) Token() (*oauth2.Token, error) { token := &oauth2.Token{ AccessToken: t.AccessToken, } return token, nil } func sshKeys(ids []int) []godo.DropletCreateSSHKey { var out []godo.DropletCreateSSHKey for _, id := range ids { out = append(out, godo.DropletCreateSSHKey{ID: id}) } return out } func volumes(names []string) []godo.DropletCreateVolume { var out []godo.DropletCreateVolume for _, name := range names { out = append(out, godo.DropletCreateVolume{Name: name}) } return out } type DropletHandler struct { client *godo.Client settings Settings } func NewDropletHandler(settings Settings) *DropletHandler { retval := new(DropletHandler) retval.settings = settings // setup DO client tokenSource := &tokenSource{settings.ApiToken} oauthClient := oauth2.NewClient(oauth2.NoContext, tokenSource) retval.client = godo.NewClient(oauthClient) return retval } /* Gets the Droplet if it already exists, instantiates it if it does not. */ func (me *DropletHandler) Spinup(name string) (string, error) { vd := me.settings.Servers[name] if droplet, err := me.getDroplet(name); err == nil { if vd.UsePublicIP { return droplet.PublicIPv4() } else { return droplet.PrivateIPv4() } } else { // create the droplet var image godo.DropletCreateImage if vd.ImageSlug == "" { snapshot, err := me.getSnapshot(name) if err != nil { return "", err } image = godo.DropletCreateImage{ ID: snapshot.ID, } } else { image = godo.DropletCreateImage{ Slug: vd.ImageSlug, } } // determine droplet size var size string if vd.InitialSize != "" { size = vd.InitialSize } else { size = vd.Size } createRequest := &godo.DropletCreateRequest{ Name: DROPLET_NS + name, Region: vd.Region, Size: size, PrivateNetworking: true, SSHKeys: sshKeys(vd.SshKeys), Volumes: volumes(vd.Volumes), UserData: vd.UserData, Image: image, } log.Println("Spinup: Creating " + name) droplet, _, err := me.client.Droplets.Create(createRequest) if err != nil { log.Println(err) if droplet == nil { return "", err } } // wait until machine is ready for { d, _, err := me.client.Droplets.Get(droplet.ID) if err != nil { log.Println(err) return "", err } else if d.Status == "active" { break } time.Sleep(500 * time.Millisecond) } log.Println("Spinup: Created " + name) // resize if necessary if vd.InitialSize != "" && vd.InitialSize != vd.Size { // power off me.poweroff(name) // resize log.Println("Spinup: Resizing " + name) action, _, err := me.client.DropletActions.Resize(droplet.ID, vd.Size, false) if err != nil || !me.actionWait(action.ID) { return "", err } log.Println("Spinup: Resized " + name) // power back on log.Println("Spinup: Powering on " + name) action, _, err = me.client.DropletActions.PowerOn(droplet.ID) if err != nil || !me.actionWait(action.ID) { return "", err } log.Println("Spinup: Powered on " + name) } // delete the image log.Println("Spinup: Deleting image " + name) if image.ID > -1 { _, err = me.client.Images.Delete(image.ID) if err != nil { log.Println("Spinup: Could not delete image: ", err) } else { log.Println("Spinup: Deleted image " + name) } } // get the private IP and return it // get new copy of droplet that has IP droplet, _, err = me.client.Droplets.Get(droplet.ID) if err == nil { if vd.UsePublicIP { return droplet.PublicIPv4() } else { return droplet.PrivateIPv4() } } else { return "", err } } } func (me *DropletHandler) Spindown(name string) error { droplet, err := me.getDroplet(name) if err != nil { return err } // power off err = me.poweroff(name) if err != nil { return err } // snapshot existing droplet log.Println("Spindown: Creating image " + name) action, _, err := me.client.DropletActions.Snapshot(droplet.ID, DROPLET_NS+name) if err != nil || !me.actionWait(action.ID) { return err } log.Println("Spindown: Created image " + name) // delete droplet log.Println("Spindown: Deleting droplet " + name) _, err = me.client.Droplets.Delete(droplet.ID) if err != nil { return err } log.Println("Spindown: Deleted droplet " + name) return err } func (me *DropletHandler) poweroff(name string) error { droplet, err := me.getDroplet(name) if err != nil { return err } if droplet.Status != "off" { log.Println("Powering down " + name) // wait until machine is off for { droplet, err = me.getDroplet(name) if err != nil { log.Println("Power down of", name, "failed:", err) if droplet.ID < 1 { return err } } else if droplet.Status == "off" { break } time.Sleep(100 * time.Millisecond) _, _, err = me.client.DropletActions.Shutdown(droplet.ID) if err != nil { log.Println("Power down of", name, "failed:", err) } } log.Println("Powered down", name) } return err } func (me *DropletHandler) getDroplet(name string) (godo.Droplet, error) { name = DROPLET_NS + name page := 0 perPage := 200 var droplet godo.Droplet for { page++ // get list of droplets opt := &godo.ListOptions{ Page: page, PerPage: perPage, } images, _, err := me.client.Droplets.List(opt) if err != nil { break } // find droplet for _, a := range images { if a.Name == name { return a, nil } } // check next page? if len(images) < perPage { break } } return droplet, errors.New("Could not find droplet: " + name) } func (me *DropletHandler) getSnapshot(name string) (godo.Image, error) { name = DROPLET_NS + name page := 0 perPage := 200 var image godo.Image var err error for { page++ // get list of images opt := &godo.ListOptions{ Page: page, PerPage: perPage, } images, _, err := me.client.Images.ListUser(opt) if err != nil { break } // find image for _, a := range images { if a.Name == name { return a, nil } } // check next page? if len(images) < perPage { err = errors.New("Could not find image: " + name) break } } return image, err } func (me *DropletHandler) actionWait(actionId int) bool { for { a, _, err := me.client.Actions.Get(actionId) if err != nil { log.Println("Action retrieval failed: ", err) } else if a.Status == "completed" { return true } else if a.Status == "errored" { log.Println("Action failed: ", a.Type, " on ", a.ResourceID) return false } time.Sleep(1000 * time.Millisecond) } }