blob: 55a0c44ba721a9b0d749d27373fd6843382520b3 [file] [log] [blame]
Jiri Simsad7616c92015-03-24 23:44:30 -07001// Copyright 2015 The Vanadium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
Todd Wang7a6a4c22015-05-12 23:57:16 -07005// The following enables go generate to generate the doc.go file.
6//go:generate go run $V23_ROOT/release/go/src/v.io/x/lib/cmdline/testdata/gendoc.go . -help
7
Jiri Simsad5aae832014-05-10 09:56:38 -07008package main
9
10import (
11 "errors"
Jiri Simsad5aae832014-05-10 09:56:38 -070012 "fmt"
13 "net"
Jiri Simsad5aae832014-05-10 09:56:38 -070014 "strings"
Jiri Simsad5aae832014-05-10 09:56:38 -070015
Todd Wang7a6a4c22015-05-12 23:57:16 -070016 "v.io/x/lib/cmdline"
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -070017
18 "v.io/v23/context"
19
Todd Wangaffd04d2015-04-08 10:25:35 -070020 "v.io/x/ref/examples/tunnel"
21 "v.io/x/ref/examples/tunnel/internal"
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -070022 "v.io/x/ref/internal/logger"
Todd Wangaffd04d2015-04-08 10:25:35 -070023 "v.io/x/ref/lib/signals"
Todd Wang7a6a4c22015-05-12 23:57:16 -070024 "v.io/x/ref/lib/v23cmd"
Suharsh Sivakumardcc11d72015-05-11 12:19:20 -070025 _ "v.io/x/ref/runtime/factories/generic"
Jiri Simsad5aae832014-05-10 09:56:38 -070026)
27
28var (
Todd Wang7a6a4c22015-05-12 23:57:16 -070029 disablePty, forcePty, noshell bool
30 portforward, lprotocol, rprotocol string
Jiri Simsad5aae832014-05-10 09:56:38 -070031)
32
Todd Wang7a6a4c22015-05-12 23:57:16 -070033func main() {
34 cmdVsh.Flags.BoolVar(&disablePty, "T", false, "Disable pseudo-terminal allocation.")
35 cmdVsh.Flags.BoolVar(&forcePty, "t", false, "Force allocation of pseudo-terminal.")
36 cmdVsh.Flags.BoolVar(&noshell, "N", false, "Do not execute a shell. Only do port forwarding.")
37 cmdVsh.Flags.StringVar(&portforward, "L", "", `Forward local to remote, format is "localaddr,remoteaddr".`)
38 cmdVsh.Flags.StringVar(&lprotocol, "local_protocol", "tcp", "Local network protocol for port forwarding.")
39 cmdVsh.Flags.StringVar(&rprotocol, "remote_protocol", "tcp", "Remote network protocol for port forwarding.")
40 cmdline.HideGlobalFlagsExcept()
41 cmdline.Main(cmdVsh)
42}
Jiri Simsad5aae832014-05-10 09:56:38 -070043
Todd Wang7a6a4c22015-05-12 23:57:16 -070044var cmdVsh = &cmdline.Command{
45 Runner: v23cmd.RunnerFunc(runVsh),
46 Name: "vsh",
47 Short: "Vanadium shell",
48 Long: `
49Command vsh runs the Vanadium shell, a Tunnel client that can be used to run
50shell commands or start an interactive shell on a remote tunneld server.
Jiri Simsad5aae832014-05-10 09:56:38 -070051
52To open an interactive shell, use:
Todd Wang7a6a4c22015-05-12 23:57:16 -070053 vsh <object name>
Jiri Simsad5aae832014-05-10 09:56:38 -070054
55To run a shell command, use:
Todd Wang7a6a4c22015-05-12 23:57:16 -070056 vsh <object name> <command to run>
Jiri Simsad5aae832014-05-10 09:56:38 -070057
58The -L flag will forward connections from a local port to a remote address
59through the tunneld service. The flag value is localaddr,remoteaddr. E.g.
60 -L :14141,www.google.com:80
61
Todd Wang7a6a4c22015-05-12 23:57:16 -070062vsh can't be used directly with tools like rsync because vanadium object names
Asim Shankarf9a41682014-07-08 17:14:01 -070063don't look like traditional hostnames, which rsync doesn't understand. For
Todd Wang7a6a4c22015-05-12 23:57:16 -070064compatibility with such tools, vsh has a special feature that allows passing the
Robin Thellend320af422015-03-09 17:18:15 -070065vanadium object name via the VSH_NAME environment variable.
Jiri Simsad5aae832014-05-10 09:56:38 -070066
Todd Wang3e389922015-05-14 23:09:38 -070067 $ VSH_NAME=<object name> rsync -avh -e vsh /foo/* v23:/foo/
Jiri Simsad5aae832014-05-10 09:56:38 -070068
Todd Wang3e389922015-05-14 23:09:38 -070069In this example, the "v23" host will be substituted with $VSH_NAME by vsh and
Robin Thellend320af422015-03-09 17:18:15 -070070rsync will work as expected.
Todd Wang7a6a4c22015-05-12 23:57:16 -070071`,
72 ArgsName: "<object name> [command]",
73 ArgsLong: `
74<object name> is the Vanadium object name to connect to.
Jiri Simsad5aae832014-05-10 09:56:38 -070075
Todd Wang7a6a4c22015-05-12 23:57:16 -070076[command] is the shell command and args to run, for non-interactive vsh.
77`,
Jiri Simsad5aae832014-05-10 09:56:38 -070078}
79
Todd Wang7a6a4c22015-05-12 23:57:16 -070080func runVsh(ctx *context.T, env *cmdline.Env, args []string) error {
Todd Wang3e389922015-05-14 23:09:38 -070081 oname, cmd, err := objectNameAndCommandLine(env, args)
Jiri Simsad5aae832014-05-10 09:56:38 -070082 if err != nil {
Todd Wang7a6a4c22015-05-12 23:57:16 -070083 return env.UsageErrorf("%v", err)
Jiri Simsad5aae832014-05-10 09:56:38 -070084 }
85
Todd Wangd8cb55b2014-11-07 00:58:32 -080086 t := tunnel.TunnelClient(oname)
Matt Rosencrantz7a923782014-08-18 09:56:15 -070087
Todd Wang7a6a4c22015-05-12 23:57:16 -070088 if len(portforward) > 0 {
Matt Rosencrantz7a923782014-08-18 09:56:15 -070089 go runPortForwarding(ctx, t, oname)
Jiri Simsad5aae832014-05-10 09:56:38 -070090 }
91
Todd Wang7a6a4c22015-05-12 23:57:16 -070092 if noshell {
Suharsh Sivakumar546e1292015-01-07 15:04:50 -080093 <-signals.ShutdownOnSignals(ctx)
Todd Wang7a6a4c22015-05-12 23:57:16 -070094 return nil
Jiri Simsad5aae832014-05-10 09:56:38 -070095 }
96
Todd Wang3e389922015-05-14 23:09:38 -070097 opts := shellOptions(env, cmd)
Jiri Simsad5aae832014-05-10 09:56:38 -070098
Matt Rosencrantz7a923782014-08-18 09:56:15 -070099 stream, err := t.Shell(ctx, cmd, opts)
Jiri Simsad5aae832014-05-10 09:56:38 -0700100 if err != nil {
Todd Wang7a6a4c22015-05-12 23:57:16 -0700101 return err
Jiri Simsad5aae832014-05-10 09:56:38 -0700102 }
Robin Thellend9faaef32014-08-01 19:45:40 -0700103 if opts.UsePty {
Todd Wangaffd04d2015-04-08 10:25:35 -0700104 saved := internal.EnterRawTerminalMode()
105 defer internal.RestoreTerminalSettings(saved)
Robin Thellend9faaef32014-08-01 19:45:40 -0700106 }
Todd Wang7a6a4c22015-05-12 23:57:16 -0700107 runIOManager(env.Stdin, env.Stdout, env.Stderr, stream)
Jiri Simsad5aae832014-05-10 09:56:38 -0700108
Asim Shankarf9a41682014-07-08 17:14:01 -0700109 exitMsg := fmt.Sprintf("Connection to %s closed.", oname)
Jiri Simsad5aae832014-05-10 09:56:38 -0700110 exitStatus, err := stream.Finish()
111 if err != nil {
112 exitMsg += fmt.Sprintf(" (%v)", err)
113 }
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700114 ctx.VI(1).Info(exitMsg)
Jiri Simsad5aae832014-05-10 09:56:38 -0700115 // Only show the exit message on stdout for interactive shells.
116 // Otherwise, the exit message might get confused with the output
117 // of the command that was run.
118 if err != nil {
Todd Wang7a6a4c22015-05-12 23:57:16 -0700119 fmt.Fprintln(env.Stderr, exitMsg)
Jiri Simsad5aae832014-05-10 09:56:38 -0700120 } else if len(cmd) == 0 {
121 fmt.Println(exitMsg)
122 }
Todd Wang7a6a4c22015-05-12 23:57:16 -0700123 return cmdline.ErrExitCode(exitStatus)
Jiri Simsad5aae832014-05-10 09:56:38 -0700124}
125
Todd Wang3e389922015-05-14 23:09:38 -0700126func shellOptions(env *cmdline.Env, cmd string) (opts tunnel.ShellOpts) {
Todd Wang7a6a4c22015-05-12 23:57:16 -0700127 opts.UsePty = (len(cmd) == 0 || forcePty) && !disablePty
Todd Wang3e389922015-05-14 23:09:38 -0700128 opts.Environment = environment(env.Vars)
Todd Wangaffd04d2015-04-08 10:25:35 -0700129 ws, err := internal.GetWindowSize()
Jiri Simsad5aae832014-05-10 09:56:38 -0700130 if err != nil {
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700131 logger.Global().VI(1).Infof("GetWindowSize failed: %v", err)
Jiri Simsad5aae832014-05-10 09:56:38 -0700132 } else {
Robin Thellendc14adc92015-02-11 13:02:19 -0800133 opts.WinSize.Rows = ws.Row
134 opts.WinSize.Cols = ws.Col
Jiri Simsad5aae832014-05-10 09:56:38 -0700135 }
136 return
137}
138
Todd Wang3e389922015-05-14 23:09:38 -0700139func environment(vars map[string]string) []string {
Jiri Simsad5aae832014-05-10 09:56:38 -0700140 env := []string{}
141 for _, name := range []string{"TERM", "COLORTERM"} {
Todd Wang3e389922015-05-14 23:09:38 -0700142 if value := vars[name]; value != "" {
143 env = append(env, name+"="+value)
Jiri Simsad5aae832014-05-10 09:56:38 -0700144 }
145 }
146 return env
147}
148
Asim Shankarf9a41682014-07-08 17:14:01 -0700149// objectNameAndCommandLine extracts the object name and the remote command to
150// send to the server. The object name is the first non-flag argument.
151// The command line is the concatenation of all non-flag arguments excluding
152// the object name.
Todd Wang3e389922015-05-14 23:09:38 -0700153func objectNameAndCommandLine(env *cmdline.Env, args []string) (string, string, error) {
Asim Shankarf9a41682014-07-08 17:14:01 -0700154 if len(args) < 1 {
Bogdan Caprita87851242014-07-02 14:40:39 -0700155 return "", "", errors.New("object name missing")
Jiri Simsad5aae832014-05-10 09:56:38 -0700156 }
Asim Shankarf9a41682014-07-08 17:14:01 -0700157 name := args[0]
158 args = args[1:]
159 // For compatibility with tools like rsync. Because object names
Jiri Simsad5aae832014-05-10 09:56:38 -0700160 // don't look like traditional hostnames, tools that work with rsh and
161 // ssh can't work directly with vsh. This trick makes the following
162 // possible:
Robin Thellend320af422015-03-09 17:18:15 -0700163 // $ VSH_NAME=<object name> rsync -avh -e vsh /foo/* v23:/foo/
164 // The "v23" host will be substituted with <object name>.
Todd Wang3e389922015-05-14 23:09:38 -0700165 if envName := env.Vars["VSH_NAME"]; envName != "" && name == "v23" {
Jiri Simsad5aae832014-05-10 09:56:38 -0700166 name = envName
167 }
168 cmd := strings.Join(args, " ")
169 return name, cmd, nil
170}
171
Matt Rosencrantz77f4b932014-12-29 11:28:49 -0800172func runPortForwarding(ctx *context.T, t tunnel.TunnelClientMethods, oname string) {
Todd Wang7a6a4c22015-05-12 23:57:16 -0700173 // portforward is localaddr,remoteaddr
174 parts := strings.Split(portforward, ",")
Jiri Simsad5aae832014-05-10 09:56:38 -0700175 var laddr, raddr string
176 if len(parts) != 2 {
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700177 ctx.Fatalf("-L flag expects 2 values separated by a comma")
Jiri Simsad5aae832014-05-10 09:56:38 -0700178 }
179 laddr = parts[0]
180 raddr = parts[1]
181
Todd Wang7a6a4c22015-05-12 23:57:16 -0700182 ln, err := net.Listen(lprotocol, laddr)
Jiri Simsad5aae832014-05-10 09:56:38 -0700183 if err != nil {
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700184 ctx.Fatalf("net.Listen(%q, %q) failed: %v", lprotocol, laddr, err)
Jiri Simsad5aae832014-05-10 09:56:38 -0700185 }
186 defer ln.Close()
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700187 ctx.VI(1).Infof("Listening on %q", ln.Addr())
Jiri Simsad5aae832014-05-10 09:56:38 -0700188 for {
189 conn, err := ln.Accept()
190 if err != nil {
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700191 ctx.Infof("Accept failed: %v", err)
Jiri Simsad5aae832014-05-10 09:56:38 -0700192 continue
193 }
Todd Wang7a6a4c22015-05-12 23:57:16 -0700194 stream, err := t.Forward(ctx, rprotocol, raddr)
Jiri Simsad5aae832014-05-10 09:56:38 -0700195 if err != nil {
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700196 ctx.Infof("Tunnel(%q, %q) failed: %v", rprotocol, raddr, err)
Jiri Simsad5aae832014-05-10 09:56:38 -0700197 conn.Close()
198 continue
199 }
Asim Shankarf9a41682014-07-08 17:14:01 -0700200 name := fmt.Sprintf("%v-->%v-->(%v)-->%v", conn.RemoteAddr(), conn.LocalAddr(), oname, raddr)
Jiri Simsad5aae832014-05-10 09:56:38 -0700201 go func() {
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700202 ctx.VI(1).Infof("TUNNEL START: %v", name)
Todd Wangaffd04d2015-04-08 10:25:35 -0700203 errf := internal.Forward(conn, stream.SendStream(), stream.RecvStream())
Jiri Simsad5aae832014-05-10 09:56:38 -0700204 err := stream.Finish()
Cosmos Nicolaoue6fbf982015-06-18 16:37:19 -0700205 ctx.VI(1).Infof("TUNNEL END : %v (%v, %v)", name, errf, err)
Jiri Simsad5aae832014-05-10 09:56:38 -0700206 }()
207 }
208}