2024-03-04 20:18:49 +00:00
|
|
|
use std::{
|
2024-03-24 17:38:18 +00:00
|
|
|
collections::HashSet,
|
2024-03-24 13:31:11 +00:00
|
|
|
fs::{read_to_string, write},
|
2024-03-04 20:18:49 +00:00
|
|
|
path::{Component, Path, PathBuf},
|
|
|
|
};
|
|
|
|
|
|
|
|
use full_moon::{
|
|
|
|
ast::types::ExportedTypeDeclaration,
|
|
|
|
parse,
|
|
|
|
visitors::{Visit, Visitor},
|
|
|
|
};
|
|
|
|
use log::debug;
|
|
|
|
use semver::Version;
|
|
|
|
use thiserror::Error;
|
|
|
|
|
|
|
|
use crate::{
|
2024-03-25 16:29:31 +00:00
|
|
|
dependencies::resolution::{packages_folder, ResolvedPackage, RootLockfileNode},
|
2024-03-04 20:18:49 +00:00
|
|
|
manifest::{Manifest, ManifestReadError, PathStyle, Realm},
|
|
|
|
package_name::PackageName,
|
|
|
|
project::Project,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TypeVisitor {
|
|
|
|
pub(crate) types: Vec<String>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Visitor for TypeVisitor {
|
|
|
|
fn visit_exported_type_declaration(&mut self, node: &ExportedTypeDeclaration) {
|
|
|
|
let name = node.type_declaration().type_name().to_string();
|
|
|
|
|
|
|
|
let (declaration_generics, generics) =
|
|
|
|
if let Some(declaration) = node.type_declaration().generics() {
|
|
|
|
let mut declaration_generics = vec![];
|
|
|
|
let mut generics = vec![];
|
|
|
|
|
|
|
|
for generic in declaration.generics().iter() {
|
|
|
|
declaration_generics.push(generic.to_string());
|
|
|
|
|
|
|
|
if generic.default_type().is_some() {
|
|
|
|
generics.push(generic.parameter().to_string())
|
|
|
|
} else {
|
|
|
|
generics.push(generic.to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
(
|
|
|
|
format!("<{}>", declaration_generics.join(", ")),
|
|
|
|
format!("<{}>", generics.join(", ")),
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
("".to_string(), "".to_string())
|
|
|
|
};
|
|
|
|
|
|
|
|
self.types.push(format!(
|
|
|
|
"export type {name}{declaration_generics} = module.{name}{generics}\n"
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Generates the contents of a linking file, given the require path, and the contents of the target file
|
|
|
|
/// The contents will be scanned for type exports, and the linking file will be generated accordingly
|
|
|
|
pub fn linking_file(content: &str, path: &str) -> Result<String, full_moon::Error> {
|
|
|
|
let mut linker = format!("local module = require({path})\n");
|
|
|
|
let mut visitor = TypeVisitor { types: vec![] };
|
|
|
|
|
|
|
|
parse(content)?.nodes().visit(&mut visitor);
|
|
|
|
|
|
|
|
for ty in visitor.types {
|
|
|
|
linker.push_str(&ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
linker.push_str("return module");
|
|
|
|
|
|
|
|
Ok(linker)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Error)]
|
|
|
|
/// An error that occurred while linking dependencies
|
|
|
|
pub enum LinkingError {
|
|
|
|
#[error("error interacting with the file system")]
|
|
|
|
/// An error that occurred while interacting with the file system
|
|
|
|
Io(#[from] std::io::Error),
|
|
|
|
|
|
|
|
#[error("failed getting file name from {0}")]
|
|
|
|
/// An error that occurred while getting a file name
|
|
|
|
FileNameFail(PathBuf),
|
|
|
|
|
|
|
|
#[error("failed converting file name to string")]
|
|
|
|
/// An error that occurred while converting a file name to a string
|
|
|
|
FileNameToStringFail,
|
|
|
|
|
|
|
|
#[error("failed getting relative path from {0} to {1}")]
|
|
|
|
/// An error that occurred while getting a relative path
|
|
|
|
RelativePathFail(PathBuf, PathBuf),
|
|
|
|
|
|
|
|
#[error("failed getting path parent of {0}")]
|
|
|
|
/// An error that occurred while getting a path parent
|
|
|
|
ParentFail(PathBuf),
|
|
|
|
|
|
|
|
#[error("failed to convert path component to string")]
|
|
|
|
/// An error that occurred while converting a path component to a string
|
|
|
|
ComponentToStringFail,
|
|
|
|
|
|
|
|
#[error("failed to get path string")]
|
|
|
|
/// An error that occurred while getting a path string
|
|
|
|
PathToStringFail,
|
|
|
|
|
|
|
|
#[error("error encoding utf-8 string")]
|
|
|
|
/// An error that occurred while converting a byte slice to a string
|
|
|
|
Utf8(#[from] std::str::Utf8Error),
|
|
|
|
|
|
|
|
#[error("error reading manifest")]
|
|
|
|
/// An error that occurred while reading the manifest of a package
|
|
|
|
ManifestRead(#[from] ManifestReadError),
|
|
|
|
|
|
|
|
#[error("missing realm {0} in-game path")]
|
|
|
|
/// An error that occurred while getting the in-game path for a realm
|
|
|
|
MissingRealmInGamePath(Realm),
|
|
|
|
|
|
|
|
#[error("library source is not valid Luau")]
|
|
|
|
/// An error that occurred because the library source is not valid Luau
|
|
|
|
InvalidLuau(#[from] full_moon::Error),
|
|
|
|
}
|
|
|
|
|
2024-03-24 13:31:11 +00:00
|
|
|
pub(crate) fn link<P: AsRef<Path>, Q: AsRef<Path>>(
|
|
|
|
project: &Project,
|
2024-03-04 20:18:49 +00:00
|
|
|
resolved_pkg: &ResolvedPackage,
|
2024-03-25 16:29:31 +00:00
|
|
|
lockfile: &RootLockfileNode,
|
2024-03-04 20:18:49 +00:00
|
|
|
destination_dir: P,
|
|
|
|
parent_dependency_packages_dir: Q,
|
2024-03-24 17:38:18 +00:00
|
|
|
only_name: bool,
|
2024-03-04 20:18:49 +00:00
|
|
|
) -> Result<(), LinkingError> {
|
|
|
|
let (_, source_dir) = resolved_pkg.directory(project.path());
|
|
|
|
let file = Manifest::from_path(&source_dir)?;
|
|
|
|
|
2024-03-24 13:31:11 +00:00
|
|
|
let Some(relative_lib_export) = file.exports.lib else {
|
2024-03-04 20:18:49 +00:00
|
|
|
return Ok(());
|
|
|
|
};
|
|
|
|
|
2024-03-24 13:31:11 +00:00
|
|
|
let lib_export = relative_lib_export.to_path(&source_dir);
|
2024-03-04 20:18:49 +00:00
|
|
|
|
|
|
|
let path_style = &project.manifest().path_style;
|
|
|
|
let PathStyle::Roblox { place } = &path_style;
|
|
|
|
|
|
|
|
debug!("linking {resolved_pkg} using `{}` path style", path_style);
|
|
|
|
|
2024-03-24 13:31:11 +00:00
|
|
|
let pkg_name = resolved_pkg.pkg_ref.name();
|
|
|
|
let name = pkg_name.name();
|
2024-03-04 20:18:49 +00:00
|
|
|
|
2024-03-25 16:29:31 +00:00
|
|
|
let destination_dir = match lockfile
|
|
|
|
.specifiers
|
|
|
|
.get(&pkg_name)
|
|
|
|
.and_then(|v| v.get(resolved_pkg.pkg_ref.version()))
|
|
|
|
{
|
|
|
|
Some(specifier) => project.path().join(packages_folder(
|
|
|
|
specifier.realm().copied().unwrap_or_default(),
|
|
|
|
)),
|
|
|
|
None => destination_dir.as_ref().to_path_buf(),
|
2024-03-04 20:18:49 +00:00
|
|
|
};
|
|
|
|
|
2024-03-24 17:38:18 +00:00
|
|
|
let destination_file = destination_dir.join(format!(
|
|
|
|
"{}{}.lua",
|
|
|
|
if only_name { "" } else { pkg_name.prefix() },
|
|
|
|
name
|
|
|
|
));
|
2024-03-04 20:18:49 +00:00
|
|
|
|
|
|
|
let realm_folder = project.path().join(resolved_pkg.packages_folder());
|
|
|
|
let in_different_folders = realm_folder != parent_dependency_packages_dir.as_ref();
|
|
|
|
|
|
|
|
let mut path = if in_different_folders {
|
|
|
|
pathdiff::diff_paths(&source_dir, &realm_folder)
|
|
|
|
.ok_or_else(|| LinkingError::RelativePathFail(source_dir.clone(), realm_folder))?
|
|
|
|
} else {
|
|
|
|
pathdiff::diff_paths(&source_dir, &destination_dir).ok_or_else(|| {
|
|
|
|
LinkingError::RelativePathFail(source_dir.clone(), destination_dir.to_path_buf())
|
|
|
|
})?
|
|
|
|
};
|
|
|
|
path.set_extension("");
|
|
|
|
|
|
|
|
let beginning = if in_different_folders {
|
|
|
|
place
|
|
|
|
.get(&resolved_pkg.realm)
|
|
|
|
.ok_or_else(|| LinkingError::MissingRealmInGamePath(resolved_pkg.realm))?
|
|
|
|
.clone()
|
|
|
|
} else if name == "init" {
|
|
|
|
"script".to_string()
|
|
|
|
} else {
|
|
|
|
"script.Parent".to_string()
|
|
|
|
};
|
|
|
|
|
2024-03-24 18:27:10 +00:00
|
|
|
let mut components = path
|
|
|
|
.components()
|
|
|
|
.map(|component| {
|
2024-03-04 20:18:49 +00:00
|
|
|
Ok(match component {
|
|
|
|
Component::ParentDir => ".Parent".to_string(),
|
|
|
|
Component::Normal(part) => format!(
|
|
|
|
"[{:?}]",
|
|
|
|
part.to_str().ok_or(LinkingError::ComponentToStringFail)?
|
|
|
|
),
|
|
|
|
_ => unreachable!("invalid path component"),
|
|
|
|
})
|
2024-03-24 18:27:10 +00:00
|
|
|
})
|
|
|
|
.collect::<Result<Vec<_>, LinkingError>>()?;
|
|
|
|
components.pop();
|
|
|
|
|
|
|
|
let path = beginning + &components.join("") + &format!("[{name:?}]");
|
2024-03-04 20:18:49 +00:00
|
|
|
|
|
|
|
debug!(
|
|
|
|
"writing linking file for {} with import `{path}` to {}",
|
|
|
|
source_dir.display(),
|
|
|
|
destination_file.display()
|
|
|
|
);
|
|
|
|
|
2024-03-24 13:31:11 +00:00
|
|
|
let file_contents = match relative_lib_export.as_str() {
|
|
|
|
"true" => "".to_string(),
|
|
|
|
_ => read_to_string(lib_export)?,
|
|
|
|
};
|
2024-03-04 20:18:49 +00:00
|
|
|
|
2024-03-24 13:31:11 +00:00
|
|
|
let linking_file_contents = linking_file(&file_contents, &path)?;
|
2024-03-04 20:18:49 +00:00
|
|
|
|
|
|
|
write(&destination_file, linking_file_contents)?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Error)]
|
|
|
|
#[error("error linking {1}@{2} to {3}@{4}")]
|
|
|
|
/// An error that occurred while linking the dependencies
|
|
|
|
pub struct LinkingDependenciesError(
|
|
|
|
#[source] LinkingError,
|
|
|
|
PackageName,
|
|
|
|
Version,
|
|
|
|
PackageName,
|
|
|
|
Version,
|
|
|
|
);
|
|
|
|
|
2024-03-25 16:29:31 +00:00
|
|
|
fn is_duplicate_in<T: PartialEq>(item: T, items: &[T]) -> bool {
|
|
|
|
let mut count = 0u8;
|
|
|
|
items.iter().any(|i| {
|
|
|
|
if i == &item {
|
|
|
|
count += 1;
|
|
|
|
}
|
|
|
|
count > 1
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-03-24 13:31:11 +00:00
|
|
|
impl Project {
|
2024-03-04 20:18:49 +00:00
|
|
|
/// Links the dependencies of the project
|
|
|
|
pub fn link_dependencies(
|
|
|
|
&self,
|
2024-03-25 16:29:31 +00:00
|
|
|
lockfile: &RootLockfileNode,
|
2024-03-04 20:18:49 +00:00
|
|
|
) -> Result<(), LinkingDependenciesError> {
|
2024-03-25 16:29:31 +00:00
|
|
|
let root_deps = lockfile.specifiers.keys().collect::<HashSet<_>>();
|
|
|
|
let root_dep_names = root_deps.iter().map(|n| n.name()).collect::<Vec<_>>();
|
2024-03-24 17:38:18 +00:00
|
|
|
|
2024-03-25 16:29:31 +00:00
|
|
|
for (name, versions) in &lockfile.children {
|
2024-03-04 20:18:49 +00:00
|
|
|
for (version, resolved_pkg) in versions {
|
|
|
|
let (container_dir, _) = resolved_pkg.directory(self.path());
|
|
|
|
|
|
|
|
debug!(
|
|
|
|
"linking package {name}@{version}'s dependencies to directory {}",
|
|
|
|
container_dir.display()
|
|
|
|
);
|
|
|
|
|
2024-03-25 16:29:31 +00:00
|
|
|
let resolved_pkg_dep_names = resolved_pkg
|
|
|
|
.dependencies
|
|
|
|
.iter()
|
|
|
|
.map(|(n, _)| n.name())
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
2024-03-04 20:18:49 +00:00
|
|
|
for (dep_name, dep_version) in &resolved_pkg.dependencies {
|
2024-03-25 16:29:31 +00:00
|
|
|
let dep = lockfile
|
|
|
|
.children
|
2024-03-04 20:18:49 +00:00
|
|
|
.get(dep_name)
|
|
|
|
.and_then(|versions| versions.get(dep_version))
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
link(
|
|
|
|
self,
|
|
|
|
dep,
|
2024-03-25 16:29:31 +00:00
|
|
|
lockfile,
|
2024-03-04 20:18:49 +00:00
|
|
|
&container_dir,
|
|
|
|
&self.path().join(resolved_pkg.packages_folder()),
|
2024-03-25 16:29:31 +00:00
|
|
|
!is_duplicate_in(dep_name.name(), &resolved_pkg_dep_names),
|
2024-03-04 20:18:49 +00:00
|
|
|
)
|
|
|
|
.map_err(|e| {
|
|
|
|
LinkingDependenciesError(
|
|
|
|
e,
|
|
|
|
dep_name.clone(),
|
|
|
|
dep_version.clone(),
|
|
|
|
name.clone(),
|
|
|
|
version.clone(),
|
|
|
|
)
|
|
|
|
})?;
|
|
|
|
}
|
|
|
|
|
2024-03-25 16:29:31 +00:00
|
|
|
if root_deps.contains(&name) {
|
2024-03-04 20:18:49 +00:00
|
|
|
let linking_dir = &self.path().join(resolved_pkg.packages_folder());
|
|
|
|
|
|
|
|
debug!(
|
|
|
|
"linking root package {name}@{version} to directory {}",
|
|
|
|
linking_dir.display()
|
|
|
|
);
|
|
|
|
|
2024-03-24 17:38:18 +00:00
|
|
|
link(
|
|
|
|
self,
|
|
|
|
resolved_pkg,
|
2024-03-25 16:29:31 +00:00
|
|
|
lockfile,
|
2024-03-24 17:38:18 +00:00
|
|
|
linking_dir,
|
|
|
|
linking_dir,
|
2024-03-25 16:29:31 +00:00
|
|
|
!is_duplicate_in(name.name(), &root_dep_names),
|
2024-03-24 17:38:18 +00:00
|
|
|
)
|
|
|
|
.map_err(|e| {
|
2024-03-04 20:18:49 +00:00
|
|
|
LinkingDependenciesError(
|
|
|
|
e,
|
|
|
|
name.clone(),
|
|
|
|
version.clone(),
|
|
|
|
name.clone(),
|
|
|
|
version.clone(),
|
|
|
|
)
|
|
|
|
})?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|