mirror of
https://github.com/denoland/deno.git
synced 2025-01-22 15:10:44 -05:00
4e1abb4f3a
To better reflect changes in error types in JS from #3662 this PR changes default error type used in ops from "ErrBox" to "OpError". "OpError" is a type that can be sent over to JSON; it has all information needed to construct error in JavaScript. That made "GetErrorKind" trait useless and so it was removed altogether. To provide compatibility with previous use of "ErrBox" an implementation of "From<ErrBox> for OpError" was added, however, it is an escape hatch and ops implementors should strive to use "OpError" directly.
217 lines
5.1 KiB
Rust
217 lines
5.1 KiB
Rust
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
|
use super::dispatch_json::{Deserialize, JsonOp, Value};
|
|
use super::io::StreamResource;
|
|
use crate::fs as deno_fs;
|
|
use crate::op_error::OpError;
|
|
use crate::ops::json_op;
|
|
use crate::state::State;
|
|
use deno_core::*;
|
|
use futures::future::FutureExt;
|
|
use std;
|
|
use std::convert::From;
|
|
use std::io::SeekFrom;
|
|
use std::path::Path;
|
|
use tokio;
|
|
|
|
pub fn init(i: &mut Isolate, s: &State) {
|
|
i.register_op("open", s.core_op(json_op(s.stateful_op(op_open))));
|
|
i.register_op("close", s.core_op(json_op(s.stateful_op(op_close))));
|
|
i.register_op("seek", s.core_op(json_op(s.stateful_op(op_seek))));
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
struct OpenArgs {
|
|
promise_id: Option<u64>,
|
|
filename: String,
|
|
options: Option<OpenOptions>,
|
|
mode: Option<String>,
|
|
}
|
|
|
|
#[derive(Deserialize, Default, Debug)]
|
|
#[serde(rename_all = "camelCase")]
|
|
#[serde(default)]
|
|
struct OpenOptions {
|
|
read: bool,
|
|
write: bool,
|
|
create: bool,
|
|
truncate: bool,
|
|
append: bool,
|
|
create_new: bool,
|
|
}
|
|
|
|
fn op_open(
|
|
state: &State,
|
|
args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
let args: OpenArgs = serde_json::from_value(args)?;
|
|
let filename = deno_fs::resolve_from_cwd(Path::new(&args.filename))?;
|
|
let state_ = state.clone();
|
|
let mut open_options = tokio::fs::OpenOptions::new();
|
|
|
|
if let Some(options) = args.options {
|
|
if options.read {
|
|
state.check_read(&filename)?;
|
|
}
|
|
|
|
if options.write || options.append {
|
|
state.check_write(&filename)?;
|
|
}
|
|
|
|
open_options
|
|
.read(options.read)
|
|
.create(options.create)
|
|
.write(options.write)
|
|
.truncate(options.truncate)
|
|
.append(options.append)
|
|
.create_new(options.create_new);
|
|
} else if let Some(mode) = args.mode {
|
|
let mode = mode.as_ref();
|
|
match mode {
|
|
"r" => {
|
|
state.check_read(&filename)?;
|
|
}
|
|
"w" | "a" | "x" => {
|
|
state.check_write(&filename)?;
|
|
}
|
|
&_ => {
|
|
state.check_read(&filename)?;
|
|
state.check_write(&filename)?;
|
|
}
|
|
};
|
|
|
|
match mode {
|
|
"r" => {
|
|
open_options.read(true);
|
|
}
|
|
"r+" => {
|
|
open_options.read(true).write(true);
|
|
}
|
|
"w" => {
|
|
open_options.create(true).write(true).truncate(true);
|
|
}
|
|
"w+" => {
|
|
open_options
|
|
.read(true)
|
|
.create(true)
|
|
.write(true)
|
|
.truncate(true);
|
|
}
|
|
"a" => {
|
|
open_options.create(true).append(true);
|
|
}
|
|
"a+" => {
|
|
open_options.read(true).create(true).append(true);
|
|
}
|
|
"x" => {
|
|
open_options.create_new(true).write(true);
|
|
}
|
|
"x+" => {
|
|
open_options.create_new(true).read(true).write(true);
|
|
}
|
|
&_ => {
|
|
// TODO: this should be type error
|
|
return Err(OpError::other("Unknown open mode.".to_string()));
|
|
}
|
|
}
|
|
} else {
|
|
return Err(OpError::other(
|
|
"Open requires either mode or options.".to_string(),
|
|
));
|
|
};
|
|
|
|
let is_sync = args.promise_id.is_none();
|
|
|
|
let fut = async move {
|
|
let fs_file = open_options.open(filename).await?;
|
|
let mut state = state_.borrow_mut();
|
|
let rid = state
|
|
.resource_table
|
|
.add("fsFile", Box::new(StreamResource::FsFile(fs_file)));
|
|
Ok(json!(rid))
|
|
};
|
|
|
|
if is_sync {
|
|
let buf = futures::executor::block_on(fut)?;
|
|
Ok(JsonOp::Sync(buf))
|
|
} else {
|
|
Ok(JsonOp::Async(fut.boxed_local()))
|
|
}
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct CloseArgs {
|
|
rid: i32,
|
|
}
|
|
|
|
fn op_close(
|
|
state: &State,
|
|
args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
let args: CloseArgs = serde_json::from_value(args)?;
|
|
|
|
let mut state = state.borrow_mut();
|
|
state
|
|
.resource_table
|
|
.close(args.rid as u32)
|
|
.ok_or_else(OpError::bad_resource)?;
|
|
Ok(JsonOp::Sync(json!({})))
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
struct SeekArgs {
|
|
promise_id: Option<u64>,
|
|
rid: i32,
|
|
offset: i32,
|
|
whence: i32,
|
|
}
|
|
|
|
fn op_seek(
|
|
state: &State,
|
|
args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
let args: SeekArgs = serde_json::from_value(args)?;
|
|
let rid = args.rid as u32;
|
|
let offset = args.offset;
|
|
let whence = args.whence as u32;
|
|
// Translate seek mode to Rust repr.
|
|
let seek_from = match whence {
|
|
0 => SeekFrom::Start(offset as u64),
|
|
1 => SeekFrom::Current(i64::from(offset)),
|
|
2 => SeekFrom::End(i64::from(offset)),
|
|
_ => {
|
|
return Err(OpError::type_error(format!(
|
|
"Invalid seek mode: {}",
|
|
whence
|
|
)));
|
|
}
|
|
};
|
|
|
|
let state = state.borrow();
|
|
let resource = state
|
|
.resource_table
|
|
.get::<StreamResource>(rid)
|
|
.ok_or_else(OpError::bad_resource)?;
|
|
|
|
let tokio_file = match resource {
|
|
StreamResource::FsFile(ref file) => file,
|
|
_ => return Err(OpError::bad_resource()),
|
|
};
|
|
let mut file = futures::executor::block_on(tokio_file.try_clone())?;
|
|
|
|
let fut = async move {
|
|
file.seek(seek_from).await?;
|
|
Ok(json!({}))
|
|
};
|
|
|
|
if args.promise_id.is_none() {
|
|
let buf = futures::executor::block_on(fut)?;
|
|
Ok(JsonOp::Sync(buf))
|
|
} else {
|
|
Ok(JsonOp::Async(fut.boxed_local()))
|
|
}
|
|
}
|