170 lines
5.2 KiB
Go
170 lines
5.2 KiB
Go
package classic
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net/url"
|
|
"os"
|
|
"regexp"
|
|
"time"
|
|
|
|
"github.com/hashicorp/packer/common"
|
|
"github.com/hashicorp/packer/helper/communicator"
|
|
"github.com/hashicorp/packer/helper/config"
|
|
"github.com/hashicorp/packer/packer"
|
|
"github.com/hashicorp/packer/template/interpolate"
|
|
)
|
|
|
|
type Config struct {
|
|
common.PackerConfig `mapstructure:",squash"`
|
|
Comm communicator.Config `mapstructure:",squash"`
|
|
attribs map[string]interface{}
|
|
|
|
// Access config overrides
|
|
Username string `mapstructure:"username"`
|
|
Password string `mapstructure:"password"`
|
|
IdentityDomain string `mapstructure:"identity_domain"`
|
|
APIEndpoint string `mapstructure:"api_endpoint"`
|
|
apiEndpointURL *url.URL
|
|
|
|
// Image
|
|
// PersistentVolumeSize lets us control the volume size by using persistent boot storage
|
|
PersistentVolumeSize int `mapstructure:"persistent_volume_size"`
|
|
/* TODO:
|
|
builder image list
|
|
default to OL image
|
|
make sure if set then PVS is above
|
|
some way to choose which connection to use for master
|
|
possible ignore everything for builder and always use SSH keys
|
|
*/
|
|
|
|
ImageName string `mapstructure:"image_name"`
|
|
Shape string `mapstructure:"shape"`
|
|
SourceImageList string `mapstructure:"source_image_list"`
|
|
SnapshotTimeout time.Duration `mapstructure:"snapshot_timeout"`
|
|
DestImageList string `mapstructure:"dest_image_list"`
|
|
// Attributes and Attributes file are both optional and mutually exclusive.
|
|
Attributes string `mapstructure:"attributes"`
|
|
AttributesFile string `mapstructure:"attributes_file"`
|
|
// Optional; if you don't enter anything, the image list description
|
|
// will read "Packer-built image list"
|
|
DestImageListDescription string `mapstructure:"image_description"`
|
|
// Optional. Describes what computers are allowed to reach your instance
|
|
// via SSH. This whitelist must contain the computer you're running Packer
|
|
// from. It defaults to public-internet, meaning that you can SSH into your
|
|
// instance from anywhere as long as you have the right keys
|
|
SSHSourceList string `mapstructure:"ssh_source_list"`
|
|
|
|
ctx interpolate.Context
|
|
}
|
|
|
|
func (c *Config) Identifier(s string) string {
|
|
return fmt.Sprintf("/Compute-%s/%s/%s", c.IdentityDomain, c.Username, s)
|
|
}
|
|
|
|
func NewConfig(raws ...interface{}) (*Config, error) {
|
|
c := &Config{}
|
|
|
|
// Decode from template
|
|
err := config.Decode(c, &config.DecodeOpts{
|
|
Interpolate: true,
|
|
InterpolateContext: &c.ctx,
|
|
}, raws...)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to mapstructure Config: %+v", err)
|
|
}
|
|
|
|
c.apiEndpointURL, err = url.Parse(c.APIEndpoint)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Error parsing API Endpoint: %s", err)
|
|
}
|
|
// Set default source list
|
|
if c.SSHSourceList == "" {
|
|
c.SSHSourceList = "seciplist:/oracle/public/public-internet"
|
|
}
|
|
// Use default oracle username with sudo privileges
|
|
if c.Comm.SSHUsername == "" {
|
|
c.Comm.SSHUsername = "opc"
|
|
}
|
|
|
|
if c.SnapshotTimeout == 0 {
|
|
c.SnapshotTimeout = 20 * time.Minute
|
|
}
|
|
|
|
// Validate that all required fields are present
|
|
var errs *packer.MultiError
|
|
required := map[string]string{
|
|
"username": c.Username,
|
|
"password": c.Password,
|
|
"api_endpoint": c.APIEndpoint,
|
|
"identity_domain": c.IdentityDomain,
|
|
"source_image_list": c.SourceImageList,
|
|
"dest_image_list": c.DestImageList,
|
|
"shape": c.Shape,
|
|
}
|
|
for k, v := range required {
|
|
if v == "" {
|
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("You must specify a %s.", k))
|
|
}
|
|
}
|
|
|
|
// Object names can contain only alphanumeric characters, hyphens, underscores, and periods
|
|
reValidObject := regexp.MustCompile("^[a-zA-Z0-9-._/]+$")
|
|
var objectValidation = []struct {
|
|
name string
|
|
value string
|
|
}{
|
|
{"dest_image_list", c.DestImageList},
|
|
{"image_name", c.ImageName},
|
|
}
|
|
for _, ov := range objectValidation {
|
|
if !reValidObject.MatchString(ov.value) {
|
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("%s can contain only alphanumeric characters, hyphens, underscores, and periods.", ov.name))
|
|
}
|
|
}
|
|
|
|
if c.Attributes != "" && c.AttributesFile != "" {
|
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("Only one of user_data or user_data_file can be specified."))
|
|
} else if c.AttributesFile != "" {
|
|
if _, err := os.Stat(c.AttributesFile); err != nil {
|
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("attributes_file not found: %s", c.AttributesFile))
|
|
}
|
|
}
|
|
|
|
if es := c.Comm.Prepare(&c.ctx); len(es) > 0 {
|
|
errs = packer.MultiErrorAppend(errs, es...)
|
|
}
|
|
|
|
if errs != nil && len(errs.Errors) > 0 {
|
|
return nil, errs
|
|
}
|
|
|
|
// unpack attributes from json into config
|
|
var data map[string]interface{}
|
|
|
|
if c.Attributes != "" {
|
|
err := json.Unmarshal([]byte(c.Attributes), &data)
|
|
if err != nil {
|
|
err = fmt.Errorf("Problem parsing json from attributes: %s", err)
|
|
packer.MultiErrorAppend(errs, err)
|
|
}
|
|
c.attribs = data
|
|
} else if c.AttributesFile != "" {
|
|
fidata, err := ioutil.ReadFile(c.AttributesFile)
|
|
if err != nil {
|
|
err = fmt.Errorf("Problem reading attributes_file: %s", err)
|
|
packer.MultiErrorAppend(errs, err)
|
|
}
|
|
err = json.Unmarshal(fidata, &data)
|
|
c.attribs = data
|
|
if err != nil {
|
|
err = fmt.Errorf("Problem parsing json from attributes_file: %s", err)
|
|
packer.MultiErrorAppend(errs, err)
|
|
}
|
|
c.attribs = data
|
|
}
|
|
|
|
return c, nil
|
|
}
|