blob: c6e958a359301be9863ac94091711585e8cc21b5 [file] [log] [blame]
Jiri Simsa5293dcb2014-05-10 09:56:38 -07001package main
2
3import (
4 "errors"
5 "flag"
6 "fmt"
7 "net"
8 "os"
9 "path"
10 "strings"
11 "time"
12
13 "veyron/examples/tunnel"
14 "veyron/examples/tunnel/lib"
15 "veyron/lib/signals"
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070016 "veyron2/context"
Jiri Simsa5293dcb2014-05-10 09:56:38 -070017 "veyron2/rt"
18 "veyron2/vlog"
19)
20
21var (
22 disablePty = flag.Bool("T", false, "Disable pseudo-terminal allocation.")
23 forcePty = flag.Bool("t", false, "Force allocation of pseudo-terminal.")
Jiri Simsa5293dcb2014-05-10 09:56:38 -070024
25 portforward = flag.String("L", "", "localaddr,remoteaddr Forward local 'localaddr' to 'remoteaddr'")
26 lprotocol = flag.String("local_protocol", "tcp", "Local network protocol for port forwarding")
27 rprotocol = flag.String("remote_protocol", "tcp", "Remote network protocol for port forwarding")
28
29 noshell = flag.Bool("N", false, "Do not execute a shell. Only do port forwarding.")
30)
31
32func init() {
33 flag.Usage = func() {
34 bname := path.Base(os.Args[0])
35 fmt.Fprintf(os.Stderr, `%s: Veyron SHell.
36
37This tool is used to run shell commands or an interactive shell on a remote
38tunneld service.
39
40To open an interactive shell, use:
Asim Shankar2a46dba2014-07-08 17:14:01 -070041 %s <object name>
Jiri Simsa5293dcb2014-05-10 09:56:38 -070042
43To run a shell command, use:
Asim Shankar2a46dba2014-07-08 17:14:01 -070044 %s <object name> <command to run>
Jiri Simsa5293dcb2014-05-10 09:56:38 -070045
46The -L flag will forward connections from a local port to a remote address
47through the tunneld service. The flag value is localaddr,remoteaddr. E.g.
48 -L :14141,www.google.com:80
49
Asim Shankar2a46dba2014-07-08 17:14:01 -070050%s can't be used directly with tools like rsync because veyron object names
51don't look like traditional hostnames, which rsync doesn't understand. For
Jiri Simsa5293dcb2014-05-10 09:56:38 -070052compatibility with such tools, %s has a special feature that allows passing the
Asim Shankar2a46dba2014-07-08 17:14:01 -070053veyron object name via the VSH_NAME environment variable.
Jiri Simsa5293dcb2014-05-10 09:56:38 -070054
Asim Shankar2a46dba2014-07-08 17:14:01 -070055 $ VSH_NAME=<object name> rsync -avh -e %s /foo/* veyron:/foo/
Jiri Simsa5293dcb2014-05-10 09:56:38 -070056
57In this example, the "veyron" host will be substituted with $VSH_NAME by %s
58and rsync will work as expected.
59
60Full flags:
61`, os.Args[0], bname, bname, bname, bname, os.Args[0], bname)
62 flag.PrintDefaults()
63 }
64}
65
66func main() {
67 // Work around the fact that os.Exit doesn't run deferred functions.
68 os.Exit(realMain())
69}
70
71func realMain() int {
72 r := rt.Init()
Bogdan Caprita4258d882014-07-02 09:15:22 -070073 defer r.Cleanup()
Jiri Simsa5293dcb2014-05-10 09:56:38 -070074
Asim Shankar2a46dba2014-07-08 17:14:01 -070075 oname, cmd, err := objectNameAndCommandLine()
Jiri Simsa5293dcb2014-05-10 09:56:38 -070076 if err != nil {
77 flag.Usage()
78 fmt.Fprintf(os.Stderr, "\n%v\n", err)
79 return 1
80 }
81
Asim Shankar2a46dba2014-07-08 17:14:01 -070082 t, err := tunnel.BindTunnel(oname)
Jiri Simsa5293dcb2014-05-10 09:56:38 -070083 if err != nil {
Asim Shankar2a46dba2014-07-08 17:14:01 -070084 vlog.Fatalf("BindTunnel(%q) failed: %v", oname, err)
Jiri Simsa5293dcb2014-05-10 09:56:38 -070085 }
86
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070087 ctx, _ := rt.R().NewContext().WithTimeout(24 * time.Hour)
88
Jiri Simsa5293dcb2014-05-10 09:56:38 -070089 if len(*portforward) > 0 {
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070090 go runPortForwarding(ctx, t, oname)
Jiri Simsa5293dcb2014-05-10 09:56:38 -070091 }
92
93 if *noshell {
94 <-signals.ShutdownOnSignals()
95 return 0
96 }
97
98 opts := shellOptions(cmd)
99
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700100 stream, err := t.Shell(ctx, cmd, opts)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700101 if err != nil {
102 fmt.Fprintf(os.Stderr, "Error: %v\n", err)
103 return 1
104 }
Robin Thellendad0a9652014-08-01 19:45:40 -0700105 if opts.UsePty {
106 saved := lib.EnterRawTerminalMode()
107 defer lib.RestoreTerminalSettings(saved)
108 }
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700109 runIOManager(os.Stdin, os.Stdout, os.Stderr, stream)
110
Asim Shankar2a46dba2014-07-08 17:14:01 -0700111 exitMsg := fmt.Sprintf("Connection to %s closed.", oname)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700112 exitStatus, err := stream.Finish()
113 if err != nil {
114 exitMsg += fmt.Sprintf(" (%v)", err)
115 }
116 vlog.VI(1).Info(exitMsg)
117 // Only show the exit message on stdout for interactive shells.
118 // Otherwise, the exit message might get confused with the output
119 // of the command that was run.
120 if err != nil {
121 fmt.Fprintln(os.Stderr, exitMsg)
122 } else if len(cmd) == 0 {
123 fmt.Println(exitMsg)
124 }
125 return int(exitStatus)
126}
127
128func shellOptions(cmd string) (opts tunnel.ShellOpts) {
129 opts.UsePty = (len(cmd) == 0 || *forcePty) && !*disablePty
130 opts.Environment = environment()
131 ws, err := lib.GetWindowSize()
132 if err != nil {
133 vlog.VI(1).Infof("GetWindowSize failed: %v", err)
134 } else {
135 opts.Rows = uint32(ws.Row)
136 opts.Cols = uint32(ws.Col)
137 }
138 return
139}
140
141func environment() []string {
142 env := []string{}
143 for _, name := range []string{"TERM", "COLORTERM"} {
144 if value := os.Getenv(name); value != "" {
145 env = append(env, fmt.Sprintf("%s=%s", name, value))
146 }
147 }
148 return env
149}
150
Asim Shankar2a46dba2014-07-08 17:14:01 -0700151// objectNameAndCommandLine extracts the object name and the remote command to
152// send to the server. The object name is the first non-flag argument.
153// The command line is the concatenation of all non-flag arguments excluding
154// the object name.
155func objectNameAndCommandLine() (string, string, error) {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700156 args := flag.Args()
Asim Shankar2a46dba2014-07-08 17:14:01 -0700157 if len(args) < 1 {
Bogdan Capritad9281a32014-07-02 14:40:39 -0700158 return "", "", errors.New("object name missing")
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700159 }
Asim Shankar2a46dba2014-07-08 17:14:01 -0700160 name := args[0]
161 args = args[1:]
162 // For compatibility with tools like rsync. Because object names
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700163 // don't look like traditional hostnames, tools that work with rsh and
164 // ssh can't work directly with vsh. This trick makes the following
165 // possible:
Asim Shankar2a46dba2014-07-08 17:14:01 -0700166 // $ VSH_NAME=<object name> rsync -avh -e vsh /foo/* veyron:/foo/
167 // The "veyron" host will be substituted with <object name>.
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700168 if envName := os.Getenv("VSH_NAME"); len(envName) > 0 && name == "veyron" {
169 name = envName
170 }
171 cmd := strings.Join(args, " ")
172 return name, cmd, nil
173}
174
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700175func runPortForwarding(ctx context.T, t tunnel.Tunnel, oname string) {
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700176 // *portforward is localaddr,remoteaddr
177 parts := strings.Split(*portforward, ",")
178 var laddr, raddr string
179 if len(parts) != 2 {
180 vlog.Fatalf("-L flag expects 2 values separated by a comma")
181 }
182 laddr = parts[0]
183 raddr = parts[1]
184
185 ln, err := net.Listen(*lprotocol, laddr)
186 if err != nil {
187 vlog.Fatalf("net.Listen(%q, %q) failed: %v", *lprotocol, laddr, err)
188 }
189 defer ln.Close()
190 vlog.VI(1).Infof("Listening on %q", ln.Addr())
191 for {
192 conn, err := ln.Accept()
193 if err != nil {
194 vlog.Infof("Accept failed: %v", err)
195 continue
196 }
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700197 stream, err := t.Forward(ctx, *rprotocol, raddr)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700198 if err != nil {
199 vlog.Infof("Tunnel(%q, %q) failed: %v", *rprotocol, raddr, err)
200 conn.Close()
201 continue
202 }
Asim Shankar2a46dba2014-07-08 17:14:01 -0700203 name := fmt.Sprintf("%v-->%v-->(%v)-->%v", conn.RemoteAddr(), conn.LocalAddr(), oname, raddr)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700204 go func() {
205 vlog.VI(1).Infof("TUNNEL START: %v", name)
Shyam Jayaraman97b9dca2014-07-31 13:30:46 -0700206 errf := lib.Forward(conn, stream.SendStream(), stream.RecvStream())
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700207 err := stream.Finish()
208 vlog.VI(1).Infof("TUNNEL END : %v (%v, %v)", name, errf, err)
209 }()
210 }
211}