175 lines
4.9 KiB
Go
175 lines
4.9 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"github.com/mitchellh/osext"
|
|
"github.com/mitchellh/packer/packer"
|
|
"github.com/mitchellh/packer/packer/plugin"
|
|
"io"
|
|
"log"
|
|
"os/exec"
|
|
"path/filepath"
|
|
)
|
|
|
|
// This is the default, built-in configuration that ships with
|
|
// Packer.
|
|
const defaultConfig = `
|
|
{
|
|
"plugin_min_port": 10000,
|
|
"plugin_max_port": 25000,
|
|
|
|
"builders": {
|
|
"amazon-ebs": "packer-builder-amazon-ebs",
|
|
"amazon-chroot": "packer-builder-amazon-chroot",
|
|
"amazon-instance": "packer-builder-amazon-instance",
|
|
"digitalocean": "packer-builder-digitalocean",
|
|
"docker": "packer-builder-docker",
|
|
"googlecompute": "packer-builder-googlecompute",
|
|
"openstack": "packer-builder-openstack",
|
|
"qemu": "packer-builder-qemu",
|
|
"virtualbox": "packer-builder-virtualbox",
|
|
"virtualbox-ovf": "packer-builder-virtualbox-ovf",
|
|
"vmware": "packer-builder-vmware"
|
|
},
|
|
|
|
"commands": {
|
|
"build": "packer-command-build",
|
|
"fix": "packer-command-fix",
|
|
"inspect": "packer-command-inspect",
|
|
"validate": "packer-command-validate"
|
|
},
|
|
|
|
"post-processors": {
|
|
"vagrant": "packer-post-processor-vagrant",
|
|
"vsphere": "packer-post-processor-vsphere"
|
|
},
|
|
|
|
"provisioners": {
|
|
"ansible-local": "packer-provisioner-ansible-local",
|
|
"chef-solo": "packer-provisioner-chef-solo",
|
|
"file": "packer-provisioner-file",
|
|
"puppet-masterless": "packer-provisioner-puppet-masterless",
|
|
"shell": "packer-provisioner-shell",
|
|
"salt-masterless": "packer-provisioner-salt-masterless"
|
|
}
|
|
}
|
|
`
|
|
|
|
type config struct {
|
|
PluginMinPort uint
|
|
PluginMaxPort uint
|
|
|
|
Builders map[string]string
|
|
Commands map[string]string
|
|
PostProcessors map[string]string `json:"post-processors"`
|
|
Provisioners map[string]string
|
|
}
|
|
|
|
// Decodes configuration in JSON format from the given io.Reader into
|
|
// the config object pointed to.
|
|
func decodeConfig(r io.Reader, c *config) error {
|
|
decoder := json.NewDecoder(r)
|
|
return decoder.Decode(c)
|
|
}
|
|
|
|
// Returns an array of defined command names.
|
|
func (c *config) CommandNames() (result []string) {
|
|
result = make([]string, 0, len(c.Commands))
|
|
for name := range c.Commands {
|
|
result = append(result, name)
|
|
}
|
|
return
|
|
}
|
|
|
|
// This is a proper packer.BuilderFunc that can be used to load packer.Builder
|
|
// implementations from the defined plugins.
|
|
func (c *config) LoadBuilder(name string) (packer.Builder, error) {
|
|
log.Printf("Loading builder: %s\n", name)
|
|
bin, ok := c.Builders[name]
|
|
if !ok {
|
|
log.Printf("Builder not found: %s\n", name)
|
|
return nil, nil
|
|
}
|
|
|
|
return c.pluginClient(bin).Builder()
|
|
}
|
|
|
|
// This is a proper packer.CommandFunc that can be used to load packer.Command
|
|
// implementations from the defined plugins.
|
|
func (c *config) LoadCommand(name string) (packer.Command, error) {
|
|
log.Printf("Loading command: %s\n", name)
|
|
bin, ok := c.Commands[name]
|
|
if !ok {
|
|
log.Printf("Command not found: %s\n", name)
|
|
return nil, nil
|
|
}
|
|
|
|
return c.pluginClient(bin).Command()
|
|
}
|
|
|
|
// This is a proper implementation of packer.HookFunc that can be used
|
|
// to load packer.Hook implementations from the defined plugins.
|
|
func (c *config) LoadHook(name string) (packer.Hook, error) {
|
|
log.Printf("Loading hook: %s\n", name)
|
|
return c.pluginClient(name).Hook()
|
|
}
|
|
|
|
// This is a proper packer.PostProcessorFunc that can be used to load
|
|
// packer.PostProcessor implementations from defined plugins.
|
|
func (c *config) LoadPostProcessor(name string) (packer.PostProcessor, error) {
|
|
log.Printf("Loading post-processor: %s", name)
|
|
bin, ok := c.PostProcessors[name]
|
|
if !ok {
|
|
log.Printf("Post-processor not found: %s", name)
|
|
return nil, nil
|
|
}
|
|
|
|
return c.pluginClient(bin).PostProcessor()
|
|
}
|
|
|
|
// This is a proper packer.ProvisionerFunc that can be used to load
|
|
// packer.Provisioner implementations from defined plugins.
|
|
func (c *config) LoadProvisioner(name string) (packer.Provisioner, error) {
|
|
log.Printf("Loading provisioner: %s\n", name)
|
|
bin, ok := c.Provisioners[name]
|
|
if !ok {
|
|
log.Printf("Provisioner not found: %s\n", name)
|
|
return nil, nil
|
|
}
|
|
|
|
return c.pluginClient(bin).Provisioner()
|
|
}
|
|
|
|
func (c *config) pluginClient(path string) *plugin.Client {
|
|
originalPath := path
|
|
|
|
// First attempt to find the executable by consulting the PATH.
|
|
path, err := exec.LookPath(path)
|
|
if err != nil {
|
|
// If that doesn't work, look for it in the same directory
|
|
// as the `packer` executable (us).
|
|
log.Printf("Plugin could not be found. Checking same directory as executable.")
|
|
exePath, err := osext.Executable()
|
|
if err != nil {
|
|
log.Printf("Couldn't get current exe path: %s", err)
|
|
} else {
|
|
log.Printf("Current exe path: %s", exePath)
|
|
path = filepath.Join(filepath.Dir(exePath), filepath.Base(originalPath))
|
|
}
|
|
}
|
|
|
|
// If everything failed, just use the original path and let the error
|
|
// bubble through.
|
|
if path == "" {
|
|
path = originalPath
|
|
}
|
|
|
|
log.Printf("Creating plugin client for path: %s", path)
|
|
var config plugin.ClientConfig
|
|
config.Cmd = exec.Command(path)
|
|
config.Managed = true
|
|
config.MinPort = c.PluginMinPort
|
|
config.MaxPort = c.PluginMaxPort
|
|
return plugin.NewClient(&config)
|
|
}
|