a850d3d4f0
Signed-off-by: Phus Lu <phus.lu@citrix.com>
229 lines
5.2 KiB
Go
229 lines
5.2 KiB
Go
package common
|
|
|
|
import (
|
|
"crypto/tls"
|
|
"fmt"
|
|
"github.com/mitchellh/multistep"
|
|
"github.com/mitchellh/packer/packer"
|
|
xsclient "github.com/xenserver/go-xenserver-client"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
"os/exec"
|
|
)
|
|
|
|
type StepExport struct{}
|
|
|
|
func downloadFile(url, filename string, ui packer.Ui) (err error) {
|
|
|
|
// Create the file
|
|
fh, err := os.Create(filename)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer fh.Close()
|
|
|
|
// Define a new transport which allows self-signed certs
|
|
tr := &http.Transport{
|
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
|
|
}
|
|
|
|
// Create a client
|
|
client := &http.Client{Transport: tr}
|
|
|
|
// Create request and download file
|
|
|
|
resp, err := client.Get(url)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
var progress uint
|
|
var total uint
|
|
var percentage uint
|
|
var marker_len uint
|
|
|
|
progress = uint(0)
|
|
total = uint(resp.ContentLength)
|
|
percentage = uint(0)
|
|
marker_len = uint(5)
|
|
|
|
var buffer [4096]byte
|
|
for {
|
|
n, err := resp.Body.Read(buffer[:])
|
|
if err != nil && err != io.EOF {
|
|
return err
|
|
}
|
|
|
|
progress += uint(n)
|
|
|
|
if _, write_err := fh.Write(buffer[:n]); write_err != nil {
|
|
return write_err
|
|
}
|
|
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
|
|
// Increment percentage in multiples of marker_len
|
|
cur_percentage := ((progress * 100 / total) / marker_len) * marker_len
|
|
if cur_percentage > percentage {
|
|
percentage = cur_percentage
|
|
ui.Message(fmt.Sprintf("Downloading... %d%%", percentage))
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (StepExport) Run(state multistep.StateBag) multistep.StepAction {
|
|
config := state.Get("commonconfig").(CommonConfig)
|
|
ui := state.Get("ui").(packer.Ui)
|
|
client := state.Get("client").(xsclient.XenAPIClient)
|
|
instance_uuid := state.Get("instance_uuid").(string)
|
|
suffix := ".vhd"
|
|
extrauri := "&format=vhd"
|
|
|
|
instance, err := client.GetVMByUuid(instance_uuid)
|
|
if err != nil {
|
|
ui.Error(fmt.Sprintf("Could not get VM with UUID '%s': %s", instance_uuid, err.Error()))
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
ui.Say("Step: export artifact")
|
|
|
|
switch config.Format {
|
|
case "none":
|
|
ui.Say("Skipping export")
|
|
return multistep.ActionContinue
|
|
|
|
case "xva":
|
|
// export the VM
|
|
|
|
export_filename := fmt.Sprintf("%s/%s.xva", config.OutputDir, config.VMName)
|
|
|
|
use_xe := os.Getenv("USE_XE") == "1"
|
|
if xe, e := exec.LookPath("xe"); e == nil && use_xe {
|
|
cmd := exec.Command(
|
|
xe,
|
|
"-s", client.Host,
|
|
"-p", "443",
|
|
"-u", client.Username,
|
|
"-pw", client.Password,
|
|
"vm-export",
|
|
"vm="+instance_uuid,
|
|
"compress=true",
|
|
"filename="+export_filename,
|
|
)
|
|
|
|
ui.Say(fmt.Sprintf("Getting XVA %+v %+v", cmd.Path, cmd.Args))
|
|
|
|
err = cmd.Run()
|
|
} else {
|
|
export_url := fmt.Sprintf("https://%s/export?uuid=%s&session_id=%s",
|
|
client.Host,
|
|
instance_uuid,
|
|
client.Session.(string),
|
|
)
|
|
|
|
ui.Say("Getting XVA " + export_url)
|
|
err = downloadFile(export_url, export_filename, ui)
|
|
}
|
|
|
|
if err != nil {
|
|
ui.Error(fmt.Sprintf("Could not download XVA: %s", err.Error()))
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
case "vdi_raw":
|
|
suffix = ".raw"
|
|
extrauri = ""
|
|
fallthrough
|
|
case "vdi_vhd":
|
|
// export the disks
|
|
|
|
disks, err := instance.GetDisks()
|
|
if err != nil {
|
|
ui.Error(fmt.Sprintf("Could not get VM disks: %s", err.Error()))
|
|
return multistep.ActionHalt
|
|
}
|
|
for _, disk := range disks {
|
|
disk_uuid, err := disk.GetUuid()
|
|
if err != nil {
|
|
ui.Error(fmt.Sprintf("Could not get disk with UUID '%s': %s", disk_uuid, err.Error()))
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
// Work out XenServer version
|
|
hosts, err := client.GetHosts()
|
|
|
|
if err != nil {
|
|
ui.Error(fmt.Sprintf("Could not retrieve hosts in the pool: %s", err.Error()))
|
|
return multistep.ActionHalt
|
|
}
|
|
host := hosts[0]
|
|
host_software_versions, err := host.GetSoftwareVersion()
|
|
xs_version := host_software_versions["product_version"].(string)
|
|
|
|
if err != nil {
|
|
ui.Error(fmt.Sprintf("Could not get the software version: %s", err.Error()))
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
var disk_export_url string
|
|
|
|
// @todo: check for 6.5 SP1
|
|
if xs_version <= "6.5.0" && config.Format == "vdi_vhd" {
|
|
// Export the VHD using a Transfer VM
|
|
|
|
disk_export_url, err = disk.Expose("vhd")
|
|
|
|
if err != nil {
|
|
ui.Error(fmt.Sprintf("Failed to expose disk %s: %s", disk_uuid, err.Error()))
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
} else {
|
|
|
|
// Use the preferred direct export from XAPI
|
|
// Basic auth in URL request is required as session token is not
|
|
// accepted for some reason.
|
|
// @todo: raise with XAPI team.
|
|
disk_export_url = fmt.Sprintf("https://%s:%s@%s/export_raw_vdi?vdi=%s%s",
|
|
client.Username,
|
|
client.Password,
|
|
client.Host,
|
|
disk_uuid,
|
|
extrauri)
|
|
|
|
}
|
|
|
|
disk_export_filename := fmt.Sprintf("%s/%s%s", config.OutputDir, disk_uuid, suffix)
|
|
|
|
ui.Say("Getting VDI " + disk_export_url)
|
|
err = downloadFile(disk_export_url, disk_export_filename, ui)
|
|
if err != nil {
|
|
ui.Error(fmt.Sprintf("Could not download VDI: %s", err.Error()))
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
// Call unexpose in case a TVM was used. The call is harmless
|
|
// if that is not the case.
|
|
disk.Unexpose()
|
|
|
|
}
|
|
|
|
default:
|
|
panic(fmt.Sprintf("Unknown export format '%s'", config.Format))
|
|
}
|
|
|
|
ui.Say("Download completed: " + config.OutputDir)
|
|
|
|
return multistep.ActionContinue
|
|
}
|
|
|
|
func (StepExport) Cleanup(state multistep.StateBag) {}
|