use std::cmp;
use std::collections::HashMap;
use std::fs::{File, OpenOptions};
use std::io::{self, BufReader, ErrorKind, Read, Write};
use std::os::unix::io::{AsRawFd, IntoRawFd, RawFd};
use std::os::unix::net::UnixStream;
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::mpsc::{self, SyncSender};
use std::sync::{Arc, Mutex};
use log::{debug, warn};
use nix::errno::Errno;
use nix::poll::{self, PollFlags};
use nix::sys::select::{self, FdSet};
use nix::sys::termios::{self, SetArg, SpecialCharacterIndices as SCI, Termios};
use nix::unistd::{close, isatty, read, write};
use unicode_segmentation::UnicodeSegmentation;
use utf8parse::{Parser, Receiver};
use super::{width, Event, RawMode, RawReader, Renderer, Term};
use crate::config::{Behavior, BellStyle, ColorMode, Config};
use crate::highlight::Highlighter;
use crate::keys::{KeyCode as K, KeyEvent, KeyEvent as E, Modifiers as M};
use crate::layout::{Layout, Position};
use crate::line_buffer::LineBuffer;
use crate::{error, Cmd, ReadlineError, Result};
const UNSUPPORTED_TERM: [&str; 3] = ["dumb", "cons25", "emacs"];
const BRACKETED_PASTE_ON: &str = "\x1b[?2004h";
const BRACKETED_PASTE_OFF: &str = "\x1b[?2004l";
nix::ioctl_read_bad!(win_size, libc::TIOCGWINSZ, libc::winsize);
#[allow(clippy::useless_conversion)]
fn get_win_size(fd: RawFd) -> (usize, usize) {
use std::mem::zeroed;
if cfg!(test) {
return (80, 24);
}
unsafe {
let mut size: libc::winsize = zeroed();
match win_size(fd, &mut size) {
Ok(0) => {
let cols = if size.ws_col == 0 {
80
} else {
size.ws_col as usize
};
let rows = if size.ws_row == 0 {
usize::MAX
} else {
size.ws_row as usize
};
(cols, rows)
}
_ => (80, 24),
}
}
}
fn is_unsupported_term() -> bool {
match std::env::var("TERM") {
Ok(term) => {
for iter in &UNSUPPORTED_TERM {
if (*iter).eq_ignore_ascii_case(&term) {
return true;
}
}
false
}
Err(_) => false,
}
}
fn is_a_tty(fd: RawFd) -> bool {
isatty(fd).unwrap_or(false)
}
pub type PosixKeyMap = HashMap<KeyEvent, Cmd>;
#[cfg(not(test))]
pub type KeyMap = PosixKeyMap;
#[must_use = "You must restore default mode (disable_raw_mode)"]
pub struct PosixMode {
termios: Termios,
tty_in: RawFd,
tty_out: Option<RawFd>,
raw_mode: Arc<AtomicBool>,
}
#[cfg(not(test))]
pub type Mode = PosixMode;
impl RawMode for PosixMode {
fn disable_raw_mode(&self) -> Result<()> {
termios::tcsetattr(self.tty_in, SetArg::TCSADRAIN, &self.termios)?;
if let Some(out) = self.tty_out {
write_all(out, BRACKETED_PASTE_OFF)?;
}
self.raw_mode.store(false, Ordering::SeqCst);
Ok(())
}
}
struct TtyIn {
fd: RawFd,
sigwinch_pipe: Option<RawFd>,
}
impl Read for TtyIn {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
loop {
let res = unsafe {
libc::read(
self.fd,
buf.as_mut_ptr() as *mut libc::c_void,
buf.len() as libc::size_t,
)
};
if res == -1 {
let error = io::Error::last_os_error();
if error.kind() == ErrorKind::Interrupted && self.sigwinch()? {
return Err(io::Error::new(
ErrorKind::Interrupted,
error::WindowResizedError,
));
} else if error.kind() != ErrorKind::Interrupted {
return Err(error);
}
} else {
#[allow(clippy::cast_sign_loss)]
return Ok(res as usize);
}
}
}
}
impl TtyIn {
fn sigwinch(&self) -> nix::Result<bool> {
if let Some(pipe) = self.sigwinch_pipe {
let mut buf = [0u8; 64];
match read(pipe, &mut buf) {
Ok(0) => Ok(false),
Ok(_) => Ok(true),
Err(e) if e == Errno::EWOULDBLOCK || e == Errno::EINTR => Ok(false),
Err(e) => Err(e),
}
} else {
Ok(false)
}
}
}
type PipeReader = Arc<Mutex<(File, mpsc::Receiver<String>)>>;
type PipeWriter = (Arc<Mutex<File>>, SyncSender<String>);
pub struct PosixRawReader {
tty_in: BufReader<TtyIn>,
timeout_ms: i32,
parser: Parser,
key_map: PosixKeyMap,
pipe_reader: Option<PipeReader>,
fds: FdSet,
}
impl AsRawFd for PosixRawReader {
fn as_raw_fd(&self) -> RawFd {
self.tty_in.get_ref().fd
}
}
struct Utf8 {
c: Option<char>,
valid: bool,
}
const UP: char = 'A'; const DOWN: char = 'B'; const RIGHT: char = 'C'; const LEFT: char = 'D'; const END: char = 'F'; const HOME: char = 'H'; const INSERT: char = '2'; const DELETE: char = '3'; const PAGE_UP: char = '5'; const PAGE_DOWN: char = '6'; const RXVT_HOME: char = '7';
const RXVT_END: char = '8';
const SHIFT: char = '2';
const ALT: char = '3';
const ALT_SHIFT: char = '4';
const CTRL: char = '5';
const CTRL_SHIFT: char = '6';
const CTRL_ALT: char = '7';
const CTRL_ALT_SHIFT: char = '8';
const RXVT_SHIFT: char = '$';
const RXVT_CTRL: char = '\x1e';
const RXVT_CTRL_SHIFT: char = '@';
impl PosixRawReader {
fn new(
fd: RawFd,
sigwinch_pipe: Option<RawFd>,
config: &Config,
key_map: PosixKeyMap,
pipe_reader: Option<PipeReader>,
) -> Self {
Self {
tty_in: BufReader::with_capacity(1024, TtyIn { fd, sigwinch_pipe }),
timeout_ms: config.keyseq_timeout(),
parser: Parser::new(),
key_map,
pipe_reader,
fds: FdSet::new(),
}
}
fn escape_sequence(&mut self) -> Result<KeyEvent> {
self._do_escape_sequence(true)
}
fn _do_escape_sequence(&mut self, allow_recurse: bool) -> Result<KeyEvent> {
let seq1 = self.next_char()?;
if seq1 == '[' {
self.escape_csi()
} else if seq1 == 'O' {
self.escape_o()
} else if seq1 == '\x1b' {
if !allow_recurse {
return Ok(E::ESC);
}
let timeout = if self.timeout_ms < 0 {
100
} else {
self.timeout_ms
};
match self.poll(timeout) {
Ok(0) | Err(_) => Ok(E::ESC),
Ok(n) => {
debug_assert!(n > 0, "{}", n);
let E(k, m) = self._do_escape_sequence(false)?;
Ok(E(k, m | M::ALT))
}
}
} else {
Ok(E::alt(seq1))
}
}
fn escape_csi(&mut self) -> Result<KeyEvent> {
let seq2 = self.next_char()?;
if seq2.is_ascii_digit() {
match seq2 {
'0' | '9' => {
debug!(target: "rustyline", "unsupported esc sequence: \\E[{:?}", seq2);
Ok(E(K::UnknownEscSeq, M::NONE))
}
_ => {
self.extended_escape(seq2)
}
}
} else if seq2 == '[' {
let seq3 = self.next_char()?;
Ok(match seq3 {
'A' => E(K::F(1), M::NONE),
'B' => E(K::F(2), M::NONE),
'C' => E(K::F(3), M::NONE),
'D' => E(K::F(4), M::NONE),
'E' => E(K::F(5), M::NONE),
_ => {
debug!(target: "rustyline", "unsupported esc sequence: \\E[[{:?}", seq3);
E(K::UnknownEscSeq, M::NONE)
}
})
} else {
Ok(match seq2 {
UP => E(K::Up, M::NONE),
DOWN => E(K::Down, M::NONE),
RIGHT => E(K::Right, M::NONE),
LEFT => E(K::Left, M::NONE),
END => E(K::End, M::NONE),
HOME => E(K::Home, M::NONE), 'Z' => E(K::BackTab, M::NONE),
'a' => E(K::Up, M::SHIFT), 'b' => E(K::Down, M::SHIFT), 'c' => E(K::Right, M::SHIFT), 'd' => E(K::Left, M::SHIFT), _ => {
debug!(target: "rustyline", "unsupported esc sequence: \\E[{:?}", seq2);
E(K::UnknownEscSeq, M::NONE)
}
})
}
}
#[allow(clippy::cognitive_complexity)]
fn extended_escape(&mut self, seq2: char) -> Result<KeyEvent> {
let seq3 = self.next_char()?;
if seq3 == '~' {
Ok(match seq2 {
'1' | RXVT_HOME => E(K::Home, M::NONE), INSERT => E(K::Insert, M::NONE),
DELETE => E(K::Delete, M::NONE),
'4' | RXVT_END => E(K::End, M::NONE), PAGE_UP => E(K::PageUp, M::NONE),
PAGE_DOWN => E(K::PageDown, M::NONE),
_ => {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}~", seq2);
E(K::UnknownEscSeq, M::NONE)
}
})
} else if seq3.is_ascii_digit() {
let seq4 = self.next_char()?;
if seq4 == '~' {
Ok(match (seq2, seq3) {
('1', '1') => E(K::F(1), M::NONE), ('1', '2') => E(K::F(2), M::NONE), ('1', '3') => E(K::F(3), M::NONE), ('1', '4') => E(K::F(4), M::NONE), ('1', '5') => E(K::F(5), M::NONE), ('1', '7') => E(K::F(6), M::NONE), ('1', '8') => E(K::F(7), M::NONE), ('1', '9') => E(K::F(8), M::NONE), ('2', '0') => E(K::F(9), M::NONE), ('2', '1') => E(K::F(10), M::NONE), ('2', '3') => E(K::F(11), M::NONE), ('2', '4') => E(K::F(12), M::NONE), _ => {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}{}~", seq2, seq3);
E(K::UnknownEscSeq, M::NONE)
}
})
} else if seq4 == ';' {
let seq5 = self.next_char()?;
if seq5.is_ascii_digit() {
let seq6 = self.next_char()?;
if seq6.is_ascii_digit() {
self.next_char()?; Ok(E(K::UnknownEscSeq, M::NONE))
} else if seq6 == 'R' {
Ok(E(K::UnknownEscSeq, M::NONE))
} else if seq6 == '~' {
Ok(match (seq2, seq3, seq5) {
('1', '5', CTRL) => E(K::F(5), M::CTRL),
('1', '7', CTRL) => E(K::F(6), M::CTRL),
('1', '8', CTRL) => E(K::F(7), M::CTRL),
('1', '9', CTRL) => E(K::F(8), M::CTRL),
('2', '0', CTRL) => E(K::F(9), M::CTRL),
('2', '1', CTRL) => E(K::F(10), M::CTRL),
('2', '3', CTRL) => E(K::F(11), M::CTRL),
('2', '4', CTRL) => E(K::F(12), M::CTRL),
_ => {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}{};{}~", seq2, seq3, seq5);
E(K::UnknownEscSeq, M::NONE)
}
})
} else {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}{};{}{}", seq2, seq3, seq5, seq6);
Ok(E(K::UnknownEscSeq, M::NONE))
}
} else {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}{};{:?}", seq2, seq3, seq5);
Ok(E(K::UnknownEscSeq, M::NONE))
}
} else if seq4.is_ascii_digit() {
let seq5 = self.next_char()?;
if seq5 == '~' {
Ok(match (seq2, seq3, seq4) {
('2', '0', '0') => E(K::BracketedPasteStart, M::NONE),
('2', '0', '1') => E(K::BracketedPasteEnd, M::NONE),
_ => {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}{}{}~", seq2, seq3, seq4);
E(K::UnknownEscSeq, M::NONE)
}
})
} else {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}{}{}{}", seq2, seq3, seq4, seq5);
Ok(E(K::UnknownEscSeq, M::NONE))
}
} else {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}{}{:?}", seq2, seq3, seq4);
Ok(E(K::UnknownEscSeq, M::NONE))
}
} else if seq3 == ';' {
let seq4 = self.next_char()?;
if seq4.is_ascii_digit() {
let seq5 = self.next_char()?;
if seq5.is_ascii_digit() {
self.next_char()?; Ok(E(K::UnknownEscSeq, M::NONE))
} else if seq2 == '1' {
Ok(match (seq4, seq5) {
(SHIFT, UP) => E(K::Up, M::SHIFT), (SHIFT, DOWN) => E(K::Down, M::SHIFT), (SHIFT, RIGHT) => E(K::Right, M::SHIFT),
(SHIFT, LEFT) => E(K::Left, M::SHIFT),
(SHIFT, END) => E(K::End, M::SHIFT), (SHIFT, HOME) => E(K::Home, M::SHIFT), (ALT, UP) => E(K::Up, M::ALT),
(ALT, DOWN) => E(K::Down, M::ALT),
(ALT, RIGHT) => E(K::Right, M::ALT),
(ALT, LEFT) => E(K::Left, M::ALT),
(ALT, END) => E(K::End, M::ALT),
(ALT, HOME) => E(K::Home, M::ALT),
(ALT_SHIFT, UP) => E(K::Up, M::ALT_SHIFT),
(ALT_SHIFT, DOWN) => E(K::Down, M::ALT_SHIFT),
(ALT_SHIFT, RIGHT) => E(K::Right, M::ALT_SHIFT),
(ALT_SHIFT, LEFT) => E(K::Left, M::ALT_SHIFT),
(ALT_SHIFT, END) => E(K::End, M::ALT_SHIFT),
(ALT_SHIFT, HOME) => E(K::Home, M::ALT_SHIFT),
(CTRL, UP) => E(K::Up, M::CTRL),
(CTRL, DOWN) => E(K::Down, M::CTRL),
(CTRL, RIGHT) => E(K::Right, M::CTRL),
(CTRL, LEFT) => E(K::Left, M::CTRL),
(CTRL, END) => E(K::End, M::CTRL),
(CTRL, HOME) => E(K::Home, M::CTRL),
(CTRL, 'P') => E(K::F(1), M::CTRL),
(CTRL, 'Q') => E(K::F(2), M::CTRL),
(CTRL, 'S') => E(K::F(4), M::CTRL),
(CTRL, 'p') => E(K::Char('0'), M::CTRL),
(CTRL, 'q') => E(K::Char('1'), M::CTRL),
(CTRL, 'r') => E(K::Char('2'), M::CTRL),
(CTRL, 's') => E(K::Char('3'), M::CTRL),
(CTRL, 't') => E(K::Char('4'), M::CTRL),
(CTRL, 'u') => E(K::Char('5'), M::CTRL),
(CTRL, 'v') => E(K::Char('6'), M::CTRL),
(CTRL, 'w') => E(K::Char('7'), M::CTRL),
(CTRL, 'x') => E(K::Char('8'), M::CTRL),
(CTRL, 'y') => E(K::Char('9'), M::CTRL),
(CTRL_SHIFT, UP) => E(K::Up, M::CTRL_SHIFT),
(CTRL_SHIFT, DOWN) => E(K::Down, M::CTRL_SHIFT),
(CTRL_SHIFT, RIGHT) => E(K::Right, M::CTRL_SHIFT),
(CTRL_SHIFT, LEFT) => E(K::Left, M::CTRL_SHIFT),
(CTRL_SHIFT, END) => E(K::End, M::CTRL_SHIFT),
(CTRL_SHIFT, HOME) => E(K::Home, M::CTRL_SHIFT),
(CTRL_SHIFT, 'p') => E(K::Char('0'), M::CTRL_SHIFT),
(CTRL_SHIFT, 'q') => E(K::Char('1'), M::CTRL_SHIFT),
(CTRL_SHIFT, 'r') => E(K::Char('2'), M::CTRL_SHIFT),
(CTRL_SHIFT, 's') => E(K::Char('3'), M::CTRL_SHIFT),
(CTRL_SHIFT, 't') => E(K::Char('4'), M::CTRL_SHIFT),
(CTRL_SHIFT, 'u') => E(K::Char('5'), M::CTRL_SHIFT),
(CTRL_SHIFT, 'v') => E(K::Char('6'), M::CTRL_SHIFT),
(CTRL_SHIFT, 'w') => E(K::Char('7'), M::CTRL_SHIFT),
(CTRL_SHIFT, 'x') => E(K::Char('8'), M::CTRL_SHIFT),
(CTRL_SHIFT, 'y') => E(K::Char('9'), M::CTRL_SHIFT),
(CTRL_ALT, UP) => E(K::Up, M::CTRL_ALT),
(CTRL_ALT, DOWN) => E(K::Down, M::CTRL_ALT),
(CTRL_ALT, RIGHT) => E(K::Right, M::CTRL_ALT),
(CTRL_ALT, LEFT) => E(K::Left, M::CTRL_ALT),
(CTRL_ALT, END) => E(K::End, M::CTRL_ALT),
(CTRL_ALT, HOME) => E(K::Home, M::CTRL_ALT),
(CTRL_ALT, 'p') => E(K::Char('0'), M::CTRL_ALT),
(CTRL_ALT, 'q') => E(K::Char('1'), M::CTRL_ALT),
(CTRL_ALT, 'r') => E(K::Char('2'), M::CTRL_ALT),
(CTRL_ALT, 's') => E(K::Char('3'), M::CTRL_ALT),
(CTRL_ALT, 't') => E(K::Char('4'), M::CTRL_ALT),
(CTRL_ALT, 'u') => E(K::Char('5'), M::CTRL_ALT),
(CTRL_ALT, 'v') => E(K::Char('6'), M::CTRL_ALT),
(CTRL_ALT, 'w') => E(K::Char('7'), M::CTRL_ALT),
(CTRL_ALT, 'x') => E(K::Char('8'), M::CTRL_ALT),
(CTRL_ALT, 'y') => E(K::Char('9'), M::CTRL_ALT),
(CTRL_ALT_SHIFT, UP) => E(K::Up, M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, DOWN) => E(K::Down, M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, RIGHT) => E(K::Right, M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, LEFT) => E(K::Left, M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, END) => E(K::End, M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, HOME) => E(K::Home, M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 'p') => E(K::Char('0'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 'q') => E(K::Char('1'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 'r') => E(K::Char('2'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 's') => E(K::Char('3'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 't') => E(K::Char('4'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 'u') => E(K::Char('5'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 'v') => E(K::Char('6'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 'w') => E(K::Char('7'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 'x') => E(K::Char('8'), M::CTRL_ALT_SHIFT),
(CTRL_ALT_SHIFT, 'y') => E(K::Char('9'), M::CTRL_ALT_SHIFT),
('9', UP) => E(K::Up, M::ALT),
('9', DOWN) => E(K::Down, M::ALT),
('9', RIGHT) => E(K::Right, M::ALT),
('9', LEFT) => E(K::Left, M::ALT),
_ => {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[1;{}{:?}", seq4, seq5);
E(K::UnknownEscSeq, M::NONE)
}
})
} else if seq5 == '~' {
Ok(match (seq2, seq4) {
(INSERT, SHIFT) => E(K::Insert, M::SHIFT),
(INSERT, ALT) => E(K::Insert, M::ALT),
(INSERT, ALT_SHIFT) => E(K::Insert, M::ALT_SHIFT),
(INSERT, CTRL) => E(K::Insert, M::CTRL),
(INSERT, CTRL_SHIFT) => E(K::Insert, M::CTRL_SHIFT),
(INSERT, CTRL_ALT) => E(K::Insert, M::CTRL_ALT),
(INSERT, CTRL_ALT_SHIFT) => E(K::Insert, M::CTRL_ALT_SHIFT),
(DELETE, SHIFT) => E(K::Delete, M::SHIFT),
(DELETE, ALT) => E(K::Delete, M::ALT),
(DELETE, ALT_SHIFT) => E(K::Delete, M::ALT_SHIFT),
(DELETE, CTRL) => E(K::Delete, M::CTRL),
(DELETE, CTRL_SHIFT) => E(K::Delete, M::CTRL_SHIFT),
(DELETE, CTRL_ALT) => E(K::Delete, M::CTRL_ALT),
(DELETE, CTRL_ALT_SHIFT) => E(K::Delete, M::CTRL_ALT_SHIFT),
(PAGE_UP, SHIFT) => E(K::PageUp, M::SHIFT),
(PAGE_UP, ALT) => E(K::PageUp, M::ALT),
(PAGE_UP, ALT_SHIFT) => E(K::PageUp, M::ALT_SHIFT),
(PAGE_UP, CTRL) => E(K::PageUp, M::CTRL),
(PAGE_UP, CTRL_SHIFT) => E(K::PageUp, M::CTRL_SHIFT),
(PAGE_UP, CTRL_ALT) => E(K::PageUp, M::CTRL_ALT),
(PAGE_UP, CTRL_ALT_SHIFT) => E(K::PageUp, M::CTRL_ALT_SHIFT),
(PAGE_DOWN, SHIFT) => E(K::PageDown, M::SHIFT),
(PAGE_DOWN, ALT) => E(K::PageDown, M::ALT),
(PAGE_DOWN, ALT_SHIFT) => E(K::PageDown, M::ALT_SHIFT),
(PAGE_DOWN, CTRL) => E(K::PageDown, M::CTRL),
(PAGE_DOWN, CTRL_SHIFT) => E(K::PageDown, M::CTRL_SHIFT),
(PAGE_DOWN, CTRL_ALT) => E(K::PageDown, M::CTRL_ALT),
(PAGE_DOWN, CTRL_ALT_SHIFT) => E(K::PageDown, M::CTRL_ALT_SHIFT),
_ => {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{};{:?}~", seq2, seq4);
E(K::UnknownEscSeq, M::NONE)
}
})
} else {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{};{}{:?}", seq2, seq4, seq5);
Ok(E(K::UnknownEscSeq, M::NONE))
}
} else {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{};{:?}", seq2, seq4);
Ok(E(K::UnknownEscSeq, M::NONE))
}
} else {
Ok(match (seq2, seq3) {
(DELETE, RXVT_CTRL) => E(K::Delete, M::CTRL),
(DELETE, RXVT_CTRL_SHIFT) => E(K::Delete, M::CTRL_SHIFT),
(CTRL, UP) => E(K::Up, M::CTRL),
(CTRL, DOWN) => E(K::Down, M::CTRL),
(CTRL, RIGHT) => E(K::Right, M::CTRL),
(CTRL, LEFT) => E(K::Left, M::CTRL),
(PAGE_UP, RXVT_CTRL) => E(K::PageUp, M::CTRL),
(PAGE_UP, RXVT_SHIFT) => E(K::PageUp, M::SHIFT),
(PAGE_UP, RXVT_CTRL_SHIFT) => E(K::PageUp, M::CTRL_SHIFT),
(PAGE_DOWN, RXVT_CTRL) => E(K::PageDown, M::CTRL),
(PAGE_DOWN, RXVT_SHIFT) => E(K::PageDown, M::SHIFT),
(PAGE_DOWN, RXVT_CTRL_SHIFT) => E(K::PageDown, M::CTRL_SHIFT),
(RXVT_HOME, RXVT_CTRL) => E(K::Home, M::CTRL),
(RXVT_HOME, RXVT_SHIFT) => E(K::Home, M::SHIFT),
(RXVT_HOME, RXVT_CTRL_SHIFT) => E(K::Home, M::CTRL_SHIFT),
(RXVT_END, RXVT_CTRL) => E(K::End, M::CTRL), (RXVT_END, RXVT_SHIFT) => E(K::End, M::SHIFT),
(RXVT_END, RXVT_CTRL_SHIFT) => E(K::End, M::CTRL_SHIFT),
_ => {
debug!(target: "rustyline",
"unsupported esc sequence: \\E[{}{:?}", seq2, seq3);
E(K::UnknownEscSeq, M::NONE)
}
})
}
}
fn escape_o(&mut self) -> Result<KeyEvent> {
let seq2 = self.next_char()?;
Ok(match seq2 {
UP => E(K::Up, M::NONE),
DOWN => E(K::Down, M::NONE),
RIGHT => E(K::Right, M::NONE),
LEFT => E(K::Left, M::NONE),
END => E(K::End, M::NONE), HOME => E(K::Home, M::NONE), 'M' => E::ENTER, 'P' => E(K::F(1), M::NONE), 'Q' => E(K::F(2), M::NONE), 'R' => E(K::F(3), M::NONE), 'S' => E(K::F(4), M::NONE), 'a' => E(K::Up, M::CTRL),
'b' => E(K::Down, M::CTRL),
'c' => E(K::Right, M::CTRL), 'd' => E(K::Left, M::CTRL), 'l' => E(K::F(8), M::NONE),
't' => E(K::F(5), M::NONE), 'u' => E(K::F(6), M::NONE), 'v' => E(K::F(7), M::NONE), 'w' => E(K::F(9), M::NONE), 'x' => E(K::F(10), M::NONE), _ => {
debug!(target: "rustyline", "unsupported esc sequence: \\EO{:?}", seq2);
E(K::UnknownEscSeq, M::NONE)
}
})
}
fn poll(&mut self, timeout_ms: i32) -> Result<i32> {
let n = self.tty_in.buffer().len();
if n > 0 {
return Ok(n as i32);
}
let mut fds = [poll::PollFd::new(self.as_raw_fd(), PollFlags::POLLIN)];
let r = poll::poll(&mut fds, timeout_ms);
match r {
Ok(n) => Ok(n),
Err(Errno::EINTR) => {
if self.tty_in.get_ref().sigwinch()? {
Err(ReadlineError::WindowResized)
} else {
Ok(0) }
}
Err(e) => Err(e.into()),
}
}
fn select(&mut self, single_esc_abort: bool) -> Result<Event> {
let tty_in = self.as_raw_fd();
let sigwinch_pipe = self.tty_in.get_ref().sigwinch_pipe;
let pipe_reader = self
.pipe_reader
.as_ref()
.map(|pr| pr.lock().unwrap().0.as_raw_fd());
loop {
let mut readfds = self.fds;
readfds.clear();
if let Some(sigwinch_pipe) = sigwinch_pipe {
readfds.insert(sigwinch_pipe);
}
readfds.insert(tty_in);
if let Some(pipe_reader) = pipe_reader {
readfds.insert(pipe_reader);
}
if let Err(err) = select::select(
readfds.highest().map(|h| h + 1),
Some(&mut readfds),
None,
None,
None,
) {
if err == Errno::EINTR && self.tty_in.get_ref().sigwinch()? {
return Err(ReadlineError::WindowResized);
} else if err != Errno::EINTR {
return Err(err.into());
} else {
continue;
}
};
if sigwinch_pipe.map_or(false, |fd| readfds.contains(fd)) {
self.tty_in.get_ref().sigwinch()?;
return Err(ReadlineError::WindowResized);
} else if readfds.contains(tty_in) {
return self.next_key(single_esc_abort).map(Event::KeyPress);
} else if let Some(ref pipe_reader) = self.pipe_reader {
let mut guard = pipe_reader.lock().unwrap();
let mut buf = [0; 1];
guard.0.read_exact(&mut buf)?;
if let Ok(msg) = guard.1.try_recv() {
return Ok(Event::ExternalPrint(msg));
}
}
}
}
}
impl RawReader for PosixRawReader {
#[cfg(not(feature = "signal-hook"))]
fn wait_for_input(&mut self, single_esc_abort: bool) -> Result<Event> {
match self.pipe_reader {
Some(_) => self.select(single_esc_abort),
None => self.next_key(single_esc_abort).map(Event::KeyPress),
}
}
#[cfg(feature = "signal-hook")]
fn wait_for_input(&mut self, single_esc_abort: bool) -> Result<Event> {
self.select(single_esc_abort)
}
fn next_key(&mut self, single_esc_abort: bool) -> Result<KeyEvent> {
let c = self.next_char()?;
let mut key = KeyEvent::new(c, M::NONE);
if key == E::ESC {
if !self.tty_in.buffer().is_empty() {
debug!(target: "rustyline", "read buffer {:?}", self.tty_in.buffer());
}
let timeout_ms = if single_esc_abort && self.timeout_ms == -1 {
0
} else {
self.timeout_ms
};
match self.poll(timeout_ms) {
Ok(n) if n == 0 => {
}
Ok(_) => {
key = self.escape_sequence()?
}
Err(e) => return Err(e),
}
}
debug!(target: "rustyline", "c: {:?} => key: {:?}", c, key);
Ok(key)
}
fn next_char(&mut self) -> Result<char> {
let mut buf = [0; 1];
let mut receiver = Utf8 {
c: None,
valid: true,
};
loop {
let n = self.tty_in.read(&mut buf)?;
if n == 0 {
return Err(error::ReadlineError::Eof);
}
let b = buf[0];
self.parser.advance(&mut receiver, b);
if !receiver.valid {
return Err(error::ReadlineError::from(ErrorKind::InvalidData));
} else if let Some(c) = receiver.c.take() {
return Ok(c);
}
}
}
fn read_pasted_text(&mut self) -> Result<String> {
let mut buffer = String::new();
loop {
match self.next_char()? {
'\x1b' => {
let key = self.escape_sequence()?;
if key == E(K::BracketedPasteEnd, M::NONE) {
break;
} else {
continue; }
}
c => buffer.push(c),
};
}
let buffer = buffer.replace("\r\n", "\n");
let buffer = buffer.replace('\r', "\n");
Ok(buffer)
}
fn find_binding(&self, key: &KeyEvent) -> Option<Cmd> {
let cmd = self.key_map.get(key).cloned();
if let Some(ref cmd) = cmd {
debug!(target: "rustyline", "terminal key binding: {:?} => {:?}", key, cmd);
}
cmd
}
}
impl Receiver for Utf8 {
fn codepoint(&mut self, c: char) {
self.c = Some(c);
self.valid = true;
}
fn invalid_sequence(&mut self) {
self.c = None;
self.valid = false;
}
}
pub struct PosixRenderer {
out: RawFd,
cols: usize, buffer: String,
tab_stop: usize,
colors_enabled: bool,
bell_style: BellStyle,
}
impl PosixRenderer {
fn new(out: RawFd, tab_stop: usize, colors_enabled: bool, bell_style: BellStyle) -> Self {
let (cols, _) = get_win_size(out);
Self {
out,
cols,
buffer: String::with_capacity(1024),
tab_stop,
colors_enabled,
bell_style,
}
}
fn clear_old_rows(&mut self, layout: &Layout) {
use std::fmt::Write;
let current_row = layout.cursor.row;
let old_rows = layout.end.row;
let cursor_row_movement = old_rows.saturating_sub(current_row);
if cursor_row_movement > 0 {
write!(self.buffer, "\x1b[{}B", cursor_row_movement).unwrap();
}
for _ in 0..old_rows {
self.buffer.push_str("\r\x1b[K\x1b[A");
}
self.buffer.push_str("\r\x1b[K");
}
}
impl Renderer for PosixRenderer {
type Reader = PosixRawReader;
fn move_cursor(&mut self, old: Position, new: Position) -> Result<()> {
use std::fmt::Write;
self.buffer.clear();
let row_ordering = new.row.cmp(&old.row);
if row_ordering == cmp::Ordering::Greater {
let row_shift = new.row - old.row;
if row_shift == 1 {
self.buffer.push_str("\x1b[B");
} else {
write!(self.buffer, "\x1b[{}B", row_shift)?;
}
} else if row_ordering == cmp::Ordering::Less {
let row_shift = old.row - new.row;
if row_shift == 1 {
self.buffer.push_str("\x1b[A");
} else {
write!(self.buffer, "\x1b[{}A", row_shift)?;
}
}
let col_ordering = new.col.cmp(&old.col);
if col_ordering == cmp::Ordering::Greater {
let col_shift = new.col - old.col;
if col_shift == 1 {
self.buffer.push_str("\x1b[C");
} else {
write!(self.buffer, "\x1b[{}C", col_shift)?;
}
} else if col_ordering == cmp::Ordering::Less {
let col_shift = old.col - new.col;
if col_shift == 1 {
self.buffer.push_str("\x1b[D");
} else {
write!(self.buffer, "\x1b[{}D", col_shift)?;
}
}
write_all(self.out, self.buffer.as_str())?;
Ok(())
}
fn refresh_line(
&mut self,
prompt: &str,
line: &LineBuffer,
hint: Option<&str>,
old_layout: &Layout,
new_layout: &Layout,
highlighter: Option<&dyn Highlighter>,
) -> Result<()> {
use std::fmt::Write;
self.buffer.clear();
let default_prompt = new_layout.default_prompt;
let cursor = new_layout.cursor;
let end_pos = new_layout.end;
self.clear_old_rows(old_layout);
if let Some(highlighter) = highlighter {
self.buffer
.push_str(&highlighter.highlight_prompt(prompt, default_prompt));
self.buffer
.push_str(&highlighter.highlight(line, line.pos()));
} else {
self.buffer.push_str(prompt);
self.buffer.push_str(line);
}
if let Some(hint) = hint {
if let Some(highlighter) = highlighter {
self.buffer.push_str(&highlighter.highlight_hint(hint));
} else {
self.buffer.push_str(hint);
}
}
if end_pos.col == 0
&& end_pos.row > 0
&& !hint
.map(|h| h.ends_with('\n'))
.unwrap_or_else(|| line.ends_with('\n'))
{
self.buffer.push('\n');
}
let new_cursor_row_movement = end_pos.row - cursor.row;
if new_cursor_row_movement > 0 {
write!(self.buffer, "\x1b[{}A", new_cursor_row_movement)?;
}
if cursor.col > 0 {
write!(self.buffer, "\r\x1b[{}C", cursor.col).unwrap();
} else {
self.buffer.push('\r');
}
write_all(self.out, self.buffer.as_str())?;
Ok(())
}
fn write_and_flush(&mut self, buf: &str) -> Result<()> {
write_all(self.out, buf)?;
Ok(())
}
fn calculate_position(&self, s: &str, orig: Position) -> Position {
let mut pos = orig;
let mut esc_seq = 0;
for c in s.graphemes(true) {
if c == "\n" {
pos.row += 1;
pos.col = 0;
continue;
}
let cw = if c == "\t" {
self.tab_stop - (pos.col % self.tab_stop)
} else {
width(c, &mut esc_seq)
};
pos.col += cw;
if pos.col > self.cols {
pos.row += 1;
pos.col = cw;
}
}
if pos.col == self.cols {
pos.col = 0;
pos.row += 1;
}
pos
}
fn beep(&mut self) -> Result<()> {
match self.bell_style {
BellStyle::Audible => self.write_and_flush("\x07"),
_ => Ok(()),
}
}
fn clear_screen(&mut self) -> Result<()> {
self.write_and_flush("\x1b[H\x1b[J")
}
fn clear_rows(&mut self, layout: &Layout) -> Result<()> {
self.buffer.clear();
self.clear_old_rows(layout);
write_all(self.out, self.buffer.as_str())?;
Ok(())
}
fn update_size(&mut self) {
let (cols, _) = get_win_size(self.out);
self.cols = cols;
}
fn get_columns(&self) -> usize {
self.cols
}
fn get_rows(&self) -> usize {
let (_, rows) = get_win_size(self.out);
rows
}
fn colors_enabled(&self) -> bool {
self.colors_enabled
}
fn move_cursor_at_leftmost(&mut self, rdr: &mut PosixRawReader) -> Result<()> {
if rdr.poll(0)? != 0 {
debug!(target: "rustyline", "cannot request cursor location");
return Ok(());
}
self.write_and_flush("\x1b[6n")?;
if rdr.poll(100)? == 0
|| rdr.next_char()? != '\x1b'
|| rdr.next_char()? != '['
|| read_digits_until(rdr, ';')?.is_none()
{
warn!(target: "rustyline", "cannot read initial cursor location");
return Ok(());
}
let col = read_digits_until(rdr, 'R')?;
debug!(target: "rustyline", "initial cursor location: {:?}", col);
if col != Some(1) {
self.write_and_flush("\n")?;
}
Ok(())
}
}
fn read_digits_until(rdr: &mut PosixRawReader, sep: char) -> Result<Option<u32>> {
let mut num: u32 = 0;
loop {
match rdr.next_char()? {
digit @ '0'..='9' => {
num = num
.saturating_mul(10)
.saturating_add(digit.to_digit(10).unwrap());
continue;
}
c if c == sep => break,
_ => return Ok(None),
}
}
Ok(Some(num))
}
fn write_all(fd: RawFd, buf: &str) -> nix::Result<()> {
let mut bytes = buf.as_bytes();
while !bytes.is_empty() {
match write(fd, bytes) {
Ok(0) => return Err(Errno::EIO),
Ok(n) => bytes = &bytes[n..],
Err(Errno::EINTR) => {}
Err(r) => return Err(r),
}
}
Ok(())
}
#[cfg(not(feature = "signal-hook"))]
static mut SIGWINCH_PIPE: RawFd = -1;
#[cfg(not(feature = "signal-hook"))]
extern "C" fn sigwinch_handler(_: libc::c_int) {
let _ = unsafe { write(SIGWINCH_PIPE, &[b's']) };
}
#[derive(Clone, Debug)]
struct SigWinCh {
pipe: RawFd,
#[cfg(not(feature = "signal-hook"))]
original: nix::sys::signal::SigAction,
#[cfg(feature = "signal-hook")]
id: signal_hook::SigId,
}
impl SigWinCh {
#[cfg(not(feature = "signal-hook"))]
fn install_sigwinch_handler() -> Result<SigWinCh> {
use nix::sys::signal;
let (pipe, pipe_write) = UnixStream::pair()?;
pipe.set_nonblocking(true)?;
unsafe { SIGWINCH_PIPE = pipe_write.into_raw_fd() };
let sigwinch = signal::SigAction::new(
signal::SigHandler::Handler(sigwinch_handler),
signal::SaFlags::empty(),
signal::SigSet::empty(),
);
let original = unsafe { signal::sigaction(signal::SIGWINCH, &sigwinch)? };
Ok(SigWinCh {
pipe: pipe.into_raw_fd(),
original,
})
}
#[cfg(feature = "signal-hook")]
fn install_sigwinch_handler() -> Result<SigWinCh> {
let (pipe, pipe_write) = UnixStream::pair()?;
pipe.set_nonblocking(true)?;
let id = signal_hook::low_level::pipe::register(libc::SIGWINCH, pipe_write)?;
Ok(SigWinCh {
pipe: pipe.into_raw_fd(),
id,
})
}
#[cfg(not(feature = "signal-hook"))]
fn uninstall_sigwinch_handler(self) -> Result<()> {
use nix::sys::signal;
let _ = unsafe { signal::sigaction(signal::SIGWINCH, &self.original)? };
close(self.pipe)?;
unsafe { close(SIGWINCH_PIPE)? };
unsafe { SIGWINCH_PIPE = -1 };
Ok(())
}
#[cfg(feature = "signal-hook")]
fn uninstall_sigwinch_handler(self) -> Result<()> {
signal_hook::low_level::unregister(self.id);
close(self.pipe)?;
Ok(())
}
}
fn map_key(key_map: &mut HashMap<KeyEvent, Cmd>, raw: &Termios, index: SCI, name: &str, cmd: Cmd) {
let cc = char::from(raw.control_chars[index as usize]);
let key = KeyEvent::new(cc, M::NONE);
debug!(target: "rustyline", "{}: {:?}", name, key);
key_map.insert(key, cmd);
}
#[cfg(not(test))]
pub type Terminal = PosixTerminal;
#[derive(Clone, Debug)]
pub struct PosixTerminal {
unsupported: bool,
tty_in: RawFd,
is_in_a_tty: bool,
tty_out: RawFd,
is_out_a_tty: bool,
close_on_drop: bool,
pub(crate) color_mode: ColorMode,
tab_stop: usize,
bell_style: BellStyle,
enable_bracketed_paste: bool,
raw_mode: Arc<AtomicBool>,
pipe_reader: Option<PipeReader>,
pipe_writer: Option<PipeWriter>,
sigwinch: Option<SigWinCh>,
}
impl PosixTerminal {
fn colors_enabled(&self) -> bool {
match self.color_mode {
ColorMode::Enabled => self.is_out_a_tty,
ColorMode::Forced => true,
ColorMode::Disabled => false,
}
}
}
impl Term for PosixTerminal {
type ExternalPrinter = ExternalPrinter;
type KeyMap = PosixKeyMap;
type Mode = PosixMode;
type Reader = PosixRawReader;
type Writer = PosixRenderer;
fn new(
color_mode: ColorMode,
behavior: Behavior,
tab_stop: usize,
bell_style: BellStyle,
enable_bracketed_paste: bool,
) -> Result<Self> {
let (tty_in, is_in_a_tty, tty_out, is_out_a_tty, close_on_drop) =
if behavior == Behavior::PreferTerm {
let tty = OpenOptions::new().read(true).write(true).open("/dev/tty");
if let Ok(tty) = tty {
let fd = tty.into_raw_fd();
let is_a_tty = is_a_tty(fd); (fd, is_a_tty, fd, is_a_tty, true)
} else {
(
libc::STDIN_FILENO,
is_a_tty(libc::STDIN_FILENO),
libc::STDOUT_FILENO,
is_a_tty(libc::STDOUT_FILENO),
false,
)
}
} else {
(
libc::STDIN_FILENO,
is_a_tty(libc::STDIN_FILENO),
libc::STDOUT_FILENO,
is_a_tty(libc::STDOUT_FILENO),
false,
)
};
let unsupported = is_unsupported_term();
#[allow(unused_variables)]
let sigwinch = if !unsupported && is_in_a_tty && is_out_a_tty {
Some(SigWinCh::install_sigwinch_handler()?)
} else {
None
};
Ok(Self {
unsupported,
tty_in,
is_in_a_tty,
tty_out,
is_out_a_tty,
close_on_drop,
color_mode,
tab_stop,
bell_style,
enable_bracketed_paste,
raw_mode: Arc::new(AtomicBool::new(false)),
pipe_reader: None,
pipe_writer: None,
sigwinch,
})
}
fn is_unsupported(&self) -> bool {
self.unsupported
}
fn is_input_tty(&self) -> bool {
self.is_in_a_tty
}
fn is_output_tty(&self) -> bool {
self.is_out_a_tty
}
fn enable_raw_mode(&mut self) -> Result<(Self::Mode, PosixKeyMap)> {
use nix::errno::Errno::ENOTTY;
use nix::sys::termios::{ControlFlags, InputFlags, LocalFlags};
if !self.is_in_a_tty {
return Err(ENOTTY.into());
}
let original_mode = termios::tcgetattr(self.tty_in)?;
let mut raw = original_mode.clone();
raw.input_flags &= !(InputFlags::BRKINT
| InputFlags::ICRNL
| InputFlags::INPCK
| InputFlags::ISTRIP
| InputFlags::IXON);
raw.control_flags |= ControlFlags::CS8;
raw.local_flags &=
!(LocalFlags::ECHO | LocalFlags::ICANON | LocalFlags::IEXTEN | LocalFlags::ISIG);
raw.control_chars[SCI::VMIN as usize] = 1; raw.control_chars[SCI::VTIME as usize] = 0; let mut key_map: HashMap<KeyEvent, Cmd> = HashMap::with_capacity(4);
map_key(&mut key_map, &raw, SCI::VEOF, "VEOF", Cmd::EndOfFile);
map_key(&mut key_map, &raw, SCI::VINTR, "VINTR", Cmd::Interrupt);
map_key(&mut key_map, &raw, SCI::VQUIT, "VQUIT", Cmd::Interrupt);
map_key(&mut key_map, &raw, SCI::VSUSP, "VSUSP", Cmd::Suspend);
termios::tcsetattr(self.tty_in, SetArg::TCSADRAIN, &raw)?;
self.raw_mode.store(true, Ordering::SeqCst);
let out = if !self.enable_bracketed_paste {
None
} else if let Err(e) = write_all(self.tty_out, BRACKETED_PASTE_ON) {
debug!(target: "rustyline", "Cannot enable bracketed paste: {}", e);
None
} else {
Some(self.tty_out)
};
if Arc::strong_count(&self.raw_mode) == 1 {
self.pipe_writer = None;
self.pipe_reader = None;
}
Ok((
PosixMode {
termios: original_mode,
tty_in: self.tty_in,
tty_out: out,
raw_mode: self.raw_mode.clone(),
},
key_map,
))
}
fn create_reader(&self, config: &Config, key_map: PosixKeyMap) -> PosixRawReader {
PosixRawReader::new(
self.tty_in,
self.sigwinch.as_ref().map(|s| s.pipe),
config,
key_map,
self.pipe_reader.clone(),
)
}
fn create_writer(&self) -> PosixRenderer {
PosixRenderer::new(
self.tty_out,
self.tab_stop,
self.colors_enabled(),
self.bell_style,
)
}
fn writeln(&self) -> Result<()> {
write_all(self.tty_out, "\n")?;
Ok(())
}
fn create_external_printer(&mut self) -> Result<ExternalPrinter> {
if let Some(ref writer) = self.pipe_writer {
return Ok(ExternalPrinter {
writer: writer.clone(),
raw_mode: self.raw_mode.clone(),
tty_out: self.tty_out,
});
}
if self.unsupported || !self.is_input_tty() || !self.is_output_tty() {
return Err(nix::Error::ENOTTY.into());
}
use nix::unistd::pipe;
use std::os::unix::io::FromRawFd;
let (sender, receiver) = mpsc::sync_channel(1); let (r, w) = pipe()?;
let reader = Arc::new(Mutex::new((unsafe { File::from_raw_fd(r) }, receiver)));
let writer = (
Arc::new(Mutex::new(unsafe { File::from_raw_fd(w) })),
sender,
);
self.pipe_reader.replace(reader);
self.pipe_writer.replace(writer.clone());
Ok(ExternalPrinter {
writer,
raw_mode: self.raw_mode.clone(),
tty_out: self.tty_out,
})
}
}
#[allow(unused_must_use)]
impl Drop for PosixTerminal {
fn drop(&mut self) {
if self.close_on_drop {
close(self.tty_in);
debug_assert_eq!(self.tty_in, self.tty_out);
}
if let Some(sigwinch) = self.sigwinch.take() {
sigwinch.uninstall_sigwinch_handler();
}
}
}
#[derive(Debug)]
pub struct ExternalPrinter {
writer: PipeWriter,
raw_mode: Arc<AtomicBool>,
tty_out: RawFd,
}
impl super::ExternalPrinter for ExternalPrinter {
fn print(&mut self, msg: String) -> Result<()> {
if !self.raw_mode.load(Ordering::SeqCst) {
write_all(self.tty_out, msg.as_str())?;
} else if let Ok(mut writer) = self.writer.0.lock() {
self.writer
.1
.send(msg)
.map_err(|_| io::Error::from(ErrorKind::Other))?; writer.write_all(&[b'm'])?;
writer.flush()?;
} else {
return Err(io::Error::from(ErrorKind::Other).into()); }
Ok(())
}
}
#[cfg(not(test))]
pub fn suspend() -> Result<()> {
use nix::sys::signal;
use nix::unistd::Pid;
signal::kill(Pid::from_raw(0), signal::SIGTSTP)?;
Ok(())
}
#[cfg(test)]
mod test {
use super::{Position, PosixRenderer, PosixTerminal, Renderer};
use crate::config::BellStyle;
use crate::line_buffer::LineBuffer;
#[test]
#[ignore]
fn prompt_with_ansi_escape_codes() {
let out = PosixRenderer::new(libc::STDOUT_FILENO, 4, true, BellStyle::default());
let pos = out.calculate_position("\x1b[1;32m>>\x1b[0m ", Position::default());
assert_eq!(3, pos.col);
assert_eq!(0, pos.row);
}
#[test]
fn test_unsupported_term() {
std::env::set_var("TERM", "xterm");
assert!(!super::is_unsupported_term());
std::env::set_var("TERM", "dumb");
assert!(super::is_unsupported_term());
}
#[test]
fn test_send() {
fn assert_send<T: Send>() {}
assert_send::<PosixTerminal>();
}
#[test]
fn test_sync() {
fn assert_sync<T: Sync>() {}
assert_sync::<PosixTerminal>();
}
#[test]
fn test_line_wrap() {
let mut out = PosixRenderer::new(libc::STDOUT_FILENO, 4, true, BellStyle::default());
let prompt = "> ";
let default_prompt = true;
let prompt_size = out.calculate_position(prompt, Position::default());
let mut line = LineBuffer::init("", 0, None);
let old_layout = out.compute_layout(prompt_size, default_prompt, &line, None);
assert_eq!(Position { col: 2, row: 0 }, old_layout.cursor);
assert_eq!(old_layout.cursor, old_layout.end);
assert_eq!(Some(true), line.insert('a', out.cols - prompt_size.col + 1));
let new_layout = out.compute_layout(prompt_size, default_prompt, &line, None);
assert_eq!(Position { col: 1, row: 1 }, new_layout.cursor);
assert_eq!(new_layout.cursor, new_layout.end);
out.refresh_line(prompt, &line, None, &old_layout, &new_layout, None)
.unwrap();
#[rustfmt::skip]
assert_eq!(
"\r\u{1b}[K> aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa\r\u{1b}[1C",
out.buffer
);
}
}