mirror of
https://github.com/denoland/deno.git
synced 2025-01-22 23:19:55 -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.
190 lines
4.8 KiB
Rust
190 lines
4.8 KiB
Rust
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
|
use super::dispatch_json::{Deserialize, JsonOp, Value};
|
|
use crate::op_error::OpError;
|
|
use crate::ops::json_op;
|
|
use crate::state::State;
|
|
use atty;
|
|
use deno_core::*;
|
|
use std::collections::HashMap;
|
|
use std::env;
|
|
use std::io::{Error, ErrorKind};
|
|
use sys_info;
|
|
use url::Url;
|
|
|
|
pub fn init(i: &mut Isolate, s: &State) {
|
|
i.register_op("exit", s.core_op(json_op(s.stateful_op(op_exit))));
|
|
i.register_op("is_tty", s.core_op(json_op(s.stateful_op(op_is_tty))));
|
|
i.register_op("env", s.core_op(json_op(s.stateful_op(op_env))));
|
|
i.register_op("exec_path", s.core_op(json_op(s.stateful_op(op_exec_path))));
|
|
i.register_op("set_env", s.core_op(json_op(s.stateful_op(op_set_env))));
|
|
i.register_op("get_env", s.core_op(json_op(s.stateful_op(op_get_env))));
|
|
i.register_op("get_dir", s.core_op(json_op(s.stateful_op(op_get_dir))));
|
|
i.register_op("hostname", s.core_op(json_op(s.stateful_op(op_hostname))));
|
|
i.register_op("loadavg", s.core_op(json_op(s.stateful_op(op_loadavg))));
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct GetDirArgs {
|
|
kind: std::string::String,
|
|
}
|
|
|
|
fn op_get_dir(
|
|
state: &State,
|
|
args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
state.check_env()?;
|
|
let args: GetDirArgs = serde_json::from_value(args)?;
|
|
|
|
let path = match args.kind.as_str() {
|
|
"home" => dirs::home_dir(),
|
|
"config" => dirs::config_dir(),
|
|
"cache" => dirs::cache_dir(),
|
|
"executable" => dirs::executable_dir(),
|
|
"data" => dirs::data_dir(),
|
|
"data_local" => dirs::data_local_dir(),
|
|
"audio" => dirs::audio_dir(),
|
|
"desktop" => dirs::desktop_dir(),
|
|
"document" => dirs::document_dir(),
|
|
"download" => dirs::download_dir(),
|
|
"font" => dirs::font_dir(),
|
|
"picture" => dirs::picture_dir(),
|
|
"public" => dirs::public_dir(),
|
|
"template" => dirs::template_dir(),
|
|
"video" => dirs::video_dir(),
|
|
_ => {
|
|
return Err(
|
|
Error::new(
|
|
ErrorKind::InvalidInput,
|
|
format!("Invalid dir type `{}`", args.kind.as_str()),
|
|
)
|
|
.into(),
|
|
)
|
|
}
|
|
};
|
|
|
|
if path == None {
|
|
Err(OpError::not_found(format!(
|
|
"Could not get user {} directory.",
|
|
args.kind.as_str()
|
|
)))
|
|
} else {
|
|
Ok(JsonOp::Sync(json!(path
|
|
.unwrap_or_default()
|
|
.into_os_string()
|
|
.into_string()
|
|
.unwrap_or_default())))
|
|
}
|
|
}
|
|
|
|
fn op_exec_path(
|
|
state: &State,
|
|
_args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
state.check_env()?;
|
|
let current_exe = env::current_exe().unwrap();
|
|
// Now apply URL parser to current exe to get fully resolved path, otherwise
|
|
// we might get `./` and `../` bits in `exec_path`
|
|
let exe_url = Url::from_file_path(current_exe).unwrap();
|
|
let path = exe_url.to_file_path().unwrap();
|
|
Ok(JsonOp::Sync(json!(path)))
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct SetEnv {
|
|
key: String,
|
|
value: String,
|
|
}
|
|
|
|
fn op_set_env(
|
|
state: &State,
|
|
args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
let args: SetEnv = serde_json::from_value(args)?;
|
|
state.check_env()?;
|
|
env::set_var(args.key, args.value);
|
|
Ok(JsonOp::Sync(json!({})))
|
|
}
|
|
|
|
fn op_env(
|
|
state: &State,
|
|
_args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
state.check_env()?;
|
|
let v = env::vars().collect::<HashMap<String, String>>();
|
|
Ok(JsonOp::Sync(json!(v)))
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct GetEnv {
|
|
key: String,
|
|
}
|
|
|
|
fn op_get_env(
|
|
state: &State,
|
|
args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
let args: GetEnv = serde_json::from_value(args)?;
|
|
state.check_env()?;
|
|
let r = match env::var(args.key) {
|
|
Err(env::VarError::NotPresent) => json!([]),
|
|
v => json!([v?]),
|
|
};
|
|
Ok(JsonOp::Sync(r))
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct Exit {
|
|
code: i32,
|
|
}
|
|
|
|
fn op_exit(
|
|
_s: &State,
|
|
args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
let args: Exit = serde_json::from_value(args)?;
|
|
std::process::exit(args.code)
|
|
}
|
|
|
|
fn op_is_tty(
|
|
_s: &State,
|
|
_args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
Ok(JsonOp::Sync(json!({
|
|
"stdin": atty::is(atty::Stream::Stdin),
|
|
"stdout": atty::is(atty::Stream::Stdout),
|
|
"stderr": atty::is(atty::Stream::Stderr),
|
|
})))
|
|
}
|
|
|
|
fn op_loadavg(
|
|
state: &State,
|
|
_args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
state.check_env()?;
|
|
match sys_info::loadavg() {
|
|
Ok(loadavg) => Ok(JsonOp::Sync(json!([
|
|
loadavg.one,
|
|
loadavg.five,
|
|
loadavg.fifteen
|
|
]))),
|
|
Err(_) => Ok(JsonOp::Sync(json!([0f64, 0f64, 0f64]))),
|
|
}
|
|
}
|
|
|
|
fn op_hostname(
|
|
state: &State,
|
|
_args: Value,
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
) -> Result<JsonOp, OpError> {
|
|
state.check_env()?;
|
|
let hostname = sys_info::hostname().unwrap_or_else(|_| "".to_owned());
|
|
Ok(JsonOp::Sync(json!(hostname)))
|
|
}
|