1
0
Fork 0
mirror of https://github.com/Luzifer/vault-totp.git synced 2024-12-23 14:21:18 +00:00
vault-totp/vendor/github.com/kless/term/term_unix.go

186 lines
4.8 KiB
Go
Raw Normal View History

2017-01-04 21:50:12 +00:00
// Copyright 2010 Jonas mg
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
// +build !plan9,!windows
package term
import (
"io"
"os"
"github.com/tredoe/term/sys"
"golang.org/x/sys/unix"
)
// Default values for input and output.
var (
InputFD int = unix.Stdin
Input io.Reader = os.Stdin
Output io.Writer = os.Stdout
)
// A Terminal represents a general terminal interface.
type Terminal struct {
mode modeType
// Contain the state of a terminal, enabling to restore the original settings
oldState, lastState sys.Termios
// Window size
size sys.Winsize
fd int // File descriptor
}
// New creates a new terminal interface in the file descriptor InputFD.
func New() (*Terminal, error) {
var t Terminal
// Get the actual state
if err := sys.Getattr(InputFD, &t.lastState); err != nil {
return nil, os.NewSyscallError("sys.Getattr", err)
}
t.oldState = t.lastState // the actual state is copied to another one
t.fd = InputFD
return &t, nil
}
// == Restore
//
type State struct {
wrap sys.Termios
}
// OriginalState returns the terminal's original state.
func (t *Terminal) OriginalState() State {
return State{t.oldState}
}
// Restore restores the original settings for the term.
func (t *Terminal) Restore() error {
if t.mode != 0 {
if err := sys.Setattr(t.fd, sys.TCSANOW, &t.oldState); err != nil {
return os.NewSyscallError("sys.Setattr", err)
}
t.lastState = t.oldState
t.mode = 0
}
return nil
}
// Restore restores the settings from State.
func Restore(fd int, st State) error {
if err := sys.Setattr(fd, sys.TCSANOW, &st.wrap); err != nil {
return os.NewSyscallError("sys.Setattr", err)
}
return nil
}
// == Modes
//
// RawMode sets the terminal to something like the "raw" mode. Input is available
// character by character, echoing is disabled, and all special processing of
// terminal input and output characters is disabled.
//
// NOTE: in tty "raw mode", CR+LF is used for output and CR is used for input.
func (t *Terminal) RawMode() error {
// Input modes - no break, no CR to NL, no NL to CR, no carriage return,
// no strip char, no start/stop output control, no parity check.
t.lastState.Iflag &^= (sys.BRKINT | sys.IGNBRK | sys.ICRNL | sys.INLCR |
sys.IGNCR | sys.ISTRIP | sys.IXON | sys.PARMRK)
// Output modes - disable post processing.
t.lastState.Oflag &^= sys.OPOST
// Local modes - echoing off, canonical off, no extended functions,
// no signal chars (^Z,^C).
t.lastState.Lflag &^= (sys.ECHO | sys.ECHONL | sys.ICANON | sys.IEXTEN | sys.ISIG)
// Control modes - set 8 bit chars.
t.lastState.Cflag &^= (sys.CSIZE | sys.PARENB)
t.lastState.Cflag |= sys.CS8
// Control chars - set return condition: min number of bytes and timer.
// We want read to return every single byte, without timeout.
t.lastState.Cc[sys.VMIN] = 1 // Read returns when one char is available.
t.lastState.Cc[sys.VTIME] = 0
// Put the terminal in raw mode after flushing
if err := sys.Setattr(t.fd, sys.TCSAFLUSH, &t.lastState); err != nil {
return os.NewSyscallError("sys.Setattr", err)
}
t.mode |= RawMode
return nil
}
// EchoMode turns the echo mode.
func (t *Terminal) EchoMode(echo bool) error {
if !echo {
//t.lastState.Lflag &^= (sys.ECHO | sys.ECHOE | sys.ECHOK | sys.ECHONL)
t.lastState.Lflag &^= sys.ECHO
} else {
//t.lastState.Lflag |= (sys.ECHO | sys.ECHOE | sys.ECHOK | sys.ECHONL)
t.lastState.Lflag |= sys.ECHO
}
if err := sys.Setattr(t.fd, sys.TCSANOW, &t.lastState); err != nil {
return os.NewSyscallError("sys.Setattr", err)
}
if echo {
t.mode |= EchoMode
} else {
t.mode &^= EchoMode
}
return nil
}
// CharMode sets the terminal to single-character mode.
func (t *Terminal) CharMode() error {
// Disable canonical mode, and set buffer size to 1 byte.
t.lastState.Lflag &^= sys.ICANON
t.lastState.Cc[sys.VTIME] = 0
t.lastState.Cc[sys.VMIN] = 1
if err := sys.Setattr(t.fd, sys.TCSANOW, &t.lastState); err != nil {
return os.NewSyscallError("sys.Setattr", err)
}
t.mode |= CharMode
return nil
}
// SetMode sets the terminal attributes given by state.
// Warning: The use of this function is not cross-system.
func (t *Terminal) SetMode(state sys.Termios) error {
if err := sys.Setattr(t.fd, sys.TCSANOW, &state); err != nil {
return os.NewSyscallError("sys.Setattr", err)
}
t.lastState = state
t.mode |= OtherMode
return nil
}
// == Utility
//
// Fd returns the file descriptor referencing the term.
func (t *Terminal) Fd() int {
return t.fd
}
// GetSize returns the size of the term.
func (t *Terminal) GetSize() (row, column int, err error) {
if err = sys.GetWinsize(unix.Stdout, &t.size); err != nil {
return
}
return int(t.size.Row), int(t.size.Col), nil
}