Skip to content

Commit

Permalink
✨ feat: Add deansi tty
Browse files Browse the repository at this point in the history
  • Loading branch information
wychlw committed Sep 18, 2024
1 parent 8c24f4d commit 6f84f5f
Show file tree
Hide file tree
Showing 4 changed files with 305 additions and 0 deletions.
36 changes: 36 additions & 0 deletions src/pythonapi/deansi.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
use pyo3::{exceptions::PyRuntimeError, pyclass, pymethods, PyResult};

use crate::{term::tty::Tty, util::anybase::heap_raw};

use super::shell_like::{handle_wrap, PyTty, PyTtyWrapper, TtyType};

pub fn handle_deansi(inner: &mut Option<PyTtyWrapper>) -> PyResult<()> {
if inner.is_none() {
return Err(PyRuntimeError::new_err(
"You must define at least one valid object",
));
}
let mut be_wrapped = inner.take().unwrap();
let be_wrapped = be_wrapped.safe_take()?;
let be_wrapped = Box::into_inner(be_wrapped);
let dean = Box::new(crate::term::deansi::DeANSI::build(be_wrapped));
let dean: Box<dyn Tty + Send> = dean as TtyType;
*inner = Some(PyTtyWrapper {
tty: heap_raw(dean),
});
Ok(())
}

#[pyclass(extends=PyTty, subclass)]
pub struct DeANSI {}

#[pymethods]
impl DeANSI {
#[new]
fn py_new(be_wrapped: &mut PyTty) -> PyResult<(Self, PyTty)> {
let mut inner = None;
handle_wrap(&mut inner, Some(be_wrapped))?;
handle_deansi(&mut inner)?;
Ok((DeANSI {}, PyTty::build(inner.unwrap())))
}
}
53 changes: 53 additions & 0 deletions src/term/deansi.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
use std::{any::Any, error::Error};

use crate::{util::anybase::AnyBase, vendor::strip_ansi_escapes};

use super::tty::{DynTty, Tty, WrapperTty};

pub struct DeANSI {
inner: DynTty,
}

impl DeANSI {
pub fn build(inner: DynTty) -> DeANSI {
DeANSI { inner }
}
}

impl AnyBase for DeANSI {
fn as_any(&self) -> &dyn Any {
self
}
fn as_any_mut(&mut self) -> &mut dyn Any {
self
}
fn into_any(self: Box<Self>) -> Box<dyn Any> {
self
}
}

impl Tty for DeANSI {
fn read(&mut self) -> Result<Vec<u8>, Box<dyn Error>> {
// Due to the escape sequences may be cut off in the middle of the buffer,
// read the buffer in line is needed.
let data = self.inner.read_line()?;
let data = strip_ansi_escapes::strip(&data);
Ok(data)
}
fn read_line(&mut self) -> Result<Vec<u8>, Box<dyn Error>> {
let data = self.inner.read_line()?;
let data = strip_ansi_escapes::strip(&data);
Ok(data)
}
fn write(&mut self, data: &[u8]) -> Result<(), Box<dyn Error>> {
let data = strip_ansi_escapes::strip(&data);
self.inner.write(&data)?;
Ok(())
}
}

impl WrapperTty for DeANSI {
fn exit(self) -> DynTty {
self.inner
}
}
1 change: 1 addition & 0 deletions src/vendor/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
pub mod strip_ansi_escapes;
215 changes: 215 additions & 0 deletions src/vendor/strip_ansi_escapes.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,215 @@
//! Originally from https://github.com/luser/strip-ansi-escapes
//! License: MIT/Apache-2.0
//! A crate for stripping ANSI escape sequences from byte sequences.
//!
//! This can be used to take output from a program that includes escape sequences and write
//! it somewhere that does not easily support them, such as a log file.
//!
//! The simplest interface provided is the [`strip`] function, which takes a byte slice and returns
//! a `Vec` of bytes with escape sequences removed. For writing bytes directly to a writer, you
//! may prefer using the [`Writer`] struct, which implements `Write` and strips escape sequences
//! as they are written.
//!
//! [`strip`]: fn.strip.html
//! [`Writer`]: struct.Writer.html
//!
//! # Example
//!
//! ```
//! use std::io::{self, Write};
//!
//! # fn foo() -> io::Result<()> {
//! let bytes_with_colors = b"\x1b[32mfoo\x1b[m bar";
//! let plain_bytes = strip_ansi_escapes::strip(&bytes_with_colors);
//! io::stdout().write_all(&plain_bytes)?;
//! # Ok(())
//! # }
//! ```
use std::io::{self, Cursor, IntoInnerError, LineWriter, Write};
use vte::{Parser, Perform};

/// `Writer` wraps an underlying type that implements `Write`, stripping ANSI escape sequences
/// from bytes written to it before passing them to the underlying writer.
///
/// # Example
/// ```
/// use std::io::{self, Write};
/// use strip_ansi_escapes::Writer;
///
/// # fn foo() -> io::Result<()> {
/// let bytes_with_colors = b"\x1b[32mfoo\x1b[m bar";
/// let mut writer = Writer::new(io::stdout());
/// // Only `foo bar` will be written to stdout
/// writer.write_all(bytes_with_colors)?;
/// # Ok(())
/// # }
/// ```
pub struct Writer<W>
where
W: Write,
{
performer: Performer<W>,
parser: Parser,
}

/// Strip ANSI escapes from `data` and return the remaining bytes as a `Vec<u8>`.
///
/// See [the module documentation][mod] for an example.
///
/// [mod]: index.html
pub fn strip<T>(data: T) -> Vec<u8>
where
T: AsRef<[u8]>,
{
fn strip_impl(data: &[u8]) -> io::Result<Vec<u8>> {
let c = Cursor::new(Vec::new());
let mut writer = Writer::new(c);
writer.write_all(data.as_ref())?;
Ok(writer.into_inner()?.into_inner())
}

strip_impl(data.as_ref()).expect("writing to a Cursor<Vec<u8>> cannot fail")
}

/// Strip ANSI escapes from `data` and return the remaining contents as a `String`.
///
/// # Example
///
/// ```
/// let str_with_colors = "\x1b[32mfoo\x1b[m bar";
/// let string_without_colors = strip_ansi_escapes::strip_str(str_with_colors);
/// assert_eq!(string_without_colors, "foo bar");
/// ```
pub fn strip_str<T>(data: T) -> String
where
T: AsRef<str>,
{
let bytes = strip(data.as_ref());
String::from_utf8(bytes)
.expect("stripping ANSI escapes from a UTF-8 string always results in UTF-8")
}

struct Performer<W>
where
W: Write,
{
writer: LineWriter<W>,
err: Option<io::Error>,
}

impl<W> Writer<W>
where
W: Write,
{
/// Create a new `Writer` that writes to `inner`.
pub fn new(inner: W) -> Writer<W> {
Writer {
performer: Performer {
writer: LineWriter::new(inner),
err: None,
},
parser: Parser::new(),
}
}

/// Unwraps this `Writer`, returning the underlying writer.
///
/// The internal buffer is written out before returning the writer, which
/// may produce an [`IntoInnerError`].
///
/// [IntoInnerError]: https://doc.rust-lang.org/std/io/struct.IntoInnerError.html
pub fn into_inner(self) -> Result<W, IntoInnerError<LineWriter<W>>> {
self.performer.into_inner()
}
}

impl<W> Write for Writer<W>
where
W: Write,
{
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
for b in buf.iter() {
self.parser.advance(&mut self.performer, *b)
}
match self.performer.err.take() {
Some(e) => Err(e),
None => Ok(buf.len()),
}
}

fn flush(&mut self) -> io::Result<()> {
self.performer.flush()
}
}

impl<W> Performer<W>
where
W: Write,
{
pub fn flush(&mut self) -> io::Result<()> {
self.writer.flush()
}

pub fn into_inner(self) -> Result<W, IntoInnerError<LineWriter<W>>> {
self.writer.into_inner()
}
}

impl<W> Perform for Performer<W>
where
W: Write,
{
fn print(&mut self, c: char) {
// Just print bytes to the inner writer.
self.err = write!(self.writer, "{}", c).err();
}
fn execute(&mut self, byte: u8) {
// We only care about executing linefeeds.
if byte == b'\n' {
self.err = writeln!(self.writer).err();
}
}
}

#[cfg(doctest)]
extern crate doc_comment;

#[cfg(doctest)]
doc_comment::doctest!("../README.md", readme);

#[cfg(test)]
mod tests {
use super::*;

fn assert_parsed(input: &[u8], expected: &[u8]) {
let bytes = strip(input);
assert_eq!(bytes, expected);
}

#[test]
fn test_simple() {
assert_parsed(b"\x1b[m\x1b[m\x1b[32m\x1b[1m Finished\x1b[m dev [unoptimized + debuginfo] target(s) in 0.0 secs",
b" Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs");
}

#[test]
fn test_newlines() {
assert_parsed(b"foo\nbar\n", b"foo\nbar\n");
}

#[test]
fn test_escapes_newlines() {
assert_parsed(b"\x1b[m\x1b[m\x1b[32m\x1b[1m Compiling\x1b[m utf8parse v0.1.0
\x1b[m\x1b[m\x1b[32m\x1b[1m Compiling\x1b[m vte v0.3.2
\x1b[m\x1b[m\x1b[32m\x1b[1m Compiling\x1b[m strip-ansi-escapes v0.1.0-pre (file:///build/strip-ansi-escapes)
\x1b[m\x1b[m\x1b[32m\x1b[1m Finished\x1b[m dev [unoptimized + debuginfo] target(s) in 0.66 secs
",
b" Compiling utf8parse v0.1.0
Compiling vte v0.3.2
Compiling strip-ansi-escapes v0.1.0-pre (file:///build/strip-ansi-escapes)
Finished dev [unoptimized + debuginfo] target(s) in 0.66 secs
");
}
}

0 comments on commit 6f84f5f

Please sign in to comment.