2018-03-29 23:50:58 -04:00
|
|
|
package googlecomputeimport
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"google.golang.org/api/compute/v1"
|
|
|
|
"google.golang.org/api/storage/v1"
|
|
|
|
|
|
|
|
"github.com/hashicorp/packer/builder/googlecompute"
|
|
|
|
"github.com/hashicorp/packer/common"
|
|
|
|
"github.com/hashicorp/packer/helper/config"
|
|
|
|
"github.com/hashicorp/packer/helper/multistep"
|
|
|
|
"github.com/hashicorp/packer/packer"
|
|
|
|
"github.com/hashicorp/packer/post-processor/compress"
|
|
|
|
"github.com/hashicorp/packer/template/interpolate"
|
|
|
|
|
|
|
|
"golang.org/x/oauth2"
|
|
|
|
"golang.org/x/oauth2/jwt"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Config struct {
|
|
|
|
common.PackerConfig `mapstructure:",squash"`
|
|
|
|
|
2019-01-21 18:09:29 -05:00
|
|
|
AccountFile string `mapstructure:"account_file"`
|
|
|
|
ProjectId string `mapstructure:"project_id"`
|
|
|
|
|
|
|
|
Bucket string `mapstructure:"bucket"`
|
|
|
|
GCSObjectName string `mapstructure:"gcs_object_name"`
|
|
|
|
ImageDescription string `mapstructure:"image_description"`
|
|
|
|
ImageFamily string `mapstructure:"image_family"`
|
|
|
|
ImageGuestOsFeatures []string `mapstructure:"image_guest_os_features"`
|
|
|
|
ImageLabels map[string]string `mapstructure:"image_labels"`
|
|
|
|
ImageName string `mapstructure:"image_name"`
|
|
|
|
KeepOriginalImage bool `mapstructure:"keep_input_artifact"`
|
|
|
|
SkipClean bool `mapstructure:"skip_clean"`
|
2018-03-29 23:50:58 -04:00
|
|
|
|
|
|
|
ctx interpolate.Context
|
|
|
|
}
|
|
|
|
|
|
|
|
type PostProcessor struct {
|
|
|
|
config Config
|
|
|
|
runner multistep.Runner
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *PostProcessor) Configure(raws ...interface{}) error {
|
|
|
|
err := config.Decode(&p.config, &config.DecodeOpts{
|
|
|
|
Interpolate: true,
|
|
|
|
InterpolateContext: &p.config.ctx,
|
|
|
|
InterpolateFilter: &interpolate.RenderFilter{
|
|
|
|
Exclude: []string{
|
|
|
|
"gcs_object_name",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}, raws...)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set defaults
|
|
|
|
if p.config.GCSObjectName == "" {
|
|
|
|
p.config.GCSObjectName = "packer-import-{{timestamp}}.tar.gz"
|
|
|
|
}
|
|
|
|
|
|
|
|
errs := new(packer.MultiError)
|
|
|
|
|
|
|
|
// Check and render gcs_object_name
|
|
|
|
if err = interpolate.Validate(p.config.GCSObjectName, &p.config.ctx); err != nil {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, fmt.Errorf("Error parsing gcs_object_name template: %s", err))
|
|
|
|
}
|
|
|
|
|
|
|
|
templates := map[string]*string{
|
|
|
|
"bucket": &p.config.Bucket,
|
|
|
|
"image_name": &p.config.ImageName,
|
|
|
|
"project_id": &p.config.ProjectId,
|
|
|
|
"account_file": &p.config.AccountFile,
|
|
|
|
}
|
|
|
|
for key, ptr := range templates {
|
|
|
|
if *ptr == "" {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, fmt.Errorf("%s must be set", key))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(errs.Errors) > 0 {
|
|
|
|
return errs
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *PostProcessor) PostProcess(ui packer.Ui, artifact packer.Artifact) (packer.Artifact, bool, error) {
|
|
|
|
var err error
|
|
|
|
|
|
|
|
if artifact.BuilderId() != compress.BuilderId {
|
|
|
|
err = fmt.Errorf(
|
|
|
|
"incompatible artifact type: %s\nCan only import from Compress post-processor artifacts",
|
|
|
|
artifact.BuilderId())
|
|
|
|
return nil, false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
p.config.GCSObjectName, err = interpolate.Render(p.config.GCSObjectName, &p.config.ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, false, fmt.Errorf("Error rendering gcs_object_name template: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
rawImageGcsPath, err := UploadToBucket(p.config.AccountFile, ui, artifact, p.config.Bucket, p.config.GCSObjectName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, p.config.KeepOriginalImage, err
|
|
|
|
}
|
|
|
|
|
2019-01-21 18:09:29 -05:00
|
|
|
gceImageArtifact, err := CreateGceImage(p.config.AccountFile, ui, p.config.ProjectId, rawImageGcsPath, p.config.ImageName, p.config.ImageDescription, p.config.ImageFamily, p.config.ImageLabels, p.config.ImageGuestOsFeatures)
|
2018-03-29 23:50:58 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, p.config.KeepOriginalImage, err
|
|
|
|
}
|
|
|
|
|
2018-07-02 22:44:30 -04:00
|
|
|
if !p.config.SkipClean {
|
|
|
|
err = DeleteFromBucket(p.config.AccountFile, ui, p.config.Bucket, p.config.GCSObjectName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, p.config.KeepOriginalImage, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-29 23:50:58 -04:00
|
|
|
return gceImageArtifact, p.config.KeepOriginalImage, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func UploadToBucket(accountFile string, ui packer.Ui, artifact packer.Artifact, bucket string, gcsObjectName string) (string, error) {
|
|
|
|
var client *http.Client
|
|
|
|
var account googlecompute.AccountFile
|
|
|
|
|
|
|
|
err := googlecompute.ProcessAccountFile(&account, accountFile)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
var DriverScopes = []string{"https://www.googleapis.com/auth/devstorage.full_control"}
|
|
|
|
conf := jwt.Config{
|
|
|
|
Email: account.ClientEmail,
|
|
|
|
PrivateKey: []byte(account.PrivateKey),
|
|
|
|
Scopes: DriverScopes,
|
|
|
|
TokenURL: "https://accounts.google.com/o/oauth2/token",
|
|
|
|
}
|
|
|
|
|
|
|
|
client = conf.Client(oauth2.NoContext)
|
|
|
|
service, err := storage.New(client)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say("Looking for tar.gz file in list of artifacts...")
|
|
|
|
source := ""
|
|
|
|
for _, path := range artifact.Files() {
|
|
|
|
ui.Say(fmt.Sprintf("Found artifact %v...", path))
|
|
|
|
if strings.HasSuffix(path, ".tar.gz") {
|
|
|
|
source = path
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if source == "" {
|
|
|
|
return "", fmt.Errorf("No tar.gz file found in list of articats")
|
|
|
|
}
|
|
|
|
|
|
|
|
artifactFile, err := os.Open(source)
|
|
|
|
if err != nil {
|
|
|
|
err := fmt.Errorf("error opening %v", source)
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say(fmt.Sprintf("Uploading file %v to GCS bucket %v/%v...", source, bucket, gcsObjectName))
|
|
|
|
storageObject, err := service.Objects.Insert(bucket, &storage.Object{Name: gcsObjectName}).Media(artifactFile).Do()
|
|
|
|
if err != nil {
|
|
|
|
ui.Say(fmt.Sprintf("Failed to upload: %v", storageObject))
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return "https://storage.googleapis.com/" + bucket + "/" + gcsObjectName, nil
|
|
|
|
}
|
|
|
|
|
2019-01-21 18:09:29 -05:00
|
|
|
func CreateGceImage(accountFile string, ui packer.Ui, project string, rawImageURL string, imageName string, imageDescription string, imageFamily string, imageLabels map[string]string, imageGuestOsFeatures []string) (packer.Artifact, error) {
|
2018-03-29 23:50:58 -04:00
|
|
|
var client *http.Client
|
|
|
|
var account googlecompute.AccountFile
|
|
|
|
|
|
|
|
err := googlecompute.ProcessAccountFile(&account, accountFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var DriverScopes = []string{"https://www.googleapis.com/auth/compute", "https://www.googleapis.com/auth/devstorage.full_control"}
|
|
|
|
conf := jwt.Config{
|
|
|
|
Email: account.ClientEmail,
|
|
|
|
PrivateKey: []byte(account.PrivateKey),
|
|
|
|
Scopes: DriverScopes,
|
|
|
|
TokenURL: "https://accounts.google.com/o/oauth2/token",
|
|
|
|
}
|
|
|
|
|
|
|
|
client = conf.Client(oauth2.NoContext)
|
|
|
|
|
|
|
|
service, err := compute.New(client)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-01-21 18:09:29 -05:00
|
|
|
// Build up the imageFeatures
|
|
|
|
imageFeatures := make([]*compute.GuestOsFeature, len(imageGuestOsFeatures))
|
|
|
|
for _, v := range imageGuestOsFeatures {
|
|
|
|
imageFeatures = append(imageFeatures, &compute.GuestOsFeature{
|
|
|
|
Type: v,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-03-29 23:50:58 -04:00
|
|
|
gceImage := &compute.Image{
|
2019-01-21 18:09:29 -05:00
|
|
|
Description: imageDescription,
|
|
|
|
Family: imageFamily,
|
|
|
|
GuestOsFeatures: imageFeatures,
|
|
|
|
Labels: imageLabels,
|
|
|
|
Name: imageName,
|
|
|
|
RawDisk: &compute.ImageRawDisk{Source: rawImageURL},
|
|
|
|
SourceType: "RAW",
|
2018-03-29 23:50:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say(fmt.Sprintf("Creating GCE image %v...", imageName))
|
|
|
|
op, err := service.Images.Insert(project, gceImage).Do()
|
|
|
|
if err != nil {
|
|
|
|
ui.Say("Error creating GCE image")
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say("Waiting for GCE image creation operation to complete...")
|
|
|
|
for op.Status != "DONE" {
|
|
|
|
op, err = service.GlobalOperations.Get(project, op.Name).Do()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
time.Sleep(5 * time.Second)
|
|
|
|
}
|
|
|
|
|
|
|
|
// fail if image creation operation has an error
|
|
|
|
if op.Error != nil {
|
|
|
|
var imageError string
|
|
|
|
for _, error := range op.Error.Errors {
|
|
|
|
imageError += error.Message
|
|
|
|
}
|
|
|
|
err = fmt.Errorf("failed to create GCE image %s: %s", imageName, imageError)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Artifact{paths: []string{op.TargetLink}}, nil
|
|
|
|
}
|
2018-07-02 22:44:30 -04:00
|
|
|
|
|
|
|
func DeleteFromBucket(accountFile string, ui packer.Ui, bucket string, gcsObjectName string) error {
|
|
|
|
var client *http.Client
|
|
|
|
var account googlecompute.AccountFile
|
|
|
|
|
|
|
|
err := googlecompute.ProcessAccountFile(&account, accountFile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var DriverScopes = []string{"https://www.googleapis.com/auth/devstorage.full_control"}
|
|
|
|
conf := jwt.Config{
|
|
|
|
Email: account.ClientEmail,
|
|
|
|
PrivateKey: []byte(account.PrivateKey),
|
|
|
|
Scopes: DriverScopes,
|
|
|
|
TokenURL: "https://accounts.google.com/o/oauth2/token",
|
|
|
|
}
|
|
|
|
|
|
|
|
client = conf.Client(oauth2.NoContext)
|
|
|
|
service, err := storage.New(client)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say(fmt.Sprintf("Deleting import source from GCS %s/%s...", bucket, gcsObjectName))
|
|
|
|
err = service.Objects.Delete(bucket, gcsObjectName).Do()
|
|
|
|
if err != nil {
|
|
|
|
ui.Say(fmt.Sprintf("Failed to delete: %v/%v", bucket, gcsObjectName))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|