mirror of
https://github.com/denoland/deno.git
synced 2025-01-21 21:50:00 -05:00
327 lines
8.8 KiB
Rust
327 lines
8.8 KiB
Rust
use super::dispatch_minimal::MinimalOp;
|
|
use crate::http_util::HttpBody;
|
|
use crate::op_error::OpError;
|
|
use crate::ops::minimal_op;
|
|
use crate::state::State;
|
|
use deno_core::*;
|
|
use futures::future::FutureExt;
|
|
use futures::ready;
|
|
use std::future::Future;
|
|
use std::pin::Pin;
|
|
use std::task::Context;
|
|
use std::task::Poll;
|
|
use tokio::io::{AsyncRead, AsyncWrite};
|
|
use tokio::net::TcpStream;
|
|
use tokio_rustls::client::TlsStream as ClientTlsStream;
|
|
use tokio_rustls::server::TlsStream as ServerTlsStream;
|
|
|
|
#[cfg(not(windows))]
|
|
use std::os::unix::io::FromRawFd;
|
|
|
|
#[cfg(windows)]
|
|
use std::os::windows::io::FromRawHandle;
|
|
|
|
#[cfg(windows)]
|
|
extern crate winapi;
|
|
|
|
lazy_static! {
|
|
/// Due to portability issues on Windows handle to stdout is created from raw file descriptor.
|
|
/// The caveat of that approach is fact that when this handle is dropped underlying
|
|
/// file descriptor is closed - that is highly not desirable in case of stdout.
|
|
/// That's why we store this global handle that is then cloned when obtaining stdio
|
|
/// for process. In turn when resource table is dropped storing reference to that handle,
|
|
/// the handle itself won't be closed (so Deno.core.print) will still work.
|
|
static ref STDOUT_HANDLE: std::fs::File = {
|
|
#[cfg(not(windows))]
|
|
let stdout = unsafe { std::fs::File::from_raw_fd(1) };
|
|
#[cfg(windows)]
|
|
let stdout = unsafe {
|
|
std::fs::File::from_raw_handle(winapi::um::processenv::GetStdHandle(
|
|
winapi::um::winbase::STD_OUTPUT_HANDLE,
|
|
))
|
|
};
|
|
|
|
stdout
|
|
};
|
|
}
|
|
|
|
pub fn init(i: &mut Isolate, s: &State) {
|
|
i.register_op(
|
|
"op_read",
|
|
s.core_op(minimal_op(s.stateful_minimal_op(op_read))),
|
|
);
|
|
i.register_op(
|
|
"op_write",
|
|
s.core_op(minimal_op(s.stateful_minimal_op(op_write))),
|
|
);
|
|
}
|
|
|
|
pub fn get_stdio() -> (StreamResource, StreamResource, StreamResource) {
|
|
let stdin = StreamResource::Stdin(tokio::io::stdin(), TTYMetadata::default());
|
|
let stdout = StreamResource::Stdout({
|
|
let stdout = STDOUT_HANDLE
|
|
.try_clone()
|
|
.expect("Unable to clone stdout handle");
|
|
tokio::fs::File::from_std(stdout)
|
|
});
|
|
let stderr = StreamResource::Stderr(tokio::io::stderr());
|
|
|
|
(stdin, stdout, stderr)
|
|
}
|
|
|
|
#[cfg(unix)]
|
|
use nix::sys::termios;
|
|
|
|
#[derive(Default)]
|
|
pub struct TTYMetadata {
|
|
#[cfg(unix)]
|
|
pub mode: Option<termios::Termios>,
|
|
}
|
|
|
|
#[derive(Default)]
|
|
pub struct FileMetadata {
|
|
pub tty: TTYMetadata,
|
|
}
|
|
|
|
pub enum StreamResource {
|
|
Stdin(tokio::io::Stdin, TTYMetadata),
|
|
Stdout(tokio::fs::File),
|
|
Stderr(tokio::io::Stderr),
|
|
FsFile(tokio::fs::File, FileMetadata),
|
|
TcpStream(tokio::net::TcpStream),
|
|
ServerTlsStream(Box<ServerTlsStream<TcpStream>>),
|
|
ClientTlsStream(Box<ClientTlsStream<TcpStream>>),
|
|
HttpBody(Box<HttpBody>),
|
|
ChildStdin(tokio::process::ChildStdin),
|
|
ChildStdout(tokio::process::ChildStdout),
|
|
ChildStderr(tokio::process::ChildStderr),
|
|
}
|
|
|
|
/// `DenoAsyncRead` is the same as the `tokio_io::AsyncRead` trait
|
|
/// but uses an `OpError` error instead of `std::io:Error`
|
|
pub trait DenoAsyncRead {
|
|
fn poll_read(
|
|
&mut self,
|
|
cx: &mut Context,
|
|
buf: &mut [u8],
|
|
) -> Poll<Result<usize, OpError>>;
|
|
}
|
|
|
|
impl DenoAsyncRead for StreamResource {
|
|
fn poll_read(
|
|
&mut self,
|
|
cx: &mut Context,
|
|
buf: &mut [u8],
|
|
) -> Poll<Result<usize, OpError>> {
|
|
use StreamResource::*;
|
|
let mut f: Pin<Box<dyn AsyncRead>> = match self {
|
|
FsFile(f, _) => Box::pin(f),
|
|
Stdin(f, _) => Box::pin(f),
|
|
TcpStream(f) => Box::pin(f),
|
|
ClientTlsStream(f) => Box::pin(f),
|
|
ServerTlsStream(f) => Box::pin(f),
|
|
ChildStdout(f) => Box::pin(f),
|
|
ChildStderr(f) => Box::pin(f),
|
|
HttpBody(f) => Box::pin(f),
|
|
_ => return Err(OpError::bad_resource()).into(),
|
|
};
|
|
|
|
let v = ready!(f.as_mut().poll_read(cx, buf))?;
|
|
Ok(v).into()
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, PartialEq)]
|
|
enum IoState {
|
|
Pending,
|
|
Flush,
|
|
Done,
|
|
}
|
|
|
|
/// A future which can be used to easily read available number of bytes to fill
|
|
/// a buffer.
|
|
///
|
|
/// Created by the [`read`] function.
|
|
pub struct Read<T> {
|
|
rid: ResourceId,
|
|
buf: T,
|
|
io_state: IoState,
|
|
state: State,
|
|
}
|
|
|
|
impl<T> Future for Read<T>
|
|
where
|
|
T: AsMut<[u8]> + Unpin,
|
|
{
|
|
type Output = Result<i32, OpError>;
|
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context) -> Poll<Self::Output> {
|
|
let inner = self.get_mut();
|
|
if inner.io_state == IoState::Done {
|
|
panic!("poll a Read after it's done");
|
|
}
|
|
|
|
let mut state = inner.state.borrow_mut();
|
|
let resource = state
|
|
.resource_table
|
|
.get_mut::<StreamResource>(inner.rid)
|
|
.ok_or_else(OpError::bad_resource)?;
|
|
let nread = ready!(resource.poll_read(cx, &mut inner.buf.as_mut()[..]))?;
|
|
inner.io_state = IoState::Done;
|
|
Poll::Ready(Ok(nread as i32))
|
|
}
|
|
}
|
|
|
|
fn no_buffer_specified() -> OpError {
|
|
OpError::type_error("no buffer specified".to_string())
|
|
}
|
|
|
|
pub fn op_read(
|
|
state: &State,
|
|
rid: i32,
|
|
zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Pin<Box<MinimalOp>> {
|
|
debug!("read rid={}", rid);
|
|
if zero_copy.is_none() {
|
|
return futures::future::err(no_buffer_specified()).boxed_local();
|
|
}
|
|
// TODO(ry) Probably poll_fn can be used here and the Read struct eliminated.
|
|
Read {
|
|
rid: rid as u32,
|
|
buf: zero_copy.unwrap(),
|
|
io_state: IoState::Pending,
|
|
state: state.clone(),
|
|
}
|
|
.boxed_local()
|
|
}
|
|
|
|
/// `DenoAsyncWrite` is the same as the `tokio_io::AsyncWrite` trait
|
|
/// but uses an `OpError` error instead of `std::io:Error`
|
|
pub trait DenoAsyncWrite {
|
|
fn poll_write(
|
|
&mut self,
|
|
cx: &mut Context,
|
|
buf: &[u8],
|
|
) -> Poll<Result<usize, OpError>>;
|
|
|
|
fn poll_close(&mut self, cx: &mut Context) -> Poll<Result<(), OpError>>;
|
|
|
|
fn poll_flush(&mut self, cx: &mut Context) -> Poll<Result<(), OpError>>;
|
|
}
|
|
|
|
impl DenoAsyncWrite for StreamResource {
|
|
fn poll_write(
|
|
&mut self,
|
|
cx: &mut Context,
|
|
buf: &[u8],
|
|
) -> Poll<Result<usize, OpError>> {
|
|
use StreamResource::*;
|
|
let mut f: Pin<Box<dyn AsyncWrite>> = match self {
|
|
FsFile(f, _) => Box::pin(f),
|
|
Stdout(f) => Box::pin(f),
|
|
Stderr(f) => Box::pin(f),
|
|
TcpStream(f) => Box::pin(f),
|
|
ClientTlsStream(f) => Box::pin(f),
|
|
ServerTlsStream(f) => Box::pin(f),
|
|
ChildStdin(f) => Box::pin(f),
|
|
_ => return Err(OpError::bad_resource()).into(),
|
|
};
|
|
|
|
let v = ready!(f.as_mut().poll_write(cx, buf))?;
|
|
Ok(v).into()
|
|
}
|
|
|
|
fn poll_flush(&mut self, cx: &mut Context) -> Poll<Result<(), OpError>> {
|
|
use StreamResource::*;
|
|
let mut f: Pin<Box<dyn AsyncWrite>> = match self {
|
|
FsFile(f, _) => Box::pin(f),
|
|
Stdout(f) => Box::pin(f),
|
|
Stderr(f) => Box::pin(f),
|
|
TcpStream(f) => Box::pin(f),
|
|
ClientTlsStream(f) => Box::pin(f),
|
|
ServerTlsStream(f) => Box::pin(f),
|
|
ChildStdin(f) => Box::pin(f),
|
|
_ => return Err(OpError::bad_resource()).into(),
|
|
};
|
|
|
|
ready!(f.as_mut().poll_flush(cx))?;
|
|
Ok(()).into()
|
|
}
|
|
|
|
fn poll_close(&mut self, _cx: &mut Context) -> Poll<Result<(), OpError>> {
|
|
unimplemented!()
|
|
}
|
|
}
|
|
|
|
/// A future used to write some data to a stream.
|
|
pub struct Write<T> {
|
|
rid: ResourceId,
|
|
buf: T,
|
|
io_state: IoState,
|
|
state: State,
|
|
nwritten: i32,
|
|
}
|
|
|
|
/// This is almost the same implementation as in tokio, difference is
|
|
/// that error type is `OpError` instead of `std::io::Error`.
|
|
impl<T> Future for Write<T>
|
|
where
|
|
T: AsRef<[u8]> + Unpin,
|
|
{
|
|
type Output = Result<i32, OpError>;
|
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context) -> Poll<Self::Output> {
|
|
let inner = self.get_mut();
|
|
if inner.io_state == IoState::Done {
|
|
panic!("poll a Read after it's done");
|
|
}
|
|
|
|
if inner.io_state == IoState::Pending {
|
|
let mut state = inner.state.borrow_mut();
|
|
let resource = state
|
|
.resource_table
|
|
.get_mut::<StreamResource>(inner.rid)
|
|
.ok_or_else(OpError::bad_resource)?;
|
|
|
|
let nwritten = ready!(resource.poll_write(cx, inner.buf.as_ref()))?;
|
|
inner.io_state = IoState::Flush;
|
|
inner.nwritten = nwritten as i32;
|
|
}
|
|
|
|
// TODO(bartlomieju): this step was added during upgrade to Tokio 0.2
|
|
// and the reasons for the need to explicitly flush are not fully known.
|
|
// Figure out why it's needed and preferably remove it.
|
|
// https://github.com/denoland/deno/issues/3565
|
|
if inner.io_state == IoState::Flush {
|
|
let mut state = inner.state.borrow_mut();
|
|
let resource = state
|
|
.resource_table
|
|
.get_mut::<StreamResource>(inner.rid)
|
|
.ok_or_else(OpError::bad_resource)?;
|
|
ready!(resource.poll_flush(cx))?;
|
|
inner.io_state = IoState::Done;
|
|
}
|
|
|
|
Poll::Ready(Ok(inner.nwritten))
|
|
}
|
|
}
|
|
|
|
pub fn op_write(
|
|
state: &State,
|
|
rid: i32,
|
|
zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Pin<Box<MinimalOp>> {
|
|
debug!("write rid={}", rid);
|
|
if zero_copy.is_none() {
|
|
return futures::future::err(no_buffer_specified()).boxed_local();
|
|
}
|
|
// TODO(ry) Probably poll_fn can be used here and the Write struct eliminated.
|
|
Write {
|
|
rid: rid as u32,
|
|
buf: zero_copy.unwrap(),
|
|
io_state: IoState::Pending,
|
|
state: state.clone(),
|
|
nwritten: 0,
|
|
}
|
|
.boxed_local()
|
|
}
|