mirror of
https://github.com/denoland/deno.git
synced 2025-01-18 11:53:59 -05:00
2bbfef137c
This resurrects the `--unstable-detect-cjs` flag (which became stable), and repurposes it to attempt loading .js/.jsx/.ts/.tsx files as CJS in the following additional scenarios: 1. There is no package.json 1. There is a package.json without a "type" field Also cleans up the implementation of this in the LSP a lot by hanging `resolution_mode()` off `Document` (didn't think about doing that until now).
277 lines
8.5 KiB
Rust
277 lines
8.5 KiB
Rust
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
|
|
use std::sync::Arc;
|
|
|
|
use dashmap::DashMap;
|
|
use deno_media_type::MediaType;
|
|
use node_resolver::env::NodeResolverEnv;
|
|
use node_resolver::errors::ClosestPkgJsonError;
|
|
use node_resolver::InNpmPackageChecker;
|
|
use node_resolver::PackageJsonResolver;
|
|
use node_resolver::ResolutionMode;
|
|
use url::Url;
|
|
|
|
/// Keeps track of what module specifiers were resolved as CJS.
|
|
///
|
|
/// Modules that are `.js`, `.ts`, `.jsx`, and `tsx` are only known to
|
|
/// be CJS or ESM after they're loaded based on their contents. So these
|
|
/// files will be "maybe CJS" until they're loaded.
|
|
#[derive(Debug)]
|
|
pub struct CjsTracker<TEnv: NodeResolverEnv> {
|
|
is_cjs_resolver: IsCjsResolver<TEnv>,
|
|
known: DashMap<Url, ResolutionMode>,
|
|
}
|
|
|
|
impl<TEnv: NodeResolverEnv> CjsTracker<TEnv> {
|
|
pub fn new(
|
|
in_npm_pkg_checker: Arc<dyn InNpmPackageChecker>,
|
|
pkg_json_resolver: Arc<PackageJsonResolver<TEnv>>,
|
|
mode: IsCjsResolutionMode,
|
|
) -> Self {
|
|
Self {
|
|
is_cjs_resolver: IsCjsResolver::new(
|
|
in_npm_pkg_checker,
|
|
pkg_json_resolver,
|
|
mode,
|
|
),
|
|
known: Default::default(),
|
|
}
|
|
}
|
|
|
|
/// Checks whether the file might be treated as CJS, but it's not for sure
|
|
/// yet because the source hasn't been loaded to see whether it contains
|
|
/// imports or exports.
|
|
pub fn is_maybe_cjs(
|
|
&self,
|
|
specifier: &Url,
|
|
media_type: MediaType,
|
|
) -> Result<bool, ClosestPkgJsonError> {
|
|
self.treat_as_cjs_with_is_script(specifier, media_type, None)
|
|
}
|
|
|
|
/// Gets whether the file is CJS. If true, this is for sure
|
|
/// cjs because `is_script` is provided.
|
|
///
|
|
/// `is_script` should be `true` when the contents of the file at the
|
|
/// provided specifier are known to be a script and not an ES module.
|
|
pub fn is_cjs_with_known_is_script(
|
|
&self,
|
|
specifier: &Url,
|
|
media_type: MediaType,
|
|
is_script: bool,
|
|
) -> Result<bool, ClosestPkgJsonError> {
|
|
self.treat_as_cjs_with_is_script(specifier, media_type, Some(is_script))
|
|
}
|
|
|
|
fn treat_as_cjs_with_is_script(
|
|
&self,
|
|
specifier: &Url,
|
|
media_type: MediaType,
|
|
is_script: Option<bool>,
|
|
) -> Result<bool, ClosestPkgJsonError> {
|
|
let kind = match self
|
|
.get_known_mode_with_is_script(specifier, media_type, is_script)
|
|
{
|
|
Some(kind) => kind,
|
|
None => self.is_cjs_resolver.check_based_on_pkg_json(specifier)?,
|
|
};
|
|
Ok(kind == ResolutionMode::Require)
|
|
}
|
|
|
|
/// Gets the referrer for the specified module specifier.
|
|
///
|
|
/// Generally the referrer should already be tracked by calling
|
|
/// `is_cjs_with_known_is_script` before calling this method.
|
|
pub fn get_referrer_kind(&self, specifier: &Url) -> ResolutionMode {
|
|
if specifier.scheme() != "file" {
|
|
return ResolutionMode::Import;
|
|
}
|
|
self
|
|
.get_known_mode(specifier, MediaType::from_specifier(specifier))
|
|
.unwrap_or(ResolutionMode::Import)
|
|
}
|
|
|
|
fn get_known_mode(
|
|
&self,
|
|
specifier: &Url,
|
|
media_type: MediaType,
|
|
) -> Option<ResolutionMode> {
|
|
self.get_known_mode_with_is_script(specifier, media_type, None)
|
|
}
|
|
|
|
fn get_known_mode_with_is_script(
|
|
&self,
|
|
specifier: &Url,
|
|
media_type: MediaType,
|
|
is_script: Option<bool>,
|
|
) -> Option<ResolutionMode> {
|
|
self.is_cjs_resolver.get_known_mode_with_is_script(
|
|
specifier,
|
|
media_type,
|
|
is_script,
|
|
&self.known,
|
|
)
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
|
pub enum IsCjsResolutionMode {
|
|
/// Requires an explicit `"type": "commonjs"` in the package.json.
|
|
ExplicitTypeCommonJs,
|
|
/// Implicitly uses `"type": "commonjs"` if no `"type"` is specified.
|
|
ImplicitTypeCommonJs,
|
|
/// Does not respect `"type": "commonjs"` and always treats ambiguous files as ESM.
|
|
Disabled,
|
|
}
|
|
|
|
/// Resolves whether a module is CJS or ESM.
|
|
#[derive(Debug)]
|
|
pub struct IsCjsResolver<TEnv: NodeResolverEnv> {
|
|
in_npm_pkg_checker: Arc<dyn InNpmPackageChecker>,
|
|
pkg_json_resolver: Arc<PackageJsonResolver<TEnv>>,
|
|
mode: IsCjsResolutionMode,
|
|
}
|
|
|
|
impl<TEnv: NodeResolverEnv> IsCjsResolver<TEnv> {
|
|
pub fn new(
|
|
in_npm_pkg_checker: Arc<dyn InNpmPackageChecker>,
|
|
pkg_json_resolver: Arc<PackageJsonResolver<TEnv>>,
|
|
mode: IsCjsResolutionMode,
|
|
) -> Self {
|
|
Self {
|
|
in_npm_pkg_checker,
|
|
pkg_json_resolver,
|
|
mode,
|
|
}
|
|
}
|
|
|
|
/// Gets the resolution mode for a module in the LSP.
|
|
pub fn get_lsp_resolution_mode(
|
|
&self,
|
|
specifier: &Url,
|
|
is_script: Option<bool>,
|
|
) -> ResolutionMode {
|
|
if specifier.scheme() != "file" {
|
|
return ResolutionMode::Import;
|
|
}
|
|
match MediaType::from_specifier(specifier) {
|
|
MediaType::Mts | MediaType::Mjs | MediaType::Dmts => ResolutionMode::Import,
|
|
MediaType::Cjs | MediaType::Cts | MediaType::Dcts => ResolutionMode::Require,
|
|
MediaType::Dts => {
|
|
// dts files are always determined based on the package.json because
|
|
// they contain imports/exports even when considered CJS
|
|
self.check_based_on_pkg_json(specifier).unwrap_or(ResolutionMode::Import)
|
|
}
|
|
MediaType::Wasm |
|
|
MediaType::Json => ResolutionMode::Import,
|
|
MediaType::JavaScript
|
|
| MediaType::Jsx
|
|
| MediaType::TypeScript
|
|
| MediaType::Tsx
|
|
// treat these as unknown
|
|
| MediaType::Css
|
|
| MediaType::SourceMap
|
|
| MediaType::Unknown => {
|
|
match is_script {
|
|
Some(true) => self.check_based_on_pkg_json(specifier).unwrap_or(ResolutionMode::Import),
|
|
Some(false) | None => ResolutionMode::Import,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn get_known_mode_with_is_script(
|
|
&self,
|
|
specifier: &Url,
|
|
media_type: MediaType,
|
|
is_script: Option<bool>,
|
|
known_cache: &DashMap<Url, ResolutionMode>,
|
|
) -> Option<ResolutionMode> {
|
|
if specifier.scheme() != "file" {
|
|
return Some(ResolutionMode::Import);
|
|
}
|
|
|
|
match media_type {
|
|
MediaType::Mts | MediaType::Mjs | MediaType::Dmts => Some(ResolutionMode::Import),
|
|
MediaType::Cjs | MediaType::Cts | MediaType::Dcts => Some(ResolutionMode::Require),
|
|
MediaType::Dts => {
|
|
// dts files are always determined based on the package.json because
|
|
// they contain imports/exports even when considered CJS
|
|
if let Some(value) = known_cache.get(specifier).map(|v| *v) {
|
|
Some(value)
|
|
} else {
|
|
let value = self.check_based_on_pkg_json(specifier).ok();
|
|
if let Some(value) = value {
|
|
known_cache.insert(specifier.clone(), value);
|
|
}
|
|
Some(value.unwrap_or(ResolutionMode::Import))
|
|
}
|
|
}
|
|
MediaType::Wasm |
|
|
MediaType::Json => Some(ResolutionMode::Import),
|
|
MediaType::JavaScript
|
|
| MediaType::Jsx
|
|
| MediaType::TypeScript
|
|
| MediaType::Tsx
|
|
// treat these as unknown
|
|
| MediaType::Css
|
|
| MediaType::SourceMap
|
|
| MediaType::Unknown => {
|
|
if let Some(value) = known_cache.get(specifier).map(|v| *v) {
|
|
if value == ResolutionMode::Require && is_script == Some(false) {
|
|
// we now know this is actually esm
|
|
known_cache.insert(specifier.clone(), ResolutionMode::Import);
|
|
Some(ResolutionMode::Import)
|
|
} else {
|
|
Some(value)
|
|
}
|
|
} else if is_script == Some(false) {
|
|
// we know this is esm
|
|
known_cache.insert(specifier.clone(), ResolutionMode::Import);
|
|
Some(ResolutionMode::Import)
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn check_based_on_pkg_json(
|
|
&self,
|
|
specifier: &Url,
|
|
) -> Result<ResolutionMode, ClosestPkgJsonError> {
|
|
if self.in_npm_pkg_checker.in_npm_package(specifier) {
|
|
if let Some(pkg_json) =
|
|
self.pkg_json_resolver.get_closest_package_json(specifier)?
|
|
{
|
|
let is_file_location_cjs = pkg_json.typ != "module";
|
|
Ok(if is_file_location_cjs {
|
|
ResolutionMode::Require
|
|
} else {
|
|
ResolutionMode::Import
|
|
})
|
|
} else {
|
|
Ok(ResolutionMode::Require)
|
|
}
|
|
} else if self.mode != IsCjsResolutionMode::Disabled {
|
|
if let Some(pkg_json) =
|
|
self.pkg_json_resolver.get_closest_package_json(specifier)?
|
|
{
|
|
let is_cjs_type = pkg_json.typ == "commonjs"
|
|
|| self.mode == IsCjsResolutionMode::ImplicitTypeCommonJs
|
|
&& pkg_json.typ == "none";
|
|
Ok(if is_cjs_type {
|
|
ResolutionMode::Require
|
|
} else {
|
|
ResolutionMode::Import
|
|
})
|
|
} else if self.mode == IsCjsResolutionMode::ImplicitTypeCommonJs {
|
|
Ok(ResolutionMode::Require)
|
|
} else {
|
|
Ok(ResolutionMode::Import)
|
|
}
|
|
} else {
|
|
Ok(ResolutionMode::Import)
|
|
}
|
|
}
|
|
}
|