mirror of
https://github.com/denoland/deno.git
synced 2025-01-12 00:54:02 -05:00
fix(npm): improved optional dependency support (#19135)
Note: If the package information has already been cached, then this requires running with `--reload` or for the registry information to be fetched some other way (ex. the cache busting). Closes #15544 --------- Co-authored-by: Bartek Iwańczuk <biwanczuk@gmail.com>
This commit is contained in:
parent
ad22336245
commit
41f618a1df
25 changed files with 300 additions and 61 deletions
5
Cargo.lock
generated
5
Cargo.lock
generated
|
@ -1217,9 +1217,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "deno_npm"
|
name = "deno_npm"
|
||||||
version = "0.3.0"
|
version = "0.4.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "eab571ab55247090bb78bc0466fdf01bd331c98e9e2c4a236bfa93479d906c5c"
|
checksum = "007cdc59079448de7af94510f7e9652bc6f17e9029741b26f3754f86a3cc2dd0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
|
@ -1228,7 +1228,6 @@ dependencies = [
|
||||||
"log",
|
"log",
|
||||||
"monch",
|
"monch",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"parking_lot 0.12.1",
|
|
||||||
"serde",
|
"serde",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
]
|
]
|
||||||
|
|
|
@ -53,7 +53,7 @@ deno_bench_util = { version = "0.98.0", path = "./bench_util" }
|
||||||
test_util = { path = "./test_util" }
|
test_util = { path = "./test_util" }
|
||||||
deno_lockfile = "0.14.0"
|
deno_lockfile = "0.14.0"
|
||||||
deno_media_type = { version = "0.1.0", features = ["module_specifier"] }
|
deno_media_type = { version = "0.1.0", features = ["module_specifier"] }
|
||||||
deno_npm = "0.3.0"
|
deno_npm = "0.4.0"
|
||||||
deno_semver = "0.2.1"
|
deno_semver = "0.2.1"
|
||||||
|
|
||||||
# exts
|
# exts
|
||||||
|
|
|
@ -107,8 +107,12 @@ pub async fn snapshot_from_lockfile(
|
||||||
|
|
||||||
packages.push(SerializedNpmResolutionSnapshotPackage {
|
packages.push(SerializedNpmResolutionSnapshotPackage {
|
||||||
pkg_id,
|
pkg_id,
|
||||||
dist: Default::default(), // temporarily empty
|
|
||||||
dependencies,
|
dependencies,
|
||||||
|
// temporarily empty
|
||||||
|
os: Default::default(),
|
||||||
|
cpu: Default::default(),
|
||||||
|
dist: Default::default(),
|
||||||
|
optional_dependencies: Default::default(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
(root_packages, packages)
|
(root_packages, packages)
|
||||||
|
@ -131,11 +135,17 @@ pub async fn snapshot_from_lockfile(
|
||||||
}))
|
}))
|
||||||
};
|
};
|
||||||
let mut version_infos = get_version_infos();
|
let mut version_infos = get_version_infos();
|
||||||
|
|
||||||
let mut i = 0;
|
let mut i = 0;
|
||||||
while let Some(result) = version_infos.next().await {
|
while let Some(result) = version_infos.next().await {
|
||||||
packages[i].dist = match result {
|
match result {
|
||||||
Ok(version_info) => version_info.dist,
|
Ok(version_info) => {
|
||||||
|
let mut package = &mut packages[i];
|
||||||
|
package.dist = version_info.dist;
|
||||||
|
package.cpu = version_info.cpu;
|
||||||
|
package.os = version_info.os;
|
||||||
|
package.optional_dependencies =
|
||||||
|
version_info.optional_dependencies.into_keys().collect();
|
||||||
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
if api.mark_force_reload() {
|
if api.mark_force_reload() {
|
||||||
// reset and try again
|
// reset and try again
|
||||||
|
@ -146,7 +156,7 @@ pub async fn snapshot_from_lockfile(
|
||||||
return Err(err);
|
return Err(err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
i += 1;
|
i += 1;
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,6 +13,7 @@ use self::package_json::PackageJsonDeps;
|
||||||
use ::import_map::ImportMap;
|
use ::import_map::ImportMap;
|
||||||
use deno_core::resolve_url_or_path;
|
use deno_core::resolve_url_or_path;
|
||||||
use deno_npm::resolution::ValidSerializedNpmResolutionSnapshot;
|
use deno_npm::resolution::ValidSerializedNpmResolutionSnapshot;
|
||||||
|
use deno_npm::NpmSystemInfo;
|
||||||
use deno_runtime::deno_tls::RootCertStoreProvider;
|
use deno_runtime::deno_tls::RootCertStoreProvider;
|
||||||
use deno_semver::npm::NpmPackageReqReference;
|
use deno_semver::npm::NpmPackageReqReference;
|
||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
|
@ -688,6 +689,42 @@ impl CliOptions {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn npm_system_info(&self) -> NpmSystemInfo {
|
||||||
|
match self.sub_command() {
|
||||||
|
DenoSubcommand::Compile(CompileFlags {
|
||||||
|
target: Some(target),
|
||||||
|
..
|
||||||
|
}) => {
|
||||||
|
// the values of NpmSystemInfo align with the possible values for the
|
||||||
|
// `arch` and `platform` fields of Node.js' `process` global:
|
||||||
|
// https://nodejs.org/api/process.html
|
||||||
|
match target.as_str() {
|
||||||
|
"aarch64-apple-darwin" => NpmSystemInfo {
|
||||||
|
os: "darwin".to_string(),
|
||||||
|
cpu: "arm64".to_string(),
|
||||||
|
},
|
||||||
|
"x86_64-apple-darwin" => NpmSystemInfo {
|
||||||
|
os: "darwin".to_string(),
|
||||||
|
cpu: "x64".to_string(),
|
||||||
|
},
|
||||||
|
"x86_64-unknown-linux-gnu" => NpmSystemInfo {
|
||||||
|
os: "linux".to_string(),
|
||||||
|
cpu: "x64".to_string(),
|
||||||
|
},
|
||||||
|
"x86_64-pc-windows-msvc" => NpmSystemInfo {
|
||||||
|
os: "win32".to_string(),
|
||||||
|
cpu: "x64".to_string(),
|
||||||
|
},
|
||||||
|
value => {
|
||||||
|
log::warn!("Not implemented NPM system info for target '{value}'. Using current system default. This may impact NPM ");
|
||||||
|
NpmSystemInfo::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => NpmSystemInfo::default(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn resolve_deno_dir(&self) -> Result<DenoDir, AnyError> {
|
pub fn resolve_deno_dir(&self) -> Result<DenoDir, AnyError> {
|
||||||
Ok(DenoDir::new(self.maybe_custom_root())?)
|
Ok(DenoDir::new(self.maybe_custom_root())?)
|
||||||
}
|
}
|
||||||
|
|
|
@ -313,6 +313,7 @@ impl CliFactory {
|
||||||
CliNpmRegistryApi::default_url().to_owned(),
|
CliNpmRegistryApi::default_url().to_owned(),
|
||||||
npm_resolution.clone(),
|
npm_resolution.clone(),
|
||||||
self.options.node_modules_dir_path(),
|
self.options.node_modules_dir_path(),
|
||||||
|
self.options.npm_system_info(),
|
||||||
);
|
);
|
||||||
Ok(Arc::new(CliNpmResolver::new(
|
Ok(Arc::new(CliNpmResolver::new(
|
||||||
fs.clone(),
|
fs.clone(),
|
||||||
|
@ -557,8 +558,9 @@ impl CliFactory {
|
||||||
self.deno_dir()?,
|
self.deno_dir()?,
|
||||||
self.npm_api()?,
|
self.npm_api()?,
|
||||||
self.npm_cache()?,
|
self.npm_cache()?,
|
||||||
self.npm_resolver().await?,
|
|
||||||
self.npm_resolution().await?,
|
self.npm_resolution().await?,
|
||||||
|
self.npm_resolver().await?,
|
||||||
|
self.options.npm_system_info(),
|
||||||
self.package_json_deps_provider(),
|
self.package_json_deps_provider(),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@ use deno_core::serde_json::json;
|
||||||
use deno_core::serde_json::Value;
|
use deno_core::serde_json::Value;
|
||||||
use deno_core::task::spawn;
|
use deno_core::task::spawn;
|
||||||
use deno_core::ModuleSpecifier;
|
use deno_core::ModuleSpecifier;
|
||||||
|
use deno_npm::NpmSystemInfo;
|
||||||
use deno_runtime::deno_fs;
|
use deno_runtime::deno_fs;
|
||||||
use deno_runtime::deno_node::NodeResolver;
|
use deno_runtime::deno_node::NodeResolver;
|
||||||
use deno_runtime::deno_node::PackageJson;
|
use deno_runtime::deno_node::PackageJson;
|
||||||
|
@ -467,6 +468,7 @@ fn create_lsp_structs(
|
||||||
registry_url.clone(),
|
registry_url.clone(),
|
||||||
resolution.clone(),
|
resolution.clone(),
|
||||||
None,
|
None,
|
||||||
|
NpmSystemInfo::default(),
|
||||||
);
|
);
|
||||||
(
|
(
|
||||||
api,
|
api,
|
||||||
|
@ -725,6 +727,7 @@ impl Inner {
|
||||||
self.npm_api.base_url().clone(),
|
self.npm_api.base_url().clone(),
|
||||||
npm_resolution,
|
npm_resolution,
|
||||||
None,
|
None,
|
||||||
|
NpmSystemInfo::default(),
|
||||||
),
|
),
|
||||||
None,
|
None,
|
||||||
));
|
));
|
||||||
|
@ -1242,11 +1245,12 @@ impl Inner {
|
||||||
|
|
||||||
async fn refresh_npm_specifiers(&mut self) {
|
async fn refresh_npm_specifiers(&mut self) {
|
||||||
let package_reqs = self.documents.npm_package_reqs();
|
let package_reqs = self.documents.npm_package_reqs();
|
||||||
if let Err(err) = self
|
let npm_resolver = self.npm_resolver.clone();
|
||||||
.npm_resolver
|
// spawn to avoid the LSP's Send requirements
|
||||||
.set_package_reqs((*package_reqs).clone())
|
let handle = spawn(async move {
|
||||||
.await
|
npm_resolver.set_package_reqs((*package_reqs).clone()).await
|
||||||
{
|
});
|
||||||
|
if let Err(err) = handle.await.unwrap() {
|
||||||
lsp_warn!("Could not set npm package requirements. {:#}", err);
|
lsp_warn!("Could not set npm package requirements. {:#}", err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,6 +23,7 @@ use deno_npm::resolution::ValidSerializedNpmResolutionSnapshot;
|
||||||
use deno_npm::NpmPackageCacheFolderId;
|
use deno_npm::NpmPackageCacheFolderId;
|
||||||
use deno_npm::NpmPackageId;
|
use deno_npm::NpmPackageId;
|
||||||
use deno_npm::NpmResolutionPackage;
|
use deno_npm::NpmResolutionPackage;
|
||||||
|
use deno_npm::NpmSystemInfo;
|
||||||
use deno_semver::npm::NpmPackageNv;
|
use deno_semver::npm::NpmPackageNv;
|
||||||
use deno_semver::npm::NpmPackageNvReference;
|
use deno_semver::npm::NpmPackageNvReference;
|
||||||
use deno_semver::npm::NpmPackageReq;
|
use deno_semver::npm::NpmPackageReq;
|
||||||
|
@ -237,12 +238,21 @@ impl NpmResolution {
|
||||||
Ok(nv)
|
Ok(nv)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn all_packages(&self) -> Vec<NpmResolutionPackage> {
|
pub fn all_system_packages(
|
||||||
self.snapshot.read().all_packages()
|
&self,
|
||||||
|
system_info: &NpmSystemInfo,
|
||||||
|
) -> Vec<NpmResolutionPackage> {
|
||||||
|
self.snapshot.read().all_system_packages(system_info)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn all_packages_partitioned(&self) -> NpmPackagesPartitioned {
|
pub fn all_system_packages_partitioned(
|
||||||
self.snapshot.read().all_packages_partitioned()
|
&self,
|
||||||
|
system_info: &NpmSystemInfo,
|
||||||
|
) -> NpmPackagesPartitioned {
|
||||||
|
self
|
||||||
|
.snapshot
|
||||||
|
.read()
|
||||||
|
.all_system_packages_partitioned(system_info)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn has_packages(&self) -> bool {
|
pub fn has_packages(&self) -> bool {
|
||||||
|
@ -322,7 +332,7 @@ fn populate_lockfile_from_snapshot(
|
||||||
.as_serialized(),
|
.as_serialized(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
for package in snapshot.all_packages() {
|
for package in snapshot.all_packages_for_every_system() {
|
||||||
lockfile
|
lockfile
|
||||||
.check_or_insert_npm_package(npm_package_to_lockfile_info(package))?;
|
.check_or_insert_npm_package(npm_package_to_lockfile_info(package))?;
|
||||||
}
|
}
|
||||||
|
@ -330,13 +340,13 @@ fn populate_lockfile_from_snapshot(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn npm_package_to_lockfile_info(
|
fn npm_package_to_lockfile_info(
|
||||||
pkg: NpmResolutionPackage,
|
pkg: &NpmResolutionPackage,
|
||||||
) -> NpmPackageLockfileInfo {
|
) -> NpmPackageLockfileInfo {
|
||||||
let dependencies = pkg
|
let dependencies = pkg
|
||||||
.dependencies
|
.dependencies
|
||||||
.into_iter()
|
.iter()
|
||||||
.map(|(name, id)| NpmPackageDependencyLockfileInfo {
|
.map(|(name, id)| NpmPackageDependencyLockfileInfo {
|
||||||
name,
|
name: name.clone(),
|
||||||
id: id.as_serialized(),
|
id: id.as_serialized(),
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
|
@ -69,7 +69,6 @@ pub async fn cache_packages(
|
||||||
|
|
||||||
let mut handles = Vec::with_capacity(packages.len());
|
let mut handles = Vec::with_capacity(packages.len());
|
||||||
for package in packages {
|
for package in packages {
|
||||||
assert_eq!(package.copy_index, 0); // the caller should not provide any of these
|
|
||||||
let cache = cache.clone();
|
let cache = cache.clone();
|
||||||
let registry_url = registry_url.clone();
|
let registry_url = registry_url.clone();
|
||||||
let handle = spawn(async move {
|
let handle = spawn(async move {
|
||||||
|
|
|
@ -14,6 +14,7 @@ use deno_npm::resolution::PackageNotFoundFromReferrerError;
|
||||||
use deno_npm::NpmPackageCacheFolderId;
|
use deno_npm::NpmPackageCacheFolderId;
|
||||||
use deno_npm::NpmPackageId;
|
use deno_npm::NpmPackageId;
|
||||||
use deno_npm::NpmResolutionPackage;
|
use deno_npm::NpmResolutionPackage;
|
||||||
|
use deno_npm::NpmSystemInfo;
|
||||||
use deno_runtime::deno_fs::FileSystem;
|
use deno_runtime::deno_fs::FileSystem;
|
||||||
use deno_runtime::deno_node::NodePermissions;
|
use deno_runtime::deno_node::NodePermissions;
|
||||||
use deno_runtime::deno_node::NodeResolutionMode;
|
use deno_runtime::deno_node::NodeResolutionMode;
|
||||||
|
@ -33,6 +34,7 @@ pub struct GlobalNpmPackageResolver {
|
||||||
cache: Arc<NpmCache>,
|
cache: Arc<NpmCache>,
|
||||||
resolution: Arc<NpmResolution>,
|
resolution: Arc<NpmResolution>,
|
||||||
registry_url: Url,
|
registry_url: Url,
|
||||||
|
system_info: NpmSystemInfo,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GlobalNpmPackageResolver {
|
impl GlobalNpmPackageResolver {
|
||||||
|
@ -41,12 +43,14 @@ impl GlobalNpmPackageResolver {
|
||||||
cache: Arc<NpmCache>,
|
cache: Arc<NpmCache>,
|
||||||
registry_url: Url,
|
registry_url: Url,
|
||||||
resolution: Arc<NpmResolution>,
|
resolution: Arc<NpmResolution>,
|
||||||
|
system_info: NpmSystemInfo,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
fs,
|
fs,
|
||||||
cache,
|
cache,
|
||||||
resolution,
|
resolution,
|
||||||
registry_url,
|
registry_url,
|
||||||
|
system_info,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -125,7 +129,26 @@ impl NpmPackageFsResolver for GlobalNpmPackageResolver {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn cache_packages(&self) -> Result<(), AnyError> {
|
async fn cache_packages(&self) -> Result<(), AnyError> {
|
||||||
cache_packages_in_resolver(self).await
|
let package_partitions = self
|
||||||
|
.resolution
|
||||||
|
.all_system_packages_partitioned(&self.system_info);
|
||||||
|
|
||||||
|
cache_packages(
|
||||||
|
package_partitions.packages,
|
||||||
|
&self.cache,
|
||||||
|
&self.registry_url,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
// create the copy package folders
|
||||||
|
for copy in package_partitions.copy_packages {
|
||||||
|
self.cache.ensure_copy_package(
|
||||||
|
©.get_package_cache_folder_id(),
|
||||||
|
&self.registry_url,
|
||||||
|
)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn ensure_read_permission(
|
fn ensure_read_permission(
|
||||||
|
@ -137,26 +160,3 @@ impl NpmPackageFsResolver for GlobalNpmPackageResolver {
|
||||||
ensure_registry_read_permission(&self.fs, permissions, ®istry_path, path)
|
ensure_registry_read_permission(&self.fs, permissions, ®istry_path, path)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn cache_packages_in_resolver(
|
|
||||||
resolver: &GlobalNpmPackageResolver,
|
|
||||||
) -> Result<(), AnyError> {
|
|
||||||
let package_partitions = resolver.resolution.all_packages_partitioned();
|
|
||||||
|
|
||||||
cache_packages(
|
|
||||||
package_partitions.packages,
|
|
||||||
&resolver.cache,
|
|
||||||
&resolver.registry_url,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
// create the copy package folders
|
|
||||||
for copy in package_partitions.copy_packages {
|
|
||||||
resolver.cache.ensure_copy_package(
|
|
||||||
©.get_package_cache_folder_id(),
|
|
||||||
&resolver.registry_url,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
|
@ -24,6 +24,7 @@ use deno_core::url::Url;
|
||||||
use deno_npm::resolution::NpmResolutionSnapshot;
|
use deno_npm::resolution::NpmResolutionSnapshot;
|
||||||
use deno_npm::NpmPackageCacheFolderId;
|
use deno_npm::NpmPackageCacheFolderId;
|
||||||
use deno_npm::NpmPackageId;
|
use deno_npm::NpmPackageId;
|
||||||
|
use deno_npm::NpmSystemInfo;
|
||||||
use deno_runtime::deno_core::futures;
|
use deno_runtime::deno_core::futures;
|
||||||
use deno_runtime::deno_fs;
|
use deno_runtime::deno_fs;
|
||||||
use deno_runtime::deno_node::NodePermissions;
|
use deno_runtime::deno_node::NodePermissions;
|
||||||
|
@ -52,6 +53,7 @@ pub struct LocalNpmPackageResolver {
|
||||||
registry_url: Url,
|
registry_url: Url,
|
||||||
root_node_modules_path: PathBuf,
|
root_node_modules_path: PathBuf,
|
||||||
root_node_modules_url: Url,
|
root_node_modules_url: Url,
|
||||||
|
system_info: NpmSystemInfo,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl LocalNpmPackageResolver {
|
impl LocalNpmPackageResolver {
|
||||||
|
@ -62,6 +64,7 @@ impl LocalNpmPackageResolver {
|
||||||
registry_url: Url,
|
registry_url: Url,
|
||||||
node_modules_folder: PathBuf,
|
node_modules_folder: PathBuf,
|
||||||
resolution: Arc<NpmResolution>,
|
resolution: Arc<NpmResolution>,
|
||||||
|
system_info: NpmSystemInfo,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
fs,
|
fs,
|
||||||
|
@ -72,6 +75,7 @@ impl LocalNpmPackageResolver {
|
||||||
root_node_modules_url: Url::from_directory_path(&node_modules_folder)
|
root_node_modules_url: Url::from_directory_path(&node_modules_folder)
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
root_node_modules_path: node_modules_folder,
|
root_node_modules_path: node_modules_folder,
|
||||||
|
system_info,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -205,6 +209,7 @@ impl NpmPackageFsResolver for LocalNpmPackageResolver {
|
||||||
&self.progress_bar,
|
&self.progress_bar,
|
||||||
&self.registry_url,
|
&self.registry_url,
|
||||||
&self.root_node_modules_path,
|
&self.root_node_modules_path,
|
||||||
|
&self.system_info,
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
|
@ -230,6 +235,7 @@ async fn sync_resolution_with_fs(
|
||||||
progress_bar: &ProgressBar,
|
progress_bar: &ProgressBar,
|
||||||
registry_url: &Url,
|
registry_url: &Url,
|
||||||
root_node_modules_dir_path: &Path,
|
root_node_modules_dir_path: &Path,
|
||||||
|
system_info: &NpmSystemInfo,
|
||||||
) -> Result<(), AnyError> {
|
) -> Result<(), AnyError> {
|
||||||
if snapshot.is_empty() {
|
if snapshot.is_empty() {
|
||||||
return Ok(()); // don't create the directory
|
return Ok(()); // don't create the directory
|
||||||
|
@ -254,7 +260,8 @@ async fn sync_resolution_with_fs(
|
||||||
// Copy (hardlink in future) <global_registry_cache>/<package_id>/ to
|
// Copy (hardlink in future) <global_registry_cache>/<package_id>/ to
|
||||||
// node_modules/.deno/<package_folder_id_folder_name>/node_modules/<package_name>
|
// node_modules/.deno/<package_folder_id_folder_name>/node_modules/<package_name>
|
||||||
let sync_download = should_sync_download();
|
let sync_download = should_sync_download();
|
||||||
let mut package_partitions = snapshot.all_packages_partitioned();
|
let mut package_partitions =
|
||||||
|
snapshot.all_system_packages_partitioned(system_info);
|
||||||
if sync_download {
|
if sync_download {
|
||||||
// we're running the tests not with --quiet
|
// we're running the tests not with --quiet
|
||||||
// and we want the output to be deterministic
|
// and we want the output to be deterministic
|
||||||
|
|
|
@ -18,6 +18,7 @@ use deno_npm::resolution::NpmResolutionSnapshot;
|
||||||
use deno_npm::resolution::PackageReqNotFoundError;
|
use deno_npm::resolution::PackageReqNotFoundError;
|
||||||
use deno_npm::resolution::SerializedNpmResolutionSnapshot;
|
use deno_npm::resolution::SerializedNpmResolutionSnapshot;
|
||||||
use deno_npm::NpmPackageId;
|
use deno_npm::NpmPackageId;
|
||||||
|
use deno_npm::NpmSystemInfo;
|
||||||
use deno_runtime::deno_fs::FileSystem;
|
use deno_runtime::deno_fs::FileSystem;
|
||||||
use deno_runtime::deno_node::NodePermissions;
|
use deno_runtime::deno_node::NodePermissions;
|
||||||
use deno_runtime::deno_node::NodeResolutionMode;
|
use deno_runtime::deno_node::NodeResolutionMode;
|
||||||
|
@ -289,6 +290,7 @@ pub fn create_npm_fs_resolver(
|
||||||
registry_url: Url,
|
registry_url: Url,
|
||||||
resolution: Arc<NpmResolution>,
|
resolution: Arc<NpmResolution>,
|
||||||
maybe_node_modules_path: Option<PathBuf>,
|
maybe_node_modules_path: Option<PathBuf>,
|
||||||
|
system_info: NpmSystemInfo,
|
||||||
) -> Arc<dyn NpmPackageFsResolver> {
|
) -> Arc<dyn NpmPackageFsResolver> {
|
||||||
match maybe_node_modules_path {
|
match maybe_node_modules_path {
|
||||||
Some(node_modules_folder) => Arc::new(LocalNpmPackageResolver::new(
|
Some(node_modules_folder) => Arc::new(LocalNpmPackageResolver::new(
|
||||||
|
@ -298,12 +300,14 @@ pub fn create_npm_fs_resolver(
|
||||||
registry_url,
|
registry_url,
|
||||||
node_modules_folder,
|
node_modules_folder,
|
||||||
resolution,
|
resolution,
|
||||||
|
system_info,
|
||||||
)),
|
)),
|
||||||
None => Arc::new(GlobalNpmPackageResolver::new(
|
None => Arc::new(GlobalNpmPackageResolver::new(
|
||||||
fs,
|
fs,
|
||||||
cache,
|
cache,
|
||||||
registry_url,
|
registry_url,
|
||||||
resolution,
|
resolution,
|
||||||
|
system_info,
|
||||||
)),
|
)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,7 @@ use deno_core::serde_json;
|
||||||
use deno_core::url::Url;
|
use deno_core::url::Url;
|
||||||
use deno_npm::registry::PackageDepNpmSchemeValueParseError;
|
use deno_npm::registry::PackageDepNpmSchemeValueParseError;
|
||||||
use deno_npm::resolution::SerializedNpmResolutionSnapshot;
|
use deno_npm::resolution::SerializedNpmResolutionSnapshot;
|
||||||
|
use deno_npm::NpmSystemInfo;
|
||||||
use deno_runtime::permissions::PermissionsOptions;
|
use deno_runtime::permissions::PermissionsOptions;
|
||||||
use deno_semver::npm::NpmPackageReq;
|
use deno_semver::npm::NpmPackageReq;
|
||||||
use deno_semver::npm::NpmVersionReqSpecifierParseError;
|
use deno_semver::npm::NpmVersionReqSpecifierParseError;
|
||||||
|
@ -343,8 +344,9 @@ pub struct DenoCompileBinaryWriter<'a> {
|
||||||
deno_dir: &'a DenoDir,
|
deno_dir: &'a DenoDir,
|
||||||
npm_api: &'a CliNpmRegistryApi,
|
npm_api: &'a CliNpmRegistryApi,
|
||||||
npm_cache: &'a NpmCache,
|
npm_cache: &'a NpmCache,
|
||||||
|
npm_resolution: &'a NpmResolution,
|
||||||
npm_resolver: &'a CliNpmResolver,
|
npm_resolver: &'a CliNpmResolver,
|
||||||
resolution: &'a NpmResolution,
|
npm_system_info: NpmSystemInfo,
|
||||||
package_json_deps_provider: &'a PackageJsonDepsProvider,
|
package_json_deps_provider: &'a PackageJsonDepsProvider,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -356,8 +358,9 @@ impl<'a> DenoCompileBinaryWriter<'a> {
|
||||||
deno_dir: &'a DenoDir,
|
deno_dir: &'a DenoDir,
|
||||||
npm_api: &'a CliNpmRegistryApi,
|
npm_api: &'a CliNpmRegistryApi,
|
||||||
npm_cache: &'a NpmCache,
|
npm_cache: &'a NpmCache,
|
||||||
|
npm_resolution: &'a NpmResolution,
|
||||||
npm_resolver: &'a CliNpmResolver,
|
npm_resolver: &'a CliNpmResolver,
|
||||||
resolution: &'a NpmResolution,
|
npm_system_info: NpmSystemInfo,
|
||||||
package_json_deps_provider: &'a PackageJsonDepsProvider,
|
package_json_deps_provider: &'a PackageJsonDepsProvider,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
|
@ -367,7 +370,8 @@ impl<'a> DenoCompileBinaryWriter<'a> {
|
||||||
npm_api,
|
npm_api,
|
||||||
npm_cache,
|
npm_cache,
|
||||||
npm_resolver,
|
npm_resolver,
|
||||||
resolution,
|
npm_system_info,
|
||||||
|
npm_resolution,
|
||||||
package_json_deps_provider,
|
package_json_deps_provider,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -488,9 +492,10 @@ impl<'a> DenoCompileBinaryWriter<'a> {
|
||||||
.resolve_import_map(self.file_fetcher)
|
.resolve_import_map(self.file_fetcher)
|
||||||
.await?
|
.await?
|
||||||
.map(|import_map| (import_map.base_url().clone(), import_map.to_json()));
|
.map(|import_map| (import_map.base_url().clone(), import_map.to_json()));
|
||||||
let (npm_snapshot, npm_vfs, npm_files) = if self.resolution.has_packages() {
|
let (npm_snapshot, npm_vfs, npm_files) =
|
||||||
|
if self.npm_resolution.has_packages() {
|
||||||
let (root_dir, files) = self.build_vfs()?.into_dir_and_files();
|
let (root_dir, files) = self.build_vfs()?.into_dir_and_files();
|
||||||
let snapshot = self.resolution.serialized_snapshot();
|
let snapshot = self.npm_resolution.serialized_snapshot();
|
||||||
(Some(snapshot), Some(root_dir), files)
|
(Some(snapshot), Some(root_dir), files)
|
||||||
} else {
|
} else {
|
||||||
(None, None, Vec::new())
|
(None, None, Vec::new())
|
||||||
|
@ -540,7 +545,10 @@ impl<'a> DenoCompileBinaryWriter<'a> {
|
||||||
let registry_url = self.npm_api.base_url();
|
let registry_url = self.npm_api.base_url();
|
||||||
let root_path = self.npm_cache.registry_folder(registry_url);
|
let root_path = self.npm_cache.registry_folder(registry_url);
|
||||||
let mut builder = VfsBuilder::new(root_path);
|
let mut builder = VfsBuilder::new(root_path);
|
||||||
for package in self.resolution.all_packages() {
|
for package in self
|
||||||
|
.npm_resolution
|
||||||
|
.all_system_packages(&self.npm_system_info)
|
||||||
|
{
|
||||||
let folder = self
|
let folder = self
|
||||||
.npm_resolver
|
.npm_resolver
|
||||||
.resolve_pkg_folder_from_pkg_id(&package.pkg_id)?;
|
.resolve_pkg_folder_from_pkg_id(&package.pkg_id)?;
|
||||||
|
|
|
@ -37,6 +37,7 @@ use deno_core::ModuleLoader;
|
||||||
use deno_core::ModuleSpecifier;
|
use deno_core::ModuleSpecifier;
|
||||||
use deno_core::ModuleType;
|
use deno_core::ModuleType;
|
||||||
use deno_core::ResolutionKind;
|
use deno_core::ResolutionKind;
|
||||||
|
use deno_npm::NpmSystemInfo;
|
||||||
use deno_runtime::deno_fs;
|
use deno_runtime::deno_fs;
|
||||||
use deno_runtime::deno_node;
|
use deno_runtime::deno_node;
|
||||||
use deno_runtime::deno_node::analyze::NodeCodeTranslator;
|
use deno_runtime::deno_node::analyze::NodeCodeTranslator;
|
||||||
|
@ -350,6 +351,7 @@ pub async fn run(
|
||||||
npm_registry_url,
|
npm_registry_url,
|
||||||
npm_resolution.clone(),
|
npm_resolution.clone(),
|
||||||
node_modules_path,
|
node_modules_path,
|
||||||
|
NpmSystemInfo::default(),
|
||||||
);
|
);
|
||||||
let npm_resolver = Arc::new(CliNpmResolver::new(
|
let npm_resolver = Arc::new(CliNpmResolver::new(
|
||||||
fs.clone(),
|
fs.clone(),
|
||||||
|
|
|
@ -1799,3 +1799,84 @@ fn reload_info_not_found_cache_but_exists_remote() {
|
||||||
output.assert_exit_code(0);
|
output.assert_exit_code(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn binary_package_with_optional_dependencies() {
|
||||||
|
let context = TestContextBuilder::for_npm()
|
||||||
|
.use_sync_npm_download()
|
||||||
|
.use_separate_deno_dir() // the "npm" folder means something in the deno dir, so use a separate folder
|
||||||
|
.use_copy_temp_dir("npm/binary_package")
|
||||||
|
.cwd("npm/binary_package")
|
||||||
|
.build();
|
||||||
|
|
||||||
|
let temp_dir = context.temp_dir();
|
||||||
|
let temp_dir_path = temp_dir.path();
|
||||||
|
let project_path = temp_dir_path.join("npm/binary_package");
|
||||||
|
|
||||||
|
// write empty config file so a lockfile gets created
|
||||||
|
temp_dir.write("npm/binary_package/deno.json", "{}");
|
||||||
|
|
||||||
|
// run it twice, with the first time creating the lockfile and the second using it
|
||||||
|
for i in 0..2 {
|
||||||
|
if i == 1 {
|
||||||
|
assert!(project_path.join("deno.lock").exists());
|
||||||
|
}
|
||||||
|
|
||||||
|
let output = context
|
||||||
|
.new_command()
|
||||||
|
.args("run -A --node-modules-dir main.js")
|
||||||
|
.run();
|
||||||
|
|
||||||
|
#[cfg(target_os = "windows")]
|
||||||
|
{
|
||||||
|
output.assert_exit_code(0);
|
||||||
|
output.assert_matches_text(
|
||||||
|
"[WILDCARD]Hello from binary package on windows[WILDCARD]",
|
||||||
|
);
|
||||||
|
assert!(project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-windows@1.0.0")
|
||||||
|
.exists());
|
||||||
|
assert!(!project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-linux@1.0.0")
|
||||||
|
.exists());
|
||||||
|
assert!(!project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-mac@1.0.0")
|
||||||
|
.exists());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(target_os = "macos")]
|
||||||
|
{
|
||||||
|
output.assert_exit_code(0);
|
||||||
|
output.assert_matches_text(
|
||||||
|
"[WILDCARD]Hello from binary package on mac[WILDCARD]",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(!project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-windows@1.0.0")
|
||||||
|
.exists());
|
||||||
|
assert!(!project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-linux@1.0.0")
|
||||||
|
.exists());
|
||||||
|
assert!(project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-mac@1.0.0")
|
||||||
|
.exists());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(target_os = "linux")]
|
||||||
|
{
|
||||||
|
output.assert_exit_code(0);
|
||||||
|
output.assert_matches_text(
|
||||||
|
"[WILDCARD]Hello from binary package on linux[WILDCARD]",
|
||||||
|
);
|
||||||
|
assert!(!project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-windows@1.0.0")
|
||||||
|
.exists());
|
||||||
|
assert!(project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-linux@1.0.0")
|
||||||
|
.exists());
|
||||||
|
assert!(!project_path
|
||||||
|
.join("node_modules/.deno/@denotest+binary-package-mac@1.0.0")
|
||||||
|
.exists());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
1
cli/tests/testdata/npm/binary_package/main.js
vendored
Normal file
1
cli/tests/testdata/npm/binary_package/main.js
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
import "npm:@denotest/binary-package";
|
1
cli/tests/testdata/npm/registry/@denotest/binary-package-linux/1.0.0/index.js
vendored
Normal file
1
cli/tests/testdata/npm/registry/@denotest/binary-package-linux/1.0.0/index.js
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
console.log("Hello from binary package on linux");
|
8
cli/tests/testdata/npm/registry/@denotest/binary-package-linux/1.0.0/package.json
vendored
Normal file
8
cli/tests/testdata/npm/registry/@denotest/binary-package-linux/1.0.0/package.json
vendored
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
{
|
||||||
|
"name": "@denotest/binary-package-linux",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"main": "index.js",
|
||||||
|
"os": [
|
||||||
|
"linux"
|
||||||
|
]
|
||||||
|
}
|
1
cli/tests/testdata/npm/registry/@denotest/binary-package-mac/1.0.0/index.js
vendored
Normal file
1
cli/tests/testdata/npm/registry/@denotest/binary-package-mac/1.0.0/index.js
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
console.log("Hello from binary package on mac");
|
8
cli/tests/testdata/npm/registry/@denotest/binary-package-mac/1.0.0/package.json
vendored
Normal file
8
cli/tests/testdata/npm/registry/@denotest/binary-package-mac/1.0.0/package.json
vendored
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
{
|
||||||
|
"name": "@denotest/binary-package-linux",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"main": "index.js",
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
]
|
||||||
|
}
|
1
cli/tests/testdata/npm/registry/@denotest/binary-package-windows/1.0.0/index.js
vendored
Normal file
1
cli/tests/testdata/npm/registry/@denotest/binary-package-windows/1.0.0/index.js
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
console.log("Hello from binary package on windows");
|
8
cli/tests/testdata/npm/registry/@denotest/binary-package-windows/1.0.0/package.json
vendored
Normal file
8
cli/tests/testdata/npm/registry/@denotest/binary-package-windows/1.0.0/package.json
vendored
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
{
|
||||||
|
"name": "@denotest/binary-package-windows",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"main": "index.js",
|
||||||
|
"os": [
|
||||||
|
"win32"
|
||||||
|
]
|
||||||
|
}
|
13
cli/tests/testdata/npm/registry/@denotest/binary-package/1.0.0/index.js
vendored
Normal file
13
cli/tests/testdata/npm/registry/@denotest/binary-package/1.0.0/index.js
vendored
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
const packageByOs = {
|
||||||
|
"darwin": "@denotest/binary-package-mac",
|
||||||
|
"linux": "@denotest/binary-package-linux",
|
||||||
|
"win32": "@denotest/binary-package-windows",
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectedPackage = packageByOs[process.platform];
|
||||||
|
|
||||||
|
if (!selectedPackage) {
|
||||||
|
throw new Error("trying to run on unsupported platform");
|
||||||
|
}
|
||||||
|
|
||||||
|
require(selectedPackage);
|
10
cli/tests/testdata/npm/registry/@denotest/binary-package/1.0.0/package.json
vendored
Normal file
10
cli/tests/testdata/npm/registry/@denotest/binary-package/1.0.0/package.json
vendored
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
{
|
||||||
|
"name": "@denotest/binary-package",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"main": "index.js",
|
||||||
|
"optionalDependencies": {
|
||||||
|
"@denotest/binary-package-linux": "1.0.0",
|
||||||
|
"@denotest/binary-package-mac": "1.0.0",
|
||||||
|
"@denotest/binary-package-windows": "1.0.0"
|
||||||
|
}
|
||||||
|
}
|
|
@ -217,7 +217,8 @@ fn add_npm_packages_to_json(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut sorted_packages = snapshot.all_packages();
|
let mut sorted_packages =
|
||||||
|
snapshot.all_packages_for_every_system().collect::<Vec<_>>();
|
||||||
sorted_packages.sort_by(|a, b| a.pkg_id.cmp(&b.pkg_id));
|
sorted_packages.sort_by(|a, b| a.pkg_id.cmp(&b.pkg_id));
|
||||||
let mut json_packages = serde_json::Map::with_capacity(sorted_packages.len());
|
let mut json_packages = serde_json::Map::with_capacity(sorted_packages.len());
|
||||||
for pkg in sorted_packages {
|
for pkg in sorted_packages {
|
||||||
|
|
|
@ -136,6 +136,30 @@ fn get_npm_package(package_name: &str) -> Result<Option<CustomNpmPackage>> {
|
||||||
let mut version_info: serde_json::Map<String, serde_json::Value> =
|
let mut version_info: serde_json::Map<String, serde_json::Value> =
|
||||||
serde_json::from_str(&package_json_text)?;
|
serde_json::from_str(&package_json_text)?;
|
||||||
version_info.insert("dist".to_string(), dist.into());
|
version_info.insert("dist".to_string(), dist.into());
|
||||||
|
|
||||||
|
if let Some(maybe_optional_deps) = version_info.get("optionalDependencies")
|
||||||
|
{
|
||||||
|
if let Some(optional_deps) = maybe_optional_deps.as_object() {
|
||||||
|
if let Some(maybe_deps) = version_info.get("dependencies") {
|
||||||
|
if let Some(deps) = maybe_deps.as_object() {
|
||||||
|
let mut cloned_deps = deps.to_owned();
|
||||||
|
for (key, value) in optional_deps {
|
||||||
|
cloned_deps.insert(key.to_string(), value.to_owned());
|
||||||
|
}
|
||||||
|
version_info.insert(
|
||||||
|
"dependencies".to_string(),
|
||||||
|
serde_json::to_value(cloned_deps).unwrap(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
version_info.insert(
|
||||||
|
"dependencies".to_string(),
|
||||||
|
serde_json::to_value(optional_deps).unwrap(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
versions.insert(version.clone(), version_info.into());
|
versions.insert(version.clone(), version_info.into());
|
||||||
let version = semver::Version::parse(&version)?;
|
let version = semver::Version::parse(&version)?;
|
||||||
if version.cmp(&latest_version).is_gt() {
|
if version.cmp(&latest_version).is_gt() {
|
||||||
|
|
Loading…
Reference in a new issue