1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-03 12:58:54 -05:00

fix(cli): handle extensionless imports better (#13548)

Fixes #13526
This commit is contained in:
Kitson Kelly 2022-01-31 20:32:49 +11:00 committed by GitHub
parent 49a0db0d2a
commit 68c8c66b0f
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 87 additions and 23 deletions

View file

@ -1488,6 +1488,12 @@ itest!(import_file_with_colon {
http_server: true, http_server: true,
}); });
itest!(import_extensionless {
args: "run --quiet --reload import_extensionless.ts",
output: "import_extensionless.ts.out",
http_server: true,
});
itest!(classic_workers_event_loop { itest!(classic_workers_event_loop {
args: args:
"run --enable-testing-features-do-not-use classic_workers_event_loop.js", "run --enable-testing-features-do-not-use classic_workers_event_loop.js",

View file

@ -0,0 +1,3 @@
import { printHello3 } from "http://localhost:4545/v1/extensionless";
printHello3();

View file

@ -0,0 +1,2 @@
[WILDCARD]
Hello

View file

@ -141,6 +141,30 @@ fn hash_url(specifier: &ModuleSpecifier, media_type: &MediaType) -> String {
) )
} }
/// If the provided URLs derivable tsc media type doesn't match the media type,
/// we will add an extension to the output. This is to avoid issues with
/// specifiers that don't have extensions, that tsc refuses to emit because they
/// think a `.js` version exists, when it doesn't.
fn maybe_remap_specifier(
specifier: &ModuleSpecifier,
media_type: &MediaType,
) -> Option<String> {
let path = if specifier.scheme() == "file" {
if let Ok(path) = specifier.to_file_path() {
path
} else {
PathBuf::from(specifier.path())
}
} else {
PathBuf::from(specifier.path())
};
if path.extension().is_none() {
Some(format!("{}{}", specifier, media_type.as_ts_extension()))
} else {
None
}
}
/// tsc only supports `.ts`, `.tsx`, `.d.ts`, `.js`, or `.jsx` as root modules /// tsc only supports `.ts`, `.tsx`, `.d.ts`, `.js`, or `.jsx` as root modules
/// and so we have to detect the apparent media type based on extensions it /// and so we have to detect the apparent media type based on extensions it
/// supports. /// supports.
@ -235,13 +259,13 @@ pub(crate) struct Response {
#[derive(Debug)] #[derive(Debug)]
struct State { struct State {
data_url_map: HashMap<String, ModuleSpecifier>,
hash_data: Vec<Vec<u8>>, hash_data: Vec<Vec<u8>>,
emitted_files: Vec<EmittedFile>, emitted_files: Vec<EmittedFile>,
graph_data: Arc<RwLock<GraphData>>, graph_data: Arc<RwLock<GraphData>>,
maybe_config_specifier: Option<ModuleSpecifier>, maybe_config_specifier: Option<ModuleSpecifier>,
maybe_tsbuildinfo: Option<String>, maybe_tsbuildinfo: Option<String>,
maybe_response: Option<RespondArgs>, maybe_response: Option<RespondArgs>,
remapped_specifiers: HashMap<String, ModuleSpecifier>,
root_map: HashMap<String, ModuleSpecifier>, root_map: HashMap<String, ModuleSpecifier>,
} }
@ -252,16 +276,16 @@ impl State {
maybe_config_specifier: Option<ModuleSpecifier>, maybe_config_specifier: Option<ModuleSpecifier>,
maybe_tsbuildinfo: Option<String>, maybe_tsbuildinfo: Option<String>,
root_map: HashMap<String, ModuleSpecifier>, root_map: HashMap<String, ModuleSpecifier>,
data_url_map: HashMap<String, ModuleSpecifier>, remapped_specifiers: HashMap<String, ModuleSpecifier>,
) -> Self { ) -> Self {
State { State {
data_url_map,
hash_data, hash_data,
emitted_files: Default::default(), emitted_files: Default::default(),
graph_data, graph_data,
maybe_config_specifier, maybe_config_specifier,
maybe_tsbuildinfo, maybe_tsbuildinfo,
maybe_response: None, maybe_response: None,
remapped_specifiers,
root_map, root_map,
} }
} }
@ -335,7 +359,7 @@ fn op_emit(state: &mut State, args: Value) -> Result<Value, AnyError> {
let specifiers = specifiers let specifiers = specifiers
.iter() .iter()
.map(|s| { .map(|s| {
if let Some(data_specifier) = state.data_url_map.get(s) { if let Some(data_specifier) = state.remapped_specifiers.get(s) {
data_specifier.clone() data_specifier.clone()
} else if let Some(remapped_specifier) = state.root_map.get(s) { } else if let Some(remapped_specifier) = state.root_map.get(s) {
remapped_specifier.clone() remapped_specifier.clone()
@ -423,10 +447,10 @@ fn op_load(state: &mut State, args: Value) -> Result<Value, AnyError> {
media_type = MediaType::from(&v.specifier); media_type = MediaType::from(&v.specifier);
maybe_source maybe_source
} else { } else {
let specifier = if let Some(data_specifier) = let specifier = if let Some(remapped_specifier) =
state.data_url_map.get(&v.specifier) state.remapped_specifiers.get(&v.specifier)
{ {
data_specifier.clone() remapped_specifier.clone()
} else if let Some(remapped_specifier) = state.root_map.get(&v.specifier) { } else if let Some(remapped_specifier) = state.root_map.get(&v.specifier) {
remapped_specifier.clone() remapped_specifier.clone()
} else { } else {
@ -465,20 +489,20 @@ pub struct ResolveArgs {
pub specifiers: Vec<String>, pub specifiers: Vec<String>,
} }
fn op_resolve(state: &mut State, args: Value) -> Result<Value, AnyError> { fn op_resolve(state: &mut State, args: ResolveArgs) -> Result<Value, AnyError> {
let v: ResolveArgs = serde_json::from_value(args)
.context("Invalid request from JavaScript for \"op_resolve\".")?;
let mut resolved: Vec<(String, String)> = Vec::new(); let mut resolved: Vec<(String, String)> = Vec::new();
let referrer = if let Some(data_specifier) = state.data_url_map.get(&v.base) { let referrer = if let Some(remapped_specifier) =
data_specifier.clone() state.remapped_specifiers.get(&args.base)
} else if let Some(remapped_base) = state.root_map.get(&v.base) { {
remapped_specifier.clone()
} else if let Some(remapped_base) = state.root_map.get(&args.base) {
remapped_base.clone() remapped_base.clone()
} else { } else {
normalize_specifier(&v.base).context( normalize_specifier(&args.base).context(
"Error converting a string module specifier for \"op_resolve\".", "Error converting a string module specifier for \"op_resolve\".",
)? )?
}; };
for specifier in &v.specifiers { for specifier in &args.specifiers {
if specifier.starts_with("asset:///") { if specifier.starts_with("asset:///") {
resolved.push(( resolved.push((
specifier.clone(), specifier.clone(),
@ -528,10 +552,23 @@ fn op_resolve(state: &mut State, args: Value) -> Result<Value, AnyError> {
let specifier_str = match specifier.scheme() { let specifier_str = match specifier.scheme() {
"data" | "blob" => { "data" | "blob" => {
let specifier_str = hash_url(&specifier, media_type); let specifier_str = hash_url(&specifier, media_type);
state.data_url_map.insert(specifier_str.clone(), specifier); state
.remapped_specifiers
.insert(specifier_str.clone(), specifier);
specifier_str specifier_str
} }
_ => specifier.to_string(), _ => {
if let Some(specifier_str) =
maybe_remap_specifier(&specifier, media_type)
{
state
.remapped_specifiers
.insert(specifier_str.clone(), specifier);
specifier_str
} else {
specifier.to_string()
}
}
}; };
(specifier_str, media_type.as_ts_extension().into()) (specifier_str, media_type.as_ts_extension().into())
} }
@ -573,14 +610,14 @@ pub(crate) fn exec(request: Request) -> Result<Response, AnyError> {
// extensions and remap any that are unacceptable to tsc and add them to the // extensions and remap any that are unacceptable to tsc and add them to the
// op state so when requested, we can remap to the original specifier. // op state so when requested, we can remap to the original specifier.
let mut root_map = HashMap::new(); let mut root_map = HashMap::new();
let mut data_url_map = HashMap::new(); let mut remapped_specifiers = HashMap::new();
let root_names: Vec<String> = request let root_names: Vec<String> = request
.root_names .root_names
.iter() .iter()
.map(|(s, mt)| match s.scheme() { .map(|(s, mt)| match s.scheme() {
"data" | "blob" => { "data" | "blob" => {
let specifier_str = hash_url(s, mt); let specifier_str = hash_url(s, mt);
data_url_map.insert(specifier_str.clone(), s.clone()); remapped_specifiers.insert(specifier_str.clone(), s.clone());
specifier_str specifier_str
} }
_ => { _ => {
@ -605,7 +642,7 @@ pub(crate) fn exec(request: Request) -> Result<Response, AnyError> {
request.maybe_config_specifier.clone(), request.maybe_config_specifier.clone(),
request.maybe_tsbuildinfo.clone(), request.maybe_tsbuildinfo.clone(),
root_map, root_map,
data_url_map, remapped_specifiers,
)); ));
} }
@ -981,7 +1018,10 @@ mod tests {
.await; .await;
let actual = op_resolve( let actual = op_resolve(
&mut state, &mut state,
json!({ "base": "https://deno.land/x/a.ts", "specifiers": [ "./b.ts" ]}), ResolveArgs {
base: "https://deno.land/x/a.ts".to_string(),
specifiers: vec!["./b.ts".to_string()],
},
) )
.expect("should have invoked op"); .expect("should have invoked op");
assert_eq!(actual, json!([["https://deno.land/x/b.ts", ".ts"]])); assert_eq!(actual, json!([["https://deno.land/x/b.ts", ".ts"]]));
@ -997,8 +1037,12 @@ mod tests {
.await; .await;
let actual = op_resolve( let actual = op_resolve(
&mut state, &mut state,
json!({ "base": "https://deno.land/x/a.ts", "specifiers": [ "./bad.ts" ]}), ResolveArgs {
).expect("should have not errored"); base: "https://deno.land/x/a.ts".to_string(),
specifiers: vec!["./bad.ts".to_string()],
},
)
.expect("should have not errored");
assert_eq!( assert_eq!(
actual, actual,
json!([["deno:///missing_dependency.d.ts", ".d.ts"]]) json!([["deno:///missing_dependency.d.ts", ".d.ts"]])

View file

@ -840,6 +840,15 @@ async fn main_server(
); );
Ok(res) Ok(res)
} }
(_, "/v1/extensionless") => {
let mut res =
Response::new(Body::from(r#"export * from "/subdir/mod1.ts";"#));
res.headers_mut().insert(
"content-type",
HeaderValue::from_static("application/typescript"),
);
Ok(res)
}
(_, "/subdir/no_js_ext@1.0.0") => { (_, "/subdir/no_js_ext@1.0.0") => {
let mut res = Response::new(Body::from( let mut res = Response::new(Body::from(
r#"import { printHello } from "./mod2.ts"; r#"import { printHello } from "./mod2.ts";