mirror of
https://github.com/fluencelabs/wasmer
synced 2025-04-04 00:31:07 +00:00
implement a cross-platform stdout pipe
This commit is contained in:
parent
dac0bd74c5
commit
3783b923d1
28
src/common/file_descriptor.rs
Normal file
28
src/common/file_descriptor.rs
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
use std::io::Read;
|
||||||
|
use std::io;
|
||||||
|
use std::io::Error;
|
||||||
|
use std::io::ErrorKind;
|
||||||
|
use std::ffi::CString;
|
||||||
|
|
||||||
|
pub struct FileDescriptor(libc::c_int);
|
||||||
|
|
||||||
|
impl FileDescriptor {
|
||||||
|
pub fn new(file_descriptor_number: libc::c_int) -> FileDescriptor {
|
||||||
|
FileDescriptor(file_descriptor_number)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Read for FileDescriptor {
|
||||||
|
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
||||||
|
let file_descriptor: libc::c_int = self.0;
|
||||||
|
let count = unsafe {
|
||||||
|
libc::read(file_descriptor, buf.as_mut_ptr() as *mut libc::c_void, 1)
|
||||||
|
};
|
||||||
|
if count < 0 {
|
||||||
|
Err(Error::new(ErrorKind::Other, "read error"))
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
Ok(count as usize)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -3,3 +3,4 @@ pub mod slice;
|
|||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
pub mod stdio;
|
pub mod stdio;
|
||||||
|
mod file_descriptor;
|
@ -1,7 +1,8 @@
|
|||||||
use libc;
|
use libc;
|
||||||
use std::fs::File;
|
use crate::file_descriptor::FileDescriptor;
|
||||||
|
use std::io::BufReader;
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
use std::os::unix::io::FromRawFd;
|
use crate::common::file_descriptor::FileDescriptor;
|
||||||
|
|
||||||
// A struct to hold the references to the base stdout and the captured one
|
// A struct to hold the references to the base stdout and the captured one
|
||||||
pub struct StdioCapturer {
|
pub struct StdioCapturer {
|
||||||
@ -17,7 +18,12 @@ pub struct StdioCapturer {
|
|||||||
impl StdioCapturer {
|
impl StdioCapturer {
|
||||||
fn pipe() -> (libc::c_int, libc::c_int) {
|
fn pipe() -> (libc::c_int, libc::c_int) {
|
||||||
let mut fds = [0; 2];
|
let mut fds = [0; 2];
|
||||||
|
|
||||||
|
#[cfg(not(target_os = "windows"))]
|
||||||
assert_eq!(unsafe { libc::pipe(fds.as_mut_ptr()) }, 0);
|
assert_eq!(unsafe { libc::pipe(fds.as_mut_ptr()) }, 0);
|
||||||
|
#[cfg(target_os = "windows")]
|
||||||
|
assert_eq!(unsafe { libc::pipe(fds.as_mut_ptr(), 1000, libc::O_TEXT) }, 0);
|
||||||
|
|
||||||
(fds[0], fds[1])
|
(fds[0], fds[1])
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -28,9 +34,6 @@ impl StdioCapturer {
|
|||||||
let (stdout_reader, stdout_writer) = Self::pipe();
|
let (stdout_reader, stdout_writer) = Self::pipe();
|
||||||
let (stderr_reader, stderr_writer) = Self::pipe();
|
let (stderr_reader, stderr_writer) = Self::pipe();
|
||||||
|
|
||||||
// std::io::stdout().flush().unwrap();
|
|
||||||
// std::io::stderr().flush().unwrap();
|
|
||||||
|
|
||||||
assert!(unsafe { libc::dup2(stdout_writer, libc::STDOUT_FILENO) } > -1);
|
assert!(unsafe { libc::dup2(stdout_writer, libc::STDOUT_FILENO) } > -1);
|
||||||
assert!(unsafe { libc::dup2(stderr_writer, libc::STDERR_FILENO) } > -1);
|
assert!(unsafe { libc::dup2(stderr_writer, libc::STDERR_FILENO) } > -1);
|
||||||
|
|
||||||
@ -48,28 +51,23 @@ impl StdioCapturer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn end(self) -> (String, String) {
|
pub fn end(self) -> Result<(String, String), std::io::Error> {
|
||||||
// The Stdio passed into the Command took over (and closed) std{out, err}
|
// The Stdio passed into the Command took over (and closed) std{out, err}
|
||||||
// so we should restore them as they were.
|
// so we should restore them as they were.
|
||||||
|
|
||||||
assert!(unsafe { libc::dup2(self.stdout_backup, libc::STDOUT_FILENO) } > -1);
|
assert!(unsafe { libc::dup2(self.stdout_backup, libc::STDOUT_FILENO) } > -1);
|
||||||
assert!(unsafe { libc::dup2(self.stderr_backup, libc::STDERR_FILENO) } > -1);
|
assert!(unsafe { libc::dup2(self.stderr_backup, libc::STDERR_FILENO) } > -1);
|
||||||
|
|
||||||
// assert_eq!(unsafe { libc::close(self.stdout_backup) }, 0);
|
let fd = FileDescriptor::new(self.stdout_reader);
|
||||||
// assert_eq!(unsafe { libc::close(self.stderr_backup) }, 0);
|
let mut reader = BufReader::new(fd);
|
||||||
|
let mut stdout_read = "".to_string();
|
||||||
|
let _ = reader.read_to_string(&mut stdout_read)?;
|
||||||
|
|
||||||
let mut stdout_read = String::new();
|
let fd = FileDescriptor::new(self.stderr_reader);
|
||||||
let mut stdout_file: File = unsafe { FromRawFd::from_raw_fd(self.stdout_reader) };
|
let mut reader = BufReader::new(fd);
|
||||||
stdout_file
|
let mut stderr_read = "".to_string();
|
||||||
.read_to_string(&mut stdout_read)
|
let _ = reader.read_to_string(&mut stderr_read)?;
|
||||||
.expect("failed to read from stdout file");
|
|
||||||
|
|
||||||
let mut stderr_read = String::new();
|
Ok((stdout_read, stderr_read))
|
||||||
let mut stderr_file: File = unsafe { FromRawFd::from_raw_fd(self.stderr_reader) };
|
|
||||||
stderr_file
|
|
||||||
.read_to_string(&mut stderr_read)
|
|
||||||
.expect("failed to read from stdout file");
|
|
||||||
|
|
||||||
(stdout_read, stderr_read)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user