external.go 1.61 KB
Newer Older
Jeromy's avatar
Jeromy committed
1 2 3 4 5 6 7 8 9 10
package commands

import (
	"bytes"
	"fmt"
	"io"
	"os"
	"os/exec"
	"strings"

Jakub Sztandera's avatar
Jakub Sztandera committed
11
	cmds "github.com/ipfs/go-ipfs-cmds"
Jeromy's avatar
Jeromy committed
12 13
)

14
func ExternalBinary(instructions string) *cmds.Command {
Jeromy's avatar
Jeromy committed
15
	return &cmds.Command{
Steven Allen's avatar
Steven Allen committed
16 17
		Arguments: []cmds.Argument{
			cmds.StringArg("args", false, true, "Arguments for subcommand."),
Jeromy's avatar
Jeromy committed
18 19
		},
		External: true,
Overbool's avatar
Overbool committed
20 21
		Run: func(req *cmds.Request, res cmds.ResponseEmitter, env cmds.Environment) error {
			binname := strings.Join(append([]string{"ipfs"}, req.Path...), "-")
Jeromy's avatar
Jeromy committed
22 23 24
			_, err := exec.LookPath(binname)
			if err != nil {
				// special case for '--help' on uninstalled binaries.
Overbool's avatar
Overbool committed
25
				for _, arg := range req.Arguments {
Jeromy's avatar
Jeromy committed
26 27 28
					if arg == "--help" || arg == "-h" {
						buf := new(bytes.Buffer)
						fmt.Fprintf(buf, "%s is an 'external' command.\n", binname)
29
						fmt.Fprintf(buf, "It does not currently appear to be installed.\n")
30
						fmt.Fprintf(buf, "%s\n", instructions)
Overbool's avatar
Overbool committed
31
						return res.Emit(buf)
Jeromy's avatar
Jeromy committed
32 33 34
					}
				}

Overbool's avatar
Overbool committed
35
				return fmt.Errorf("%s not installed", binname)
Jeromy's avatar
Jeromy committed
36 37 38 39
			}

			r, w := io.Pipe()

Overbool's avatar
Overbool committed
40
			cmd := exec.Command(binname, req.Arguments...)
Jeromy's avatar
Jeromy committed
41 42 43 44 45 46 47 48

			// TODO: make commands lib be able to pass stdin through daemon
			//cmd.Stdin = req.Stdin()
			cmd.Stdin = io.LimitReader(nil, 0)
			cmd.Stdout = w
			cmd.Stderr = w

			// setup env of child program
Overbool's avatar
Overbool committed
49
			osenv := os.Environ()
Jeromy's avatar
Jeromy committed
50

Overbool's avatar
Overbool committed
51
			cmd.Env = osenv
Jeromy's avatar
Jeromy committed
52 53 54

			err = cmd.Start()
			if err != nil {
Overbool's avatar
Overbool committed
55
				return fmt.Errorf("failed to start subcommand: %s", err)
Jeromy's avatar
Jeromy committed
56 57
			}

Overbool's avatar
Overbool committed
58
			errC := make(chan error)
Jeromy's avatar
Jeromy committed
59 60

			go func() {
Overbool's avatar
Overbool committed
61 62
				var err error
				defer func() { errC <- err }()
Jeromy's avatar
Jeromy committed
63 64 65
				err = cmd.Wait()
				w.Close()
			}()
Overbool's avatar
Overbool committed
66 67 68 69 70 71 72

			err = res.Emit(r)
			if err != nil {
				return err
			}

			return <-errC
Jeromy's avatar
Jeromy committed
73 74 75
		},
	}
}