2024-01-01 14:58:21 -05:00
|
|
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
2023-04-06 18:46:44 -04:00
|
|
|
|
2023-01-23 23:41:02 +01:00
|
|
|
use std::path::PathBuf;
|
2023-04-06 18:46:44 -04:00
|
|
|
|
2024-05-28 14:58:43 -04:00
|
|
|
use deno_core::anyhow::Context;
|
2023-04-06 18:46:44 -04:00
|
|
|
use deno_core::error::AnyError;
|
2024-06-28 17:18:21 -07:00
|
|
|
use deno_core::parking_lot::Mutex;
|
|
|
|
use deno_core::parking_lot::MutexGuard;
|
2024-05-23 12:31:05 -07:00
|
|
|
use deno_runtime::deno_node::PackageJson;
|
2019-11-03 10:39:27 -05:00
|
|
|
|
2022-11-02 16:32:30 +01:00
|
|
|
use crate::args::ConfigFile;
|
2024-05-28 14:58:43 -04:00
|
|
|
use crate::cache;
|
2024-05-31 23:25:08 -04:00
|
|
|
use crate::util::fs::atomic_write_file_with_retries;
|
2022-11-02 16:32:30 +01:00
|
|
|
use crate::Flags;
|
2021-12-07 18:21:04 -06:00
|
|
|
|
2024-06-28 17:18:21 -07:00
|
|
|
use crate::args::DenoSubcommand;
|
|
|
|
use crate::args::InstallFlags;
|
|
|
|
use crate::args::InstallKind;
|
|
|
|
|
|
|
|
use deno_lockfile::Lockfile;
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct CliLockfile {
|
|
|
|
lockfile: Mutex<Lockfile>,
|
|
|
|
pub filename: PathBuf,
|
2024-07-02 15:00:16 -07:00
|
|
|
pub frozen: bool,
|
2024-06-28 17:18:21 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Guard<'a, T> {
|
|
|
|
guard: MutexGuard<'a, T>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, T> std::ops::Deref for Guard<'a, T> {
|
|
|
|
type Target = T;
|
|
|
|
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.guard
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, T> std::ops::DerefMut for Guard<'a, T> {
|
|
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
|
|
&mut self.guard
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CliLockfile {
|
2024-07-02 15:00:16 -07:00
|
|
|
pub fn new(lockfile: Lockfile, frozen: bool) -> Self {
|
2024-06-28 17:18:21 -07:00
|
|
|
let filename = lockfile.filename.clone();
|
|
|
|
Self {
|
|
|
|
lockfile: Mutex::new(lockfile),
|
|
|
|
filename,
|
2024-07-02 15:00:16 -07:00
|
|
|
frozen,
|
2024-06-28 17:18:21 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the inner deno_lockfile::Lockfile.
|
|
|
|
pub fn lock(&self) -> Guard<Lockfile> {
|
|
|
|
Guard {
|
|
|
|
guard: self.lockfile.lock(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_workspace_config(
|
|
|
|
&self,
|
|
|
|
options: deno_lockfile::SetWorkspaceConfigOptions,
|
|
|
|
) {
|
|
|
|
self.lockfile.lock().set_workspace_config(options);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn overwrite(&self) -> bool {
|
|
|
|
self.lockfile.lock().overwrite
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn write_if_changed(&self) -> Result<(), AnyError> {
|
2024-07-02 15:00:16 -07:00
|
|
|
self.error_if_changed()?;
|
2024-06-28 17:18:21 -07:00
|
|
|
let mut lockfile = self.lockfile.lock();
|
|
|
|
let Some(bytes) = lockfile.resolve_write_bytes() else {
|
|
|
|
return Ok(()); // nothing to do
|
|
|
|
};
|
|
|
|
// do an atomic write to reduce the chance of multiple deno
|
|
|
|
// processes corrupting the file
|
|
|
|
atomic_write_file_with_retries(
|
|
|
|
&lockfile.filename,
|
|
|
|
bytes,
|
|
|
|
cache::CACHE_PERM,
|
2023-01-23 23:41:02 +01:00
|
|
|
)
|
2024-06-28 17:18:21 -07:00
|
|
|
.context("Failed writing lockfile.")?;
|
|
|
|
lockfile.has_content_changed = false;
|
|
|
|
Ok(())
|
2023-01-23 23:41:02 +01:00
|
|
|
}
|
|
|
|
|
2024-06-28 17:18:21 -07:00
|
|
|
pub fn discover(
|
|
|
|
flags: &Flags,
|
|
|
|
maybe_config_file: Option<&ConfigFile>,
|
|
|
|
maybe_package_json: Option<&PackageJson>,
|
|
|
|
) -> Result<Option<CliLockfile>, AnyError> {
|
|
|
|
if flags.no_lock
|
|
|
|
|| matches!(
|
|
|
|
flags.subcommand,
|
|
|
|
DenoSubcommand::Install(InstallFlags {
|
|
|
|
kind: InstallKind::Global(..),
|
|
|
|
..
|
|
|
|
}) | DenoSubcommand::Uninstall(_)
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
|
|
|
|
let filename = match flags.lock {
|
|
|
|
Some(ref lock) => PathBuf::from(lock),
|
|
|
|
None => match maybe_config_file {
|
|
|
|
Some(config_file) => {
|
|
|
|
if config_file.specifier.scheme() == "file" {
|
|
|
|
match config_file.resolve_lockfile_path()? {
|
|
|
|
Some(path) => path,
|
|
|
|
None => return Ok(None),
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return Ok(None);
|
2022-11-02 16:32:30 +01:00
|
|
|
}
|
|
|
|
}
|
2024-06-28 17:18:21 -07:00
|
|
|
None => match maybe_package_json {
|
|
|
|
Some(package_json) => {
|
|
|
|
package_json.path.parent().unwrap().join("deno.lock")
|
|
|
|
}
|
|
|
|
None => return Ok(None),
|
|
|
|
},
|
2024-05-23 12:31:05 -07:00
|
|
|
},
|
2024-06-28 17:18:21 -07:00
|
|
|
};
|
2024-05-28 14:58:43 -04:00
|
|
|
|
2024-06-28 17:18:21 -07:00
|
|
|
let lockfile = if flags.lock_write {
|
2024-07-02 15:00:16 -07:00
|
|
|
CliLockfile::new(
|
|
|
|
Lockfile::new_empty(filename, true),
|
|
|
|
flags.frozen_lockfile,
|
|
|
|
)
|
2024-06-28 17:18:21 -07:00
|
|
|
} else {
|
2024-07-02 15:00:16 -07:00
|
|
|
Self::read_from_path(filename, flags.frozen_lockfile)?
|
2024-06-28 17:18:21 -07:00
|
|
|
};
|
|
|
|
Ok(Some(lockfile))
|
|
|
|
}
|
2024-07-02 15:00:16 -07:00
|
|
|
pub fn read_from_path(
|
|
|
|
filename: PathBuf,
|
|
|
|
frozen: bool,
|
|
|
|
) -> Result<CliLockfile, AnyError> {
|
2024-06-28 17:18:21 -07:00
|
|
|
match std::fs::read_to_string(&filename) {
|
2024-07-02 15:00:16 -07:00
|
|
|
Ok(text) => Ok(CliLockfile::new(
|
|
|
|
Lockfile::with_lockfile_content(filename, &text, false)?,
|
|
|
|
frozen,
|
|
|
|
)),
|
|
|
|
Err(err) if err.kind() == std::io::ErrorKind::NotFound => Ok(
|
|
|
|
CliLockfile::new(Lockfile::new_empty(filename, false), frozen),
|
|
|
|
),
|
2024-06-28 17:18:21 -07:00
|
|
|
Err(err) => Err(err).with_context(|| {
|
|
|
|
format!("Failed reading lockfile '{}'", filename.display())
|
|
|
|
}),
|
2024-05-28 14:58:43 -04:00
|
|
|
}
|
|
|
|
}
|
2024-07-02 15:00:16 -07:00
|
|
|
pub fn error_if_changed(&self) -> Result<(), AnyError> {
|
|
|
|
if !self.frozen {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
let lockfile = self.lockfile.lock();
|
|
|
|
if lockfile.has_content_changed {
|
|
|
|
let suggested = if *super::DENO_FUTURE {
|
|
|
|
"`deno cache --frozen=false`, `deno install --frozen=false`,"
|
|
|
|
} else {
|
|
|
|
"`deno cache --frozen=false`"
|
|
|
|
};
|
|
|
|
|
|
|
|
let contents =
|
|
|
|
std::fs::read_to_string(&lockfile.filename).unwrap_or_default();
|
|
|
|
let new_contents = lockfile.as_json_string();
|
|
|
|
let diff = crate::util::diff::diff(&contents, &new_contents);
|
|
|
|
// has an extra newline at the end
|
|
|
|
let diff = diff.trim_end();
|
|
|
|
Err(deno_core::anyhow::anyhow!(
|
|
|
|
"The lockfile is out of date. Run {suggested} or rerun with `--frozen=false` to update it.\nchanges:\n{diff}"
|
|
|
|
))
|
|
|
|
} else {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
2024-05-28 14:58:43 -04:00
|
|
|
}
|