From eec68e319ead2b934db8f5996e167a96d1ef0c18 Mon Sep 17 00:00:00 2001 From: Ali Rizvi-Santiago Date: Sat, 24 Nov 2018 00:38:33 -0600 Subject: [PATCH 1/3] Replaced unsafe method of determining a user's home directory with an implementation based on os/user. --- packer/config_file_unix.go | 38 +++++++++++++++++++++----------------- 1 file changed, 21 insertions(+), 17 deletions(-) diff --git a/packer/config_file_unix.go b/packer/config_file_unix.go index 512ac42ed..10337202d 100644 --- a/packer/config_file_unix.go +++ b/packer/config_file_unix.go @@ -3,13 +3,10 @@ package packer import ( - "bytes" - "errors" "log" "os" - "os/exec" + "os/user" "path/filepath" - "strings" ) func configFile() (string, error) { @@ -31,24 +28,31 @@ func configDir() (string, error) { } func homeDir() (string, error) { - // First prefer the HOME environmental variable - if home := os.Getenv("HOME"); home != "" { + var u *user.User + + /// First prefer the HOME environmental variable + if home, ok := os.LookupEnv("HOME"); ok { log.Printf("Detected home directory from env var: %s", home) return home, nil } - // If that fails, try the shell - var stdout bytes.Buffer - cmd := exec.Command("sh", "-c", "eval echo ~$USER") - cmd.Stdout = &stdout - if err := cmd.Run(); err != nil { + /// Fall back to the passwd database if not found which follows + /// the same semantics as bourne shell + var err error + + // Check username specified in the environment first + if username, ok := os.LookupEnv("USER"); ok { + u, err = user.Lookup(username) + + } else { + // Otherwise we assume the current user + u, err = user.Current() + } + + // Fail if we were unable to read the record + if err != nil { return "", err } - result := strings.TrimSpace(stdout.String()) - if result == "" { - return "", errors.New("blank output") - } - - return result, nil + return u.HomeDir, nil } From 5147ac036412d9dc7e9f8897e6f2f91f081f1baa Mon Sep 17 00:00:00 2001 From: Adrien Delorme Date: Fri, 30 Nov 2018 14:56:53 +0100 Subject: [PATCH 2/3] use Getenv instead of LookupEnv so users can set USER="" and HOME="" + lookup different username only if it is different thant the current username --- packer/config_file_unix.go | 20 ++++++++------------ 1 file changed, 8 insertions(+), 12 deletions(-) diff --git a/packer/config_file_unix.go b/packer/config_file_unix.go index 10337202d..5c7a31760 100644 --- a/packer/config_file_unix.go +++ b/packer/config_file_unix.go @@ -28,25 +28,21 @@ func configDir() (string, error) { } func homeDir() (string, error) { - var u *user.User - /// First prefer the HOME environmental variable - if home, ok := os.LookupEnv("HOME"); ok { + // First prefer the HOME environmental variable + if home := os.Getenv("HOME"); home != "" { log.Printf("Detected home directory from env var: %s", home) return home, nil } - /// Fall back to the passwd database if not found which follows - /// the same semantics as bourne shell - var err error + // Fall back to the passwd database if not found which follows + // the same semantics as bourne shell + u, err := user.Current() - // Check username specified in the environment first - if username, ok := os.LookupEnv("USER"); ok { + // Get homedir from specified username + // if it is set and different than what we have + if username := os.Getenv("USER"); username != "" && err == nil && u.Username != username { u, err = user.Lookup(username) - - } else { - // Otherwise we assume the current user - u, err = user.Current() } // Fail if we were unable to read the record From 1055c119bc8a77704f4a1f5cc5920a946c0773dd Mon Sep 17 00:00:00 2001 From: Adrien Delorme Date: Fri, 30 Nov 2018 15:47:43 +0100 Subject: [PATCH 3/3] make code of homeDir, configFile & configDir not os specific anymore --- packer/config_file.go | 46 +++++++++++++++++++++++++++++++ packer/config_file_unix.go | 52 ++--------------------------------- packer/config_file_windows.go | 44 ++--------------------------- 3 files changed, 52 insertions(+), 90 deletions(-) diff --git a/packer/config_file.go b/packer/config_file.go index edd10edee..043b40ae8 100644 --- a/packer/config_file.go +++ b/packer/config_file.go @@ -1,7 +1,9 @@ package packer import ( + "log" "os" + "os/user" "path/filepath" ) @@ -38,3 +40,47 @@ func ConfigTmpDir() (string, error) { } return td, nil } + +func homeDir() (string, error) { + + // First prefer the HOME environmental variable + if home := os.Getenv("HOME"); home != "" { + log.Printf("Detected home directory from env var: %s", home) + return home, nil + } + + // Fall back to the passwd database if not found which follows + // the same semantics as bourne shell + u, err := user.Current() + + // Get homedir from specified username + // if it is set and different than what we have + if username := os.Getenv("USER"); username != "" && err == nil && u.Username != username { + u, err = user.Lookup(username) + } + + // Fail if we were unable to read the record + if err != nil { + return "", err + } + + return u.HomeDir, nil +} + +func configFile() (string, error) { + dir, err := homeDir() + if err != nil { + return "", err + } + + return filepath.Join(dir, defaultConfigFile), nil +} + +func configDir() (string, error) { + dir, err := homeDir() + if err != nil { + return "", err + } + + return filepath.Join(dir, defaultConfigDir), nil +} diff --git a/packer/config_file_unix.go b/packer/config_file_unix.go index 5c7a31760..ab3ac73be 100644 --- a/packer/config_file_unix.go +++ b/packer/config_file_unix.go @@ -2,53 +2,7 @@ package packer -import ( - "log" - "os" - "os/user" - "path/filepath" +const ( + defaultConfigFile = ".packerconfig" + defaultConfigDir = ".packer.d" ) - -func configFile() (string, error) { - dir, err := homeDir() - if err != nil { - return "", err - } - - return filepath.Join(dir, ".packerconfig"), nil -} - -func configDir() (string, error) { - dir, err := homeDir() - if err != nil { - return "", err - } - - return filepath.Join(dir, ".packer.d"), nil -} - -func homeDir() (string, error) { - - // First prefer the HOME environmental variable - if home := os.Getenv("HOME"); home != "" { - log.Printf("Detected home directory from env var: %s", home) - return home, nil - } - - // Fall back to the passwd database if not found which follows - // the same semantics as bourne shell - u, err := user.Current() - - // Get homedir from specified username - // if it is set and different than what we have - if username := os.Getenv("USER"); username != "" && err == nil && u.Username != username { - u, err = user.Lookup(username) - } - - // Fail if we were unable to read the record - if err != nil { - return "", err - } - - return u.HomeDir, nil -} diff --git a/packer/config_file_windows.go b/packer/config_file_windows.go index d0bcc1c50..b249c492d 100644 --- a/packer/config_file_windows.go +++ b/packer/config_file_windows.go @@ -2,45 +2,7 @@ package packer -import ( - "path/filepath" - "syscall" - "unsafe" +const ( + defaultConfigFile = "packer.config" + defaultConfigDir = "packer.d" ) - -var ( - shell = syscall.MustLoadDLL("Shell32.dll") - getFolderPath = shell.MustFindProc("SHGetFolderPathW") -) - -const CSIDL_APPDATA = 26 - -func configFile() (string, error) { - dir, err := homeDir() - if err != nil { - return "", err - } - - return filepath.Join(dir, "packer.config"), nil -} - -func configDir() (string, error) { - dir, err := homeDir() - if err != nil { - return "", err - } - - return filepath.Join(dir, "packer.d"), nil -} - -func homeDir() (string, error) { - b := make([]uint16, syscall.MAX_PATH) - - // See: http://msdn.microsoft.com/en-us/library/windows/desktop/bb762181(v=vs.85).aspx - r, _, err := getFolderPath.Call(0, CSIDL_APPDATA, 0, 0, uintptr(unsafe.Pointer(&b[0]))) - if uint32(r) != 0 { - return "", err - } - - return syscall.UTF16ToString(b), nil -}