0
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-30 11:15:13 -05:00
denoland-deno/src/main.rs

208 lines
5.3 KiB
Rust
Raw Normal View History

extern crate flatbuffers;
2018-07-26 17:54:22 -04:00
extern crate libc;
extern crate msg_rs as msg_generated;
2018-07-24 01:45:23 -04:00
extern crate sha1;
extern crate tempfile;
extern crate url;
2018-07-26 17:54:22 -04:00
#[macro_use]
extern crate log;
mod binding;
mod deno_dir;
mod fs;
pub mod handlers;
2018-07-04 14:50:28 -04:00
2018-06-16 01:43:23 +02:00
use libc::c_int;
use libc::c_void;
2018-07-25 18:27:27 -07:00
use std::env;
2018-06-16 01:43:23 +02:00
use std::ffi::CStr;
use std::ffi::CString;
2018-07-25 18:27:27 -07:00
use std::mem;
2018-06-22 23:19:43 +02:00
2018-07-25 18:27:27 -07:00
// Returns args passed to V8, followed by args passed to JS
fn parse_core_args(args: Vec<String>) -> (Vec<String>, Vec<String>) {
let mut rest = vec![];
// Filter out args that shouldn't be passed to V8
let mut args: Vec<String> = args
.into_iter()
.filter(|arg| {
if arg.as_str() == "--help" {
rest.push(arg.clone());
return false;
}
true
})
.collect();
// Replace args being sent to V8
for idx in 0..args.len() {
if args[idx] == "--v8-options" {
mem::swap(args.get_mut(idx).unwrap(), &mut String::from("--help"));
}
}
(args, rest)
}
// Pass the command line arguments to v8.
// Returns a vector of command line arguments that v8 did not understand.
2018-07-25 18:27:27 -07:00
fn set_flags(args: Vec<String>) -> Vec<String> {
// deno_set_flags(int* argc, char** argv) mutates argc and argv to remove
// flags that v8 understands.
2018-07-25 18:27:27 -07:00
// First parse core args, then converto to a vector of C strings.
let (argv, rest) = parse_core_args(args);
let mut argv = argv
.iter()
.map(|arg| CString::new(arg.as_str()).unwrap().into_bytes_with_nul())
.collect::<Vec<_>>();
2018-07-25 18:27:27 -07:00
// Make a new array, that can be modified by V8::SetFlagsFromCommandLine(),
// containing mutable raw pointers to the individual command line args.
let mut c_argv = argv
.iter_mut()
.map(|arg| arg.as_mut_ptr() as *mut i8)
.collect::<Vec<_>>();
// Store the length of the argv array in a local variable. We'll pass a
// pointer to this local variable to deno_set_flags(), which then
// updates its value.
2018-07-25 18:27:27 -07:00
let mut c_argc = c_argv.len() as c_int;
// Let v8 parse the arguments it recognizes and remove them from c_argv.
unsafe {
binding::deno_set_flags(&mut c_argc, c_argv.as_mut_ptr());
};
// If c_argc was updated we have to change the length of c_argv to match.
c_argv.truncate(c_argc as usize);
// Copy the modified arguments list into a proper rust vec and return it.
c_argv
.iter()
.map(|ptr| unsafe {
let cstr = CStr::from_ptr(*ptr as *const i8);
let slice = cstr.to_str().unwrap();
slice.to_string()
})
2018-07-25 18:27:27 -07:00
.chain(rest.into_iter())
.collect()
2018-06-16 01:43:23 +02:00
}
2018-06-29 15:07:19 -07:00
type DenoException<'a> = &'a str;
pub struct Deno {
ptr: *const binding::DenoC,
2018-07-26 17:54:22 -04:00
dir: deno_dir::DenoDir,
2018-06-22 23:19:43 +02:00
}
static DENO_INIT: std::sync::Once = std::sync::ONCE_INIT;
2018-06-22 23:19:43 +02:00
impl Deno {
fn new<'a>() -> &'a mut Deno {
DENO_INIT.call_once(|| {
unsafe { binding::deno_init() };
});
let deno_box = Box::new(Deno {
ptr: 0 as *const binding::DenoC,
2018-07-26 17:54:22 -04:00
dir: deno_dir::DenoDir::new(None).unwrap(),
});
let deno: &'a mut Deno = Box::leak(deno_box);
let external_ptr = deno as *mut _ as *const c_void;
let internal_deno_ptr = unsafe {
binding::deno_new(external_ptr, binding::deno_handle_msg_from_js)
};
deno.ptr = internal_deno_ptr;
deno
}
2018-06-22 23:19:43 +02:00
fn execute(
&mut self,
js_filename: &str,
js_source: &str,
) -> Result<(), DenoException> {
let filename = CString::new(js_filename).unwrap();
let source = CString::new(js_source).unwrap();
let r = unsafe {
binding::deno_execute(self.ptr, filename.as_ptr(), source.as_ptr())
};
if r == 0 {
let ptr = unsafe { binding::deno_last_exception(self.ptr) };
let cstr = unsafe { CStr::from_ptr(ptr) };
return Err(cstr.to_str().unwrap());
2018-06-22 23:19:43 +02:00
}
Ok(())
}
2018-06-29 15:07:19 -07:00
}
2018-06-22 23:19:43 +02:00
2018-06-29 15:07:19 -07:00
impl Drop for Deno {
fn drop(&mut self) {
unsafe { binding::deno_delete(self.ptr) }
}
2018-06-22 23:19:43 +02:00
}
2018-07-25 18:27:27 -07:00
#[test]
fn test_parse_core_args_1() {
let js_args =
parse_core_args(vec!["deno".to_string(), "--v8-options".to_string()]);
assert!(js_args == (vec!["deno".to_string(), "--help".to_string()], vec![]));
}
#[test]
fn test_parse_core_args_2() {
let js_args = parse_core_args(vec!["deno".to_string(), "--help".to_string()]);
assert!(js_args == (vec!["deno".to_string()], vec!["--help".to_string()]));
}
pub fn from_c<'a>(d: *const binding::DenoC) -> &'a mut Deno {
let ptr = unsafe { binding::deno_get_data(d) };
let deno_ptr = ptr as *mut Deno;
let deno_box = unsafe { Box::from_raw(deno_ptr) };
Box::leak(deno_box)
}
#[test]
fn test_c_to_rust() {
let d = Deno::new();
let d2 = from_c(d.ptr);
assert!(d.ptr == d2.ptr);
2018-07-26 17:54:22 -04:00
assert!(d.dir.root.join("gen") == d.dir.gen, "Sanity check");
}
static LOGGER: Logger = Logger;
struct Logger;
impl log::Log for Logger {
fn enabled(&self, metadata: &log::Metadata) -> bool {
metadata.level() <= log::Level::Info
}
fn log(&self, record: &log::Record) {
if self.enabled(record.metadata()) {
println!("{} - {}", record.level(), record.args());
}
}
fn flush(&self) {}
}
2018-06-16 01:43:23 +02:00
fn main() {
log::set_logger(&LOGGER).unwrap();
log::set_max_level(log::LevelFilter::Info);
2018-07-16 23:25:50 -04:00
2018-07-25 18:27:27 -07:00
let _js_args = set_flags(env::args().collect());
2018-06-22 23:19:43 +02:00
/*
2018-06-16 01:43:23 +02:00
let v = unsafe { deno_v8_version() };
let c_str = unsafe { CStr::from_ptr(v) };
let version = c_str.to_str().unwrap();
println!("version: {}", version);
2018-06-22 23:19:43 +02:00
*/
let d = Deno::new();
2018-06-22 23:19:43 +02:00
d.execute("deno_main.js", "denoMain();")
.unwrap_or_else(|err| {
error!("{}", err);
std::process::exit(1);
});
2018-06-16 01:43:23 +02:00
}