veyron/lib/modules: add support for per-command environment and logging.

Change-Id: Ie54dd05c161b06303cee7b878772d5e58ba31e17
diff --git a/lib/modules/exec.go b/lib/modules/exec.go
index 44ef680..35d6512 100644
--- a/lib/modules/exec.go
+++ b/lib/modules/exec.go
@@ -11,15 +11,15 @@
 	"sync"
 	"time"
 
-	"veyron.io/veyron/veyron2/vlog"
-
 	vexec "veyron.io/veyron/veyron/lib/exec"
+	"veyron.io/veyron/veyron2/vlog"
 )
 
 // execHandle implements both the command and Handle interfaces.
 type execHandle struct {
 	mu         sync.Mutex
 	cmd        *exec.Cmd
+	name       string
 	entryPoint string
 	handle     *vexec.ParentHandle
 	sh         *Shell
@@ -63,8 +63,8 @@
 	return runFlag.Value.String() == "TestHelperProcess"
 }
 
-func newExecHandle(entryPoint string) command {
-	return &execHandle{entryPoint: entryPoint}
+func newExecHandle(name string) command {
+	return &execHandle{name: name, entryPoint: ShellEntryPoint + "=" + name}
 }
 
 func (eh *execHandle) Stdout() io.Reader {
@@ -91,40 +91,14 @@
 	eh.mu.Unlock()
 }
 
-func osEnvironMap() map[string]string {
-	m := make(map[string]string)
-	for _, osv := range os.Environ() {
-		if len(osv) == 0 {
-			continue
-		}
-		parts := strings.SplitN(osv, "=", 2)
-		key := parts[0]
-		if len(parts) == 2 {
-			m[key] = parts[1]
-		} else {
-			m[key] = ""
-		}
-	}
-	return m
-}
-func (sh *Shell) mergeOSEnv() map[string]string {
-	merged := osEnvironMap()
-	sh.mu.Lock()
-	for k, v := range sh.env {
-		merged[k] = v
-	}
-	sh.mu.Unlock()
-	return merged
-}
-
-func (eh *execHandle) envelope(sh *Shell, args ...string) ([]string, []string) {
+func (eh *execHandle) envelope(sh *Shell, env []string, args ...string) ([]string, []string) {
 	newargs := []string{os.Args[0]}
 	newargs = append(newargs, testFlags()...)
 	newargs = append(newargs, args...)
-	return newargs, append(sh.MergedEnv(), eh.entryPoint)
+	return newargs, append(env, eh.entryPoint)
 }
 
-func (eh *execHandle) start(sh *Shell, args ...string) (Handle, error) {
+func (eh *execHandle) start(sh *Shell, env []string, args ...string) (Handle, error) {
 	eh.mu.Lock()
 	defer eh.mu.Unlock()
 	eh.sh = sh
@@ -133,7 +107,7 @@
 	// the flag package.
 	newargs := append(testFlags(), args[1:]...)
 	cmd := exec.Command(os.Args[0], newargs...)
-	cmd.Env = append(sh.MergedEnv(), eh.entryPoint)
+	cmd.Env = append(env, eh.entryPoint)
 	fname := strings.TrimPrefix(eh.entryPoint, ShellEntryPoint+"=")
 	stderr, err := newLogfile(strings.TrimLeft(fname, "-\n\t "))
 	if err != nil {
@@ -155,10 +129,13 @@
 	eh.stdin = stdin
 	eh.handle = handle
 	eh.cmd = cmd
+	vlog.VI(1).Infof("Start: %q args: %v", eh.name, cmd.Args)
+	vlog.VI(2).Infof("Start: %q env: %v", eh.name, cmd.Env)
 	if err := handle.Start(); err != nil {
 		return nil, err
 	}
-	err = handle.WaitForReady(10 * time.Second)
+	// TODO(cnicolaou): make this timeout configurable
+	err = handle.WaitForReady(time.Minute)
 	return eh, err
 }
 
@@ -169,6 +146,7 @@
 func (eh *execHandle) Shutdown(stdout, stderr io.Writer) error {
 	eh.mu.Lock()
 	defer eh.mu.Unlock()
+	vlog.VI(1).Infof("Shutdown: %q", eh.name)
 	eh.stdin.Close()
 	logFile := eh.stderr.Name()
 	defer eh.sh.Forget(eh)
@@ -177,14 +155,19 @@
 		os.Remove(logFile)
 	}()
 
+	// TODO(cnicolaou): make this configurable
+	timeout := 10 * time.Second
 	if stdout == nil && stderr == nil {
-		return eh.cmd.Wait()
+		return eh.handle.Wait(timeout)
 	}
-	// Read from stdin before waiting for the child process to ensure
-	// that we get to read all of its output.
-	readTo(eh.stdout, stdout)
 
-	procErr := eh.cmd.Wait()
+	if stdout != nil {
+		// Read from stdin before waiting for the child process to ensure
+		// that we get to read all of its output.
+		readTo(eh.stdout, stdout)
+	}
+
+	procErr := eh.handle.Wait(timeout)
 
 	// Stderr is buffered to a file, so we can safely read it after we
 	// wait for the process.
@@ -192,7 +175,7 @@
 	if stderr != nil {
 		stderrFile, err := os.Open(logFile)
 		if err != nil {
-			fmt.Fprintf(os.Stderr, "failed to open %q: %s\n", logFile, err)
+			vlog.VI(1).Infof("failed to open %q: %s\n", logFile, err)
 			return procErr
 		}
 		readTo(stderrFile, stderr)
@@ -216,8 +199,7 @@
 		return
 	}
 	if err := child.dispatch(); err != nil {
-		fmt.Fprintf(os.Stderr, "Failed: %s\n", err)
-		os.Exit(1)
+		vlog.Fatalf("Failed: %s", err)
 	}
 	os.Exit(0)
 }
@@ -273,15 +255,14 @@
 		for {
 			_, err := os.FindProcess(pid)
 			if err != nil {
-				fmt.Fprintf(os.Stderr, "Looks like our parent exited: %v", err)
-				os.Exit(1)
+				vlog.Fatalf("Looks like our parent exited: %v", err)
 			}
 			time.Sleep(time.Second)
 		}
 	}(os.Getppid())
 
 	args := append([]string{command}, flag.Args()...)
-	return m.fn(os.Stdin, os.Stdout, os.Stderr, osEnvironMap(), args...)
+	return m.fn(os.Stdin, os.Stdout, os.Stderr, envSliceToMap(os.Environ()), args...)
 }
 
 func (child *childRegistrar) addSubprocesses(sh *Shell, pattern string) error {