#![deny(missing_docs)] //! A package manager for the Luau programming language, supporting multiple runtimes including Roblox and Lune. //! pesde has its own registry, however it can also use Wally, and Git repositories as package sources. //! It has been designed with multiple targets in mind, namely Roblox, Lune, and Luau. use crate::{ lockfile::Lockfile, manifest::Manifest, source::{traits::PackageSource, PackageSources}, }; use async_stream::stream; use fs_err::tokio as fs; use futures::{future::try_join_all, Stream}; use gix::sec::identity::Account; use std::{ collections::{HashMap, HashSet}, fmt::Debug, path::{Path, PathBuf}, }; use tracing::instrument; use wax::Pattern; /// Downloading packages pub mod download; /// Utility for downloading and linking in the correct order pub mod download_and_link; /// Linking packages pub mod linking; /// Lockfile pub mod lockfile; /// Manifest pub mod manifest; /// Package names pub mod names; /// Patching packages #[cfg(feature = "patches")] pub mod patches; /// Resolving packages pub mod resolver; /// Running scripts pub mod scripts; /// Package sources pub mod source; pub(crate) mod util; /// The name of the manifest file pub const MANIFEST_FILE_NAME: &str = "pesde.toml"; /// The name of the lockfile pub const LOCKFILE_FILE_NAME: &str = "pesde.lock"; /// The name of the default index pub const DEFAULT_INDEX_NAME: &str = "default"; /// The name of the packages container pub const PACKAGES_CONTAINER_NAME: &str = ".pesde"; pub(crate) const LINK_LIB_NO_FILE_FOUND: &str = "____pesde_no_export_file_found"; /// The folder in which scripts are linked pub const SCRIPTS_LINK_FOLDER: &str = ".pesde"; /// Struct containing the authentication configuration #[derive(Debug, Default, Clone)] pub struct AuthConfig { tokens: HashMap, git_credentials: Option, } impl AuthConfig { /// Create a new `AuthConfig` pub fn new() -> Self { AuthConfig::default() } /// Set the tokens pub fn with_tokens, S: AsRef>( mut self, tokens: I, ) -> Self { self.tokens = tokens .into_iter() .map(|(url, s)| (url, s.as_ref().to_string())) .collect(); self } /// Set the git credentials pub fn with_git_credentials(mut self, git_credentials: Option) -> Self { self.git_credentials = git_credentials; self } /// Get the tokens pub fn tokens(&self) -> &HashMap { &self.tokens } /// Get the git credentials pub fn git_credentials(&self) -> Option<&Account> { self.git_credentials.as_ref() } } /// The main struct of the pesde library, representing a project #[derive(Debug, Clone)] pub struct Project { package_dir: PathBuf, workspace_dir: Option, data_dir: PathBuf, auth_config: AuthConfig, cas_dir: PathBuf, } impl Project { /// Create a new `Project` pub fn new, Q: AsRef, R: AsRef, S: AsRef>( package_dir: P, workspace_dir: Option, data_dir: R, cas_dir: S, auth_config: AuthConfig, ) -> Self { Project { package_dir: package_dir.as_ref().to_path_buf(), workspace_dir: workspace_dir.map(|d| d.as_ref().to_path_buf()), data_dir: data_dir.as_ref().to_path_buf(), auth_config, cas_dir: cas_dir.as_ref().to_path_buf(), } } /// The directory of the package pub fn package_dir(&self) -> &Path { &self.package_dir } /// The directory of the workspace this package belongs to, if any pub fn workspace_dir(&self) -> Option<&Path> { self.workspace_dir.as_deref() } /// The directory to store general-purpose data pub fn data_dir(&self) -> &Path { &self.data_dir } /// The authentication configuration pub fn auth_config(&self) -> &AuthConfig { &self.auth_config } /// The CAS (content-addressable storage) directory pub fn cas_dir(&self) -> &Path { &self.cas_dir } /// Read the manifest file #[instrument(skip(self), ret(level = "trace"), level = "debug")] pub async fn read_manifest(&self) -> Result { let string = fs::read_to_string(self.package_dir.join(MANIFEST_FILE_NAME)).await?; Ok(string) } // TODO: cache the manifest /// Deserialize the manifest file #[instrument(skip(self), ret(level = "trace"), level = "debug")] pub async fn deser_manifest(&self) -> Result { let string = fs::read_to_string(self.package_dir.join(MANIFEST_FILE_NAME)).await?; Ok(toml::from_str(&string)?) } /// Write the manifest file #[instrument(skip(self, manifest), level = "debug")] pub async fn write_manifest>(&self, manifest: S) -> Result<(), std::io::Error> { fs::write(self.package_dir.join(MANIFEST_FILE_NAME), manifest.as_ref()).await } /// Deserialize the lockfile #[instrument(skip(self), ret(level = "trace"), level = "debug")] pub async fn deser_lockfile(&self) -> Result { let string = fs::read_to_string(self.package_dir.join(LOCKFILE_FILE_NAME)).await?; Ok(toml::from_str(&string)?) } /// Write the lockfile #[instrument(skip(self, lockfile), level = "debug")] pub async fn write_lockfile( &self, lockfile: Lockfile, ) -> Result<(), errors::LockfileWriteError> { let string = toml::to_string(&lockfile)?; fs::write(self.package_dir.join(LOCKFILE_FILE_NAME), string).await?; Ok(()) } /// Get the workspace members #[instrument(skip(self), level = "debug")] pub async fn workspace_members + Debug>( &self, dir: P, can_ref_self: bool, ) -> Result< impl Stream>, errors::WorkspaceMembersError, > { let dir = dir.as_ref().to_path_buf(); let manifest = fs::read_to_string(dir.join(MANIFEST_FILE_NAME)) .await .map_err(errors::WorkspaceMembersError::ManifestMissing)?; let manifest = toml::from_str::(&manifest).map_err(|e| { errors::WorkspaceMembersError::ManifestDeser(dir.to_path_buf(), Box::new(e)) })?; let members = matching_globs( dir, manifest.workspace_members.iter().map(|s| s.as_str()), false, can_ref_self, ) .await?; Ok(stream! { for path in members { let manifest = fs::read_to_string(path.join(MANIFEST_FILE_NAME)) .await .map_err(errors::WorkspaceMembersError::ManifestMissing)?; let manifest = toml::from_str::(&manifest).map_err(|e| { errors::WorkspaceMembersError::ManifestDeser(path.clone(), Box::new(e)) })?; yield Ok((path, manifest)); } }) } } /// Gets all matching paths in a directory #[deprecated( since = "0.5.0-rc.13", note = "use `matching_globs` instead, which does not have the old behaviour of including whole directories by their name (`src` instead of `src/**`)" )] #[instrument(ret, level = "trace")] pub async fn matching_globs_old_behaviour< 'a, P: AsRef + Debug, I: IntoIterator + Debug, >( dir: P, globs: I, relative: bool, ) -> Result, errors::MatchingGlobsError> { let (negative_globs, positive_globs) = globs .into_iter() .partition::, _>(|glob| glob.starts_with('!')); let negative_globs = wax::any( negative_globs .into_iter() .map(|glob| wax::Glob::new(&glob[1..])) .collect::, _>>()?, )?; let (positive_globs, file_names) = positive_globs .into_iter() // only globs we can be sure of (maintaining compatibility with old "only file/dir name" system) .partition::, _>(|glob| glob.contains('/')); let file_names = file_names.into_iter().collect::>(); let positive_globs = wax::any( positive_globs .into_iter() .map(wax::Glob::new) .collect::, _>>()?, )?; let mut read_dirs = vec![(fs::read_dir(dir.as_ref().to_path_buf()).await?, false)]; let mut paths = HashSet::new(); let mut is_root = true; while let Some((mut read_dir, is_entire_dir_included)) = read_dirs.pop() { while let Some(entry) = read_dir.next_entry().await? { let path = entry.path(); let relative_path = path.strip_prefix(dir.as_ref()).unwrap(); let file_name = path.file_name().unwrap(); let is_filename_match = is_root && file_name.to_str().is_some_and(|s| file_names.contains(s)); if entry.file_type().await?.is_dir() { read_dirs.push(( fs::read_dir(&path).await?, is_entire_dir_included || is_filename_match, )); if is_filename_match { tracing::warn!("directory name usage found for {}. this is deprecated and will be removed in the future", path.display()); } } if (is_entire_dir_included || is_filename_match) || (positive_globs.is_match(relative_path) && !negative_globs.is_match(relative_path)) { paths.insert(if relative { relative_path.to_path_buf() } else { path.to_path_buf() }); } } is_root = false; } Ok(paths) } /// Gets all matching paths in a directory #[instrument(ret, level = "trace")] pub async fn matching_globs<'a, P: AsRef + Debug, I: IntoIterator + Debug>( dir: P, globs: I, relative: bool, can_ref_self: bool, ) -> Result, errors::MatchingGlobsError> { let (negative_globs, mut positive_globs): (HashSet<&str>, _) = globs.into_iter().partition(|glob| glob.starts_with('!')); let include_self = positive_globs.remove(".") && can_ref_self; let negative_globs = wax::any( negative_globs .into_iter() .map(|glob| wax::Glob::new(&glob[1..])) .collect::, _>>()?, )?; let positive_globs = wax::any( positive_globs .into_iter() .map(wax::Glob::new) .collect::, _>>()?, )?; let mut read_dirs = vec![fs::read_dir(dir.as_ref().to_path_buf()).await?]; let mut paths = HashSet::new(); if include_self { paths.insert(if relative { PathBuf::new() } else { dir.as_ref().to_path_buf() }); } while let Some(mut read_dir) = read_dirs.pop() { while let Some(entry) = read_dir.next_entry().await? { let path = entry.path(); if entry.file_type().await?.is_dir() { read_dirs.push(fs::read_dir(&path).await?); } let relative_path = path.strip_prefix(dir.as_ref()).unwrap(); if positive_globs.is_match(relative_path) && !negative_globs.is_match(relative_path) { paths.insert(if relative { relative_path.to_path_buf() } else { path.to_path_buf() }); } } } Ok(paths) } /// Refreshes the sources asynchronously pub async fn refresh_sources>( project: &Project, sources: I, refreshed_sources: &mut HashSet, ) -> Result<(), Box> { try_join_all(sources.map(|source| { let needs_refresh = refreshed_sources.insert(source.clone()); async move { if needs_refresh { source.refresh(project).await.map_err(Box::new) } else { Ok(()) } } })) .await .map(|_| ()) } /// Errors that can occur when using the pesde library pub mod errors { use std::path::PathBuf; use thiserror::Error; /// Errors that can occur when reading the manifest file #[derive(Debug, Error)] #[non_exhaustive] pub enum ManifestReadError { /// An IO error occurred #[error("io error reading manifest file")] Io(#[from] std::io::Error), /// An error occurred while deserializing the manifest file #[error("error deserializing manifest file")] Serde(#[from] toml::de::Error), } /// Errors that can occur when reading the lockfile #[derive(Debug, Error)] #[non_exhaustive] pub enum LockfileReadError { /// An IO error occurred #[error("io error reading lockfile")] Io(#[from] std::io::Error), /// An error occurred while deserializing the lockfile #[error("error deserializing lockfile")] Serde(#[from] toml::de::Error), } /// Errors that can occur when writing the lockfile #[derive(Debug, Error)] #[non_exhaustive] pub enum LockfileWriteError { /// An IO error occurred #[error("io error writing lockfile")] Io(#[from] std::io::Error), /// An error occurred while serializing the lockfile #[error("error serializing lockfile")] Serde(#[from] toml::ser::Error), } /// Errors that can occur when finding workspace members #[derive(Debug, Error)] #[non_exhaustive] pub enum WorkspaceMembersError { /// The manifest file could not be found #[error("missing manifest file")] ManifestMissing(#[source] std::io::Error), /// An error occurred deserializing the manifest file #[error("error deserializing manifest file at {0}")] ManifestDeser(PathBuf, #[source] Box), /// An error occurred interacting with the filesystem #[error("error interacting with the filesystem")] Io(#[from] std::io::Error), /// An error occurred while globbing #[error("error globbing")] Globbing(#[from] MatchingGlobsError), } /// Errors that can occur when finding matching globs #[derive(Debug, Error)] #[non_exhaustive] pub enum MatchingGlobsError { /// An error occurred interacting with the filesystem #[error("error interacting with the filesystem")] Io(#[from] std::io::Error), /// An error occurred while building a glob #[error("error building glob")] BuildGlob(#[from] wax::BuildError), } }