1use itertools::Itertools;
2use lets_find_up::{find_up_with, FindUpKind, FindUpOptions};
3use path_slash::PathBufExt;
4use project_toml::{
5 LocalProjectTomlValidationError, PartialProjectToml, RemoteProjectTomlValidationError,
6};
7use std::{
8 io,
9 ops::Deref,
10 path::{Path, PathBuf},
11 str::FromStr,
12};
13use thiserror::Error;
14use toml_edit::{DocumentMut, Item};
15
16use crate::{
17 build,
18 config::{Config, LuaVersion},
19 git::{
20 self,
21 shorthand::RemoteGitUrlShorthand,
22 url::RemoteGitUrl,
23 utils::{GitError, SemVerTagOrSha},
24 },
25 lockfile::{LockfileError, ProjectLockfile, ReadOnly},
26 lua_rockspec::{
27 LocalLuaRockspec, LuaRockspecError, LuaVersionError, PartialLuaRockspec,
28 PartialRockspecError, RemoteLuaRockspec,
29 },
30 package::SpecRev,
31 remote_package_db::RemotePackageDB,
32 rockspec::{
33 lua_dependency::{DependencyType, LuaDependencySpec, LuaDependencyType},
34 LuaVersionCompatibility,
35 },
36 tree::{Tree, TreeError},
37};
38use crate::{
39 lockfile::PinnedState,
40 package::{PackageName, PackageReq},
41};
42
43pub(crate) mod gen;
44pub mod project_toml;
45
46pub use project_toml::PROJECT_TOML;
47
48pub const EXTRA_ROCKSPEC: &str = "extra.rockspec";
49pub(crate) const LUX_DIR_NAME: &str = ".lux";
50const LUARC: &str = ".luarc.json";
51const EMMYRC: &str = ".emmyrc.json";
52
53#[derive(Error, Debug)]
54#[error(transparent)]
55pub enum ProjectError {
56 #[error("cannot get current directory: {0}")]
57 GetCwd(io::Error),
58 #[error("error reading project TOML at {0}:\n{1}")]
59 ReadProjectTOML(String, io::Error),
60 #[error("error creating project root at {0}:\n{1}")]
61 CreateProjectRoot(String, io::Error),
62 Lockfile(#[from] LockfileError),
63 Project(#[from] LocalProjectTomlValidationError),
64 Toml(#[from] toml::de::Error),
65 #[error("error when parsing `extra.rockspec`: {0}")]
66 Rockspec(#[from] PartialRockspecError),
67 #[error("not in a lux project directory")]
68 NotAProjectDir,
69}
70
71#[derive(Error, Debug)]
72#[error(transparent)]
73pub enum IntoLocalRockspecError {
74 LocalProjectTomlValidationError(#[from] LocalProjectTomlValidationError),
75 RockspecError(#[from] LuaRockspecError),
76}
77
78#[derive(Error, Debug)]
79#[error(transparent)]
80pub enum IntoRemoteRockspecError {
81 RocksTomlValidationError(#[from] RemoteProjectTomlValidationError),
82 RockspecError(#[from] LuaRockspecError),
83}
84
85#[derive(Error, Debug)]
86pub enum ProjectEditError {
87 #[error(transparent)]
88 Io(#[from] tokio::io::Error),
89 #[error(transparent)]
90 Toml(#[from] toml_edit::TomlError),
91 #[error("error parsing lux.toml after edit. This is probably a bug.")]
92 TomlDe(#[from] toml::de::Error),
93 #[error(transparent)]
94 Git(#[from] GitError),
95 #[error("unable to query latest version for {0}")]
96 LatestVersionNotFound(PackageName),
97 #[error("expected field to be a value, but got {0}")]
98 ExpectedValue(toml_edit::Item),
99 #[error("expected string, but got {0}")]
100 ExpectedString(toml_edit::Value),
101 #[error(transparent)]
102 GitUrlShorthandParse(#[from] git::shorthand::ParseError),
103}
104
105#[derive(Error, Debug)]
106#[error(transparent)]
107pub enum ProjectTreeError {
108 Tree(#[from] TreeError),
109 LuaVersionError(#[from] LuaVersionError),
110}
111
112#[derive(Error, Debug)]
113pub enum PinError {
114 #[error("package {0} not found in dependencies")]
115 PackageNotFound(PackageName),
116 #[error("dependency {dep} is already {}pinned!", if *.pin_state == PinnedState::Unpinned { "un" } else { "" })]
117 PinStateUnchanged {
118 pin_state: PinnedState,
119 dep: PackageName,
120 },
121 #[error(transparent)]
122 Toml(#[from] toml_edit::TomlError),
123 #[error("error parsing lux.toml after edit. This is probably a bug.")]
124 TomlDe(#[from] toml::de::Error),
125 #[error(transparent)]
126 Io(#[from] tokio::io::Error),
127}
128
129#[derive(Clone, Debug)]
132#[cfg_attr(test, derive(Default))]
133pub struct ProjectRoot(PathBuf);
134
135impl ProjectRoot {
136 pub(crate) fn new() -> Self {
137 Self(PathBuf::new())
138 }
139}
140
141impl AsRef<Path> for ProjectRoot {
142 fn as_ref(&self) -> &Path {
143 self.0.as_ref()
144 }
145}
146
147impl Deref for ProjectRoot {
148 type Target = PathBuf;
149
150 fn deref(&self) -> &Self::Target {
151 &self.0
152 }
153}
154
155#[derive(Clone, Debug)]
156pub struct Project {
157 root: ProjectRoot,
159 toml: PartialProjectToml,
161}
162
163impl Project {
164 pub fn current() -> Result<Option<Self>, ProjectError> {
165 let cwd = std::env::current_dir().map_err(ProjectError::GetCwd)?;
166 Self::from(&cwd)
167 }
168
169 pub fn current_or_err() -> Result<Self, ProjectError> {
170 Self::current()?.ok_or(ProjectError::NotAProjectDir)
171 }
172
173 pub fn from_exact(start: impl AsRef<Path>) -> Result<Option<Self>, ProjectError> {
174 if !start.as_ref().exists() {
175 return Ok(None);
176 }
177
178 if start.as_ref().join(PROJECT_TOML).exists() {
179 let project_toml_path = start.as_ref().join(PROJECT_TOML);
180 let toml_content = std::fs::read_to_string(&project_toml_path).map_err(|err| {
181 ProjectError::ReadProjectTOML(project_toml_path.to_string_lossy().to_string(), err)
182 })?;
183 let root = start.as_ref();
184
185 let mut project = Project {
186 root: ProjectRoot(root.to_path_buf()),
187 toml: PartialProjectToml::new(&toml_content, ProjectRoot(root.to_path_buf()))?,
188 };
189
190 if let Some(extra_rockspec) = project.extra_rockspec()? {
191 project.toml = project.toml.merge(extra_rockspec);
192 }
193
194 Ok(Some(project))
195 } else {
196 Ok(None)
197 }
198 }
199
200 pub fn from(start: impl AsRef<Path>) -> Result<Option<Self>, ProjectError> {
201 if !start.as_ref().exists() {
202 return Ok(None);
203 }
204
205 match find_up_with(
206 PROJECT_TOML,
207 FindUpOptions {
208 cwd: start.as_ref(),
209 kind: FindUpKind::File,
210 },
211 ) {
212 Ok(Some(path)) => {
213 if let Some(root) = path.parent() {
214 let toml_content = std::fs::read_to_string(&path).map_err(|err| {
215 ProjectError::ReadProjectTOML(path.to_string_lossy().to_string(), err)
216 })?;
217
218 let mut project = Project {
219 root: ProjectRoot(root.to_path_buf()),
220 toml: PartialProjectToml::new(
221 &toml_content,
222 ProjectRoot(root.to_path_buf()),
223 )?,
224 };
225
226 if let Some(extra_rockspec) = project.extra_rockspec()? {
227 project.toml = project.toml.merge(extra_rockspec);
228 }
229
230 std::fs::create_dir_all(root).map_err(|err| {
231 ProjectError::CreateProjectRoot(root.to_string_lossy().to_string(), err)
232 })?;
233
234 Ok(Some(project))
235 } else {
236 Ok(None)
237 }
238 }
239 _ => Ok(None),
243 }
244 }
245
246 pub fn toml_path(&self) -> PathBuf {
248 self.root.join(PROJECT_TOML)
249 }
250
251 pub fn luarc_path(&self) -> PathBuf {
253 let luarc_path = self.root.join(LUARC);
254 if luarc_path.is_file() {
255 luarc_path
256 } else {
257 let emmy_path = self.root.join(EMMYRC);
258 if emmy_path.is_file() {
259 emmy_path
260 } else {
261 luarc_path
262 }
263 }
264 }
265
266 pub fn extra_rockspec_path(&self) -> PathBuf {
268 self.root.join(EXTRA_ROCKSPEC)
269 }
270
271 pub fn lockfile_path(&self) -> PathBuf {
273 self.root.join("lux.lock")
274 }
275
276 pub fn lockfile(&self) -> Result<ProjectLockfile<ReadOnly>, ProjectError> {
278 Ok(ProjectLockfile::new(self.lockfile_path())?)
279 }
280
281 pub fn try_lockfile(&self) -> Result<Option<ProjectLockfile<ReadOnly>>, ProjectError> {
283 let path = self.lockfile_path();
284 if path.is_file() {
285 Ok(Some(ProjectLockfile::load(path)?))
286 } else {
287 Ok(None)
288 }
289 }
290
291 pub fn root(&self) -> &ProjectRoot {
292 &self.root
293 }
294
295 pub fn toml(&self) -> &PartialProjectToml {
296 &self.toml
297 }
298
299 pub fn local_rockspec(&self) -> Result<LocalLuaRockspec, IntoLocalRockspecError> {
300 Ok(self.toml().into_local()?.to_lua_rockspec()?)
301 }
302
303 pub fn remote_rockspec(
304 &self,
305 specrev: Option<SpecRev>,
306 ) -> Result<RemoteLuaRockspec, IntoRemoteRockspecError> {
307 Ok(self.toml().into_remote(specrev)?.to_lua_rockspec()?)
308 }
309
310 pub fn extra_rockspec(&self) -> Result<Option<PartialLuaRockspec>, PartialRockspecError> {
311 if self.extra_rockspec_path().exists() {
312 Ok(Some(PartialLuaRockspec::new(&std::fs::read_to_string(
313 self.extra_rockspec_path(),
314 )?)?))
315 } else {
316 Ok(None)
317 }
318 }
319
320 pub(crate) fn default_tree_root_dir(&self) -> PathBuf {
321 self.root.join(LUX_DIR_NAME)
322 }
323
324 pub fn tree(&self, config: &Config) -> Result<Tree, ProjectTreeError> {
325 self.lua_version_tree(self.lua_version(config)?, config)
326 }
327
328 pub(crate) fn lua_version_tree(
329 &self,
330 lua_version: LuaVersion,
331 config: &Config,
332 ) -> Result<Tree, ProjectTreeError> {
333 Ok(Tree::new(
334 self.default_tree_root_dir(),
335 lua_version,
336 config,
337 )?)
338 }
339
340 pub fn test_tree(&self, config: &Config) -> Result<Tree, ProjectTreeError> {
341 Ok(self.tree(config)?.test_tree(config)?)
342 }
343
344 pub fn build_tree(&self, config: &Config) -> Result<Tree, ProjectTreeError> {
345 Ok(self.tree(config)?.build_tree(config)?)
346 }
347
348 pub fn lua_version(&self, config: &Config) -> Result<LuaVersion, LuaVersionError> {
349 self.toml().lua_version_matches(config)
350 }
351
352 pub async fn add(
353 &mut self,
354 dependencies: DependencyType<PackageReq>,
355 package_db: &RemotePackageDB,
356 ) -> Result<(), ProjectEditError> {
357 let mut project_toml =
358 toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
359
360 prepare_dependency_tables(&mut project_toml);
361 let table = match dependencies {
362 DependencyType::Regular(_) => &mut project_toml["dependencies"],
363 DependencyType::Build(_) => &mut project_toml["build_dependencies"],
364 DependencyType::Test(_) => &mut project_toml["test_dependencies"],
365 DependencyType::External(_) => &mut project_toml["external_dependencies"],
366 };
367
368 match dependencies {
369 DependencyType::Regular(ref deps)
370 | DependencyType::Build(ref deps)
371 | DependencyType::Test(ref deps) => {
372 for dep in deps {
373 let dep_version_str = if dep.version_req().is_any() {
374 package_db
375 .latest_version(dep.name())
376 .map(|latest_version| latest_version.to_string())
377 .unwrap_or_else(|| dep.version_req().to_string())
378 } else {
379 dep.version_req().to_string()
380 };
381 table[dep.name().to_string()] = toml_edit::value(dep_version_str);
382 }
383 }
384 DependencyType::External(ref deps) => {
385 for (name, dep) in deps {
386 if let Some(path) = &dep.header {
387 table[name]["header"] = toml_edit::value(path.to_slash_lossy().to_string());
388 }
389 if let Some(path) = &dep.library {
390 table[name]["library"] =
391 toml_edit::value(path.to_slash_lossy().to_string());
392 }
393 }
394 }
395 };
396
397 let toml_content = project_toml.to_string();
398 tokio::fs::write(self.toml_path(), &toml_content).await?;
399 self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
400
401 Ok(())
402 }
403
404 pub async fn add_git(
405 &mut self,
406 dependencies: LuaDependencyType<RemoteGitUrlShorthand>,
407 ) -> Result<(), ProjectEditError> {
408 let mut project_toml =
409 toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
410
411 prepare_dependency_tables(&mut project_toml);
412 let table = match dependencies {
413 LuaDependencyType::Regular(_) => &mut project_toml["dependencies"],
414 LuaDependencyType::Build(_) => &mut project_toml["build_dependencies"],
415 LuaDependencyType::Test(_) => &mut project_toml["test_dependencies"],
416 };
417
418 match dependencies {
419 LuaDependencyType::Regular(ref urls)
420 | LuaDependencyType::Build(ref urls)
421 | LuaDependencyType::Test(ref urls) => {
422 for url in urls {
423 let git_url: RemoteGitUrl = url.clone().into();
424 let mut dep_entry = toml_edit::table();
425 match git::utils::latest_semver_tag_or_commit_sha(&git_url)? {
426 SemVerTagOrSha::SemVerTag(tag) => {
427 dep_entry["git"] = Item::Value(url.to_string().into());
428 dep_entry["version"] = Item::Value(tag.clone().into());
429 if tag.contains("-") {
430 dep_entry["rev"] = Item::Value(tag.into());
432 }
433 }
434 SemVerTagOrSha::CommitSha(sha) => {
435 dep_entry["git"] = Item::Value(url.to_string().into());
436 dep_entry["version"] = Item::Value(sha.into());
437 }
438 }
439 table[git_url.repo.clone()] = dep_entry;
440 }
441 }
442 }
443
444 let toml_content = project_toml.to_string();
445 tokio::fs::write(self.toml_path(), &toml_content).await?;
446 self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
447
448 Ok(())
449 }
450
451 pub async fn remove(
452 &mut self,
453 dependencies: DependencyType<PackageName>,
454 ) -> Result<(), ProjectEditError> {
455 let mut project_toml =
456 toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
457
458 prepare_dependency_tables(&mut project_toml);
459 let table = match dependencies {
460 DependencyType::Regular(_) => &mut project_toml["dependencies"],
461 DependencyType::Build(_) => &mut project_toml["build_dependencies"],
462 DependencyType::Test(_) => &mut project_toml["test_dependencies"],
463 DependencyType::External(_) => &mut project_toml["external_dependencies"],
464 };
465
466 match dependencies {
467 DependencyType::Regular(ref deps)
468 | DependencyType::Build(ref deps)
469 | DependencyType::Test(ref deps) => {
470 for dep in deps {
471 table[dep.to_string()] = Item::None;
472 }
473 }
474 DependencyType::External(ref deps) => {
475 for (name, dep) in deps {
476 if dep.header.is_some() {
477 table[name]["header"] = Item::None;
478 }
479 if dep.library.is_some() {
480 table[name]["library"] = Item::None;
481 }
482 }
483 }
484 };
485
486 let toml_content = project_toml.to_string();
487 tokio::fs::write(self.toml_path(), &toml_content).await?;
488 self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
489
490 Ok(())
491 }
492
493 pub async fn upgrade(
494 &mut self,
495 dependencies: LuaDependencyType<PackageName>,
496 package_db: &RemotePackageDB,
497 ) -> Result<(), ProjectEditError> {
498 let mut project_toml =
499 toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
500
501 prepare_dependency_tables(&mut project_toml);
502 let table = match dependencies {
503 LuaDependencyType::Regular(_) => &mut project_toml["dependencies"],
504 LuaDependencyType::Build(_) => &mut project_toml["build_dependencies"],
505 LuaDependencyType::Test(_) => &mut project_toml["test_dependencies"],
506 };
507
508 match dependencies {
509 LuaDependencyType::Regular(ref deps)
510 | LuaDependencyType::Build(ref deps)
511 | LuaDependencyType::Test(ref deps) => {
512 let latest_rock_version_str =
513 |dep: &PackageName| -> Result<String, ProjectEditError> {
514 Ok(package_db
515 .latest_version(dep)
516 .ok_or(ProjectEditError::LatestVersionNotFound(dep.clone()))?
517 .to_string())
518 };
519 for dep in deps {
520 let mut dep_item = table[dep.to_string()].clone();
521 match &dep_item {
522 Item::Value(_) => {
523 let dep_version_str = latest_rock_version_str(dep)?;
524 table[dep.to_string()] = toml_edit::value(dep_version_str);
525 }
526 Item::Table(tbl) => {
527 match tbl.get("git") {
528 Some(git_item) => {
529 let git_value = git_item
530 .clone()
531 .into_value()
532 .map_err(ProjectEditError::ExpectedValue)?;
533 let git_url_str = git_value.as_str().ok_or(
534 ProjectEditError::ExpectedString(git_value.clone()),
535 )?;
536 let shorthand: RemoteGitUrlShorthand = git_url_str.parse()?;
537 match git::utils::latest_semver_tag_or_commit_sha(
538 &shorthand.into(),
539 )? {
540 SemVerTagOrSha::SemVerTag(latest_tag) => {
541 table[dep.to_string()]["version"] =
542 Item::Value(latest_tag.clone().into());
543 if latest_tag.contains("-") {
544 table[dep.to_string()]["rev"] =
546 Item::Value(latest_tag.into());
547 }
548 }
549 SemVerTagOrSha::CommitSha(latest_sha) => {
550 table[dep.to_string()]["version"] =
551 Item::Value(latest_sha.into());
552 }
553 }
554 table[dep.to_string()] = dep_item;
555 }
556 None => {
557 let dep_version_str = latest_rock_version_str(dep)?;
558 dep_item["version".to_string()] =
559 toml_edit::value(dep_version_str);
560 table[dep.to_string()] = dep_item;
561 }
562 }
563 }
564 _ => {}
565 }
566 }
567 }
568 }
569
570 let toml_content = project_toml.to_string();
571 tokio::fs::write(self.toml_path(), &toml_content).await?;
572 self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
573
574 Ok(())
575 }
576
577 pub async fn upgrade_all(
578 &mut self,
579 package_db: &RemotePackageDB,
580 ) -> Result<(), ProjectEditError> {
581 if let Some(dependencies) = &self.toml().dependencies {
582 let packages = dependencies
583 .iter()
584 .map(|dep| dep.name())
585 .cloned()
586 .collect_vec();
587 self.upgrade(LuaDependencyType::Regular(packages), package_db)
588 .await?;
589 }
590 if let Some(dependencies) = &self.toml().build_dependencies {
591 let packages = dependencies
592 .iter()
593 .map(|dep| dep.name())
594 .cloned()
595 .collect_vec();
596 self.upgrade(LuaDependencyType::Build(packages), package_db)
597 .await?;
598 }
599 if let Some(dependencies) = &self.toml().test_dependencies {
600 let packages = dependencies
601 .iter()
602 .map(|dep| dep.name())
603 .cloned()
604 .collect_vec();
605 self.upgrade(LuaDependencyType::Test(packages), package_db)
606 .await?;
607 }
608 Ok(())
609 }
610
611 pub async fn set_pinned_state(
612 &mut self,
613 dependencies: LuaDependencyType<PackageName>,
614 pin: PinnedState,
615 ) -> Result<(), PinError> {
616 let mut project_toml =
617 toml_edit::DocumentMut::from_str(&tokio::fs::read_to_string(self.toml_path()).await?)?;
618
619 prepare_dependency_tables(&mut project_toml);
620 let table = match dependencies {
621 LuaDependencyType::Regular(_) => &mut project_toml["dependencies"],
622 LuaDependencyType::Build(_) => &mut project_toml["build_dependencies"],
623 LuaDependencyType::Test(_) => &mut project_toml["test_dependencies"],
624 };
625
626 match dependencies {
627 LuaDependencyType::Regular(ref _deps) => {
628 self.toml.dependencies = Some(
629 self.toml
630 .dependencies
631 .take()
632 .unwrap_or_default()
633 .into_iter()
634 .map(|dep| LuaDependencySpec { pin, ..dep })
635 .collect(),
636 )
637 }
638 LuaDependencyType::Build(ref _deps) => {
639 self.toml.build_dependencies = Some(
640 self.toml
641 .build_dependencies
642 .take()
643 .unwrap_or_default()
644 .into_iter()
645 .map(|dep| LuaDependencySpec { pin, ..dep })
646 .collect(),
647 )
648 }
649 LuaDependencyType::Test(ref _deps) => {
650 self.toml.test_dependencies = Some(
651 self.toml
652 .test_dependencies
653 .take()
654 .unwrap_or_default()
655 .into_iter()
656 .map(|dep| LuaDependencySpec { pin, ..dep })
657 .collect(),
658 )
659 }
660 }
661
662 match dependencies {
663 LuaDependencyType::Regular(ref deps)
664 | LuaDependencyType::Build(ref deps)
665 | LuaDependencyType::Test(ref deps) => {
666 for dep in deps {
667 let mut dep_item = table[dep.to_string()].clone();
668 match dep_item {
669 version @ Item::Value(_) => match &pin {
670 PinnedState::Unpinned => {}
671 PinnedState::Pinned => {
672 if let Ok(mut dep_entry) = toml_edit::table().into_table() {
673 dep_entry.set_implicit(true);
674 dep_entry["version"] = version;
675 dep_entry["pin"] = toml_edit::value(true);
676 table[dep.to_string()] = toml_edit::Item::Table(dep_entry);
677 }
678 }
679 },
680 Item::Table(_) => {
681 dep_item["pin".to_string()] = toml_edit::value(pin.as_bool());
682 table[dep.to_string()] = dep_item;
683 }
684 _ => {}
685 }
686 }
687 }
688 }
689
690 let toml_content = project_toml.to_string();
691 tokio::fs::write(self.toml_path(), &toml_content).await?;
692 self.toml = PartialProjectToml::new(&toml_content, self.root.clone())?;
693
694 Ok(())
695 }
696
697 pub fn project_files(&self) -> Vec<PathBuf> {
698 build::utils::project_files(&self.root().0)
699 }
700}
701
702fn prepare_dependency_tables(project_toml: &mut DocumentMut) {
703 if !project_toml.contains_table("dependencies") {
704 if let Ok(mut table) = toml_edit::table().into_table() {
705 table.set_implicit(true);
706 project_toml["dependencies"] = toml_edit::Item::Table(table);
707 }
708 }
709 if !project_toml.contains_table("build_dependencies") {
710 if let Ok(mut table) = toml_edit::table().into_table() {
711 table.set_implicit(true);
712 project_toml["build_dependencies"] = toml_edit::Item::Table(table);
713 }
714 }
715 if !project_toml.contains_table("test_dependencies") {
716 if let Ok(mut table) = toml_edit::table().into_table() {
717 table.set_implicit(true);
718 project_toml["test_dependencies"] = toml_edit::Item::Table(table);
719 }
720 }
721 if !project_toml.contains_table("external_dependencies") {
722 if let Ok(mut table) = toml_edit::table().into_table() {
723 table.set_implicit(true);
724 project_toml["external_dependencies"] = toml_edit::Item::Table(table);
725 }
726 }
727}
728
729#[cfg(test)]
731mod tests {
732 use std::collections::HashMap;
733
734 use assert_fs::prelude::PathCopy;
735 use url::Url;
736
737 use super::*;
738 use crate::{
739 lua_rockspec::ExternalDependencySpec,
740 manifest::{Manifest, ManifestMetadata},
741 package::PackageReq,
742 rockspec::Rockspec,
743 };
744
745 #[tokio::test]
746 async fn test_add_various_dependencies() {
747 let sample_project: PathBuf = "resources/test/sample-projects/no-build-spec/".into();
748 let project_root = assert_fs::TempDir::new().unwrap();
749 project_root.copy_from(&sample_project, &["**"]).unwrap();
750 let project_root: PathBuf = project_root.path().into();
751 let mut project = Project::from(&project_root).unwrap().unwrap();
752 let add_dependencies =
753 vec![PackageReq::new("busted".into(), Some(">= 1.0.0".into())).unwrap()];
754 let expected_dependencies = vec![PackageReq::new("busted".into(), Some(">= 1.0.0".into()))
755 .unwrap()
756 .into()];
757
758 let test_manifest_path =
759 PathBuf::from(env!("CARGO_MANIFEST_DIR")).join("resources/test/manifest-5.1");
760 let content = String::from_utf8(std::fs::read(&test_manifest_path).unwrap()).unwrap();
761 let metadata = ManifestMetadata::new(&content).unwrap();
762 let package_db = Manifest::new(Url::parse("https://example.com").unwrap(), metadata).into();
763
764 project
765 .add(
766 DependencyType::Regular(add_dependencies.clone()),
767 &package_db,
768 )
769 .await
770 .unwrap();
771
772 project
773 .add(DependencyType::Build(add_dependencies.clone()), &package_db)
774 .await
775 .unwrap();
776 project
777 .add(DependencyType::Test(add_dependencies.clone()), &package_db)
778 .await
779 .unwrap();
780
781 project
782 .add(
783 DependencyType::External(HashMap::from([(
784 "lib".into(),
785 ExternalDependencySpec {
786 library: Some("path.so".into()),
787 header: None,
788 },
789 )])),
790 &package_db,
791 )
792 .await
793 .unwrap();
794
795 let project = Project::from(&project_root).unwrap().unwrap();
798 let validated_toml = project.toml().into_remote(None).unwrap();
799
800 assert_eq!(
801 validated_toml.dependencies().current_platform(),
802 &expected_dependencies
803 );
804 assert_eq!(
805 validated_toml.build_dependencies().current_platform(),
806 &expected_dependencies
807 );
808 assert_eq!(
809 validated_toml.test_dependencies().current_platform(),
810 &expected_dependencies
811 );
812 assert_eq!(
813 validated_toml
814 .external_dependencies()
815 .current_platform()
816 .get("lib")
817 .unwrap(),
818 &ExternalDependencySpec {
819 library: Some("path.so".into()),
820 header: None
821 }
822 );
823 }
824
825 #[tokio::test]
826 async fn test_remove_dependencies() {
827 let sample_project: PathBuf = "resources/test/sample-projects/dependencies/".into();
828 let project_root = assert_fs::TempDir::new().unwrap();
829 project_root.copy_from(&sample_project, &["**"]).unwrap();
830 let project_root: PathBuf = project_root.path().into();
831 let mut project = Project::from(&project_root).unwrap().unwrap();
832 let remove_dependencies = vec!["lua-cjson".into(), "plenary.nvim".into()];
833 project
834 .remove(DependencyType::Regular(remove_dependencies.clone()))
835 .await
836 .unwrap();
837 let check = |project: &Project| {
838 for name in &remove_dependencies {
839 assert!(!project
840 .toml()
841 .dependencies
842 .clone()
843 .unwrap_or_default()
844 .iter()
845 .any(|dep| dep.name() == name));
846 }
847 };
848 check(&project);
849 let reloaded_project = Project::from(&project_root).unwrap().unwrap();
851 check(&reloaded_project);
852 }
853
854 #[tokio::test]
855 async fn test_extra_rockspec_parsing() {
856 let sample_project: PathBuf = "resources/test/sample-projects/extra-rockspec/".into();
857 let project_root = assert_fs::TempDir::new().unwrap();
858 project_root.copy_from(&sample_project, &["**"]).unwrap();
859 let project_root: PathBuf = project_root.path().into();
860 let project = Project::from(project_root).unwrap().unwrap();
861
862 let extra_rockspec = project.extra_rockspec().unwrap();
863
864 assert!(extra_rockspec.is_some());
865
866 let rocks = project.toml().into_remote(None).unwrap();
867
868 assert_eq!(rocks.package().to_string(), "custom-package");
869 }
870
871 #[tokio::test]
872 async fn test_pin_dependencies() {
873 test_pin_unpin_dependencies(PinnedState::Pinned).await
874 }
875
876 #[tokio::test]
877 async fn test_unpin_dependencies() {
878 test_pin_unpin_dependencies(PinnedState::Unpinned).await
879 }
880
881 async fn test_pin_unpin_dependencies(pin: PinnedState) {
882 let sample_project: PathBuf = "resources/test/sample-projects/dependencies/".into();
883 let project_root = assert_fs::TempDir::new().unwrap();
884 project_root.copy_from(&sample_project, &["**"]).unwrap();
885 let project_root: PathBuf = project_root.path().into();
886 let mut project = Project::from(&project_root).unwrap().unwrap();
887 let pin_dependencies = vec!["lua-cjson".into(), "plenary.nvim".into()];
888 project
889 .set_pinned_state(LuaDependencyType::Regular(pin_dependencies.clone()), pin)
890 .await
891 .unwrap();
892 let check = |project: &Project| {
893 for name in &pin_dependencies {
894 assert!(project
895 .toml()
896 .dependencies
897 .clone()
898 .unwrap_or_default()
899 .iter()
900 .any(|dep| dep.name() == name && dep.pin == pin));
901 }
902 };
903 check(&project);
904 let reloaded_project = Project::from(&project_root).unwrap().unwrap();
906 check(&reloaded_project);
907 }
908}