2013-05-08 20:28:05 -04:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2013-06-09 01:26:49 -04:00
|
|
|
"encoding/json"
|
2015-08-17 04:26:03 -04:00
|
|
|
"fmt"
|
2013-06-09 01:26:49 -04:00
|
|
|
"io"
|
2013-05-08 20:28:05 -04:00
|
|
|
"log"
|
2016-10-13 19:41:33 -04:00
|
|
|
"os"
|
2013-05-08 20:28:05 -04:00
|
|
|
"os/exec"
|
2013-06-17 18:55:21 -04:00
|
|
|
"path/filepath"
|
2015-06-10 23:23:39 -04:00
|
|
|
"runtime"
|
2019-12-17 05:25:56 -05:00
|
|
|
"sort"
|
2014-09-02 18:29:37 -04:00
|
|
|
"strings"
|
2013-05-08 20:28:05 -04:00
|
|
|
|
2017-04-04 16:39:01 -04:00
|
|
|
"github.com/hashicorp/packer/command"
|
|
|
|
"github.com/hashicorp/packer/packer"
|
2020-11-19 18:10:00 -05:00
|
|
|
packersdk "github.com/hashicorp/packer/packer-plugin-sdk/packer"
|
2017-04-04 16:39:01 -04:00
|
|
|
"github.com/hashicorp/packer/packer/plugin"
|
2014-09-02 18:29:37 -04:00
|
|
|
)
|
2013-05-08 21:13:15 -04:00
|
|
|
|
2015-08-17 23:33:50 -04:00
|
|
|
// PACKERSPACE is used to represent the spaces that separate args for a command
|
|
|
|
// without being confused with spaces in the path to the command itself.
|
|
|
|
const PACKERSPACE = "-PACKERSPACE-"
|
|
|
|
|
2013-05-08 20:28:05 -04:00
|
|
|
type config struct {
|
2014-09-08 17:20:13 -04:00
|
|
|
DisableCheckpoint bool `json:"disable_checkpoint"`
|
|
|
|
DisableCheckpointSignature bool `json:"disable_checkpoint_signature"`
|
2019-03-19 09:47:21 -04:00
|
|
|
PluginMinPort int
|
|
|
|
PluginMaxPort int
|
2020-01-07 16:15:45 -05:00
|
|
|
RawBuilders map[string]string `json:"builders"`
|
|
|
|
RawProvisioners map[string]string `json:"provisioners"`
|
|
|
|
RawPostProcessors map[string]string `json:"post-processors"`
|
|
|
|
Builders packer.MapOfBuilder `json:"-"`
|
|
|
|
Provisioners packer.MapOfProvisioner `json:"-"`
|
|
|
|
PostProcessors packer.MapOfPostProcessor `json:"-"`
|
2013-05-08 20:28:05 -04:00
|
|
|
}
|
|
|
|
|
2020-01-07 16:15:45 -05:00
|
|
|
// decodeConfig decodes configuration in JSON format from the given io.Reader into
|
2013-06-09 01:26:49 -04:00
|
|
|
// the config object pointed to.
|
|
|
|
func decodeConfig(r io.Reader, c *config) error {
|
|
|
|
decoder := json.NewDecoder(r)
|
|
|
|
return decoder.Decode(c)
|
2013-05-08 20:28:05 -04:00
|
|
|
}
|
|
|
|
|
2020-01-07 16:15:45 -05:00
|
|
|
// LoadExternalComponentsFromConfig loads plugins defined in RawBuilders, RawProvisioners, and RawPostProcessors.
|
|
|
|
func (c *config) LoadExternalComponentsFromConfig() {
|
|
|
|
// helper to build up list of plugin paths
|
|
|
|
extractPaths := func(m map[string]string) []string {
|
|
|
|
paths := make([]string, 0, len(m))
|
|
|
|
for _, v := range m {
|
|
|
|
paths = append(paths, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
return paths
|
|
|
|
}
|
|
|
|
|
|
|
|
var pluginPaths []string
|
|
|
|
pluginPaths = append(pluginPaths, extractPaths(c.RawProvisioners)...)
|
|
|
|
pluginPaths = append(pluginPaths, extractPaths(c.RawBuilders)...)
|
|
|
|
pluginPaths = append(pluginPaths, extractPaths(c.RawPostProcessors)...)
|
|
|
|
|
|
|
|
var externallyUsed = make([]string, 0, len(pluginPaths))
|
|
|
|
for _, pluginPath := range pluginPaths {
|
2020-01-17 13:33:51 -05:00
|
|
|
name, err := c.loadSingleComponent(pluginPath)
|
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
continue
|
2020-01-07 16:15:45 -05:00
|
|
|
}
|
2020-01-17 13:33:51 -05:00
|
|
|
|
|
|
|
log.Printf("loaded plugin: %s = %s", name, pluginPath)
|
|
|
|
externallyUsed = append(externallyUsed, name)
|
2020-01-07 16:15:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(externallyUsed) > 0 {
|
|
|
|
sort.Strings(externallyUsed)
|
|
|
|
log.Printf("using external plugins %v", externallyUsed)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-17 13:33:51 -05:00
|
|
|
func (c *config) loadSingleComponent(path string) (string, error) {
|
2020-01-07 16:15:45 -05:00
|
|
|
pluginName := filepath.Base(path)
|
|
|
|
|
|
|
|
// On Windows, ignore any plugins that don't end in .exe.
|
|
|
|
// We could do a full PATHEXT parse, but this is probably good enough.
|
|
|
|
if runtime.GOOS == "windows" && strings.ToLower(filepath.Ext(pluginName)) != ".exe" {
|
2020-01-17 13:33:51 -05:00
|
|
|
return "", fmt.Errorf("error loading plugin %q, no exe extension", path)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := os.Stat(path); err != nil {
|
|
|
|
return "", fmt.Errorf("error loading plugin %q: %s", path, err)
|
2020-01-07 16:15:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the filename has a ".", trim up to there
|
|
|
|
if idx := strings.Index(pluginName, "."); idx >= 0 {
|
|
|
|
pluginName = pluginName[:idx]
|
|
|
|
}
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case strings.HasPrefix(pluginName, "packer-builder-"):
|
|
|
|
pluginName = pluginName[len("packer-builder-"):]
|
2020-12-01 16:42:11 -05:00
|
|
|
c.Builders[pluginName] = func() (packersdk.Builder, error) {
|
2020-01-07 16:15:45 -05:00
|
|
|
return c.pluginClient(path).Builder()
|
|
|
|
}
|
|
|
|
case strings.HasPrefix(pluginName, "packer-post-processor-"):
|
|
|
|
pluginName = pluginName[len("packer-post-processor-"):]
|
2020-12-01 17:48:55 -05:00
|
|
|
c.PostProcessors[pluginName] = func() (packersdk.PostProcessor, error) {
|
2020-01-07 16:15:45 -05:00
|
|
|
return c.pluginClient(path).PostProcessor()
|
|
|
|
}
|
|
|
|
case strings.HasPrefix(pluginName, "packer-provisioner-"):
|
|
|
|
pluginName = pluginName[len("packer-provisioner-"):]
|
2020-12-01 17:48:55 -05:00
|
|
|
c.Provisioners[pluginName] = func() (packersdk.Provisioner, error) {
|
2020-01-07 16:15:45 -05:00
|
|
|
return c.pluginClient(path).Provisioner()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-17 13:33:51 -05:00
|
|
|
return pluginName, nil
|
2020-01-07 16:15:45 -05:00
|
|
|
}
|
|
|
|
|
2014-09-02 18:29:37 -04:00
|
|
|
// Discover discovers plugins.
|
|
|
|
//
|
2015-03-11 01:20:30 -04:00
|
|
|
// Search the directory of the executable, then the plugins directory, and
|
|
|
|
// finally the CWD, in that order. Any conflicts will overwrite previously
|
|
|
|
// found plugins, in that order.
|
|
|
|
// Hence, the priority order is the reverse of the search order - i.e., the
|
|
|
|
// CWD has the highest priority.
|
2014-09-02 18:29:37 -04:00
|
|
|
func (c *config) Discover() error {
|
2016-10-13 19:41:33 -04:00
|
|
|
// If we are already inside a plugin process we should not need to
|
|
|
|
// discover anything.
|
2016-10-13 21:14:22 -04:00
|
|
|
if os.Getenv(plugin.MagicCookieKey) == plugin.MagicCookieValue {
|
2016-10-13 19:41:33 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-16 16:02:31 -05:00
|
|
|
// Next, look in the same directory as the executable.
|
2020-10-06 04:59:32 -04:00
|
|
|
exePath, err := os.Executable()
|
2014-09-02 18:29:37 -04:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("[ERR] Error loading exe directory: %s", err)
|
|
|
|
} else {
|
2019-12-17 05:25:56 -05:00
|
|
|
if err := c.discoverExternalComponents(filepath.Dir(exePath)); err != nil {
|
2014-09-02 18:29:37 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-16 16:02:31 -05:00
|
|
|
// Next, look in the default plugins directory inside the configdir/.packer.d/plugins.
|
2015-10-16 20:32:36 -04:00
|
|
|
dir, err := packer.ConfigDir()
|
2014-09-08 16:38:35 -04:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("[ERR] Error loading config directory: %s", err)
|
|
|
|
} else {
|
2019-12-17 05:25:56 -05:00
|
|
|
if err := c.discoverExternalComponents(filepath.Join(dir, "plugins")); err != nil {
|
2014-09-08 16:38:35 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-17 04:26:03 -04:00
|
|
|
// Next, look in the CWD.
|
2019-12-17 05:25:56 -05:00
|
|
|
if err := c.discoverExternalComponents("."); err != nil {
|
2014-09-08 16:38:35 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-01-17 15:40:30 -05:00
|
|
|
// Check whether there is a custom Plugin directory defined. This gets
|
|
|
|
// absolute preference.
|
|
|
|
if packerPluginPath := os.Getenv("PACKER_PLUGIN_PATH"); packerPluginPath != "" {
|
|
|
|
sep := ":"
|
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
// on windows, PATH is semicolon-separated
|
|
|
|
sep = ";"
|
|
|
|
}
|
|
|
|
plugPaths := strings.Split(packerPluginPath, sep)
|
|
|
|
for _, plugPath := range plugPaths {
|
|
|
|
if err := c.discoverExternalComponents(plugPath); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-17 23:33:50 -04:00
|
|
|
// Finally, try to use an internal plugin. Note that this will not override
|
2015-08-17 04:26:03 -04:00
|
|
|
// any previously-loaded plugins.
|
2019-12-17 05:25:56 -05:00
|
|
|
if err := c.discoverInternalComponents(); err != nil {
|
2015-08-17 04:26:03 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-09-02 18:29:37 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-12-01 16:42:11 -05:00
|
|
|
// This is a proper packer.BuilderFunc that can be used to load packersdk.Builder
|
2013-06-09 01:26:49 -04:00
|
|
|
// implementations from the defined plugins.
|
2020-12-01 16:42:11 -05:00
|
|
|
func (c *config) StartBuilder(name string) (packersdk.Builder, error) {
|
2013-05-09 00:05:35 -04:00
|
|
|
log.Printf("Loading builder: %s\n", name)
|
2019-12-17 05:25:56 -05:00
|
|
|
return c.Builders.Start(name)
|
2013-05-09 00:05:35 -04:00
|
|
|
}
|
|
|
|
|
2013-06-09 01:26:49 -04:00
|
|
|
// This is a proper implementation of packer.HookFunc that can be used
|
2020-11-19 18:10:00 -05:00
|
|
|
// to load packersdk.Hook implementations from the defined plugins.
|
|
|
|
func (c *config) StarHook(name string) (packersdk.Hook, error) {
|
2013-05-11 14:11:40 -04:00
|
|
|
log.Printf("Loading hook: %s\n", name)
|
2013-06-11 14:06:23 -04:00
|
|
|
return c.pluginClient(name).Hook()
|
2013-05-11 14:11:40 -04:00
|
|
|
}
|
2013-05-24 00:39:00 -04:00
|
|
|
|
2020-12-01 17:48:55 -05:00
|
|
|
// This is a proper packersdk.PostProcessorFunc that can be used to load
|
|
|
|
// packersdk.PostProcessor implementations from defined plugins.
|
|
|
|
func (c *config) StartPostProcessor(name string) (packersdk.PostProcessor, error) {
|
2013-06-18 14:00:31 -04:00
|
|
|
log.Printf("Loading post-processor: %s", name)
|
2019-12-17 05:25:56 -05:00
|
|
|
return c.PostProcessors.Start(name)
|
2013-06-18 14:00:31 -04:00
|
|
|
}
|
|
|
|
|
2013-06-09 01:26:49 -04:00
|
|
|
// This is a proper packer.ProvisionerFunc that can be used to load
|
|
|
|
// packer.Provisioner implementations from defined plugins.
|
2020-12-01 17:48:55 -05:00
|
|
|
func (c *config) StartProvisioner(name string) (packersdk.Provisioner, error) {
|
2013-05-24 00:39:00 -04:00
|
|
|
log.Printf("Loading provisioner: %s\n", name)
|
2019-12-17 05:25:56 -05:00
|
|
|
return c.Provisioners.Start(name)
|
2013-06-11 14:06:23 -04:00
|
|
|
}
|
|
|
|
|
2019-12-17 05:25:56 -05:00
|
|
|
func (c *config) discoverExternalComponents(path string) error {
|
2014-09-02 18:29:37 -04:00
|
|
|
var err error
|
2014-10-14 18:45:00 -04:00
|
|
|
|
|
|
|
if !filepath.IsAbs(path) {
|
|
|
|
path, err = filepath.Abs(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2019-12-17 05:25:56 -05:00
|
|
|
externallyUsed := []string{}
|
|
|
|
|
|
|
|
pluginPaths, err := c.discoverSingle(filepath.Join(path, "packer-builder-*"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-06 15:12:23 -05:00
|
|
|
for pluginName, pluginPath := range pluginPaths {
|
|
|
|
newPath := pluginPath // this needs to be stored in a new variable for the func below
|
2020-12-01 16:42:11 -05:00
|
|
|
c.Builders[pluginName] = func() (packersdk.Builder, error) {
|
2020-01-06 14:50:12 -05:00
|
|
|
return c.pluginClient(newPath).Builder()
|
2019-12-17 05:25:56 -05:00
|
|
|
}
|
2020-01-06 15:12:23 -05:00
|
|
|
externallyUsed = append(externallyUsed, pluginName)
|
2019-12-17 05:25:56 -05:00
|
|
|
}
|
|
|
|
if len(externallyUsed) > 0 {
|
|
|
|
sort.Strings(externallyUsed)
|
|
|
|
log.Printf("using external builders %v", externallyUsed)
|
|
|
|
externallyUsed = nil
|
|
|
|
}
|
2014-10-14 18:45:00 -04:00
|
|
|
|
2019-12-17 05:25:56 -05:00
|
|
|
pluginPaths, err = c.discoverSingle(filepath.Join(path, "packer-post-processor-*"))
|
2014-09-02 18:29:37 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-06 15:12:23 -05:00
|
|
|
for pluginName, pluginPath := range pluginPaths {
|
|
|
|
newPath := pluginPath // this needs to be stored in a new variable for the func below
|
2020-12-01 17:48:55 -05:00
|
|
|
c.PostProcessors[pluginName] = func() (packersdk.PostProcessor, error) {
|
2020-01-06 14:50:12 -05:00
|
|
|
return c.pluginClient(newPath).PostProcessor()
|
2019-12-17 05:25:56 -05:00
|
|
|
}
|
2020-01-06 15:12:23 -05:00
|
|
|
externallyUsed = append(externallyUsed, pluginName)
|
2019-12-17 05:25:56 -05:00
|
|
|
}
|
|
|
|
if len(externallyUsed) > 0 {
|
|
|
|
sort.Strings(externallyUsed)
|
|
|
|
log.Printf("using external post-processors %v", externallyUsed)
|
|
|
|
externallyUsed = nil
|
|
|
|
}
|
2014-09-02 18:29:37 -04:00
|
|
|
|
2019-12-17 05:25:56 -05:00
|
|
|
pluginPaths, err = c.discoverSingle(filepath.Join(path, "packer-provisioner-*"))
|
2014-09-02 18:29:37 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-06 15:12:23 -05:00
|
|
|
for pluginName, pluginPath := range pluginPaths {
|
|
|
|
newPath := pluginPath // this needs to be stored in a new variable for the func below
|
2020-12-01 17:48:55 -05:00
|
|
|
c.Provisioners[pluginName] = func() (packersdk.Provisioner, error) {
|
2020-01-06 14:50:12 -05:00
|
|
|
return c.pluginClient(newPath).Provisioner()
|
2019-12-17 05:25:56 -05:00
|
|
|
}
|
2020-01-06 15:12:23 -05:00
|
|
|
externallyUsed = append(externallyUsed, pluginName)
|
2019-12-17 05:25:56 -05:00
|
|
|
}
|
|
|
|
if len(externallyUsed) > 0 {
|
|
|
|
sort.Strings(externallyUsed)
|
|
|
|
log.Printf("using external provisioners %v", externallyUsed)
|
|
|
|
externallyUsed = nil
|
|
|
|
}
|
2014-09-02 18:29:37 -04:00
|
|
|
|
2019-12-17 05:25:56 -05:00
|
|
|
return nil
|
2014-09-02 18:29:37 -04:00
|
|
|
}
|
|
|
|
|
2019-12-17 05:25:56 -05:00
|
|
|
func (c *config) discoverSingle(glob string) (map[string]string, error) {
|
2014-09-02 18:29:37 -04:00
|
|
|
matches, err := filepath.Glob(glob)
|
|
|
|
if err != nil {
|
2019-12-17 05:25:56 -05:00
|
|
|
return nil, err
|
2014-09-02 18:29:37 -04:00
|
|
|
}
|
|
|
|
|
2019-12-17 05:25:56 -05:00
|
|
|
res := make(map[string]string)
|
2014-09-02 18:29:37 -04:00
|
|
|
|
|
|
|
prefix := filepath.Base(glob)
|
|
|
|
prefix = prefix[:strings.Index(prefix, "*")]
|
|
|
|
for _, match := range matches {
|
|
|
|
file := filepath.Base(match)
|
|
|
|
|
2020-01-07 16:15:45 -05:00
|
|
|
// On Windows, ignore any plugins that don't end in .exe.
|
2015-06-10 23:23:39 -04:00
|
|
|
// We could do a full PATHEXT parse, but this is probably good enough.
|
2015-06-11 16:45:24 -04:00
|
|
|
if runtime.GOOS == "windows" && strings.ToLower(filepath.Ext(file)) != ".exe" {
|
2015-06-10 23:23:39 -04:00
|
|
|
log.Printf(
|
|
|
|
"[DEBUG] Ignoring plugin match %s, no exe extension",
|
|
|
|
match)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2014-09-02 18:29:37 -04:00
|
|
|
// If the filename has a ".", trim up to there
|
2020-02-13 15:59:47 -05:00
|
|
|
if idx := strings.Index(file, ".exe"); idx >= 0 {
|
2014-09-02 18:29:37 -04:00
|
|
|
file = file[:idx]
|
|
|
|
}
|
|
|
|
|
|
|
|
// Look for foo-bar-baz. The plugin name is "baz"
|
2020-01-06 15:12:23 -05:00
|
|
|
pluginName := file[len(prefix):]
|
|
|
|
log.Printf("[DEBUG] Discovered plugin: %s = %s", pluginName, match)
|
|
|
|
res[pluginName] = match
|
2014-09-02 18:29:37 -04:00
|
|
|
}
|
|
|
|
|
2019-12-17 05:25:56 -05:00
|
|
|
return res, nil
|
2014-09-02 18:29:37 -04:00
|
|
|
}
|
|
|
|
|
2019-12-17 05:25:56 -05:00
|
|
|
func (c *config) discoverInternalComponents() error {
|
2015-08-17 04:26:03 -04:00
|
|
|
// Get the packer binary path
|
2020-10-06 04:59:32 -04:00
|
|
|
packerPath, err := os.Executable()
|
2015-08-17 04:26:03 -04:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("[ERR] Error loading exe directory: %s", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for builder := range command.Builders {
|
2019-12-17 05:25:56 -05:00
|
|
|
builder := builder
|
2015-08-17 04:26:03 -04:00
|
|
|
_, found := (c.Builders)[builder]
|
|
|
|
if !found {
|
2020-12-01 16:42:11 -05:00
|
|
|
c.Builders[builder] = func() (packersdk.Builder, error) {
|
2019-12-17 05:25:56 -05:00
|
|
|
bin := fmt.Sprintf("%s%splugin%spacker-builder-%s",
|
|
|
|
packerPath, PACKERSPACE, PACKERSPACE, builder)
|
|
|
|
return c.pluginClient(bin).Builder()
|
|
|
|
}
|
2015-08-17 04:26:03 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for provisioner := range command.Provisioners {
|
2019-12-17 05:25:56 -05:00
|
|
|
provisioner := provisioner
|
2015-08-17 04:26:03 -04:00
|
|
|
_, found := (c.Provisioners)[provisioner]
|
|
|
|
if !found {
|
2020-12-01 17:48:55 -05:00
|
|
|
c.Provisioners[provisioner] = func() (packersdk.Provisioner, error) {
|
2019-12-17 05:25:56 -05:00
|
|
|
bin := fmt.Sprintf("%s%splugin%spacker-provisioner-%s",
|
|
|
|
packerPath, PACKERSPACE, PACKERSPACE, provisioner)
|
|
|
|
return c.pluginClient(bin).Provisioner()
|
|
|
|
}
|
2015-08-17 04:26:03 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for postProcessor := range command.PostProcessors {
|
2019-12-17 05:25:56 -05:00
|
|
|
postProcessor := postProcessor
|
2015-08-17 04:26:03 -04:00
|
|
|
_, found := (c.PostProcessors)[postProcessor]
|
|
|
|
if !found {
|
2020-12-01 17:48:55 -05:00
|
|
|
c.PostProcessors[postProcessor] = func() (packersdk.PostProcessor, error) {
|
2019-12-17 05:25:56 -05:00
|
|
|
bin := fmt.Sprintf("%s%splugin%spacker-post-processor-%s",
|
|
|
|
packerPath, PACKERSPACE, PACKERSPACE, postProcessor)
|
|
|
|
return c.pluginClient(bin).PostProcessor()
|
|
|
|
}
|
2015-08-17 04:26:03 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-06-11 14:06:23 -04:00
|
|
|
func (c *config) pluginClient(path string) *plugin.Client {
|
2013-06-17 18:55:21 -04:00
|
|
|
originalPath := path
|
|
|
|
|
2019-12-27 16:23:28 -05:00
|
|
|
// Check for special case using `packer plugin PLUGIN`
|
|
|
|
args := []string{}
|
|
|
|
if strings.Contains(path, PACKERSPACE) {
|
|
|
|
parts := strings.Split(path, PACKERSPACE)
|
|
|
|
path = parts[0]
|
|
|
|
args = parts[1:]
|
|
|
|
}
|
|
|
|
|
2013-06-17 18:55:21 -04:00
|
|
|
// 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).
|
2019-12-17 05:25:56 -05:00
|
|
|
log.Printf("Plugin could not be found at %s (%v). Checking same directory as executable.", originalPath, err)
|
2020-10-06 04:59:32 -04:00
|
|
|
exePath, err := os.Executable()
|
2013-06-17 18:55:21 -04:00
|
|
|
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)
|
2013-06-11 14:06:23 -04:00
|
|
|
var config plugin.ClientConfig
|
2015-08-17 04:26:03 -04:00
|
|
|
config.Cmd = exec.Command(path, args...)
|
2013-06-11 14:06:23 -04:00
|
|
|
config.Managed = true
|
2013-06-11 14:08:21 -04:00
|
|
|
config.MinPort = c.PluginMinPort
|
|
|
|
config.MaxPort = c.PluginMaxPort
|
2013-06-11 14:06:23 -04:00
|
|
|
return plugin.NewClient(&config)
|
2013-05-24 00:39:00 -04:00
|
|
|
}
|