diff --git a/cli/compiler.rs b/cli/compiler.rs index 872ed28a23c306..c827c3ca1d8b57 100644 --- a/cli/compiler.rs +++ b/cli/compiler.rs @@ -9,6 +9,7 @@ use crate::state::*; use crate::tokio_util; use crate::worker::Worker; use deno::Buf; +use deno::ModuleSpecifier; use futures::Future; use futures::Stream; use std::path::PathBuf; @@ -17,6 +18,7 @@ use std::sync::atomic::Ordering; // This corresponds to JS ModuleMetaData. // TODO Rename one or the other so they correspond. +// TODO(bartlomieju): change `*_name` to `*_url` and use Url type #[derive(Debug, Clone)] pub struct ModuleMetaData { pub module_name: String, @@ -203,6 +205,8 @@ pub fn compile_async( .and_then(move |maybe_msg: Option| { debug!("Received message from worker"); + // TODO: here TS compiler emitted the files to disc and we should signal ModuleMetaData + // cache that source code is available if let Some(msg) = maybe_msg { let json_str = std::str::from_utf8(&msg).unwrap(); debug!("Message: {}", json_str); @@ -213,8 +217,10 @@ pub fn compile_async( Ok(()) }).and_then(move |_| { + let module_specifier = ModuleSpecifier::resolve_url(&module_name) + .expect("Should be valid module specifier"); state.dir.fetch_module_meta_data_async( - &module_name, + &module_specifier, true, true, ).map_err(|e| { @@ -249,7 +255,7 @@ mod tests { tokio_util::init(|| { let specifier = "./tests/002_hello.ts"; use deno::ModuleSpecifier; - let module_name = ModuleSpecifier::resolve_root(specifier) + let module_name = ModuleSpecifier::resolve_url_or_path(specifier) .unwrap() .to_string(); @@ -296,7 +302,7 @@ mod tests { fn test_bundle_async() { let specifier = "./tests/002_hello.ts"; use deno::ModuleSpecifier; - let module_name = ModuleSpecifier::resolve_root(specifier) + let module_name = ModuleSpecifier::resolve_url_or_path(specifier) .unwrap() .to_string(); diff --git a/cli/deno_dir.rs b/cli/deno_dir.rs index 324aa0e3295829..657e6670eb270d 100644 --- a/cli/deno_dir.rs +++ b/cli/deno_dir.rs @@ -11,6 +11,7 @@ use crate::progress::Progress; use crate::source_maps::SourceMapGetter; use crate::tokio_util; use crate::version; +use deno::ModuleSpecifier; use dirs; use futures::future::{loop_fn, Either, Loop}; use futures::Future; @@ -19,17 +20,52 @@ use ring; use serde_json; use std; use std::collections::HashSet; +use std::fmt; +use std::fmt::Display; use std::fmt::Write; use std::fs; use std::path::Path; use std::path::PathBuf; use std::result::Result; use std::str; +use std::str::FromStr; use std::sync::Arc; use std::sync::Mutex; use url; use url::Url; +// TODO: DenoDirError is temporary solution, should be upgraded during rewrite +#[derive(Debug, PartialEq)] +pub enum DenoDirErrorKind { + UnsupportedFetchScheme, +} + +#[derive(Debug)] +pub struct DenoDirError { + pub message: String, + pub kind: DenoDirErrorKind, +} + +impl DenoDirError { + pub fn new(message: String, kind: DenoDirErrorKind) -> Self { + DenoDirError { message, kind } + } +} + +impl std::error::Error for DenoDirError { + fn description(&self) -> &str { + &*self.message + } +} + +impl Display for DenoDirError { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + writeln!(f, "{}", self.message) + } +} + +const SUPPORTED_URL_SCHEMES: [&str; 3] = ["http", "https", "file"]; + fn normalize_path(path: &Path) -> PathBuf { let s = String::from(path.to_str().unwrap()); let normalized_string = if cfg!(windows) { @@ -159,23 +195,18 @@ impl DenoDir { pub fn fetch_module_meta_data_async( self: &Self, - specifier: &str, + specifier: &ModuleSpecifier, use_cache: bool, no_fetch: bool, ) -> impl Future { - debug!("fetch_module_meta_data. specifier {} ", specifier); - - // TODO: rename specifier? - let specifier = specifier.to_string(); - // TODO: url resolution should happen here - // let module_name = ... + let module_url = specifier.to_url(); + debug!("fetch_module_meta_data. specifier {} ", module_url); - // TODO: this should return only deps filepath for given module URL - let result = self.resolve_module(&specifier, "."); + let result = self.url_to_deps_path(&module_url); if let Err(err) = result { return Either::A(futures::future::err(DenoError::from(err))); } - let (module_name, filepath) = result.unwrap(); + let deps_filepath = result.unwrap(); let gen = self.gen.clone(); @@ -187,8 +218,8 @@ impl DenoDir { Either::B( get_source_code_async( self, - module_name.as_str(), - filepath, + &module_url, + deps_filepath, use_cache, no_fetch, ).then(move |result| { @@ -199,7 +230,7 @@ impl DenoDir { // For NotFound, change the message to something better. return Err(deno_error::new( ErrorKind::NotFound, - format!("Cannot resolve module \"{}\"", specifier), + format!("Cannot resolve module \"{}\"", module_url.to_string()), )); } else { return Err(err); @@ -256,7 +287,7 @@ impl DenoDir { /// This function is deprecated. pub fn fetch_module_meta_data( self: &Self, - specifier: &str, + specifier: &ModuleSpecifier, use_cache: bool, no_fetch: bool, ) -> Result { @@ -265,104 +296,77 @@ impl DenoDir { ) } - // Prototype: https://github.com/denoland/deno/blob/golang/os.go#L56-L68 - // TODO: this method should take deps filepath and return URL for module - fn src_file_to_url(self: &Self, filename: &str) -> String { - let filename_path = Path::new(filename); - if filename_path.starts_with(&self.deps) { - let (rest, prefix) = if filename_path.starts_with(&self.deps_https) { - let rest = filename_path.strip_prefix(&self.deps_https).unwrap(); - let prefix = "https://".to_string(); - (rest, prefix) - } else if filename_path.starts_with(&self.deps_http) { - let rest = filename_path.strip_prefix(&self.deps_http).unwrap(); - let prefix = "http://".to_string(); - (rest, prefix) - } else { - // TODO(kevinkassimo): change this to support other protocols than http - unimplemented!() - }; - // Windows doesn't support ":" in filenames, so we represent port using a - // special string. - // TODO(ry) This current implementation will break on a URL that has - // the default port but contains "_PORT" in the path. - let rest = rest.to_str().unwrap().replacen("_PORT", ":", 1); - prefix + &rest - } else { - String::from(filename) - } - } - - /// Returns (module name, local filename) - pub fn resolve_module_url( + /// This method returns local file path for given module url that is used + /// internally by DenoDir to reference module. + /// + /// For specifiers starting with `file://` returns the input. + /// + /// For specifier starting with `http://` and `https://` it returns + /// path to DenoDir dependency directory. + pub fn url_to_deps_path( self: &Self, - specifier: &str, - referrer: &str, - ) -> Result { - debug!( - "pre-resolve_module specifier {} referrer {}", - specifier, referrer - ); - - let specifier = self.src_file_to_url(specifier); - let referrer = self.src_file_to_url(referrer); - - debug!( - "resolve_module specifier {} referrer {}", - specifier, referrer - ); + url: &Url, + ) -> Result { + let filename = match url.scheme() { + "file" => url.to_file_path().unwrap(), + "https" => get_cache_filename(self.deps_https.as_path(), &url), + "http" => get_cache_filename(self.deps_http.as_path(), &url), + scheme => { + return Err( + DenoDirError::new( + format!("Unsupported scheme \"{}\" for module \"{}\". Supported schemes: {:#?}", scheme, url, SUPPORTED_URL_SCHEMES), + DenoDirErrorKind::UnsupportedFetchScheme + ) + ); + } + }; - resolve_file_url(specifier, referrer) + debug!("deps filename: {:?}", filename); + Ok(normalize_path(&filename)) } - // TODO(bartlomieju): this method should return only `local filepath` - // it should be called with already resolved URLs - // TODO(bartlomieju): rename to url_to_deps_path - /// Returns (module name, local filename) - pub fn resolve_module( - self: &Self, - specifier: &str, - referrer: &str, - ) -> Result<(String, PathBuf), url::ParseError> { - let j = self.resolve_module_url(specifier, referrer)?; - - let module_name = j.to_string(); - let filename = match j.scheme() { - "file" => j.to_file_path().unwrap(), - "https" => get_cache_filename(self.deps_https.as_path(), &j), - "http" => get_cache_filename(self.deps_http.as_path(), &j), - // TODO(kevinkassimo): change this to support other protocols than http. - _ => unimplemented!(), - }; + // TODO: this method is only used by `SourceMapGetter` impl - can we organize it better? + fn try_resolve_and_get_module_meta_data( + &self, + script_name: &str, + ) -> Option { + // if `script_name` can't be resolved to ModuleSpecifier it's probably internal + // script (like `gen/cli/bundle/compiler.js`) so we won't be + // able to get source for it anyway + let maybe_specifier = ModuleSpecifier::resolve_url(script_name); + + if maybe_specifier.is_err() { + return None; + } - debug!("module_name: {}, filename: {:?}", module_name, filename); - Ok((module_name, normalize_path(&filename))) + let module_specifier = maybe_specifier.unwrap(); + // TODO: this method shouldn't issue `fetch_module_meta_data` - this is done for each line + // in JS stack trace so it's pretty slow - quick idea: store `ModuleMetaData` in one + // structure available to DenoDir so it's not fetched from disk everytime it's needed + match self.fetch_module_meta_data(&module_specifier, true, true) { + Err(_) => None, + Ok(out) => Some(out), + } } } impl SourceMapGetter for DenoDir { fn get_source_map(&self, script_name: &str) -> Option> { - match self.fetch_module_meta_data(script_name, true, true) { - Err(_e) => None, - Ok(out) => match out.maybe_source_map { - None => None, - Some(source_map) => Some(source_map), - }, - } + self + .try_resolve_and_get_module_meta_data(script_name) + .and_then(|out| out.maybe_source_map) } fn get_source_line(&self, script_name: &str, line: usize) -> Option { - match self.fetch_module_meta_data(script_name, true, true) { - Ok(out) => match str::from_utf8(&out.source_code) { - Ok(v) => { + self + .try_resolve_and_get_module_meta_data(script_name) + .and_then(|out| { + str::from_utf8(&out.source_code).ok().and_then(|v| { let lines: Vec<&str> = v.lines().collect(); assert!(lines.len() > line); Some(lines[line].to_string()) - } - _ => None, - }, - _ => None, - } + }) + }) } } @@ -381,14 +385,15 @@ impl SourceMapGetter for DenoDir { /// use_cache. fn get_source_code_async( deno_dir: &DenoDir, - module_name: &str, + module_url: &Url, filepath: PathBuf, use_cache: bool, no_fetch: bool, ) -> impl Future { let filename = filepath.to_str().unwrap().to_string(); - let module_name = module_name.to_string(); - let is_module_remote = is_remote(&module_name); + let module_name = module_url.to_string(); + let url_scheme = module_url.scheme(); + let is_module_remote = url_scheme == "http" || url_scheme == "https"; // We try fetch local. Three cases: // 1. Remote downloads are not allowed, we're only allowed to use cache. // 2. This is a remote module and we're allowed to use cached downloads. @@ -403,7 +408,7 @@ fn get_source_code_async( module_name, is_module_remote ); // Note that local fetch is done synchronously. - match fetch_local_source(deno_dir, &module_name, &filepath, None) { + match fetch_local_source(deno_dir, &module_url, &filepath, None) { Ok(Some(output)) => { debug!("found local source "); return Either::A(futures::future::ok(output)); @@ -445,7 +450,7 @@ fn get_source_code_async( // not cached/local, try remote. Either::B( - fetch_remote_source_async(deno_dir, &module_name, &filepath).and_then( + fetch_remote_source_async(deno_dir, &module_url, &filepath).and_then( move |maybe_remote_source| match maybe_remote_source { Some(output) => { download_cache.mark(&module_name); @@ -465,17 +470,13 @@ fn get_source_code_async( /// This function is deprecated. fn get_source_code( deno_dir: &DenoDir, - module_name: &str, + module_url: &Url, filepath: PathBuf, use_cache: bool, no_fetch: bool, ) -> DenoResult { tokio_util::block_on(get_source_code_async( - deno_dir, - module_name, - filepath, - use_cache, - no_fetch, + deno_dir, module_url, filepath, use_cache, no_fetch, )) } @@ -532,20 +533,6 @@ fn source_code_hash( out } -// TODO: module_name should be Url -fn is_remote(module_name: &str) -> bool { - module_name.starts_with("http://") || module_name.starts_with("https://") -} - -// TODO: basically parse or resolve from_file_path -fn parse_local_or_remote(p: &str) -> Result { - if is_remote(p) || p.starts_with("file:") { - Url::parse(p) - } else { - Url::from_file_path(p).map_err(|_err| url::ParseError::IdnaError) - } -} - fn map_file_extension(path: &Path) -> msg::MediaType { match path.extension() { None => msg::MediaType::Unknown, @@ -604,7 +591,7 @@ fn filter_shebang(bytes: Vec) -> Vec { /// Save source code and related headers for given module fn save_module_code_and_headers( filepath: PathBuf, - module_name: &str, + module_url: &Url, source: &str, maybe_content_type: Option, maybe_initial_filepath: Option, @@ -630,7 +617,7 @@ fn save_module_code_and_headers( save_source_code_headers( &maybe_initial_filepath.unwrap(), maybe_content_type.clone(), - Some(module_name.to_string()), + Some(module_url.to_string()), ); } } @@ -638,18 +625,22 @@ fn save_module_code_and_headers( Ok(()) } +fn url_into_uri(url: &url::Url) -> http::uri::Uri { + http::uri::Uri::from_str(&url.to_string()) + .expect("url::Url should be parseable as http::uri::Uri") +} + /// Asynchronously fetch remote source file specified by the URL `module_name` /// and write it to disk at `filename`. fn fetch_remote_source_async( deno_dir: &DenoDir, - module_name: &str, + module_url: &Url, filepath: &Path, ) -> impl Future, Error = DenoError> { use crate::http_util::FetchOnceResult; - let download_job = deno_dir.progress.add("Download", module_name); + let download_job = deno_dir.progress.add("Download", &module_url.to_string()); - let module_name = module_name.to_owned(); let filepath = filepath.to_owned(); // We write a special ".headers.json" file into the `.deno/deps` directory along side the @@ -661,27 +652,27 @@ fn fetch_remote_source_async( deno_dir.clone(), None, None, - module_name.clone(), + module_url.clone(), filepath.clone(), ), |( dir, mut maybe_initial_module_name, mut maybe_initial_filepath, - module_name, + module_url, filepath, )| { - let url = module_name.parse::().unwrap(); + let module_uri = url_into_uri(&module_url); // Single pass fetch, either yields code or yields redirect. - http_util::fetch_string_once(url).and_then(move |fetch_once_result| { + http_util::fetch_string_once(module_uri).and_then(move |fetch_once_result| { match fetch_once_result { - FetchOnceResult::Redirect(url) => { + FetchOnceResult::Redirect(uri) => { // If redirects, update module_name and filename for next looped call. - let (new_module_name, new_filepath) = dir - .resolve_module(&url.to_string(), ".")?; + let new_module_url = Url::parse(&uri.to_string()).expect("http::uri::Uri should be parseable as Url"); + let new_filepath = dir.url_to_deps_path(&new_module_url)?; if maybe_initial_module_name.is_none() { - maybe_initial_module_name = Some(module_name.clone()); + maybe_initial_module_name = Some(module_url.to_string()); maybe_initial_filepath = Some(filepath.clone()); } @@ -690,7 +681,7 @@ fn fetch_remote_source_async( dir, maybe_initial_module_name, maybe_initial_filepath, - new_module_name, + new_module_url, new_filepath, ))) } @@ -698,7 +689,7 @@ fn fetch_remote_source_async( // We land on the code. save_module_code_and_headers( filepath.clone(), - &module_name.clone(), + &module_url, &source, maybe_content_type.clone(), maybe_initial_filepath, @@ -709,8 +700,9 @@ fn fetch_remote_source_async( maybe_content_type.as_ref().map(String::as_str), ); + // TODO: module_name should be renamed to URL let module_meta_data = ModuleMetaData { - module_name: module_name.to_string(), + module_name: module_url.to_string(), module_redirect_source_name: maybe_initial_module_name, filename: filepath.clone(), media_type, @@ -737,13 +729,11 @@ fn fetch_remote_source_async( #[cfg(test)] fn fetch_remote_source( deno_dir: &DenoDir, - module_name: &str, + module_url: &Url, filepath: &Path, ) -> DenoResult> { tokio_util::block_on(fetch_remote_source_async( - deno_dir, - module_name, - filepath, + deno_dir, module_url, filepath, )) } @@ -758,7 +748,7 @@ fn fetch_remote_source( /// after following all redirections. fn fetch_local_source( deno_dir: &DenoDir, - module_name: &str, + module_url: &Url, filepath: &Path, module_initial_source_name: Option, ) -> DenoResult> { @@ -773,20 +763,21 @@ fn fetch_local_source( // redirect_to https://import-meta.now.sh/sub/final1.js // real_filename /Users/kun/Library/Caches/deno/deps/https/import-meta.now.sh/sub/final1.js // real_module_name = https://import-meta.now.sh/sub/final1.js - let (real_module_name, real_filepath) = - deno_dir.resolve_module(&redirect_to, ".")?; + let real_module_url = + Url::parse(&redirect_to).expect("Should be valid URL"); + let real_filepath = deno_dir.url_to_deps_path(&real_module_url)?; let mut module_initial_source_name = module_initial_source_name; // If this is the first redirect attempt, // then module_initial_source_name should be None. // In that case, use current module name as module_initial_source_name. if module_initial_source_name.is_none() { - module_initial_source_name = Some(module_name.to_owned()); + module_initial_source_name = Some(module_url.to_string()); } // Recurse. return fetch_local_source( deno_dir, - &real_module_name, + &real_module_url, &real_filepath, module_initial_source_name, ); @@ -804,7 +795,7 @@ fn fetch_local_source( Ok(c) => c, }; Ok(Some(ModuleMetaData { - module_name: module_name.to_string(), + module_name: module_url.to_string(), module_redirect_source_name: module_initial_source_name, filename: filepath.to_owned(), media_type: map_content_type( @@ -944,36 +935,6 @@ pub fn resolve_from_cwd(path: &str) -> Result<(PathBuf, String), DenoError> { Ok((normalized_path, path_string)) } -pub fn resolve_file_url( - specifier: String, - mut referrer: String, -) -> Result { - if referrer.starts_with('.') { - let cwd = std::env::current_dir().unwrap(); - let referrer_path = cwd.join(referrer); - referrer = referrer_path.to_str().unwrap().to_string() + "/"; - } - - // - let j = if is_remote(&specifier) - || (Path::new(&specifier).is_absolute() && !is_remote(&referrer)) - { - parse_local_or_remote(&specifier)? - } else if referrer.ends_with('/') { - let r = Url::from_directory_path(&referrer); - // TODO(ry) Properly handle error. - if r.is_err() { - error!("Url::from_directory_path error {}", referrer); - } - let base = r.unwrap(); - base.join(specifier.as_ref())? - } else { - let base = parse_local_or_remote(&referrer)?; - base.join(specifier.as_ref())? - }; - Ok(j) -} - #[cfg(test)] mod tests { use super::*; @@ -1121,14 +1082,15 @@ mod tests { let (temp_dir, deno_dir) = test_setup(); // http_util::fetch_sync_string requires tokio tokio_util::init(|| { - let module_name = "http://localhost:4545/tests/subdir/mod2.ts"; + let module_url = + Url::parse("http://localhost:4545/tests/subdir/mod2.ts").unwrap(); let filepath = deno_dir .deps_http .join("localhost_PORT4545/tests/subdir/mod2.ts"); let headers_file_name = source_code_headers_filename(&filepath); let result = - get_source_code(&deno_dir, module_name, filepath.clone(), true, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), true, false); assert!(result.is_ok()); let r = result.unwrap(); assert_eq!( @@ -1143,7 +1105,7 @@ mod tests { let _ = fs::write(&headers_file_name, "{ \"mime_type\": \"text/javascript\" }"); let result2 = - get_source_code(&deno_dir, module_name, filepath.clone(), true, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), true, false); assert!(result2.is_ok()); let r2 = result2.unwrap(); assert_eq!( @@ -1165,7 +1127,7 @@ mod tests { None, ); let result3 = - get_source_code(&deno_dir, module_name, filepath.clone(), true, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), true, false); assert!(result3.is_ok()); let r3 = result3.unwrap(); assert_eq!( @@ -1185,7 +1147,7 @@ mod tests { // and don't use cache let deno_dir = setup_deno_dir(temp_dir.path()); let result4 = - get_source_code(&deno_dir, module_name, filepath.clone(), false, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), false, false); assert!(result4.is_ok()); let r4 = result4.unwrap(); let expected4 = @@ -1202,14 +1164,16 @@ mod tests { let (temp_dir, deno_dir) = test_setup(); // http_util::fetch_sync_string requires tokio tokio_util::init(|| { - let module_name = "http://localhost:4545/tests/subdir/mismatch_ext.ts"; + let module_url = + Url::parse("http://localhost:4545/tests/subdir/mismatch_ext.ts") + .unwrap(); let filepath = deno_dir .deps_http .join("localhost_PORT4545/tests/subdir/mismatch_ext.ts"); let headers_file_name = source_code_headers_filename(&filepath); let result = - get_source_code(&deno_dir, module_name, filepath.clone(), true, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), true, false); assert!(result.is_ok()); let r = result.unwrap(); let expected = "export const loaded = true;\n".as_bytes(); @@ -1228,7 +1192,7 @@ mod tests { None, ); let result2 = - get_source_code(&deno_dir, module_name, filepath.clone(), true, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), true, false); assert!(result2.is_ok()); let r2 = result2.unwrap(); let expected2 = "export const loaded = true;\n".as_bytes(); @@ -1242,7 +1206,7 @@ mod tests { // and don't use cache let deno_dir = setup_deno_dir(temp_dir.path()); let result3 = - get_source_code(&deno_dir, module_name, filepath.clone(), false, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), false, false); assert!(result3.is_ok()); let r3 = result3.unwrap(); let expected3 = "export const loaded = true;\n".as_bytes(); @@ -1262,7 +1226,9 @@ mod tests { let (_temp_dir, deno_dir) = test_setup(); // http_util::fetch_sync_string requires tokio tokio_util::init(|| { - let module_name = "http://localhost:4545/tests/subdir/mismatch_ext.ts"; + let module_url = + Url::parse("http://localhost:4545/tests/subdir/mismatch_ext.ts") + .unwrap(); let filepath = deno_dir .deps_http .join("localhost_PORT4545/tests/subdir/mismatch_ext.ts"); @@ -1270,7 +1236,7 @@ mod tests { // first download let result = - get_source_code(&deno_dir, module_name, filepath.clone(), false, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), false, false); assert!(result.is_ok()); let result = fs::File::open(&headers_file_name); @@ -1284,7 +1250,7 @@ mod tests { // false, this can be verified using source header file creation timestamp (should be // the same as after first download) let result = - get_source_code(&deno_dir, module_name, filepath.clone(), false, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), false, false); assert!(result.is_ok()); let result = fs::File::open(&headers_file_name); @@ -1303,8 +1269,9 @@ mod tests { let (_temp_dir, deno_dir) = test_setup(); // Test basic follow and headers recording tokio_util::init(|| { - let redirect_module_name = - "http://localhost:4546/tests/subdir/redirects/redirect1.js"; + let redirect_module_url = + Url::parse("http://localhost:4546/tests/subdir/redirects/redirect1.js") + .unwrap(); let redirect_source_filepath = deno_dir .deps_http .join("localhost_PORT4546/tests/subdir/redirects/redirect1.js"); @@ -1320,7 +1287,7 @@ mod tests { let mod_meta = get_source_code( &deno_dir, - redirect_module_name, + &redirect_module_url, redirect_source_filepath.clone(), true, false, @@ -1347,7 +1314,7 @@ mod tests { assert_eq!(&mod_meta.module_name, target_module_name); assert_eq!( &mod_meta.module_redirect_source_name.clone().unwrap(), - redirect_module_name + &redirect_module_url.to_string() ); }); } @@ -1357,8 +1324,9 @@ mod tests { let (_temp_dir, deno_dir) = test_setup(); // Test double redirects and headers recording tokio_util::init(|| { - let redirect_module_name = - "http://localhost:4548/tests/subdir/redirects/redirect1.js"; + let redirect_module_url = + Url::parse("http://localhost:4548/tests/subdir/redirects/redirect1.js") + .unwrap(); let redirect_source_filepath = deno_dir .deps_http .join("localhost_PORT4548/tests/subdir/redirects/redirect1.js"); @@ -1380,7 +1348,7 @@ mod tests { let mod_meta = get_source_code( &deno_dir, - redirect_module_name, + &redirect_module_url, redirect_source_filepath.clone(), true, false, @@ -1414,7 +1382,7 @@ mod tests { assert_eq!(&mod_meta.module_name, target_module_name); assert_eq!( &mod_meta.module_redirect_source_name.clone().unwrap(), - redirect_module_name + &redirect_module_url.to_string() ); }); } @@ -1423,26 +1391,27 @@ mod tests { fn test_get_source_code_no_fetch() { let (_temp_dir, deno_dir) = test_setup(); tokio_util::init(|| { - let module_name = "http://localhost:4545/tests/002_hello.ts"; + let module_url = + Url::parse("http://localhost:4545/tests/002_hello.ts").unwrap(); let filepath = deno_dir .deps_http .join("localhost_PORT4545/tests/002_hello.ts"); // file hasn't been cached before and remote downloads are not allowed let result = - get_source_code(&deno_dir, module_name, filepath.clone(), true, true); + get_source_code(&deno_dir, &module_url, filepath.clone(), true, true); assert!(result.is_err()); let err = result.err().unwrap(); assert_eq!(err.kind(), ErrorKind::NotFound); // download and cache file let result = - get_source_code(&deno_dir, module_name, filepath.clone(), true, false); + get_source_code(&deno_dir, &module_url, filepath.clone(), true, false); assert!(result.is_ok()); // module is already cached, should be ok even with `no_fetch` let result = - get_source_code(&deno_dir, module_name, filepath.clone(), true, true); + get_source_code(&deno_dir, &module_url, filepath.clone(), true, true); assert!(result.is_ok()); }); } @@ -1453,8 +1422,9 @@ mod tests { // http_util::fetch_sync_string requires tokio tokio_util::init(|| { let (_temp_dir, deno_dir) = test_setup(); - let module_name = - "http://127.0.0.1:4545/tests/subdir/mt_video_mp2t.t3.ts".to_string(); + let module_url = + Url::parse("http://127.0.0.1:4545/tests/subdir/mt_video_mp2t.t3.ts") + .unwrap(); let filepath = deno_dir .deps_http .join("127.0.0.1_PORT4545/tests/subdir/mt_video_mp2t.t3.ts"); @@ -1462,7 +1432,7 @@ mod tests { let result = tokio_util::block_on(fetch_remote_source_async( &deno_dir, - &module_name, + &module_url, &filepath, )); assert!(result.is_ok()); @@ -1478,8 +1448,7 @@ mod tests { Some("text/javascript".to_owned()), None, ); - let result2 = - fetch_local_source(&deno_dir, &module_name, &filepath, None); + let result2 = fetch_local_source(&deno_dir, &module_url, &filepath, None); assert!(result2.is_ok()); let r2 = result2.unwrap().unwrap(); assert_eq!(r2.source_code, b"export const loaded = true;\n"); @@ -1494,14 +1463,15 @@ mod tests { // http_util::fetch_sync_string requires tokio tokio_util::init(|| { let (_temp_dir, deno_dir) = test_setup(); - let module_name = - "http://localhost:4545/tests/subdir/mt_video_mp2t.t3.ts"; + let module_url = + Url::parse("http://localhost:4545/tests/subdir/mt_video_mp2t.t3.ts") + .unwrap(); let filepath = deno_dir .deps_http .join("localhost_PORT4545/tests/subdir/mt_video_mp2t.t3.ts"); let headers_file_name = source_code_headers_filename(&filepath); - let result = fetch_remote_source(&deno_dir, module_name, &filepath); + let result = fetch_remote_source(&deno_dir, &module_url, &filepath); assert!(result.is_ok()); let r = result.unwrap().unwrap(); assert_eq!(r.source_code, "export const loaded = true;\n".as_bytes()); @@ -1515,7 +1485,7 @@ mod tests { Some("text/javascript".to_owned()), None, ); - let result2 = fetch_local_source(&deno_dir, module_name, &filepath, None); + let result2 = fetch_local_source(&deno_dir, &module_url, &filepath, None); assert!(result2.is_ok()); let r2 = result2.unwrap().unwrap(); assert_eq!(r2.source_code, "export const loaded = true;\n".as_bytes()); @@ -1530,11 +1500,12 @@ mod tests { // http_util::fetch_sync_string requires tokio tokio_util::init(|| { let (_temp_dir, deno_dir) = test_setup(); - let module_name = "http://localhost:4545/tests/subdir/no_ext"; + let module_url = + Url::parse("http://localhost:4545/tests/subdir/no_ext").unwrap(); let filepath = deno_dir .deps_http .join("localhost_PORT4545/tests/subdir/no_ext"); - let result = fetch_remote_source(&deno_dir, module_name, &filepath); + let result = fetch_remote_source(&deno_dir, &module_url, &filepath); assert!(result.is_ok()); let r = result.unwrap().unwrap(); assert_eq!(r.source_code, "export const loaded = true;\n".as_bytes()); @@ -1545,11 +1516,13 @@ mod tests { "text/typescript" ); - let module_name_2 = "http://localhost:4545/tests/subdir/mismatch_ext.ts"; + let module_url_2 = + Url::parse("http://localhost:4545/tests/subdir/mismatch_ext.ts") + .unwrap(); let filepath_2 = deno_dir .deps_http .join("localhost_PORT4545/tests/subdir/mismatch_ext.ts"); - let result_2 = fetch_remote_source(&deno_dir, module_name_2, &filepath_2); + let result_2 = fetch_remote_source(&deno_dir, &module_url_2, &filepath_2); assert!(result_2.is_ok()); let r2 = result_2.unwrap().unwrap(); assert_eq!(r2.source_code, "export const loaded = true;\n".as_bytes()); @@ -1561,11 +1534,13 @@ mod tests { ); // test unknown extension - let module_name_3 = "http://localhost:4545/tests/subdir/unknown_ext.deno"; + let module_url_3 = + Url::parse("http://localhost:4545/tests/subdir/unknown_ext.deno") + .unwrap(); let filepath_3 = deno_dir .deps_http .join("localhost_PORT4545/tests/subdir/unknown_ext.deno"); - let result_3 = fetch_remote_source(&deno_dir, module_name_3, &filepath_3); + let result_3 = fetch_remote_source(&deno_dir, &module_url_3, &filepath_3); assert!(result_3.is_ok()); let r3 = result_3.unwrap().unwrap(); assert_eq!(r3.source_code, "export const loaded = true;\n".as_bytes()); @@ -1583,10 +1558,11 @@ mod tests { // only local, no http_util::fetch_sync_string called let (_temp_dir, deno_dir) = test_setup(); let cwd = std::env::current_dir().unwrap(); - let module_name = "http://example.com/mt_text_typescript.t1.ts"; // not used + let module_url = + Url::parse("http://example.com/mt_text_typescript.t1.ts").unwrap(); let filepath = cwd.join("tests/subdir/mt_text_typescript.t1.ts"); - let result = fetch_local_source(&deno_dir, module_name, &filepath, None); + let result = fetch_local_source(&deno_dir, &module_url, &filepath, None); assert!(result.is_ok()); let r = result.unwrap().unwrap(); assert_eq!(r.source_code, "export const loaded = true;\n".as_bytes()); @@ -1597,18 +1573,17 @@ mod tests { fn test_fetch_module_meta_data() { let (_temp_dir, deno_dir) = test_setup(); - let cwd = std::env::current_dir().unwrap(); - let cwd_string = String::from(cwd.to_str().unwrap()) + "/"; - tokio_util::init(|| { // Test failure case. - let specifier = add_root!("/baddir/hello.ts"); - let r = deno_dir.fetch_module_meta_data(specifier, true, false); + let specifier = + ModuleSpecifier::resolve_url(file_url!("/baddir/hello.ts")).unwrap(); + let r = deno_dir.fetch_module_meta_data(&specifier, true, false); assert!(r.is_err()); // Assuming cwd is the deno repo root. - let specifier = &format!("{}{}", cwd_string.as_str(), "js/main.ts"); - let r = deno_dir.fetch_module_meta_data(specifier, true, false); + let specifier = + ModuleSpecifier::resolve_url_or_path("js/main.ts").unwrap(); + let r = deno_dir.fetch_module_meta_data(&specifier, true, false); assert!(r.is_ok()); }) } @@ -1618,116 +1593,58 @@ mod tests { /*recompile ts file*/ let (_temp_dir, deno_dir) = test_setup(); - let cwd = std::env::current_dir().unwrap(); - let cwd_string = String::from(cwd.to_str().unwrap()) + "/"; - tokio_util::init(|| { // Test failure case. - let specifier = add_root!("/baddir/hello.ts"); - let r = deno_dir.fetch_module_meta_data(specifier, false, false); + let specifier = + ModuleSpecifier::resolve_url(file_url!("/baddir/hello.ts")).unwrap(); + let r = deno_dir.fetch_module_meta_data(&specifier, false, false); assert!(r.is_err()); // Assuming cwd is the deno repo root. - let specifier = &format!("{}{}", cwd_string.as_str(), "js/main.ts"); - let r = deno_dir.fetch_module_meta_data(specifier, false, false); + let specifier = + ModuleSpecifier::resolve_url_or_path("js/main.ts").unwrap(); + let r = deno_dir.fetch_module_meta_data(&specifier, false, false); assert!(r.is_ok()); }) } - #[test] - fn test_src_file_to_url_1() { - let (_temp_dir, deno_dir) = test_setup(); - assert_eq!("hello", deno_dir.src_file_to_url("hello")); - assert_eq!("/hello", deno_dir.src_file_to_url("/hello")); - let x = deno_dir.deps_http.join("hello/world.txt"); - assert_eq!( - "http://hello/world.txt", - deno_dir.src_file_to_url(x.to_str().unwrap()) - ); - } - - #[test] - fn test_src_file_to_url_2() { - let (_temp_dir, deno_dir) = test_setup(); - assert_eq!("hello", deno_dir.src_file_to_url("hello")); - assert_eq!("/hello", deno_dir.src_file_to_url("/hello")); - let x = deno_dir.deps_https.join("hello/world.txt"); - assert_eq!( - "https://hello/world.txt", - deno_dir.src_file_to_url(x.to_str().unwrap()) - ); - } - - #[test] - fn test_src_file_to_url_3() { - let (_temp_dir, deno_dir) = test_setup(); - let x = deno_dir.deps_http.join("localhost_PORT4545/world.txt"); - assert_eq!( - "http://localhost:4545/world.txt", - deno_dir.src_file_to_url(x.to_str().unwrap()) - ); - } - - #[test] - fn test_src_file_to_url_4() { - let (_temp_dir, deno_dir) = test_setup(); - let x = deno_dir.deps_https.join("localhost_PORT4545/world.txt"); - assert_eq!( - "https://localhost:4545/world.txt", - deno_dir.src_file_to_url(x.to_str().unwrap()) - ); - } - // https://github.com/denoland/deno/blob/golang/os_test.go#L16-L87 #[test] - fn test_resolve_module_1() { + fn test_url_to_deps_path_1() { let (_temp_dir, deno_dir) = test_setup(); let test_cases = [ ( - "./subdir/print_hello.ts", - add_root!("/Users/rld/go/src/github.com/denoland/deno/testdata/006_url_imports.ts"), file_url!("/Users/rld/go/src/github.com/denoland/deno/testdata/subdir/print_hello.ts"), add_root!("/Users/rld/go/src/github.com/denoland/deno/testdata/subdir/print_hello.ts"), ), ( - "testdata/001_hello.js", - add_root!("/Users/rld/go/src/github.com/denoland/deno/"), file_url!("/Users/rld/go/src/github.com/denoland/deno/testdata/001_hello.js"), add_root!("/Users/rld/go/src/github.com/denoland/deno/testdata/001_hello.js"), ), ( - add_root!("/Users/rld/src/deno/hello.js"), - ".", file_url!("/Users/rld/src/deno/hello.js"), add_root!("/Users/rld/src/deno/hello.js"), ), ( - add_root!("/this/module/got/imported.js"), - add_root!("/that/module/did/it.js"), file_url!("/this/module/got/imported.js"), add_root!("/this/module/got/imported.js"), ), ]; for &test in test_cases.iter() { - let specifier = String::from(test.0); - let referrer = String::from(test.1); - let (module_name, filename) = - deno_dir.resolve_module(&specifier, &referrer).unwrap(); - assert_eq!(module_name, test.2); - assert_eq!(filename.to_str().unwrap().to_string(), test.3); + let url = Url::parse(test.0).unwrap(); + let filename = deno_dir.url_to_deps_path(&url).unwrap(); + assert_eq!(filename.to_str().unwrap().to_string(), test.1); } } #[test] - fn test_resolve_module_2() { + fn test_url_to_deps_path_2() { let (_temp_dir, deno_dir) = test_setup(); - let specifier = "http://localhost:4545/testdata/subdir/print_hello.ts"; - let referrer = add_root!("/deno/testdata/006_url_imports.ts"); - - let expected_module_name = - "http://localhost:4545/testdata/subdir/print_hello.ts"; + let specifier = + Url::parse("http://localhost:4545/testdata/subdir/print_hello.ts") + .unwrap(); let expected_filename = normalize_to_str( deno_dir .deps_http @@ -1735,9 +1652,7 @@ mod tests { .as_ref(), ); - let (module_name, filename) = - deno_dir.resolve_module(specifier, referrer).unwrap(); - assert_eq!(module_name, expected_module_name); + let filename = deno_dir.url_to_deps_path(&specifier).unwrap(); assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); } @@ -1745,171 +1660,19 @@ mod tests { fn test_resolve_module_3() { let (_temp_dir, deno_dir) = test_setup(); - let specifier_ = - deno_dir.deps_http.join("unpkg.com/liltest@0.0.5/index.ts"); - let specifier = specifier_.to_str().unwrap(); - let referrer = "."; - - let expected_module_name = "http://unpkg.com/liltest@0.0.5/index.ts"; - let expected_filename = normalize_to_str( - deno_dir - .deps_http - .join("unpkg.com/liltest@0.0.5/index.ts") - .as_ref(), - ); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, referrer).unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); - } - - #[test] - fn test_resolve_module_4() { - let (_temp_dir, deno_dir) = test_setup(); - - let specifier = "./util"; - let referrer_ = deno_dir.deps_http.join("unpkg.com/liltest@0.0.5/index.ts"); - let referrer = referrer_.to_str().unwrap(); - - // http containing files -> load relative import with http - let expected_module_name = "http://unpkg.com/liltest@0.0.5/util"; - let expected_filename = normalize_to_str( - deno_dir - .deps_http - .join("unpkg.com/liltest@0.0.5/util") - .as_ref(), - ); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, referrer).unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); - } - - #[test] - fn test_resolve_module_5() { - let (_temp_dir, deno_dir) = test_setup(); - - let specifier = "./util"; - let referrer_ = - deno_dir.deps_https.join("unpkg.com/liltest@0.0.5/index.ts"); - let referrer = referrer_.to_str().unwrap(); - - // https containing files -> load relative import with https - let expected_module_name = "https://unpkg.com/liltest@0.0.5/util"; - let expected_filename = normalize_to_str( - deno_dir - .deps_https - .join("unpkg.com/liltest@0.0.5/util") - .as_ref(), - ); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, referrer).unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); - } - - #[test] - fn test_resolve_module_6() { - let (_temp_dir, deno_dir) = test_setup(); - - let specifier = "http://localhost:4545/tests/subdir/mod2.ts"; - let referrer = add_root!("/deno/tests/006_url_imports.ts"); - let expected_module_name = "http://localhost:4545/tests/subdir/mod2.ts"; - let expected_filename = normalize_to_str( - deno_dir - .deps_http - .join("localhost_PORT4545/tests/subdir/mod2.ts") - .as_ref(), - ); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, referrer).unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); - } - - #[test] - fn test_resolve_module_7() { - let (_temp_dir, deno_dir) = test_setup(); - - let specifier = "http_test.ts"; - let referrer = add_root!("/Users/rld/src/deno_net/"); - let expected_module_name = - file_url!("/Users/rld/src/deno_net/http_test.ts"); - let expected_filename = add_root!("/Users/rld/src/deno_net/http_test.ts"); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, referrer).unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); - } - - #[test] - fn test_resolve_module_8() { - let (_temp_dir, deno_dir) = test_setup(); - - let specifier = "/util"; - let referrer_ = - deno_dir.deps_https.join("unpkg.com/liltest@0.0.5/index.ts"); - let referrer = referrer_.to_str().unwrap(); - - let expected_module_name = "https://unpkg.com/util"; - let expected_filename = - normalize_to_str(deno_dir.deps_https.join("unpkg.com/util").as_ref()); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, referrer).unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); - } - - #[test] - fn test_resolve_module_referrer_dot() { - let (_temp_dir, deno_dir) = test_setup(); - - let specifier = "tests/001_hello.js"; - - let cwd = std::env::current_dir().unwrap(); - let expected_path = cwd.join(specifier); - let expected_module_name = - Url::from_file_path(&expected_path).unwrap().to_string(); - let expected_filename = normalize_to_str(&expected_path); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, ".").unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, "./").unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); - } - - #[test] - fn test_resolve_module_referrer_dotdot() { - let (_temp_dir, deno_dir) = test_setup(); - - let specifier = "tests/001_hello.js"; - - let cwd = std::env::current_dir().unwrap(); - let expected_path = cwd.join("..").join(specifier); - let expected_module_name = - Url::from_file_path(&expected_path).unwrap().to_string(); - let expected_filename = normalize_to_str(&expected_path); - - let (module_name, filename) = - deno_dir.resolve_module(specifier, "..").unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); + // unsupported schemes + let test_cases = [ + "ftp://localhost:4545/testdata/subdir/print_hello.ts", + "blob:https://whatwg.org/d0360e2f-caee-469f-9a2f-87d5b0456f6f", + ]; - let (module_name, filename) = - deno_dir.resolve_module(specifier, "../").unwrap(); - assert_eq!(module_name, expected_module_name); - assert_eq!(filename.to_str().unwrap().to_string(), expected_filename); + for &test in test_cases.iter() { + let url = Url::parse(test).unwrap(); + assert_eq!( + deno_dir.url_to_deps_path(&url).unwrap_err().kind, + DenoDirErrorKind::UnsupportedFetchScheme + ); + } } #[test] diff --git a/cli/deno_error.rs b/cli/deno_error.rs index 2e683c504e6ac8..0410c35d2eb6c1 100644 --- a/cli/deno_error.rs +++ b/cli/deno_error.rs @@ -1,4 +1,5 @@ // Copyright 2018-2019 the Deno authors. All rights reserved. MIT license. +use crate::deno_dir; use crate::diagnostics; use crate::fmt_errors::JSErrorColor; use crate::import_map; @@ -34,6 +35,7 @@ enum Repr { ModuleResolutionErr(ModuleResolutionError), Diagnostic(diagnostics::Diagnostic), JSError(JSError), + DenoDirErr(deno_dir::DenoDirError), } /// Create a new simple DenoError. @@ -110,11 +112,13 @@ impl DenoError { use ModuleResolutionError::*; match err { InvalidUrl(err) | InvalidBaseUrl(err) => Self::url_error_kind(err), - ImportPathPrefixMissing => ErrorKind::ImportPathPrefixMissing, + InvalidPath => ErrorKind::InvalidPath, + ImportPrefixMissing => ErrorKind::ImportPrefixMissing, } } Repr::Diagnostic(ref _err) => ErrorKind::Diagnostic, Repr::JSError(ref _err) => ErrorKind::JSError, + Repr::DenoDirErr(ref _err) => ErrorKind::DenoDirError, } } @@ -140,6 +144,7 @@ impl fmt::Display for DenoError { Repr::ModuleResolutionErr(ref err) => err.fmt(f), Repr::Diagnostic(ref err) => err.fmt(f), Repr::JSError(ref err) => JSErrorColor(err).fmt(f), + Repr::DenoDirErr(ref err) => err.fmt(f), } } } @@ -155,6 +160,7 @@ impl std::error::Error for DenoError { Repr::ModuleResolutionErr(ref err) => err.description(), Repr::Diagnostic(ref err) => &err.items[0].message, Repr::JSError(ref err) => &err.description(), + Repr::DenoDirErr(ref err) => err.description(), } } @@ -168,6 +174,7 @@ impl std::error::Error for DenoError { Repr::ModuleResolutionErr(ref err) => err.source(), Repr::Diagnostic(ref _err) => None, Repr::JSError(ref err) => Some(err), + Repr::DenoDirErr(ref err) => Some(err), } } } @@ -255,6 +262,14 @@ impl From for DenoError { } } +impl From for DenoError { + fn from(err: deno_dir::DenoDirError) -> Self { + Self { + repr: Repr::DenoDirErr(err), + } + } +} + impl From for DenoError { fn from(err: ModuleResolutionError) -> Self { Self { @@ -333,6 +348,8 @@ pub fn err_check(r: Result) { mod tests { use super::*; use crate::ansi::strip_ansi_codes; + use crate::deno_dir::DenoDirError; + use crate::deno_dir::DenoDirErrorKind; use crate::diagnostics::Diagnostic; use crate::diagnostics::DiagnosticCategory; use crate::diagnostics::DiagnosticItem; @@ -449,6 +466,13 @@ mod tests { } } + fn deno_dir_error() -> DenoDirError { + DenoDirError::new( + "a deno dir error".to_string(), + DenoDirErrorKind::UnsupportedFetchScheme, + ) + } + #[test] fn test_simple_error() { let err = new(ErrorKind::NoError, "foo".to_string()); @@ -493,6 +517,13 @@ mod tests { assert_eq!(err.to_string(), "an import map error"); } + #[test] + fn test_deno_dir_error() { + let err = DenoError::from(deno_dir_error()); + assert_eq!(err.kind(), ErrorKind::DenoDirError); + assert_eq!(err.to_string(), "a deno dir error\n"); + } + #[test] fn test_bad_resource() { let err = bad_resource(); diff --git a/cli/flags.rs b/cli/flags.rs index 4d68ac726128da..b3fc11380ce673 100644 --- a/cli/flags.rs +++ b/cli/flags.rs @@ -630,7 +630,9 @@ pub enum DenoSubcommand { fn get_default_bundle_filename(source_file: &str) -> String { use deno::ModuleSpecifier; - let url = ModuleSpecifier::resolve_root(source_file).unwrap().to_url(); + let url = ModuleSpecifier::resolve_url_or_path(source_file) + .unwrap() + .to_url(); let path_segments = url.path_segments().unwrap(); let last = path_segments.last().unwrap(); String::from(last.trim_end_matches(".ts").trim_end_matches(".js")) diff --git a/cli/import_map.rs b/cli/import_map.rs index 4321cacadfd4e5..44525f7a2e5007 100644 --- a/cli/import_map.rs +++ b/cli/import_map.rs @@ -178,9 +178,7 @@ impl ImportMap { continue; } - let normalized_address = ModuleSpecifier::resolve(&url_string, ".") - .expect("Address should be valid module specifier"); - normalized_addresses.push(normalized_address); + normalized_addresses.push(url.into()); } normalized_addresses diff --git a/cli/msg.fbs b/cli/msg.fbs index da181af435d989..08da7c3b55e203 100644 --- a/cli/msg.fbs +++ b/cli/msg.fbs @@ -102,6 +102,7 @@ enum ErrorKind: byte { WouldBlock, InvalidInput, InvalidData, + InvalidPath, TimedOut, Interrupted, WriteZero, @@ -141,7 +142,8 @@ enum ErrorKind: byte { NoAsyncSupport, NoSyncSupport, ImportMapError, - ImportPathPrefixMissing, + ImportPrefixMissing, + DenoDirError, // other kinds Diagnostic, diff --git a/cli/ops.rs b/cli/ops.rs index 0664d8077f2c1d..770baf4edf579c 100644 --- a/cli/ops.rs +++ b/cli/ops.rs @@ -481,8 +481,12 @@ fn op_cache( // cache path. In the future, checksums will not be used in the cache // filenames and this requirement can be removed. See // https://github.com/denoland/deno/issues/2057 + let module_specifier = ModuleSpecifier::resolve_url(module_id) + .expect("Should be valid module specifier"); let module_meta_data = - state.dir.fetch_module_meta_data(module_id, true, true)?; + state + .dir + .fetch_module_meta_data(&module_specifier, true, true)?; let (js_cache_path, source_map_path) = state.dir.cache_path( &PathBuf::from(&module_meta_data.filename), @@ -525,11 +529,8 @@ fn op_fetch_module_meta_data( let fut = state .dir - .fetch_module_meta_data_async( - &resolved_specifier.to_string(), - use_cache, - no_fetch, - ).and_then(move |out| { + .fetch_module_meta_data_async(&resolved_specifier, use_cache, no_fetch) + .and_then(move |out| { let builder = &mut FlatBufferBuilder::new(); let data_off = builder.create_vector(out.source_code.as_slice()); let msg_args = msg::FetchModuleMetaDataResArgs { @@ -2053,7 +2054,7 @@ fn op_create_worker( err_check(worker.execute("denoMain()")); err_check(worker.execute("workerMain()")); - let module_specifier = ModuleSpecifier::resolve_root(specifier)?; + let module_specifier = ModuleSpecifier::resolve_url_or_path(specifier)?; let op = worker diff --git a/cli/state.rs b/cli/state.rs index fd209a0f23daaf..9f6c5996922b66 100644 --- a/cli/state.rs +++ b/cli/state.rs @@ -126,11 +126,8 @@ pub fn fetch_module_meta_data_and_maybe_compile_async( state_ .dir - .fetch_module_meta_data_async( - &module_specifier.to_string(), - use_cache, - no_fetch, - ).and_then(move |out| { + .fetch_module_meta_data_async(&module_specifier, use_cache, no_fetch) + .and_then(move |out| { if out.media_type == msg::MediaType::TypeScript && !out.has_output_code_and_source_map() { @@ -170,7 +167,8 @@ impl Loader for ThreadSafeState { } } - ModuleSpecifier::resolve(specifier, referrer).map_err(DenoError::from) + ModuleSpecifier::resolve_import(specifier, referrer) + .map_err(DenoError::from) } /// Given an absolute url, load its source code. @@ -252,7 +250,7 @@ impl ThreadSafeState { None } else { let root_specifier = argv_rest[1].clone(); - match ModuleSpecifier::resolve_root(&root_specifier) { + match ModuleSpecifier::resolve_url_or_path(&root_specifier) { Ok(specifier) => Some(specifier), Err(e) => { // TODO: handle unresolvable specifier diff --git a/cli/worker.rs b/cli/worker.rs index 65da666e87dbc7..1cf38e295102e8 100644 --- a/cli/worker.rs +++ b/cli/worker.rs @@ -141,7 +141,7 @@ mod tests { #[test] fn execute_mod_esm_imports_a() { let module_specifier = - ModuleSpecifier::resolve_root("tests/esm_imports_a.js").unwrap(); + ModuleSpecifier::resolve_url_or_path("tests/esm_imports_a.js").unwrap(); let argv = vec![String::from("./deno"), module_specifier.to_string()]; let state = ThreadSafeState::new( flags::DenoFlags::default(), @@ -169,7 +169,7 @@ mod tests { #[test] fn execute_mod_circular() { let module_specifier = - ModuleSpecifier::resolve_root("tests/circular1.js").unwrap(); + ModuleSpecifier::resolve_url_or_path("tests/circular1.js").unwrap(); let argv = vec![String::from("./deno"), module_specifier.to_string()]; let state = ThreadSafeState::new( flags::DenoFlags::default(), @@ -197,7 +197,7 @@ mod tests { #[test] fn execute_006_url_imports() { let module_specifier = - ModuleSpecifier::resolve_root("tests/006_url_imports.ts").unwrap(); + ModuleSpecifier::resolve_url_or_path("tests/006_url_imports.ts").unwrap(); let argv = vec![String::from("deno"), module_specifier.to_string()]; let mut flags = flags::DenoFlags::default(); flags.reload = true; @@ -327,7 +327,7 @@ mod tests { // "foo" is not a valid module specifier so this should return an error. let mut worker = create_test_worker(); let module_specifier = - ModuleSpecifier::resolve_root("does-not-exist").unwrap(); + ModuleSpecifier::resolve_url_or_path("does-not-exist").unwrap(); let result = worker.execute_mod_async(&module_specifier, false).wait(); assert!(result.is_err()); }) @@ -340,7 +340,7 @@ mod tests { // tests). let mut worker = create_test_worker(); let module_specifier = - ModuleSpecifier::resolve_root("./tests/002_hello.ts").unwrap(); + ModuleSpecifier::resolve_url_or_path("./tests/002_hello.ts").unwrap(); let result = worker.execute_mod_async(&module_specifier, false).wait(); assert!(result.is_ok()); }) diff --git a/core/module_specifier.rs b/core/module_specifier.rs index 8b8ccf4a675ee7..84c40a94e90194 100644 --- a/core/module_specifier.rs +++ b/core/module_specifier.rs @@ -1,3 +1,4 @@ +use std::env::current_dir; use std::error::Error; use std::fmt; use url::ParseError; @@ -8,7 +9,8 @@ use url::Url; pub enum ModuleResolutionError { InvalidUrl(ParseError), InvalidBaseUrl(ParseError), - ImportPathPrefixMissing, + InvalidPath, + ImportPrefixMissing, } use ModuleResolutionError::*; @@ -28,7 +30,8 @@ impl fmt::Display for ModuleResolutionError { InvalidBaseUrl(ref err) => { write!(f, "invalid base URL for relative import: {}", err) } - ImportPathPrefixMissing => { + InvalidPath => write!(f, "invalid module path"), + ImportPrefixMissing => { write!(f, "relative import path not prefixed with / or ./ or ../") } } @@ -46,7 +49,7 @@ impl ModuleSpecifier { /// Resolves module using this algorithm: /// https://html.spec.whatwg.org/multipage/webappapis.html#resolve-a-module-specifier - pub fn resolve( + pub fn resolve_import( specifier: &str, base: &str, ) -> Result { @@ -64,7 +67,7 @@ impl ModuleSpecifier { || specifier.starts_with("./") || specifier.starts_with("../")) => { - Err(ImportPathPrefixMissing)? + Err(ImportPrefixMissing)? } // 3. Return the result of applying the URL parser to specifier with base @@ -76,7 +79,7 @@ impl ModuleSpecifier { // If parsing the specifier as a URL failed for a different reason than // it being relative, always return the original error. We don't want to - // return `ImportPathPrefixMissing` or `InvalidBaseUrl` if the real + // return `ImportPrefixMissing` or `InvalidBaseUrl` if the real // problem lies somewhere else. Err(err) => Err(InvalidUrl(err))?, }; @@ -84,25 +87,70 @@ impl ModuleSpecifier { Ok(ModuleSpecifier(url)) } - /// Takes a string representing a path or URL to a module, but of the type - /// passed through the command-line interface for the main module. This is - /// slightly different than specifiers used in import statements: "foo.js" for - /// example is allowed here, whereas in import statements a leading "./" is - /// required ("./foo.js"). This function is aware of the current working - /// directory and returns an absolute URL. - pub fn resolve_root( - root_specifier: &str, + /// Converts a string representing an absulute URL into a ModuleSpecifier. + pub fn resolve_url( + url_str: &str, ) -> Result { - let url = match Url::parse(root_specifier) { - Ok(url) => url, - Err(..) => { - let cwd = std::env::current_dir().unwrap(); - let base = Url::from_directory_path(cwd).unwrap(); - base.join(&root_specifier).map_err(InvalidUrl)? - } - }; + Url::parse(url_str) + .map(ModuleSpecifier) + .map_err(ModuleResolutionError::InvalidUrl) + } - Ok(ModuleSpecifier(url)) + /// Takes a string representing either an absolute URL or a file path, + /// as it may be passed to deno as a command line argument. + /// The string is interpreted as a URL if it starts with a valid URI scheme, + /// e.g. 'http:' or 'file:' or 'git+ssh:'. If not, it's interpreted as a + /// file path; if it is a relative path it's resolved relative to the current + /// working directory. + pub fn resolve_url_or_path( + specifier: &str, + ) -> Result { + if Self::specifier_has_uri_scheme(specifier) { + Self::resolve_url(specifier) + } else { + Self::resolve_path(specifier) + } + } + + /// Converts a string representing a relative or absolute path into a + /// ModuleSpecifier. A relative path is considered relative to the current + /// working directory. + fn resolve_path( + path_str: &str, + ) -> Result { + let path = current_dir().unwrap().join(path_str); + Url::from_file_path(path) + .map(ModuleSpecifier) + .map_err(|()| ModuleResolutionError::InvalidPath) + } + + /// Returns true if the input string starts with a sequence of characters + /// that could be a valid URI scheme, like 'https:', 'git+ssh:' or 'data:'. + /// + /// According to RFC 3986 (https://tools.ietf.org/html/rfc3986#section-3.1), + /// a valid scheme has the following format: + /// scheme = ALPHA *( ALPHA / DIGIT / "+" / "-" / "." ) + /// + /// We additionally require the scheme to be at least 2 characters long, + /// because otherwise a windows path like c:/foo would be treated as a URL, + /// while no schemes with a one-letter name actually exist. + fn specifier_has_uri_scheme(specifier: &str) -> bool { + let mut chars = specifier.chars(); + let mut len = 0usize; + // THe first character must be a letter. + match chars.next() { + Some(c) if c.is_ascii_alphabetic() => len += 1, + _ => return false, + } + // Second and following characters must be either a letter, number, + // plus sign, minus sign, or dot. + loop { + match chars.next() { + Some(c) if c.is_ascii_alphanumeric() || "+-.".contains(c) => len += 1, + Some(':') if len >= 2 => return true, + _ => return false, + } + } } } @@ -123,3 +171,280 @@ impl PartialEq for ModuleSpecifier { &self.to_string() == other } } + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_resolve_import() { + let tests = vec![ + ( + "./005_more_imports.ts", + "http://deno.land/core/tests/006_url_imports.ts", + "http://deno.land/core/tests/005_more_imports.ts", + ), + ( + "../005_more_imports.ts", + "http://deno.land/core/tests/006_url_imports.ts", + "http://deno.land/core/005_more_imports.ts", + ), + ( + "http://deno.land/core/tests/005_more_imports.ts", + "http://deno.land/core/tests/006_url_imports.ts", + "http://deno.land/core/tests/005_more_imports.ts", + ), + ( + "data:text/javascript,export default 'grapes';", + "http://deno.land/core/tests/006_url_imports.ts", + "data:text/javascript,export default 'grapes';", + ), + ( + "blob:https://whatwg.org/d0360e2f-caee-469f-9a2f-87d5b0456f6f", + "http://deno.land/core/tests/006_url_imports.ts", + "blob:https://whatwg.org/d0360e2f-caee-469f-9a2f-87d5b0456f6f", + ), + ( + "javascript:export default 'artichokes';", + "http://deno.land/core/tests/006_url_imports.ts", + "javascript:export default 'artichokes';", + ), + ( + "data:text/plain,export default 'kale';", + "http://deno.land/core/tests/006_url_imports.ts", + "data:text/plain,export default 'kale';", + ), + ( + "/dev/core/tests/005_more_imports.ts", + "file:///home/yeti", + "file:///dev/core/tests/005_more_imports.ts", + ), + ( + "//zombo.com/1999.ts", + "https://cherry.dev/its/a/thing", + "https://zombo.com/1999.ts", + ), + ( + "http://deno.land/this/url/is/valid", + "base is clearly not a valid url", + "http://deno.land/this/url/is/valid", + ), + ( + "//server/some/dir/file", + "file:///home/yeti/deno", + "file://server/some/dir/file", + ), + // This test is disabled because the url crate does not follow the spec, + // dropping the server part from the final result. + // ( + // "/another/path/at/the/same/server", + // "file://server/some/dir/file", + // "file://server/another/path/at/the/same/server", + // ), + ]; + + for (specifier, base, expected_url) in tests { + let url = ModuleSpecifier::resolve_import(specifier, base) + .unwrap() + .to_string(); + assert_eq!(url, expected_url); + } + } + + #[test] + fn test_resolve_import_error() { + use url::ParseError::*; + use ModuleResolutionError::*; + + let tests = vec![ + ( + "005_more_imports.ts", + "http://deno.land/core/tests/006_url_imports.ts", + ImportPrefixMissing, + ), + ( + ".tomato", + "http://deno.land/core/tests/006_url_imports.ts", + ImportPrefixMissing, + ), + ( + "..zucchini.mjs", + "http://deno.land/core/tests/006_url_imports.ts", + ImportPrefixMissing, + ), + ( + r".\yam.es", + "http://deno.land/core/tests/006_url_imports.ts", + ImportPrefixMissing, + ), + ( + r"..\yam.es", + "http://deno.land/core/tests/006_url_imports.ts", + ImportPrefixMissing, + ), + ( + "https://eggplant:b/c", + "http://deno.land/core/tests/006_url_imports.ts", + InvalidUrl(InvalidPort), + ), + ( + "https://eggplant@/c", + "http://deno.land/core/tests/006_url_imports.ts", + InvalidUrl(EmptyHost), + ), + ( + "./foo.ts", + "/relative/base/url", + InvalidBaseUrl(RelativeUrlWithoutBase), + ), + ]; + + for (specifier, base, expected_err) in tests { + let err = ModuleSpecifier::resolve_import(specifier, base).unwrap_err(); + assert_eq!(err, expected_err); + } + } + + #[test] + fn test_resolve_url_or_path() { + // Absolute URL. + let mut tests: Vec<(&str, String)> = vec![ + ( + "http://deno.land/core/tests/006_url_imports.ts", + "http://deno.land/core/tests/006_url_imports.ts".to_string(), + ), + ( + "https://deno.land/core/tests/006_url_imports.ts", + "https://deno.land/core/tests/006_url_imports.ts".to_string(), + ), + ]; + + // The local path tests assume that the cwd is the deno repo root. + let cwd = current_dir().unwrap(); + let cwd_str = cwd.to_str().unwrap(); + + if cfg!(target_os = "windows") { + // Absolute local path. + let expected_url = "file:///C:/deno/tests/006_url_imports.ts"; + tests.extend(vec![ + ( + r"C:/deno/tests/006_url_imports.ts", + expected_url.to_string(), + ), + ( + r"C:\deno\tests\006_url_imports.ts", + expected_url.to_string(), + ), + ( + r"\\?\C:\deno\tests\006_url_imports.ts", + expected_url.to_string(), + ), + // Not supported: `Url::from_file_path()` fails. + // (r"\\.\C:\deno\tests\006_url_imports.ts", expected_url.to_string()), + // Not supported: `Url::from_file_path()` performs the wrong conversion. + // (r"//./C:/deno/tests/006_url_imports.ts", expected_url.to_string()), + ]); + + // Rooted local path without drive letter. + let expected_url = format!( + "file:///{}:/deno/tests/006_url_imports.ts", + cwd_str.get(..1).unwrap(), + ); + tests.extend(vec![ + (r"/deno/tests/006_url_imports.ts", expected_url.to_string()), + (r"\deno\tests\006_url_imports.ts", expected_url.to_string()), + ]); + + // Relative local path. + let expected_url = format!( + "file:///{}/tests/006_url_imports.ts", + cwd_str.replace("\\", "/") + ); + tests.extend(vec![ + (r"tests/006_url_imports.ts", expected_url.to_string()), + (r"tests\006_url_imports.ts", expected_url.to_string()), + (r"./tests/006_url_imports.ts", expected_url.to_string()), + (r".\tests\006_url_imports.ts", expected_url.to_string()), + ]); + + // UNC network path. + let expected_url = "file://server/share/deno/cool"; + tests.extend(vec![ + (r"\\server\share\deno\cool", expected_url.to_string()), + (r"\\server/share/deno/cool", expected_url.to_string()), + // Not supported: `Url::from_file_path()` performs the wrong conversion. + // (r"//server/share/deno/cool", expected_url.to_string()), + ]); + } else { + // Absolute local path. + let expected_url = "file:///deno/tests/006_url_imports.ts"; + tests.extend(vec![ + ("/deno/tests/006_url_imports.ts", expected_url.to_string()), + ("//deno/tests/006_url_imports.ts", expected_url.to_string()), + ]); + + // Relative local path. + let expected_url = format!("file://{}/tests/006_url_imports.ts", cwd_str); + tests.extend(vec![ + ("tests/006_url_imports.ts", expected_url.to_string()), + ("./tests/006_url_imports.ts", expected_url.to_string()), + ]); + } + + for (specifier, expected_url) in tests { + let url = ModuleSpecifier::resolve_url_or_path(specifier) + .unwrap() + .to_string(); + assert_eq!(url, expected_url); + } + } + + #[test] + fn test_resolve_url_or_path_error() { + use url::ParseError::*; + use ModuleResolutionError::*; + + let mut tests = vec![ + ("https://eggplant:b/c", InvalidUrl(InvalidPort)), + ("https://:8080/a/b/c", InvalidUrl(EmptyHost)), + ]; + if cfg!(target_os = "windows") { + tests.push((r"\\.\c:/stuff/deno/script.ts", InvalidPath)); + } + + for (specifier, expected_err) in tests { + let err = ModuleSpecifier::resolve_url_or_path(specifier).unwrap_err(); + assert_eq!(err, expected_err); + } + } + + #[test] + fn test_specifier_has_uri_scheme() { + let tests = vec![ + ("http://foo.bar/etc", true), + ("HTTP://foo.bar/etc", true), + ("http:ftp:", true), + ("http:", true), + ("hTtP:", true), + ("ftp:", true), + ("mailto:spam@please.me", true), + ("git+ssh://git@github.com/denoland/deno", true), + ("blob:https://whatwg.org/mumbojumbo", true), + ("abc.123+DEF-ghi:", true), + ("abc.123+def-ghi:@", true), + ("", false), + (":not", false), + ("http", false), + ("c:dir", false), + ("X:", false), + ("./http://not", false), + ("1abc://kinda/but/no", false), + ("schluẞ://no/more", false), + ]; + + for (specifier, expected) in tests { + let result = ModuleSpecifier::specifier_has_uri_scheme(specifier); + assert_eq!(result, expected); + } + } +} diff --git a/core/modules.rs b/core/modules.rs index 8a229e7706a90e..c1fa5d7336564a 100644 --- a/core/modules.rs +++ b/core/modules.rs @@ -643,11 +643,11 @@ mod tests { eprintln!(">> RESOLVING, S: {}, R: {}", specifier, referrer); - let output_specifier = match ModuleSpecifier::resolve(specifier, referrer) - { - Ok(specifier) => specifier, - Err(_e) => return Err(MockError::ResolveErr), - }; + let output_specifier = + match ModuleSpecifier::resolve_import(specifier, referrer) { + Ok(specifier) => specifier, + Err(..) => return Err(MockError::ResolveErr), + }; if mock_source_code(&output_specifier.to_string()).is_some() { Ok(output_specifier) diff --git a/tests/error_011_bad_module_specifier.ts.out b/tests/error_011_bad_module_specifier.ts.out index d5a3efd7e2aefb..9eec893073431e 100644 --- a/tests/error_011_bad_module_specifier.ts.out +++ b/tests/error_011_bad_module_specifier.ts.out @@ -1,4 +1,4 @@ -[WILDCARD]error: Uncaught ImportPathPrefixMissing: relative import path not prefixed with / or ./ or ../ +[WILDCARD]error: Uncaught ImportPrefixMissing: relative import path not prefixed with / or ./ or ../ [WILDCARD] js/errors.ts:[WILDCARD] at DenoError (js/errors.ts:[WILDCARD]) at maybeError (js/errors.ts:[WILDCARD]) diff --git a/tests/error_012_bad_dynamic_import_specifier.ts.out b/tests/error_012_bad_dynamic_import_specifier.ts.out index 1a20b60a1750bf..0d4abe3b6e709d 100644 --- a/tests/error_012_bad_dynamic_import_specifier.ts.out +++ b/tests/error_012_bad_dynamic_import_specifier.ts.out @@ -1,4 +1,4 @@ -[WILDCARD]error: Uncaught ImportPathPrefixMissing: relative import path not prefixed with / or ./ or ../ +[WILDCARD]error: Uncaught ImportPrefixMissing: relative import path not prefixed with / or ./ or ../ [WILDCARD] js/errors.ts:[WILDCARD] at DenoError (js/errors.ts:[WILDCARD]) at maybeError (js/errors.ts:[WILDCARD]) @@ -7,5 +7,6 @@ at fetchModuleMetaData (js/compiler.ts:[WILDCARD]) at _resolveModule (js/compiler.ts:[WILDCARD]) at js/compiler.ts:[WILDCARD] + at resolveModuleNames (js/compiler.ts:[WILDCARD]) at resolveModuleNamesWorker ([WILDCARD]) at resolveModuleNamesReusingOldState ([WILDCARD]typescript.js:[WILDCARD])