2017-09-12 11:30:39 -04:00
|
|
|
package oci
|
2017-02-13 05:35:14 -05:00
|
|
|
|
|
|
|
import (
|
2018-03-28 16:27:41 -04:00
|
|
|
"encoding/base64"
|
2017-02-13 05:35:14 -05:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2018-03-28 16:27:41 -04:00
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
2017-02-13 05:35:14 -05:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2018-06-04 09:18:18 -04:00
|
|
|
"strings"
|
2017-02-13 05:35:14 -05:00
|
|
|
|
2017-04-07 06:20:33 -04:00
|
|
|
"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"
|
2018-04-11 05:20:40 -04:00
|
|
|
ocicommon "github.com/oracle/oci-go-sdk/common"
|
2017-02-13 05:35:14 -05:00
|
|
|
|
|
|
|
"github.com/mitchellh/go-homedir"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Config struct {
|
|
|
|
common.PackerConfig `mapstructure:",squash"`
|
|
|
|
Comm communicator.Config `mapstructure:",squash"`
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
ConfigProvider ocicommon.ConfigurationProvider
|
2017-02-13 05:35:14 -05:00
|
|
|
|
|
|
|
AccessCfgFile string `mapstructure:"access_cfg_file"`
|
|
|
|
AccessCfgFileAccount string `mapstructure:"access_cfg_file_account"`
|
|
|
|
|
|
|
|
// Access config overrides
|
2018-02-13 08:20:26 -05:00
|
|
|
UserID string `mapstructure:"user_ocid"`
|
|
|
|
TenancyID string `mapstructure:"tenancy_ocid"`
|
|
|
|
Region string `mapstructure:"region"`
|
|
|
|
Fingerprint string `mapstructure:"fingerprint"`
|
|
|
|
KeyFile string `mapstructure:"key_file"`
|
|
|
|
PassPhrase string `mapstructure:"pass_phrase"`
|
|
|
|
UsePrivateIP bool `mapstructure:"use_private_ip"`
|
2017-02-13 05:35:14 -05:00
|
|
|
|
|
|
|
AvailabilityDomain string `mapstructure:"availability_domain"`
|
|
|
|
CompartmentID string `mapstructure:"compartment_ocid"`
|
|
|
|
|
|
|
|
// Image
|
|
|
|
BaseImageID string `mapstructure:"base_image_ocid"`
|
|
|
|
Shape string `mapstructure:"shape"`
|
|
|
|
ImageName string `mapstructure:"image_name"`
|
2018-06-22 07:38:25 -04:00
|
|
|
|
|
|
|
// Instance
|
|
|
|
InstanceName string `mapstructure:"instance_name"`
|
|
|
|
|
2018-07-17 11:41:19 -04:00
|
|
|
// Metadata optionally contains custom metadata key/value pairs provided in the
|
|
|
|
// configuration. While this can be used to set metadata["user_data"] the explicit
|
|
|
|
// "user_data" and "user_data_file" values will have precedence.
|
|
|
|
// An instance's metadata can be obtained from at http://169.254.169.254 on the
|
|
|
|
// launched instance.
|
|
|
|
Metadata map[string]string `mapstructure:"metadata"`
|
2018-07-02 04:48:08 -04:00
|
|
|
|
2018-03-28 16:27:41 -04:00
|
|
|
// UserData and UserDataFile file are both optional and mutually exclusive.
|
|
|
|
UserData string `mapstructure:"user_data"`
|
|
|
|
UserDataFile string `mapstructure:"user_data_file"`
|
2017-02-13 05:35:14 -05:00
|
|
|
|
|
|
|
// Networking
|
|
|
|
SubnetID string `mapstructure:"subnet_ocid"`
|
|
|
|
|
2018-06-04 08:14:59 -04:00
|
|
|
// Tagging
|
|
|
|
Tags map[string]string `mapstructure:"tags"`
|
|
|
|
|
2017-02-13 05:35:14 -05:00
|
|
|
ctx interpolate.Context
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Determine where the SDK config is located
|
2018-04-11 05:20:40 -04:00
|
|
|
if c.AccessCfgFile == "" {
|
|
|
|
c.AccessCfgFile, err = getDefaultOCISettingsPath()
|
2017-02-13 05:35:14 -05:00
|
|
|
if err != nil {
|
2018-04-11 05:20:40 -04:00
|
|
|
log.Println("Default OCI settings file not found")
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
if c.AccessCfgFileAccount == "" {
|
|
|
|
c.AccessCfgFileAccount = "DEFAULT"
|
|
|
|
}
|
2017-02-13 05:35:14 -05:00
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
var keyContent []byte
|
|
|
|
if c.KeyFile != "" {
|
|
|
|
path, err := homedir.Expand(c.KeyFile)
|
2017-02-13 05:35:14 -05:00
|
|
|
if err != nil {
|
2018-04-11 05:20:40 -04:00
|
|
|
return nil, err
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
// Read API signing key
|
|
|
|
keyContent, err = ioutil.ReadFile(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
fileProvider, _ := ocicommon.ConfigurationProviderFromFileWithProfile(c.AccessCfgFile, c.AccessCfgFileAccount, c.PassPhrase)
|
|
|
|
if c.Region == "" {
|
|
|
|
var region string
|
|
|
|
if fileProvider != nil {
|
|
|
|
region, _ = fileProvider.Region()
|
|
|
|
}
|
|
|
|
if region == "" {
|
|
|
|
c.Region = "us-phoenix-1"
|
|
|
|
}
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
providers := []ocicommon.ConfigurationProvider{
|
|
|
|
NewRawConfigurationProvider(c.TenancyID, c.UserID, c.Region, c.Fingerprint, string(keyContent), &c.PassPhrase),
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
if fileProvider != nil {
|
|
|
|
providers = append(providers, fileProvider)
|
2017-02-14 05:58:38 -05:00
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
// Load API access configuration from SDK
|
|
|
|
configProvider, err := ocicommon.ComposingConfigurationProvider(providers)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
var errs *packer.MultiError
|
|
|
|
if es := c.Comm.Prepare(&c.ctx); len(es) > 0 {
|
|
|
|
errs = packer.MultiErrorAppend(errs, es...)
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
if userOCID, _ := configProvider.UserOCID(); userOCID == "" {
|
2017-02-13 05:35:14 -05:00
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("'user_ocid' must be specified"))
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
tenancyOCID, _ := configProvider.TenancyOCID()
|
|
|
|
if tenancyOCID == "" {
|
2017-02-13 05:35:14 -05:00
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("'tenancy_ocid' must be specified"))
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
if fingerprint, _ := configProvider.KeyFingerprint(); fingerprint == "" {
|
2017-02-13 05:35:14 -05:00
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("'fingerprint' must be specified"))
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
if _, err := configProvider.PrivateRSAKey(); err != nil {
|
2017-02-13 05:35:14 -05:00
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("'key_file' must be specified"))
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
c.ConfigProvider = configProvider
|
2017-02-13 05:35:14 -05:00
|
|
|
|
|
|
|
if c.AvailabilityDomain == "" {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("'availability_domain' must be specified"))
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
if c.CompartmentID == "" && tenancyOCID != "" {
|
|
|
|
c.CompartmentID = tenancyOCID
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if c.Shape == "" {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("'shape' must be specified"))
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.SubnetID == "" {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("'subnet_ocid' must be specified"))
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.BaseImageID == "" {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("'base_image_ocid' must be specified"))
|
|
|
|
}
|
|
|
|
|
2018-06-04 09:18:18 -04:00
|
|
|
// Validate tag lengths. TODO (hlowndes) maximum number of tags allowed.
|
|
|
|
if c.Tags != nil {
|
|
|
|
for k, v := range c.Tags {
|
|
|
|
k = strings.TrimSpace(k)
|
|
|
|
v = strings.TrimSpace(v)
|
|
|
|
if len(k) > 100 {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, fmt.Errorf("Tag key length too long. Maximum 100 but found %d. Key: %s", len(k), k))
|
|
|
|
}
|
|
|
|
if len(k) == 0 {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("Tag key empty in config"))
|
|
|
|
}
|
|
|
|
if len(v) > 100 {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, fmt.Errorf("Tag value length too long. Maximum 100 but found %d. Key: %s", len(v), k))
|
|
|
|
}
|
|
|
|
if len(v) == 0 {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, errors.New("Tag value empty in config"))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-13 05:35:14 -05:00
|
|
|
if c.ImageName == "" {
|
2017-08-02 05:51:08 -04:00
|
|
|
name, err := interpolate.Render("packer-{{timestamp}}", nil)
|
|
|
|
if err != nil {
|
|
|
|
errs = packer.MultiErrorAppend(errs,
|
|
|
|
fmt.Errorf("unable to parse image name: %s", err))
|
|
|
|
} else {
|
|
|
|
c.ImageName = name
|
|
|
|
}
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
2018-03-28 16:27:41 -04:00
|
|
|
// Optional UserData config
|
|
|
|
if c.UserData != "" && c.UserDataFile != "" {
|
|
|
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("Only one of user_data or user_data_file can be specified."))
|
|
|
|
} else if c.UserDataFile != "" {
|
|
|
|
if _, err := os.Stat(c.UserDataFile); err != nil {
|
|
|
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("user_data_file not found: %s", c.UserDataFile))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// read UserDataFile into string.
|
|
|
|
if c.UserDataFile != "" {
|
|
|
|
fiData, err := ioutil.ReadFile(c.UserDataFile)
|
|
|
|
if err != nil {
|
|
|
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("Problem reading user_data_file: %s", err))
|
|
|
|
}
|
|
|
|
c.UserData = string(fiData)
|
|
|
|
}
|
|
|
|
// Test if UserData is encoded already, and if not, encode it
|
|
|
|
if c.UserData != "" {
|
|
|
|
if _, err := base64.StdEncoding.DecodeString(c.UserData); err != nil {
|
|
|
|
log.Printf("[DEBUG] base64 encoding user data...")
|
|
|
|
c.UserData = base64.StdEncoding.EncodeToString([]byte(c.UserData))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-13 05:35:14 -05:00
|
|
|
if errs != nil && len(errs.Errors) > 0 {
|
|
|
|
return nil, errs
|
|
|
|
}
|
|
|
|
|
|
|
|
return c, nil
|
|
|
|
}
|
|
|
|
|
2017-09-12 11:30:39 -04:00
|
|
|
// getDefaultOCISettingsPath uses mitchellh/go-homedir to compute the default
|
|
|
|
// config file location ($HOME/.oci/config).
|
|
|
|
func getDefaultOCISettingsPath() (string, error) {
|
2017-02-13 05:35:14 -05:00
|
|
|
home, err := homedir.Dir()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2017-09-12 11:30:39 -04:00
|
|
|
path := filepath.Join(home, ".oci", "config")
|
2017-02-13 05:35:14 -05:00
|
|
|
if _, err := os.Stat(path); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return path, nil
|
|
|
|
}
|