2014-10-23 19:44:57 -04:00
|
|
|
// +build !windows
|
|
|
|
|
2016-08-01 22:06:38 -04:00
|
|
|
// Package term provides structures and helper functions to work with
|
2015-07-25 04:35:07 -04:00
|
|
|
// terminal (state, sizes).
|
2013-03-13 03:29:40 -04:00
|
|
|
package term
|
2013-02-13 20:10:00 -05:00
|
|
|
|
|
|
|
import (
|
2013-11-29 12:52:44 -05:00
|
|
|
"errors"
|
2016-08-04 12:35:29 -04:00
|
|
|
"fmt"
|
2015-01-23 20:33:49 -05:00
|
|
|
"io"
|
2013-05-14 18:37:35 -04:00
|
|
|
"os"
|
|
|
|
"os/signal"
|
2017-05-04 20:52:19 -04:00
|
|
|
|
|
|
|
"golang.org/x/sys/unix"
|
2013-02-13 20:10:00 -05:00
|
|
|
)
|
|
|
|
|
2013-11-29 12:52:44 -05:00
|
|
|
var (
|
2015-07-25 04:35:07 -04:00
|
|
|
// ErrInvalidState is returned if the state of the terminal is invalid.
|
2013-11-29 14:08:37 -05:00
|
|
|
ErrInvalidState = errors.New("Invalid terminal state")
|
2013-11-29 12:52:44 -05:00
|
|
|
)
|
|
|
|
|
2015-07-25 04:35:07 -04:00
|
|
|
// State represents the state of the terminal.
|
2013-02-13 20:10:00 -05:00
|
|
|
type State struct {
|
2013-02-26 20:26:46 -05:00
|
|
|
termios Termios
|
2013-02-13 20:10:00 -05:00
|
|
|
}
|
|
|
|
|
2015-07-25 04:35:07 -04:00
|
|
|
// Winsize represents the size of the terminal window.
|
2013-05-24 17:44:16 -04:00
|
|
|
type Winsize struct {
|
|
|
|
Height uint16
|
2013-07-08 13:20:13 -04:00
|
|
|
Width uint16
|
2016-04-13 20:08:00 -04:00
|
|
|
x uint16
|
|
|
|
y uint16
|
2013-05-24 17:44:16 -04:00
|
|
|
}
|
|
|
|
|
2017-01-16 23:45:27 -05:00
|
|
|
// StdStreams returns the standard streams (stdin, stdout, stderr).
|
2015-03-22 12:55:21 -04:00
|
|
|
func StdStreams() (stdIn io.ReadCloser, stdOut, stdErr io.Writer) {
|
|
|
|
return os.Stdin, os.Stdout, os.Stderr
|
2015-01-23 20:33:49 -05:00
|
|
|
}
|
|
|
|
|
2015-07-25 04:35:07 -04:00
|
|
|
// GetFdInfo returns the file descriptor for an os.File and indicates whether the file represents a terminal.
|
2015-03-06 20:04:35 -05:00
|
|
|
func GetFdInfo(in interface{}) (uintptr, bool) {
|
2015-01-23 20:33:49 -05:00
|
|
|
var inFd uintptr
|
|
|
|
var isTerminalIn bool
|
|
|
|
if file, ok := in.(*os.File); ok {
|
|
|
|
inFd = file.Fd()
|
|
|
|
isTerminalIn = IsTerminal(inFd)
|
|
|
|
}
|
|
|
|
return inFd, isTerminalIn
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:10:00 -05:00
|
|
|
// IsTerminal returns true if the given file descriptor is a terminal.
|
2013-06-01 18:55:05 -04:00
|
|
|
func IsTerminal(fd uintptr) bool {
|
2013-02-26 20:26:46 -05:00
|
|
|
var termios Termios
|
2014-11-21 08:12:03 -05:00
|
|
|
return tcget(fd, &termios) == 0
|
2013-02-13 20:10:00 -05:00
|
|
|
}
|
|
|
|
|
2015-07-25 04:35:07 -04:00
|
|
|
// RestoreTerminal restores the terminal connected to the given file descriptor
|
|
|
|
// to a previous state.
|
2013-06-24 17:52:02 -04:00
|
|
|
func RestoreTerminal(fd uintptr, state *State) error {
|
2013-11-29 12:52:44 -05:00
|
|
|
if state == nil {
|
|
|
|
return ErrInvalidState
|
|
|
|
}
|
2014-11-21 08:12:03 -05:00
|
|
|
if err := tcset(fd, &state.termios); err != 0 {
|
2013-11-29 12:52:44 -05:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
2013-02-13 20:10:00 -05:00
|
|
|
}
|
2013-05-14 18:37:35 -04:00
|
|
|
|
2015-07-25 04:35:07 -04:00
|
|
|
// SaveState saves the state of the terminal connected to the given file descriptor.
|
2013-08-03 19:43:20 -04:00
|
|
|
func SaveState(fd uintptr) (*State, error) {
|
|
|
|
var oldState State
|
2014-11-21 08:12:03 -05:00
|
|
|
if err := tcget(fd, &oldState.termios); err != 0 {
|
2013-05-14 18:37:35 -04:00
|
|
|
return nil, err
|
|
|
|
}
|
2013-08-03 19:43:20 -04:00
|
|
|
|
|
|
|
return &oldState, nil
|
|
|
|
}
|
|
|
|
|
2015-07-25 04:35:07 -04:00
|
|
|
// DisableEcho applies the specified state to the terminal connected to the file
|
|
|
|
// descriptor, with echo disabled.
|
2013-08-18 00:29:37 -04:00
|
|
|
func DisableEcho(fd uintptr, state *State) error {
|
2013-08-03 19:43:20 -04:00
|
|
|
newState := state.termios
|
2017-05-04 20:52:19 -04:00
|
|
|
newState.Lflag &^= unix.ECHO
|
2013-08-03 19:43:20 -04:00
|
|
|
|
2014-11-21 08:12:03 -05:00
|
|
|
if err := tcset(fd, &newState); err != 0 {
|
2013-08-03 19:43:20 -04:00
|
|
|
return err
|
|
|
|
}
|
2013-08-18 00:29:37 -04:00
|
|
|
handleInterrupt(fd, state)
|
2013-08-03 19:43:20 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-07-25 04:35:07 -04:00
|
|
|
// SetRawTerminal puts the terminal connected to the given file descriptor into
|
2016-06-22 19:34:01 -04:00
|
|
|
// raw mode and returns the previous state. On UNIX, this puts both the input
|
|
|
|
// and output into raw mode. On Windows, it only puts the input into raw mode.
|
2013-08-18 00:29:37 -04:00
|
|
|
func SetRawTerminal(fd uintptr) (*State, error) {
|
|
|
|
oldState, err := MakeRaw(fd)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
handleInterrupt(fd, oldState)
|
|
|
|
return oldState, err
|
|
|
|
}
|
|
|
|
|
2016-06-22 19:34:01 -04:00
|
|
|
// SetRawTerminalOutput puts the output of terminal connected to the given file
|
|
|
|
// descriptor into raw mode. On UNIX, this does nothing and returns nil for the
|
|
|
|
// state. On Windows, it disables LF -> CRLF translation.
|
|
|
|
func SetRawTerminalOutput(fd uintptr) (*State, error) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2013-08-18 00:29:37 -04:00
|
|
|
func handleInterrupt(fd uintptr, state *State) {
|
2013-08-03 19:43:20 -04:00
|
|
|
sigchan := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(sigchan, os.Interrupt)
|
2013-05-14 18:37:35 -04:00
|
|
|
go func() {
|
2016-08-04 12:35:29 -04:00
|
|
|
for range sigchan {
|
|
|
|
// quit cleanly and the new terminal item is on a new line
|
|
|
|
fmt.Println()
|
|
|
|
signal.Stop(sigchan)
|
|
|
|
close(sigchan)
|
|
|
|
RestoreTerminal(fd, state)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
2013-05-14 18:37:35 -04:00
|
|
|
}()
|
2013-08-03 19:43:20 -04:00
|
|
|
}
|