2013-08-27 00:57:23 -04:00
|
|
|
package openstack
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2015-06-12 22:55:39 -04:00
|
|
|
"io/ioutil"
|
2013-08-27 00:57:23 -04:00
|
|
|
"log"
|
2014-10-27 19:40:49 -04:00
|
|
|
|
2016-11-27 19:59:26 -05:00
|
|
|
"github.com/gophercloud/gophercloud/openstack/compute/v2/extensions/keypairs"
|
|
|
|
"github.com/gophercloud/gophercloud/openstack/compute/v2/servers"
|
2017-04-04 16:39:01 -04:00
|
|
|
"github.com/hashicorp/packer/packer"
|
2015-06-12 00:16:43 -04:00
|
|
|
"github.com/mitchellh/multistep"
|
2013-08-27 00:57:23 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
type StepRunSourceServer struct {
|
2015-06-12 11:10:10 -04:00
|
|
|
Name string
|
|
|
|
SourceImage string
|
2015-08-10 09:35:56 -04:00
|
|
|
SourceImageName string
|
2015-06-12 11:10:10 -04:00
|
|
|
SecurityGroups []string
|
|
|
|
Networks []string
|
|
|
|
AvailabilityZone string
|
2015-06-12 22:55:39 -04:00
|
|
|
UserData string
|
|
|
|
UserDataFile string
|
2015-12-08 07:45:26 -05:00
|
|
|
ConfigDrive bool
|
2017-01-04 21:15:16 -05:00
|
|
|
InstanceMetadata map[string]string
|
2015-12-08 07:45:26 -05:00
|
|
|
server *servers.Server
|
2013-08-27 00:57:23 -04:00
|
|
|
}
|
|
|
|
|
2013-08-31 15:37:07 -04:00
|
|
|
func (s *StepRunSourceServer) Run(state multistep.StateBag) multistep.StepAction {
|
2015-06-12 00:16:43 -04:00
|
|
|
config := state.Get("config").(Config)
|
2015-06-12 10:32:31 -04:00
|
|
|
flavor := state.Get("flavor_id").(string)
|
2013-08-31 15:37:07 -04:00
|
|
|
ui := state.Get("ui").(packer.Ui)
|
2013-08-27 00:57:23 -04:00
|
|
|
|
2015-06-12 00:16:43 -04:00
|
|
|
// We need the v2 compute client
|
|
|
|
computeClient, err := config.computeV2Client()
|
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf("Error initializing compute client: %s", err)
|
|
|
|
state.Put("error", err)
|
|
|
|
return multistep.ActionHalt
|
2014-01-17 18:46:35 -05:00
|
|
|
}
|
|
|
|
|
2015-06-12 00:16:43 -04:00
|
|
|
networks := make([]servers.Network, len(s.Networks))
|
2014-06-15 14:56:19 -04:00
|
|
|
for i, networkUuid := range s.Networks {
|
2015-06-12 00:16:43 -04:00
|
|
|
networks[i].UUID = networkUuid
|
2014-06-15 14:56:19 -04:00
|
|
|
}
|
|
|
|
|
2015-06-12 22:55:39 -04:00
|
|
|
userData := []byte(s.UserData)
|
|
|
|
if s.UserDataFile != "" {
|
|
|
|
userData, err = ioutil.ReadFile(s.UserDataFile)
|
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf("Error reading user data file: %s", err)
|
|
|
|
state.Put("error", err)
|
|
|
|
return multistep.ActionHalt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-12 10:32:31 -04:00
|
|
|
ui.Say("Launching server...")
|
2015-06-12 00:16:43 -04:00
|
|
|
|
2016-10-08 17:18:19 -04:00
|
|
|
serverOpts := servers.CreateOpts{
|
|
|
|
Name: s.Name,
|
|
|
|
ImageRef: s.SourceImage,
|
|
|
|
ImageName: s.SourceImageName,
|
|
|
|
FlavorRef: flavor,
|
|
|
|
SecurityGroups: s.SecurityGroups,
|
|
|
|
Networks: networks,
|
|
|
|
AvailabilityZone: s.AvailabilityZone,
|
|
|
|
UserData: userData,
|
2016-11-27 19:59:26 -05:00
|
|
|
ConfigDrive: &s.ConfigDrive,
|
2016-12-19 16:47:09 -05:00
|
|
|
ServiceClient: computeClient,
|
2017-01-04 21:15:16 -05:00
|
|
|
Metadata: s.InstanceMetadata,
|
2016-10-08 17:18:19 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
var serverOptsExt servers.CreateOptsBuilder
|
|
|
|
keyName, hasKey := state.GetOk("keyPair")
|
|
|
|
if hasKey {
|
|
|
|
serverOptsExt = keypairs.CreateOptsExt{
|
|
|
|
CreateOptsBuilder: serverOpts,
|
|
|
|
KeyName: keyName.(string),
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
serverOptsExt = serverOpts
|
|
|
|
}
|
|
|
|
|
|
|
|
s.server, err = servers.Create(computeClient, serverOptsExt).Extract()
|
2013-08-27 00:57:23 -04:00
|
|
|
if err != nil {
|
|
|
|
err := fmt.Errorf("Error launching source server: %s", err)
|
2013-08-31 15:37:07 -04:00
|
|
|
state.Put("error", err)
|
2013-08-27 00:57:23 -04:00
|
|
|
ui.Error(err.Error())
|
|
|
|
return multistep.ActionHalt
|
|
|
|
}
|
|
|
|
|
2015-06-12 10:32:31 -04:00
|
|
|
ui.Message(fmt.Sprintf("Server ID: %s", s.server.ID))
|
2015-06-12 00:16:43 -04:00
|
|
|
log.Printf("server id: %s", s.server.ID)
|
2013-08-27 00:57:23 -04:00
|
|
|
|
2015-06-12 10:32:31 -04:00
|
|
|
ui.Say("Waiting for server to become ready...")
|
2013-08-27 00:57:23 -04:00
|
|
|
stateChange := StateChangeConf{
|
|
|
|
Pending: []string{"BUILD"},
|
2015-06-12 22:50:59 -04:00
|
|
|
Target: []string{"ACTIVE"},
|
2015-06-12 00:16:43 -04:00
|
|
|
Refresh: ServerStateRefreshFunc(computeClient, s.server),
|
2013-08-27 00:57:23 -04:00
|
|
|
StepState: state,
|
|
|
|
}
|
|
|
|
latestServer, err := WaitForState(&stateChange)
|
|
|
|
if err != nil {
|
2015-06-12 00:16:43 -04:00
|
|
|
err := fmt.Errorf("Error waiting for server (%s) to become ready: %s", s.server.ID, err)
|
2013-08-31 15:37:07 -04:00
|
|
|
state.Put("error", err)
|
2013-08-27 00:57:23 -04:00
|
|
|
ui.Error(err.Error())
|
|
|
|
return multistep.ActionHalt
|
|
|
|
}
|
|
|
|
|
2015-06-12 00:16:43 -04:00
|
|
|
s.server = latestServer.(*servers.Server)
|
2013-08-31 15:37:07 -04:00
|
|
|
state.Put("server", s.server)
|
2013-08-27 00:57:23 -04:00
|
|
|
|
|
|
|
return multistep.ActionContinue
|
|
|
|
}
|
|
|
|
|
2013-08-31 15:37:07 -04:00
|
|
|
func (s *StepRunSourceServer) Cleanup(state multistep.StateBag) {
|
2013-08-27 00:57:23 -04:00
|
|
|
if s.server == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-06-12 00:16:43 -04:00
|
|
|
config := state.Get("config").(Config)
|
2013-08-31 15:37:07 -04:00
|
|
|
ui := state.Get("ui").(packer.Ui)
|
2013-08-27 00:57:23 -04:00
|
|
|
|
2015-06-12 00:16:43 -04:00
|
|
|
// We need the v2 compute client
|
|
|
|
computeClient, err := config.computeV2Client()
|
|
|
|
if err != nil {
|
|
|
|
ui.Error(fmt.Sprintf("Error terminating server, may still be around: %s", err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-12-17 14:12:52 -05:00
|
|
|
ui.Say(fmt.Sprintf("Terminating the source server: %s ...", s.server.ID))
|
2015-06-12 00:16:43 -04:00
|
|
|
if err := servers.Delete(computeClient, s.server.ID).ExtractErr(); err != nil {
|
2013-08-27 00:57:23 -04:00
|
|
|
ui.Error(fmt.Sprintf("Error terminating server, may still be around: %s", err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
stateChange := StateChangeConf{
|
2015-06-12 22:50:59 -04:00
|
|
|
Pending: []string{"ACTIVE", "BUILD", "REBUILD", "SUSPENDED", "SHUTOFF", "STOPPED"},
|
2015-06-12 00:16:43 -04:00
|
|
|
Refresh: ServerStateRefreshFunc(computeClient, s.server),
|
2015-06-12 22:50:59 -04:00
|
|
|
Target: []string{"DELETED"},
|
2013-08-27 00:57:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
WaitForState(&stateChange)
|
|
|
|
}
|