1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-21 13:00:36 -05:00

fix(lsp/check): don't resolve unknown media types to a .js extension (#27631)

Fixes https://github.com/denoland/deno/issues/25762. Note that some of
the things in that issue are not resolved (vite/client types not working
properly which has other root causes), but the wildcard module
augmentation specifically is fixed by this.

We were telling TSC that files with unknown media types had an extension
of `.js`, so the ambient module declarations weren't applying. Instead,
just don't resolve them, so the ambient declaration applies.
This commit is contained in:
Nathan Whitaker 2025-01-10 19:26:01 -08:00 committed by GitHub
parent f6dcc13537
commit 70c822bfe2
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 66 additions and 18 deletions

View file

@ -4509,11 +4509,12 @@ fn op_release(
#[op2] #[op2]
#[serde] #[serde]
#[allow(clippy::type_complexity)]
fn op_resolve( fn op_resolve(
state: &mut OpState, state: &mut OpState,
#[string] base: String, #[string] base: String,
#[serde] specifiers: Vec<(bool, String)>, #[serde] specifiers: Vec<(bool, String)>,
) -> Result<Vec<Option<(String, String)>>, deno_core::url::ParseError> { ) -> Result<Vec<Option<(String, Option<String>)>>, deno_core::url::ParseError> {
op_resolve_inner(state, ResolveArgs { base, specifiers }) op_resolve_inner(state, ResolveArgs { base, specifiers })
} }
@ -4595,10 +4596,11 @@ async fn op_poll_requests(
} }
#[inline] #[inline]
#[allow(clippy::type_complexity)]
fn op_resolve_inner( fn op_resolve_inner(
state: &mut OpState, state: &mut OpState,
args: ResolveArgs, args: ResolveArgs,
) -> Result<Vec<Option<(String, String)>>, deno_core::url::ParseError> { ) -> Result<Vec<Option<(String, Option<String>)>>, deno_core::url::ParseError> {
let state = state.borrow_mut::<State>(); let state = state.borrow_mut::<State>();
let mark = state.performance.mark_with_args("tsc.op.op_resolve", &args); let mark = state.performance.mark_with_args("tsc.op.op_resolve", &args);
let referrer = state.specifier_map.normalize(&args.base)?; let referrer = state.specifier_map.normalize(&args.base)?;
@ -4611,7 +4613,11 @@ fn op_resolve_inner(
o.map(|(s, mt)| { o.map(|(s, mt)| {
( (
state.specifier_map.denormalize(&s), state.specifier_map.denormalize(&s),
mt.as_ts_extension().to_string(), if matches!(mt, MediaType::Unknown) {
None
} else {
Some(mt.as_ts_extension().to_string())
},
) )
}) })
}) })
@ -6461,7 +6467,7 @@ mod tests {
resolved, resolved,
vec![Some(( vec![Some((
temp_dir.url().join("b.ts").unwrap().to_string(), temp_dir.url().join("b.ts").unwrap().to_string(),
MediaType::TypeScript.as_ts_extension().to_string() Some(MediaType::TypeScript.as_ts_extension().to_string())
))] ))]
); );
} }

View file

@ -723,7 +723,7 @@ delete Object.prototype.__proto__;
} }
: arg; : arg;
if (fileReference.fileName.startsWith("npm:")) { if (fileReference.fileName.startsWith("npm:")) {
/** @type {[string, ts.Extension] | undefined} */ /** @type {[string, ts.Extension | null] | undefined} */
const resolved = ops.op_resolve( const resolved = ops.op_resolve(
containingFilePath, containingFilePath,
[ [
@ -735,7 +735,7 @@ delete Object.prototype.__proto__;
], ],
], ],
)?.[0]; )?.[0];
if (resolved) { if (resolved && resolved[1]) {
return { return {
resolvedTypeReferenceDirective: { resolvedTypeReferenceDirective: {
primary: true, primary: true,
@ -785,7 +785,7 @@ delete Object.prototype.__proto__;
debug(` base: ${base}`); debug(` base: ${base}`);
debug(` specifiers: ${specifiers.map((s) => s[1]).join(", ")}`); debug(` specifiers: ${specifiers.map((s) => s[1]).join(", ")}`);
} }
/** @type {Array<[string, ts.Extension] | undefined>} */ /** @type {Array<[string, ts.Extension | null] | undefined>} */
const resolved = ops.op_resolve( const resolved = ops.op_resolve(
base, base,
specifiers, specifiers,
@ -793,7 +793,7 @@ delete Object.prototype.__proto__;
if (resolved) { if (resolved) {
/** @type {Array<ts.ResolvedModuleWithFailedLookupLocations>} */ /** @type {Array<ts.ResolvedModuleWithFailedLookupLocations>} */
const result = resolved.map((item) => { const result = resolved.map((item) => {
if (item) { if (item && item[1]) {
const [resolvedFileName, extension] = item; const [resolvedFileName, extension] = item;
return { return {
resolvedModule: { resolvedModule: {

View file

@ -746,7 +746,7 @@ fn op_resolve(
state: &mut OpState, state: &mut OpState,
#[string] base: String, #[string] base: String,
#[serde] specifiers: Vec<(bool, String)>, #[serde] specifiers: Vec<(bool, String)>,
) -> Result<Vec<(String, &'static str)>, ResolveError> { ) -> Result<Vec<(String, Option<&'static str>)>, ResolveError> {
op_resolve_inner(state, ResolveArgs { base, specifiers }) op_resolve_inner(state, ResolveArgs { base, specifiers })
} }
@ -754,9 +754,9 @@ fn op_resolve(
fn op_resolve_inner( fn op_resolve_inner(
state: &mut OpState, state: &mut OpState,
args: ResolveArgs, args: ResolveArgs,
) -> Result<Vec<(String, &'static str)>, ResolveError> { ) -> Result<Vec<(String, Option<&'static str>)>, ResolveError> {
let state = state.borrow_mut::<State>(); let state = state.borrow_mut::<State>();
let mut resolved: Vec<(String, &'static str)> = let mut resolved: Vec<(String, Option<&'static str>)> =
Vec::with_capacity(args.specifiers.len()); Vec::with_capacity(args.specifiers.len());
let referrer = if let Some(remapped_specifier) = let referrer = if let Some(remapped_specifier) =
state.maybe_remapped_specifier(&args.base) state.maybe_remapped_specifier(&args.base)
@ -770,14 +770,14 @@ fn op_resolve_inner(
if specifier.starts_with("node:") { if specifier.starts_with("node:") {
resolved.push(( resolved.push((
MISSING_DEPENDENCY_SPECIFIER.to_string(), MISSING_DEPENDENCY_SPECIFIER.to_string(),
MediaType::Dts.as_ts_extension(), Some(MediaType::Dts.as_ts_extension()),
)); ));
continue; continue;
} }
if specifier.starts_with("asset:///") { if specifier.starts_with("asset:///") {
let ext = MediaType::from_str(&specifier).as_ts_extension(); let ext = MediaType::from_str(&specifier).as_ts_extension();
resolved.push((specifier, ext)); resolved.push((specifier, Some(ext)));
continue; continue;
} }
@ -857,14 +857,15 @@ fn op_resolve_inner(
( (
specifier_str, specifier_str,
match media_type { match media_type {
MediaType::Css => ".js", // surface these as .js for typescript MediaType::Css => Some(".js"), // surface these as .js for typescript
media_type => media_type.as_ts_extension(), MediaType::Unknown => None,
media_type => Some(media_type.as_ts_extension()),
}, },
) )
} }
None => ( None => (
MISSING_DEPENDENCY_SPECIFIER.to_string(), MISSING_DEPENDENCY_SPECIFIER.to_string(),
MediaType::Dts.as_ts_extension(), Some(MediaType::Dts.as_ts_extension()),
), ),
}; };
log::debug!("Resolved {} from {} to {:?}", specifier, referrer, result); log::debug!("Resolved {} from {} to {:?}", specifier, referrer, result);
@ -1441,7 +1442,10 @@ mod tests {
}, },
) )
.expect("should have invoked op"); .expect("should have invoked op");
assert_eq!(actual, vec![("https://deno.land/x/b.ts".into(), ".ts")]); assert_eq!(
actual,
vec![("https://deno.land/x/b.ts".into(), Some(".ts"))]
);
} }
#[tokio::test] #[tokio::test]
@ -1460,7 +1464,10 @@ mod tests {
}, },
) )
.expect("should have not errored"); .expect("should have not errored");
assert_eq!(actual, vec![(MISSING_DEPENDENCY_SPECIFIER.into(), ".d.ts")]); assert_eq!(
actual,
vec![(MISSING_DEPENDENCY_SPECIFIER.into(), Some(".d.ts"))]
);
} }
#[tokio::test] #[tokio::test]

View file

@ -17221,3 +17221,38 @@ fn lsp_wasm_module() {
); );
client.shutdown(); client.shutdown();
} }
#[test]
fn wildcard_augment() {
let context = TestContextBuilder::new().use_temp_cwd().build();
let mut client = context.new_lsp_command().build();
let temp_dir = context.temp_dir().path();
let source = source_file(
temp_dir.join("index.ts"),
r#"
import styles from "./hello_world.scss";
function bar(v: string): string {
return v;
}
bar(styles);
"#,
);
temp_dir.join("index.d.ts").write(
r#"
declare module '*.scss' {
const content: string;
export default content;
}
"#,
);
temp_dir
.join("hello_world.scss")
.write("body { color: red; }");
client.initialize_default();
let diagnostics = client.did_open_file(&source);
assert_eq!(diagnostics.all().len(), 0);
}