ssh/tailssh: handle terminal opcodes

Updates #3802 #4146

Signed-off-by: Maisem Ali <maisem@tailscale.com>
This commit is contained in:
Maisem Ali 2022-03-12 17:40:40 -08:00 committed by Maisem Ali
parent da6ce27416
commit 6d61b7906e
2 changed files with 95 additions and 27 deletions

View File

@ -29,6 +29,7 @@ import (
"github.com/creack/pty" "github.com/creack/pty"
"github.com/tailscale/ssh" "github.com/tailscale/ssh"
"github.com/u-root/u-root/pkg/termios" "github.com/u-root/u-root/pkg/termios"
gossh "golang.org/x/crypto/ssh"
"golang.org/x/sys/unix" "golang.org/x/sys/unix"
"tailscale.com/cmd/tailscaled/childproc" "tailscale.com/cmd/tailscaled/childproc"
"tailscale.com/types/logger" "tailscale.com/types/logger"
@ -178,7 +179,7 @@ func (srv *server) launchProcess(ctx context.Context, s ssh.Session, ci *sshConn
stdin, stdout, stderr, err = startWithStdPipes(cmd) stdin, stdout, stderr, err = startWithStdPipes(cmd)
return return
} }
pty, err := startWithPTY(cmd, ptyReq) pty, err := srv.startWithPTY(cmd, ptyReq)
if err != nil { if err != nil {
return nil, nil, nil, nil, err return nil, nil, nil, nil, err
} }
@ -196,8 +197,70 @@ func resizeWindow(f *os.File, winCh <-chan ssh.Window) {
} }
} }
// opcodeShortName is a mapping of SSH opcode
// to mnemonic names expected by the termios packaage.
// These are meant to be platform independent.
var opcodeShortName = map[uint8]string{
gossh.VINTR: "intr",
gossh.VQUIT: "quit",
gossh.VERASE: "erase",
gossh.VKILL: "kill",
gossh.VEOF: "eof",
gossh.VEOL: "eol",
gossh.VEOL2: "eol2",
gossh.VSTART: "start",
gossh.VSTOP: "stop",
gossh.VSUSP: "susp",
gossh.VDSUSP: "dsusp",
gossh.VREPRINT: "rprnt",
gossh.VWERASE: "werase",
gossh.VLNEXT: "lnext",
gossh.VFLUSH: "flush",
gossh.VSWTCH: "swtch",
gossh.VSTATUS: "status",
gossh.VDISCARD: "discard",
gossh.IGNPAR: "ignpar",
gossh.PARMRK: "parmrk",
gossh.INPCK: "inpck",
gossh.ISTRIP: "istrip",
gossh.INLCR: "inlcr",
gossh.IGNCR: "igncr",
gossh.ICRNL: "icrnl",
gossh.IUCLC: "iuclc",
gossh.IXON: "ixon",
gossh.IXANY: "ixany",
gossh.IXOFF: "ixoff",
gossh.IMAXBEL: "imaxbel",
gossh.IUTF8: "iutf8",
gossh.ISIG: "isig",
gossh.ICANON: "icanon",
gossh.XCASE: "xcase",
gossh.ECHO: "echo",
gossh.ECHOE: "echoe",
gossh.ECHOK: "echok",
gossh.ECHONL: "echonl",
gossh.NOFLSH: "noflsh",
gossh.TOSTOP: "tostop",
gossh.IEXTEN: "iexten",
gossh.ECHOCTL: "echoctl",
gossh.ECHOKE: "echoke",
gossh.PENDIN: "pendin",
gossh.OPOST: "opost",
gossh.OLCUC: "olcuc",
gossh.ONLCR: "onlcr",
gossh.OCRNL: "ocrnl",
gossh.ONOCR: "onocr",
gossh.ONLRET: "onlret",
gossh.CS7: "cs7",
gossh.CS8: "cs8",
gossh.PARENB: "parenb",
gossh.PARODD: "parodd",
gossh.TTY_OP_ISPEED: "tty_op_ispeed",
gossh.TTY_OP_OSPEED: "tty_op_ospeed",
}
// startWithPTY starts cmd with a psuedo-terminal attached to Stdin, Stdout and Stderr. // startWithPTY starts cmd with a psuedo-terminal attached to Stdin, Stdout and Stderr.
func startWithPTY(cmd *exec.Cmd, ptyReq ssh.Pty) (ptyFile *os.File, err error) { func (srv *server) startWithPTY(cmd *exec.Cmd, ptyReq ssh.Pty) (ptyFile *os.File, err error) {
var tty *os.File var tty *os.File
ptyFile, tty, err = pty.Open() ptyFile, tty, err = pty.Open()
if err != nil { if err != nil {
@ -210,7 +273,7 @@ func startWithPTY(cmd *exec.Cmd, ptyReq ssh.Pty) (ptyFile *os.File, err error) {
tty.Close() tty.Close()
} }
}() }()
ptyRawConn, err := ptyFile.SyscallConn() ptyRawConn, err := tty.SyscallConn()
if err != nil { if err != nil {
return nil, fmt.Errorf("SyscallConn: %w", err) return nil, fmt.Errorf("SyscallConn: %w", err)
} }
@ -228,21 +291,30 @@ func startWithPTY(cmd *exec.Cmd, ptyReq ssh.Pty) (ptyFile *os.File, err error) {
tios.Row = int(ptyReq.Window.Height) tios.Row = int(ptyReq.Window.Height)
tios.Col = int(ptyReq.Window.Width) tios.Col = int(ptyReq.Window.Width)
// And these are just stumbling around in the dark temporarily for c, v := range ptyReq.Modes {
// while we try to match OpenSSH settings. Empirically this makes if c == gossh.TTY_OP_ISPEED {
// stty -a output be the same, but we're still having problems: tios.Ispeed = int(v)
// https://github.com/tailscale/tailscale/issues/4146 continue
// TODO(bradfitz): figure all this out and do something more principled }
// and confident and documented, once we have a clue. if c == gossh.TTY_OP_OSPEED {
tios.Ispeed = 9600 tios.Ospeed = int(v)
tios.Ospeed = 9600 continue
tios.CC["eol"] = 255 }
tios.CC["eol2"] = 255 k, ok := opcodeShortName[c]
tios.Opts["echok"] = false if !ok {
tios.Opts["imaxbel"] = true srv.logf("unknown opcode: %d", c)
tios.Opts["iutf8"] = true continue
tios.Opts["ixany"] = true }
tios.Opts["pendin"] = true if _, ok := tios.CC[k]; ok {
tios.CC[k] = uint8(v)
continue
}
if _, ok := tios.Opts[k]; ok {
tios.Opts[k] = v > 0
continue
}
srv.logf("unsupported opcode: %v(%d)=%v", k, c, v)
}
// Save PTY settings. // Save PTY settings.
if _, err := tios.STTY(int(fd)); err != nil { if _, err := tios.STTY(int(fd)); err != nil {

View File

@ -19,7 +19,6 @@ import (
"os" "os"
"os/exec" "os/exec"
"os/user" "os/user"
"reflect"
"strings" "strings"
"sync" "sync"
"time" "time"
@ -353,6 +352,10 @@ func (srv *server) handleAcceptedSSH(ctx context.Context, s ssh.Session, ci *ssh
} }
} }
// Take control of the PTY so that we can configure it below.
// See https://github.com/tailscale/tailscale/issues/4146
s.DisablePTYEmulation()
cmd, stdin, stdout, stderr, err := srv.launchProcess(ctx, s, ci, lu) cmd, stdin, stdout, stderr, err := srv.launchProcess(ctx, s, ci, lu)
if err != nil { if err != nil {
logf("start failed: %v", err.Error()) logf("start failed: %v", err.Error())
@ -376,14 +379,7 @@ func (srv *server) handleAcceptedSSH(ctx context.Context, s ssh.Session, ci *ssh
stdin.Close() stdin.Close()
}() }()
go func() { go func() {
// Write to s.Channel directly, avoiding gliderlab/ssh's (*session).Write _, err := io.Copy(s, stdout)
// call that translates newline endings, which we don't need.
// See https://github.com/tailscale/tailscale/issues/4146.
// TODO(bradfitz,maisem): remove this reflect hackery once gliderlab/ssh changes
// are all in.
// s is an gliderlabs/ssh.(*session); write to its Channel field.
sshChan := reflect.ValueOf(s).Elem().FieldByName("Channel").Interface().(io.Writer)
_, err := io.Copy(sshChan, stdout)
if err != nil { if err != nil {
// TODO: don't log in the success case. // TODO: don't log in the success case.
logf("ssh: stdout copy: %v", err) logf("ssh: stdout copy: %v", err)