use itertools::Itertools;
use lets_find_up::{find_up_with, FindUpKind, FindUpOptions};
use mlua::{ExternalResult, UserData};
use path_slash::PathBufExt;
use project_toml::{
LocalProjectTomlValidationError, PartialProjectToml, RemoteProjectTomlValidationError,
};
use std::{
io,
ops::Deref,
path::{Path, PathBuf},
str::FromStr,
};
use thiserror::Error;
use toml_edit::{DocumentMut, Item};
use crate::{
build,
config::{Config, LuaVersion},
git::{self, shorthand::GitUrlShorthand, utils::GitError},
lockfile::{LockfileError, ProjectLockfile, ReadOnly},
lua::lua_runtime,
lua_rockspec::{
LocalLuaRockspec, LuaRockspecError, LuaVersionError, PartialLuaRockspec,
PartialRockspecError, RemoteLuaRockspec,
},
progress::Progress,
remote_package_db::RemotePackageDB,
rockspec::{
lua_dependency::{DependencyType, LuaDependencySpec, LuaDependencyType},
LuaVersionCompatibility,
},
tree::{Tree, TreeError},
};
use crate::{
lockfile::PinnedState,
package::{PackageName, PackageReq},
};
pub(crate) mod gen;
pub mod project_toml;
pub use project_toml::PROJECT_TOML;
pub const EXTRA_ROCKSPEC: &str = "extra.rockspec";
#[derive(Error, Debug)]
#[error(transparent)]
pub enum ProjectError {
Io(#[from] io::Error),
Lockfile(#[from] LockfileError),
Project(#[from] LocalProjectTomlValidationError),
Toml(#[from] toml::de::Error),
#[error("error when parsing `extra.rockspec`: {0}")]
Rockspec(#[from] PartialRockspecError),
#[error("not in a lux project directory")]
NotAProjectDir,
}
#[derive(Error, Debug)]
#[error(transparent)]
pub enum IntoLocalRockspecError {
LocalProjectTomlValidationError(#[from] LocalProjectTomlValidationError),
RockspecError(#[from] LuaRockspecError),
}
#[derive(Error, Debug)]
#[error(transparent)]
pub enum IntoRemoteRockspecError {
RocksTomlValidationError(#[from] RemoteProjectTomlValidationError),
RockspecError(#[from] LuaRockspecError),
}
#[derive(Error, Debug)]
pub enum ProjectEditError {
#[error(transparent)]
Io(#[from] tokio::io::Error),
#[error(transparent)]
Toml(#[from] toml_edit::TomlError),
#[error("error parsing lux.toml after edit. This is probably a bug.")]
TomlDe(#[from] toml::de::Error),
#[error(transparent)]
Git(#[from] GitError),
#[error("unable to query latest version for {0}")]
LatestVersionNotFound(PackageName),
#[error("expected field to be a value, but got {0}")]
ExpectedValue(toml_edit::Item),
#[error("expected string, but got {0}")]
ExpectedString(toml_edit::Value),
#[error(transparent)]
GitUrlShorthandParse(#[from] git::shorthand::ParseError),
}
#[derive(Error, Debug)]
#[error(transparent)]
pub enum ProjectTreeError {
Tree(#[from] TreeError),
LuaVersionError(#[from] LuaVersionError),
}
#[derive(Error, Debug)]
pub enum PinError {
#[error("package {0} not found in dependencies")]
PackageNotFound(PackageName),
#[error("dependency {dep} is already {}pinned!", if *.pin_state == PinnedState::Unpinned { "un" } else { "" })]
PinStateUnchanged {
pin_state: PinnedState,
dep: PackageName,
},
#[error(transparent)]
Toml(#[from] toml_edit::TomlError),
#[error("error parsing lux.toml after edit. This is probably a bug.")]
TomlDe(#[from] toml::de::Error),
#[error(transparent)]
Io(#[from] tokio::io::Error),
}
#[derive(Clone, Debug)]
#[cfg_attr(test, derive(Default))]
pub struct ProjectRoot(PathBuf);
impl ProjectRoot {
pub(crate) fn new() -> Self {
Self(PathBuf::new())
}
}
impl AsRef<Path> for ProjectRoot {
fn as_ref(&self) -> &Path {
self.0.as_ref()
}
}
impl Deref for ProjectRoot {
type Target = PathBuf;
fn deref(&self) -> &Self::Target {
&self.0
}
}
#[derive(Clone, Debug)]
pub struct Project {
root: ProjectRoot,
toml: PartialProjectToml,
}
impl UserData for Project {
fn add_methods<M: mlua::UserDataMethods<Self>>(methods: &mut M) {
methods.add_method("toml_path", |_, this, ()| Ok(this.toml_path()));
methods.add_method("extra_rockspec_path", |_, this, ()| {
Ok(this.extra_rockspec_path())
});
methods.add_method("lockfile_path", |_, this, ()| Ok(this.lockfile_path()));
methods.add_method("root", |_, this, ()| Ok(this.root().0.clone()));
methods.add_method("toml", |_, this, ()| Ok(this.toml().clone()));
methods.add_method("local_rockspec", |_, this, ()| {
this.local_rockspec().into_lua_err()
});
methods.add_method("remote_rockspec", |_, this, ()| {
this.remote_rockspec().into_lua_err()
});
methods.add_method("tree", |_, this, config: Config| {
this.tree(&config).into_lua_err()
});
methods.add_method("test_tree", |_, this, config: Config| {
this.test_tree(&config).into_lua_err()
});
methods.add_method("lua_version", |_, this, config: Config| {
this.lua_version(&config).into_lua_err()
});
methods.add_method("extra_rockspec", |_, this, ()| {
this.extra_rockspec().into_lua_err()
});
methods.add_async_method_mut(
"add",
|_, mut this, (deps, config): (DependencyType<PackageReq>, Config)| async move {
let _guard = lua_runtime().enter();
let package_db = RemotePackageDB::from_config(&config, &Progress::NoProgress)
.await
.into_lua_err()?;
this.add(deps, &package_db).await.into_lua_err()
},
);
methods.add_async_method_mut(
"remove",
|_, mut this, deps: DependencyType<PackageName>| async move {
let _guard = lua_runtime().enter();
this.remove(deps).await.into_lua_err()
},
);
methods.add_async_method_mut(
"upgrade",
|_, mut this, (deps, package_db): (LuaDependencyType<PackageName>, RemotePackageDB)| async move {
let _guard = lua_runtime().enter();
this.upgrade(deps, &package_db).await.into_lua_err()
},
);
methods.add_async_method_mut(
"upgrade_all",
|_, mut this, package_db: RemotePackageDB| async move {
let _guard = lua_runtime().enter();
this.upgrade_all(&package_db).await.into_lua_err()
},
);
methods.add_async_method_mut(
"set_pinned_state",
|_, mut this, (deps, pin): (LuaDependencyType<PackageName>, PinnedState)| async move {
let _guard = lua_runtime().enter();
this.set_pinned_state(deps, pin).await.into_lua_err()
},
);
methods.add_method("project_files", |_, this, ()| Ok(this.project_files()));
}
}
impl Project {
pub fn current() -> Result<Option<Self>, ProjectError> {
Self::from(&std::env::current_dir()?)
}
pub fn current_or_err() -> Result<Self, ProjectError> {
Self::current()?.ok_or(ProjectError::NotAProjectDir)
}
pub fn from_exact(start: impl AsRef<Path>) -> Result<Option<Self>, ProjectError> {
if !start.as_ref().exists() {
return Ok(None);
}
if start.as_ref().join(PROJECT_TOML).exists() {
let toml_content = std::fs::read_to_string(start.as_ref().join(PROJECT_TOML))?;
let root = start.as_ref();
let mut project = Project {
root: ProjectRoot(root.to_path_buf()),
toml: PartialProjectToml::new(&toml_content, ProjectRoot(root.to_path_buf()))?,
};
if let Some(extra_rockspec) = project.extra_rockspec()? {
project.toml = project.toml.merge(extra_rockspec);
}
Ok(Some(project))
} else {
Ok(None)
}
}
pub fn from(start: impl AsRef<Path>) -> Result<Option<Self>, ProjectError> {
if !start.as_ref().exists() {
return Ok(None);
}
match find_up_with(
PROJECT_TOML,
FindUpOptions {
cwd: start.as_ref(),
kind: FindUpKind::File,
},
) {
Ok(Some(path)) => {
let toml_content = std::fs::read_to_string(&path)?;
let root = path.parent().unwrap();
let mut project = Project {
root: ProjectRoot(root.to_path_buf()),
toml: PartialProjectToml::new(&toml_content, ProjectRoot(root.to_path_buf()))?,
};
if let Some(extra_rockspec) = project.extra_rockspec()? {
project.toml = project.toml.merge(extra_rockspec);
}
std::fs::create_dir_all(root)?;
Ok(Some(project))
}
_ => Ok(None),
}
}
pub fn toml_path(&self) -> PathBuf {
self.root.join(PROJECT_TOML)
}
pub fn extra_rockspec_path(&self) -> PathBuf {
self.root.join(EXTRA_ROCKSPEC)
}
pub fn lockfile_path(&self) -> PathBuf {
self.root.join("lux.lock")
}
pub fn lockfile(&self) -> Result<ProjectLockfile<ReadOnly>, ProjectError> {
Ok(ProjectLockfile::new(self.lockfile_path())?)
}
pub fn try_lockfile(&self) -> Result<Option<ProjectLockfile<ReadOnly>>, ProjectError> {
let path = self.lockfile_path();
if path.is_file() {
Ok(Some(ProjectLockfile::load(path)?))
} else {
Ok(None)
}
}
pub fn root(&self) -> &ProjectRoot {
&self.root
}
pub fn toml(&self) -> &PartialProjectToml {
&self.toml
}
pub fn local_rockspec(&self) -> Result<LocalLuaRockspec, IntoLocalRockspecError> {
Ok(self.toml().into_local()?.to_lua_rockspec()?)
}
pub fn remote_rockspec(&self) -> Result<RemoteLuaRockspec, IntoRemoteRockspecError> {
Ok(self.toml().into_remote()?.to_lua_rockspec()?)
}
pub fn extra_rockspec(&self) -> Result<Option<PartialLuaRockspec>, PartialRockspecError> {
if self.extra_rockspec_path().exists() {
Ok(Some(PartialLuaRockspec::new(&std::fs::read_to_string(
self.extra_rockspec_path(),
)?)?))
} else {
Ok(None)
}
}
pub(crate) fn default_tree_root_dir(&self) -> PathBuf {
self.root.join(".lux")
}
pub fn tree(&self, config: &Config) -> Result<Tree, ProjectTreeError> {
self.lua_version_tree(self.lua_version(config)?, config)
}
pub(crate) fn lua_version_tree(
&self,
lua_version: LuaVersion,
config: &Config,
) -> Result<Tree, ProjectTreeError> {
Ok(Tree::new(
self.default_tree_root_dir(),
lua_version,
config,
)?)
}
pub fn test_tree(&self, config: &Config) -> Result<Tree, ProjectTreeError> {
Ok(self.tree(config)?.test_tree(config)?)
}
pub fn build_tree(&self, config: &Config) -> Result<Tree, ProjectTreeError> {
Ok(self.tree(config)?.build_tree(config)?)
}
pub fn lua_version(&self, config: &Config) -> Result<LuaVersion, LuaVersionError> {
self.toml().lua_version_matches(config)
}
pub async fn add(
&mut self,
dependencies: DependencyType<PackageReq>,
package_db: &RemotePackageDB,
) -> Result<(), ProjectEditError> {
let mut project_toml =
toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
prepare_dependency_tables(&mut project_toml);
let table = match dependencies {
DependencyType::Regular(_) => &mut project_toml["dependencies"],
DependencyType::Build(_) => &mut project_toml["build_dependencies"],
DependencyType::Test(_) => &mut project_toml["test_dependencies"],
DependencyType::External(_) => &mut project_toml["external_dependencies"],
};
match dependencies {
DependencyType::Regular(ref deps)
| DependencyType::Build(ref deps)
| DependencyType::Test(ref deps) => {
for dep in deps {
let dep_version_str = if dep.version_req().is_any() {
package_db
.latest_version(dep.name())
.expect("unable to query latest version for package")
.to_string()
} else {
dep.version_req().to_string()
};
table[dep.name().to_string()] = toml_edit::value(dep_version_str);
}
}
DependencyType::External(ref deps) => {
for (name, dep) in deps {
if let Some(path) = &dep.header {
table[name]["header"] = toml_edit::value(path.to_slash_lossy().to_string());
}
if let Some(path) = &dep.library {
table[name]["library"] =
toml_edit::value(path.to_slash_lossy().to_string());
}
}
}
};
let toml_content = project_toml.to_string();
tokio::fs::write(self.toml_path(), &toml_content).await?;
self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
Ok(())
}
pub async fn add_git(
&mut self,
dependencies: LuaDependencyType<GitUrlShorthand>,
) -> Result<(), ProjectEditError> {
let mut project_toml =
toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
prepare_dependency_tables(&mut project_toml);
let table = match dependencies {
LuaDependencyType::Regular(_) => &mut project_toml["dependencies"],
LuaDependencyType::Build(_) => &mut project_toml["build_dependencies"],
LuaDependencyType::Test(_) => &mut project_toml["test_dependencies"],
};
match dependencies {
LuaDependencyType::Regular(ref urls)
| LuaDependencyType::Build(ref urls)
| LuaDependencyType::Test(ref urls) => {
for url in urls {
let git_url: git_url_parse::GitUrl = url.clone().into();
let rev = git::utils::latest_semver_tag_or_commit_sha(&git_url)?;
table[git_url.name.clone()]["version"] = Item::Value(rev.into());
table[git_url.name.clone()]["git"] = Item::Value(url.to_string().into());
}
}
}
let toml_content = project_toml.to_string();
tokio::fs::write(self.toml_path(), &toml_content).await?;
self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
Ok(())
}
pub async fn remove(
&mut self,
dependencies: DependencyType<PackageName>,
) -> Result<(), ProjectEditError> {
let mut project_toml =
toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
prepare_dependency_tables(&mut project_toml);
let table = match dependencies {
DependencyType::Regular(_) => &mut project_toml["dependencies"],
DependencyType::Build(_) => &mut project_toml["build_dependencies"],
DependencyType::Test(_) => &mut project_toml["test_dependencies"],
DependencyType::External(_) => &mut project_toml["external_dependencies"],
};
match dependencies {
DependencyType::Regular(ref deps)
| DependencyType::Build(ref deps)
| DependencyType::Test(ref deps) => {
for dep in deps {
table[dep.to_string()] = Item::None;
}
}
DependencyType::External(ref deps) => {
for (name, dep) in deps {
if dep.header.is_some() {
table[name]["header"] = Item::None;
}
if dep.library.is_some() {
table[name]["library"] = Item::None;
}
}
}
};
let toml_content = project_toml.to_string();
tokio::fs::write(self.toml_path(), &toml_content).await?;
self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
Ok(())
}
pub async fn upgrade(
&mut self,
dependencies: LuaDependencyType<PackageName>,
package_db: &RemotePackageDB,
) -> Result<(), ProjectEditError> {
let mut project_toml =
toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
prepare_dependency_tables(&mut project_toml);
let table = match dependencies {
LuaDependencyType::Regular(_) => &mut project_toml["dependencies"],
LuaDependencyType::Build(_) => &mut project_toml["build_dependencies"],
LuaDependencyType::Test(_) => &mut project_toml["test_dependencies"],
};
match dependencies {
LuaDependencyType::Regular(ref deps)
| LuaDependencyType::Build(ref deps)
| LuaDependencyType::Test(ref deps) => {
let latest_rock_version_str =
|dep: &PackageName| -> Result<String, ProjectEditError> {
Ok(package_db
.latest_version(dep)
.ok_or(ProjectEditError::LatestVersionNotFound(dep.clone()))?
.to_string())
};
for dep in deps {
let mut dep_item = table[dep.to_string()].clone();
match &dep_item {
Item::Value(_) => {
let dep_version_str = latest_rock_version_str(dep)?;
table[dep.to_string()] = toml_edit::value(dep_version_str);
}
Item::Table(tbl) => {
if tbl.contains_key("git") {
let git_value = tbl
.get("git")
.expect("expected 'git' field")
.clone()
.into_value()
.map_err(ProjectEditError::ExpectedValue)?;
let git_url_str = git_value
.as_str()
.ok_or(ProjectEditError::ExpectedString(git_value.clone()))?;
let shorthand: GitUrlShorthand = git_url_str.parse()?;
let latest_rev =
git::utils::latest_semver_tag_or_commit_sha(&shorthand.into())?;
let key = if tbl.contains_key("rev") {
"rev".to_string()
} else {
"version".to_string()
};
dep_item[key] = toml_edit::value(latest_rev);
table[dep.to_string()] = dep_item;
} else {
let dep_version_str = latest_rock_version_str(dep)?;
dep_item["version".to_string()] = toml_edit::value(dep_version_str);
table[dep.to_string()] = dep_item;
}
}
_ => {}
}
}
}
}
let toml_content = project_toml.to_string();
tokio::fs::write(self.toml_path(), &toml_content).await?;
self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
Ok(())
}
pub async fn upgrade_all(
&mut self,
package_db: &RemotePackageDB,
) -> Result<(), ProjectEditError> {
if let Some(dependencies) = &self.toml().dependencies {
let packages = dependencies
.iter()
.map(|dep| dep.name())
.cloned()
.collect_vec();
self.upgrade(LuaDependencyType::Regular(packages), package_db)
.await?;
}
if let Some(dependencies) = &self.toml().build_dependencies {
let packages = dependencies
.iter()
.map(|dep| dep.name())
.cloned()
.collect_vec();
self.upgrade(LuaDependencyType::Build(packages), package_db)
.await?;
}
if let Some(dependencies) = &self.toml().test_dependencies {
let packages = dependencies
.iter()
.map(|dep| dep.name())
.cloned()
.collect_vec();
self.upgrade(LuaDependencyType::Test(packages), package_db)
.await?;
}
Ok(())
}
pub async fn set_pinned_state(
&mut self,
dependencies: LuaDependencyType<PackageName>,
pin: PinnedState,
) -> Result<(), PinError> {
let mut project_toml =
toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
prepare_dependency_tables(&mut project_toml);
let table = match dependencies {
LuaDependencyType::Regular(_) => &mut project_toml["dependencies"],
LuaDependencyType::Build(_) => &mut project_toml["build_dependencies"],
LuaDependencyType::Test(_) => &mut project_toml["test_dependencies"],
};
match dependencies {
LuaDependencyType::Regular(ref _deps) => {
self.toml.dependencies = Some(
self.toml
.dependencies
.take()
.unwrap_or_default()
.into_iter()
.map(|dep| LuaDependencySpec { pin, ..dep })
.collect(),
)
}
LuaDependencyType::Build(ref _deps) => {
self.toml.build_dependencies = Some(
self.toml
.build_dependencies
.take()
.unwrap_or_default()
.into_iter()
.map(|dep| LuaDependencySpec { pin, ..dep })
.collect(),
)
}
LuaDependencyType::Test(ref _deps) => {
self.toml.test_dependencies = Some(
self.toml
.test_dependencies
.take()
.unwrap_or_default()
.into_iter()
.map(|dep| LuaDependencySpec { pin, ..dep })
.collect(),
)
}
}
match dependencies {
LuaDependencyType::Regular(ref deps)
| LuaDependencyType::Build(ref deps)
| LuaDependencyType::Test(ref deps) => {
for dep in deps {
let mut dep_item = table[dep.to_string()].clone();
match dep_item {
version @ Item::Value(_) => match &pin {
PinnedState::Unpinned => {}
PinnedState::Pinned => {
let mut dep_entry = toml_edit::table().into_table().unwrap();
dep_entry.set_implicit(true);
dep_entry["version"] = version;
dep_entry["pin"] = toml_edit::value(true);
table[dep.to_string()] = toml_edit::Item::Table(dep_entry);
}
},
Item::Table(_) => {
dep_item["pin".to_string()] = toml_edit::value(pin.as_bool());
table[dep.to_string()] = dep_item;
}
_ => {}
}
}
}
}
let toml_content = project_toml.to_string();
tokio::fs::write(self.toml_path(), &toml_content).await?;
self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
Ok(())
}
pub fn project_files(&self) -> Vec<PathBuf> {
build::utils::project_files(&self.root().0)
}
}
fn prepare_dependency_tables(project_toml: &mut DocumentMut) {
if !project_toml.contains_table("dependencies") {
let mut table = toml_edit::table().into_table().unwrap();
table.set_implicit(true);
project_toml["dependencies"] = toml_edit::Item::Table(table);
}
if !project_toml.contains_table("build_dependencies") {
let mut table = toml_edit::table().into_table().unwrap();
table.set_implicit(true);
project_toml["build_dependencies"] = toml_edit::Item::Table(table);
}
if !project_toml.contains_table("test_dependencies") {
let mut table = toml_edit::table().into_table().unwrap();
table.set_implicit(true);
project_toml["test_dependencies"] = toml_edit::Item::Table(table);
}
if !project_toml.contains_table("external_dependencies") {
let mut table = toml_edit::table().into_table().unwrap();
table.set_implicit(true);
project_toml["external_dependencies"] = toml_edit::Item::Table(table);
}
}
#[cfg(test)]
mod tests {
use std::collections::HashMap;
use assert_fs::prelude::PathCopy;
use url::Url;
use super::*;
use crate::{
lua_rockspec::ExternalDependencySpec,
manifest::{Manifest, ManifestMetadata},
package::PackageReq,
rockspec::Rockspec,
};
#[tokio::test]
async fn test_add_various_dependencies() {
let sample_project: PathBuf = "resources/test/sample-project-no-build-spec/".into();
let project_root = assert_fs::TempDir::new().unwrap();
project_root.copy_from(&sample_project, &["**"]).unwrap();
let project_root: PathBuf = project_root.path().into();
let mut project = Project::from(&project_root).unwrap().unwrap();
let add_dependencies =
vec![PackageReq::new("busted".into(), Some(">= 1.0.0".into())).unwrap()];
let expected_dependencies = vec![PackageReq::new("busted".into(), Some(">= 1.0.0".into()))
.unwrap()
.into()];
let test_manifest_path =
PathBuf::from(env!("CARGO_MANIFEST_DIR")).join("resources/test/manifest-5.1");
let content = String::from_utf8(std::fs::read(&test_manifest_path).unwrap()).unwrap();
let metadata = ManifestMetadata::new(&content).unwrap();
let package_db = Manifest::new(Url::parse("https://example.com").unwrap(), metadata).into();
project
.add(
DependencyType::Regular(add_dependencies.clone()),
&package_db,
)
.await
.unwrap();
project
.add(DependencyType::Build(add_dependencies.clone()), &package_db)
.await
.unwrap();
project
.add(DependencyType::Test(add_dependencies.clone()), &package_db)
.await
.unwrap();
project
.add(
DependencyType::External(HashMap::from([(
"lib".into(),
ExternalDependencySpec {
library: Some("path.so".into()),
header: None,
},
)])),
&package_db,
)
.await
.unwrap();
let project = Project::from(&project_root).unwrap().unwrap();
let validated_toml = project.toml().into_remote().unwrap();
assert_eq!(
validated_toml.dependencies().current_platform(),
&expected_dependencies
);
assert_eq!(
validated_toml.build_dependencies().current_platform(),
&expected_dependencies
);
assert_eq!(
validated_toml.test_dependencies().current_platform(),
&expected_dependencies
);
assert_eq!(
validated_toml
.external_dependencies()
.current_platform()
.get("lib")
.unwrap(),
&ExternalDependencySpec {
library: Some("path.so".into()),
header: None
}
);
}
#[tokio::test]
async fn test_remove_dependencies() {
let sample_project: PathBuf = "resources/test/sample-project-dependencies/".into();
let project_root = assert_fs::TempDir::new().unwrap();
project_root.copy_from(&sample_project, &["**"]).unwrap();
let project_root: PathBuf = project_root.path().into();
let mut project = Project::from(&project_root).unwrap().unwrap();
let remove_dependencies = vec!["lua-cjson".into(), "plenary.nvim".into()];
project
.remove(DependencyType::Regular(remove_dependencies.clone()))
.await
.unwrap();
let check = |project: &Project| {
for name in &remove_dependencies {
assert!(!project
.toml()
.dependencies
.clone()
.unwrap_or_default()
.iter()
.any(|dep| dep.name() == name));
}
};
check(&project);
let reloaded_project = Project::from(&project_root).unwrap().unwrap();
check(&reloaded_project);
}
#[tokio::test]
async fn test_extra_rockspec_parsing() {
let sample_project: PathBuf = "resources/test/sample-project-extra-rockspec".into();
let project_root = assert_fs::TempDir::new().unwrap();
project_root.copy_from(&sample_project, &["**"]).unwrap();
let project_root: PathBuf = project_root.path().into();
let project = Project::from(project_root).unwrap().unwrap();
let extra_rockspec = project.extra_rockspec().unwrap();
assert!(extra_rockspec.is_some());
let rocks = project.toml().into_remote().unwrap();
assert_eq!(rocks.package().to_string(), "custom-package");
}
#[tokio::test]
async fn test_pin_dependencies() {
test_pin_unpin_dependencies(PinnedState::Pinned).await
}
#[tokio::test]
async fn test_unpin_dependencies() {
test_pin_unpin_dependencies(PinnedState::Unpinned).await
}
async fn test_pin_unpin_dependencies(pin: PinnedState) {
let sample_project: PathBuf = "resources/test/sample-project-dependencies/".into();
let project_root = assert_fs::TempDir::new().unwrap();
project_root.copy_from(&sample_project, &["**"]).unwrap();
let project_root: PathBuf = project_root.path().into();
let mut project = Project::from(&project_root).unwrap().unwrap();
let pin_dependencies = vec!["lua-cjson".into(), "plenary.nvim".into()];
project
.set_pinned_state(LuaDependencyType::Regular(pin_dependencies.clone()), pin)
.await
.unwrap();
let check = |project: &Project| {
for name in &pin_dependencies {
assert!(project
.toml()
.dependencies
.clone()
.unwrap_or_default()
.iter()
.any(|dep| dep.name() == name && dep.pin == pin));
}
};
check(&project);
let reloaded_project = Project::from(&project_root).unwrap().unwrap();
check(&reloaded_project);
}
}