2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2019-08-23 01:30:14 -04:00
|
|
|
use super::dispatch_json::{Deserialize, JsonOp, Value};
|
2019-10-11 14:41:54 -04:00
|
|
|
use crate::ops::json_op;
|
2019-08-14 11:03:02 -04:00
|
|
|
use crate::state::ThreadSafeState;
|
|
|
|
use atty;
|
2020-01-05 11:56:18 -05:00
|
|
|
use deno_core::*;
|
2019-08-23 01:30:14 -04:00
|
|
|
use std::collections::HashMap;
|
|
|
|
use std::env;
|
2019-12-15 00:14:20 -05:00
|
|
|
use std::io::{Error, ErrorKind};
|
2019-09-27 19:09:42 -04:00
|
|
|
use sys_info;
|
2019-08-14 11:03:02 -04:00
|
|
|
use url::Url;
|
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
pub fn init(i: &mut Isolate, s: &ThreadSafeState) {
|
|
|
|
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))));
|
2019-12-15 00:14:20 -05:00
|
|
|
i.register_op("get_dir", s.core_op(json_op(s.stateful_op(op_get_dir))));
|
2019-10-11 14:41:54 -04:00
|
|
|
i.register_op("hostname", s.core_op(json_op(s.stateful_op(op_hostname))));
|
2019-08-14 11:03:02 -04:00
|
|
|
}
|
|
|
|
|
2019-12-15 00:14:20 -05:00
|
|
|
#[derive(Deserialize)]
|
|
|
|
struct GetDirArgs {
|
2019-12-18 09:29:00 -05:00
|
|
|
kind: std::string::String,
|
2019-12-15 00:14:20 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn op_get_dir(
|
2019-08-14 11:03:02 -04:00
|
|
|
state: &ThreadSafeState,
|
2019-12-15 00:14:20 -05:00
|
|
|
args: Value,
|
2020-01-24 15:10:49 -05:00
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
2019-08-26 08:50:21 -04:00
|
|
|
) -> Result<JsonOp, ErrBox> {
|
2019-08-14 11:03:02 -04:00
|
|
|
state.check_env()?;
|
2019-12-15 00:14:20 -05:00
|
|
|
let args: GetDirArgs = serde_json::from_value(args)?;
|
|
|
|
|
2019-12-18 09:29:00 -05:00
|
|
|
let path = match args.kind.as_str() {
|
2019-12-15 00:14:20 -05:00
|
|
|
"home" => dirs::home_dir(),
|
|
|
|
"config" => dirs::config_dir(),
|
|
|
|
"cache" => dirs::cache_dir(),
|
2019-12-21 06:30:13 -05:00
|
|
|
"executable" => dirs::executable_dir(),
|
2019-12-15 00:14:20 -05:00
|
|
|
"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(ErrBox::from(Error::new(
|
|
|
|
ErrorKind::InvalidInput,
|
2019-12-18 09:29:00 -05:00
|
|
|
format!("Invalid dir type `{}`", args.kind.as_str()),
|
2019-12-15 00:14:20 -05:00
|
|
|
)))
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if path == None {
|
|
|
|
Err(ErrBox::from(Error::new(
|
|
|
|
ErrorKind::NotFound,
|
2019-12-18 09:29:00 -05:00
|
|
|
format!("Could not get user {} directory.", args.kind.as_str()),
|
2019-12-15 00:14:20 -05:00
|
|
|
)))
|
|
|
|
} else {
|
|
|
|
Ok(JsonOp::Sync(json!(path
|
|
|
|
.unwrap_or_default()
|
|
|
|
.into_os_string()
|
|
|
|
.into_string()
|
|
|
|
.unwrap_or_default())))
|
|
|
|
}
|
2019-08-14 11:03:02 -04:00
|
|
|
}
|
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
fn op_exec_path(
|
2019-08-14 11:03:02 -04:00
|
|
|
state: &ThreadSafeState,
|
2019-08-23 01:30:14 -04:00
|
|
|
_args: Value,
|
2020-01-24 15:10:49 -05:00
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
2019-08-23 01:30:14 -04:00
|
|
|
) -> Result<JsonOp, ErrBox> {
|
2019-08-14 11:03:02 -04:00
|
|
|
state.check_env()?;
|
2019-08-23 01:30:14 -04:00
|
|
|
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`
|
2019-08-14 11:03:02 -04:00
|
|
|
let exe_url = Url::from_file_path(current_exe).unwrap();
|
2019-08-23 01:30:14 -04:00
|
|
|
let path = exe_url.to_file_path().unwrap();
|
|
|
|
Ok(JsonOp::Sync(json!(path)))
|
2019-08-14 11:03:02 -04:00
|
|
|
}
|
|
|
|
|
2019-08-26 08:50:21 -04:00
|
|
|
#[derive(Deserialize)]
|
|
|
|
struct SetEnv {
|
|
|
|
key: String,
|
|
|
|
value: String,
|
|
|
|
}
|
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
fn op_set_env(
|
2019-08-14 11:03:02 -04:00
|
|
|
state: &ThreadSafeState,
|
2019-08-26 08:50:21 -04:00
|
|
|
args: Value,
|
2020-01-24 15:10:49 -05:00
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
2019-08-26 08:50:21 -04:00
|
|
|
) -> Result<JsonOp, ErrBox> {
|
|
|
|
let args: SetEnv = serde_json::from_value(args)?;
|
2019-08-14 11:03:02 -04:00
|
|
|
state.check_env()?;
|
2019-08-26 08:50:21 -04:00
|
|
|
env::set_var(args.key, args.value);
|
|
|
|
Ok(JsonOp::Sync(json!({})))
|
2019-08-14 11:03:02 -04:00
|
|
|
}
|
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
fn op_env(
|
2019-08-14 11:03:02 -04:00
|
|
|
state: &ThreadSafeState,
|
2019-08-23 01:30:14 -04:00
|
|
|
_args: Value,
|
2020-01-24 15:10:49 -05:00
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
2019-08-23 01:30:14 -04:00
|
|
|
) -> Result<JsonOp, ErrBox> {
|
2019-08-14 11:03:02 -04:00
|
|
|
state.check_env()?;
|
2019-08-23 01:30:14 -04:00
|
|
|
let v = env::vars().collect::<HashMap<String, String>>();
|
|
|
|
Ok(JsonOp::Sync(json!(v)))
|
|
|
|
}
|
2019-08-14 11:03:02 -04:00
|
|
|
|
2019-10-02 11:55:28 -04:00
|
|
|
#[derive(Deserialize)]
|
|
|
|
struct GetEnv {
|
|
|
|
key: String,
|
|
|
|
}
|
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
fn op_get_env(
|
2019-10-02 11:55:28 -04:00
|
|
|
state: &ThreadSafeState,
|
|
|
|
args: Value,
|
2020-01-24 15:10:49 -05:00
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
2019-10-02 11:55:28 -04:00
|
|
|
) -> Result<JsonOp, ErrBox> {
|
|
|
|
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))
|
|
|
|
}
|
|
|
|
|
2019-08-23 01:30:14 -04:00
|
|
|
#[derive(Deserialize)]
|
|
|
|
struct Exit {
|
|
|
|
code: i32,
|
2019-08-14 11:03:02 -04:00
|
|
|
}
|
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
fn op_exit(
|
2019-08-23 01:30:14 -04:00
|
|
|
_s: &ThreadSafeState,
|
|
|
|
args: Value,
|
2020-01-24 15:10:49 -05:00
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
2019-08-23 01:30:14 -04:00
|
|
|
) -> Result<JsonOp, ErrBox> {
|
|
|
|
let args: Exit = serde_json::from_value(args)?;
|
|
|
|
std::process::exit(args.code)
|
2019-08-14 11:03:02 -04:00
|
|
|
}
|
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
fn op_is_tty(
|
2019-08-23 01:30:14 -04:00
|
|
|
_s: &ThreadSafeState,
|
|
|
|
_args: Value,
|
2020-01-24 15:10:49 -05:00
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
2019-08-23 01:30:14 -04:00
|
|
|
) -> Result<JsonOp, ErrBox> {
|
|
|
|
Ok(JsonOp::Sync(json!({
|
|
|
|
"stdin": atty::is(atty::Stream::Stdin),
|
|
|
|
"stdout": atty::is(atty::Stream::Stdout),
|
|
|
|
"stderr": atty::is(atty::Stream::Stderr),
|
|
|
|
})))
|
2019-08-14 11:03:02 -04:00
|
|
|
}
|
2019-09-27 19:09:42 -04:00
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
fn op_hostname(
|
2019-09-27 19:09:42 -04:00
|
|
|
state: &ThreadSafeState,
|
|
|
|
_args: Value,
|
2020-01-24 15:10:49 -05:00
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
2019-09-27 19:09:42 -04:00
|
|
|
) -> Result<JsonOp, ErrBox> {
|
|
|
|
state.check_env()?;
|
|
|
|
let hostname = sys_info::hostname().unwrap_or_else(|_| "".to_owned());
|
|
|
|
Ok(JsonOp::Sync(json!(hostname)))
|
|
|
|
}
|