mirror of
https://github.com/pesde-pkg/pesde.git
synced 2025-04-06 03:40:59 +01:00
feat: add alias validation
Ensures aliases don't contain characters which could cause issues. They are now also forbidden from being the same as an engine name to avoid issues.
This commit is contained in:
parent
a33302aff9
commit
5ace844035
19 changed files with 182 additions and 100 deletions
|
@ -3,7 +3,7 @@ use chrono::{DateTime, Utc};
|
||||||
use pesde::{
|
use pesde::{
|
||||||
manifest::{
|
manifest::{
|
||||||
target::{Target, TargetKind},
|
target::{Target, TargetKind},
|
||||||
DependencyType,
|
Alias, DependencyType,
|
||||||
},
|
},
|
||||||
names::PackageName,
|
names::PackageName,
|
||||||
source::{
|
source::{
|
||||||
|
@ -125,7 +125,7 @@ pub struct PackageResponseInner {
|
||||||
#[serde(skip_serializing_if = "BTreeSet::is_empty")]
|
#[serde(skip_serializing_if = "BTreeSet::is_empty")]
|
||||||
docs: BTreeSet<RegistryDocEntry>,
|
docs: BTreeSet<RegistryDocEntry>,
|
||||||
#[serde(skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
dependencies: BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
dependencies: BTreeMap<Alias, (DependencySpecifiers, DependencyType)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PackageResponseInner {
|
impl PackageResponseInner {
|
||||||
|
|
|
@ -7,7 +7,7 @@ use semver::VersionReq;
|
||||||
|
|
||||||
use crate::cli::{config::read_config, AnyPackageIdentifier, VersionedPackageName};
|
use crate::cli::{config::read_config, AnyPackageIdentifier, VersionedPackageName};
|
||||||
use pesde::{
|
use pesde::{
|
||||||
manifest::target::TargetKind,
|
manifest::{target::TargetKind, Alias},
|
||||||
names::PackageNames,
|
names::PackageNames,
|
||||||
source::{
|
source::{
|
||||||
git::{specifier::GitDependencySpecifier, GitPackageSource},
|
git::{specifier::GitDependencySpecifier, GitPackageSource},
|
||||||
|
@ -37,7 +37,7 @@ pub struct AddCommand {
|
||||||
|
|
||||||
/// The alias to use for the package
|
/// The alias to use for the package
|
||||||
#[arg(short, long)]
|
#[arg(short, long)]
|
||||||
alias: Option<String>,
|
alias: Option<Alias>,
|
||||||
|
|
||||||
/// Whether to add the package as a peer dependency
|
/// Whether to add the package as a peer dependency
|
||||||
#[arg(short, long)]
|
#[arg(short, long)]
|
||||||
|
@ -180,7 +180,9 @@ impl AddCommand {
|
||||||
"dependencies"
|
"dependencies"
|
||||||
};
|
};
|
||||||
|
|
||||||
let alias = self.alias.unwrap_or_else(|| match &self.name {
|
let alias = match self.alias {
|
||||||
|
Some(alias) => alias,
|
||||||
|
None => match &self.name {
|
||||||
AnyPackageIdentifier::PackageName(versioned) => versioned.0.name().to_string(),
|
AnyPackageIdentifier::PackageName(versioned) => versioned.0.name().to_string(),
|
||||||
AnyPackageIdentifier::Url((url, _)) => url
|
AnyPackageIdentifier::Url((url, _)) => url
|
||||||
.path
|
.path
|
||||||
|
@ -194,10 +196,13 @@ impl AddCommand {
|
||||||
.file_name()
|
.file_name()
|
||||||
.map(|s| s.to_string_lossy().to_string())
|
.map(|s| s.to_string_lossy().to_string())
|
||||||
.expect("path has no file name"),
|
.expect("path has no file name"),
|
||||||
});
|
}
|
||||||
|
.parse()
|
||||||
|
.context("auto-generated alias is invalid. use --alias to specify one")?,
|
||||||
|
};
|
||||||
|
|
||||||
let field = &mut manifest[dependency_key]
|
let field = &mut manifest[dependency_key]
|
||||||
.or_insert(toml_edit::Item::Table(toml_edit::Table::new()))[&alias];
|
.or_insert(toml_edit::Item::Table(toml_edit::Table::new()))[alias.as_str()];
|
||||||
|
|
||||||
match specifier {
|
match specifier {
|
||||||
DependencySpecifiers::Pesde(spec) => {
|
DependencySpecifiers::Pesde(spec) => {
|
||||||
|
|
|
@ -259,7 +259,7 @@ impl InitCommand {
|
||||||
continue;
|
continue;
|
||||||
};
|
};
|
||||||
|
|
||||||
let field = &mut dev_deps[alias];
|
let field = &mut dev_deps[alias.as_str()];
|
||||||
field["name"] = toml_edit::value(spec.name.to_string());
|
field["name"] = toml_edit::value(spec.name.to_string());
|
||||||
field["version"] = toml_edit::value(spec.version.to_string());
|
field["version"] = toml_edit::value(spec.version.to_string());
|
||||||
field["target"] = toml_edit::value(
|
field["target"] = toml_edit::value(
|
||||||
|
|
|
@ -12,7 +12,7 @@ use pesde::{
|
||||||
engine::EngineKind,
|
engine::EngineKind,
|
||||||
graph::{DependencyGraph, DependencyGraphWithTarget},
|
graph::{DependencyGraph, DependencyGraphWithTarget},
|
||||||
lockfile::Lockfile,
|
lockfile::Lockfile,
|
||||||
manifest::{target::TargetKind, DependencyType, Manifest},
|
manifest::{target::TargetKind, Alias, DependencyType, Manifest},
|
||||||
names::PackageNames,
|
names::PackageNames,
|
||||||
source::{pesde::PesdePackageSource, refs::PackageRefs},
|
source::{pesde::PesdePackageSource, refs::PackageRefs},
|
||||||
version_matches, Project, RefreshedSources, LOCKFILE_FILE_NAME, MANIFEST_FILE_NAME,
|
version_matches, Project, RefreshedSources, LOCKFILE_FILE_NAME, MANIFEST_FILE_NAME,
|
||||||
|
@ -25,7 +25,7 @@ use std::{
|
||||||
};
|
};
|
||||||
use tokio::task::JoinSet;
|
use tokio::task::JoinSet;
|
||||||
|
|
||||||
fn bin_link_file(alias: &str) -> String {
|
fn bin_link_file(alias: &Alias) -> String {
|
||||||
let mut all_combinations = BTreeSet::new();
|
let mut all_combinations = BTreeSet::new();
|
||||||
|
|
||||||
for a in TargetKind::VARIANTS {
|
for a in TargetKind::VARIANTS {
|
||||||
|
@ -70,23 +70,13 @@ impl DownloadAndLinkHooks for InstallHooks {
|
||||||
.values()
|
.values()
|
||||||
.filter(|node| node.target.bin_path().is_some())
|
.filter(|node| node.target.bin_path().is_some())
|
||||||
.filter_map(|node| node.node.direct.as_ref())
|
.filter_map(|node| node.node.direct.as_ref())
|
||||||
.map(|(alias, _, _)| alias)
|
.map(|(alias, _, _)| {
|
||||||
.filter(|alias| {
|
|
||||||
if *alias == env!("CARGO_BIN_NAME") {
|
|
||||||
tracing::warn!(
|
|
||||||
"package {alias} has the same name as the CLI, skipping bin link"
|
|
||||||
);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
true
|
|
||||||
})
|
|
||||||
.map(|alias| {
|
|
||||||
let bin_folder = self.bin_folder.clone();
|
let bin_folder = self.bin_folder.clone();
|
||||||
let alias = alias.clone();
|
let alias = alias.clone();
|
||||||
|
|
||||||
async move {
|
async move {
|
||||||
let bin_exec_file = bin_folder
|
let bin_exec_file = bin_folder
|
||||||
.join(&alias)
|
.join(alias.as_str())
|
||||||
.with_extension(std::env::consts::EXE_EXTENSION);
|
.with_extension(std::env::consts::EXE_EXTENSION);
|
||||||
|
|
||||||
let impl_folder = bin_folder.join(".impl");
|
let impl_folder = bin_folder.join(".impl");
|
||||||
|
@ -94,7 +84,7 @@ impl DownloadAndLinkHooks for InstallHooks {
|
||||||
.await
|
.await
|
||||||
.context("failed to create bin link folder")?;
|
.context("failed to create bin link folder")?;
|
||||||
|
|
||||||
let bin_file = impl_folder.join(&alias).with_extension("luau");
|
let bin_file = impl_folder.join(alias.as_str()).with_extension("luau");
|
||||||
fs::write(&bin_file, bin_link_file(&alias))
|
fs::write(&bin_file, bin_link_file(&alias))
|
||||||
.await
|
.await
|
||||||
.context("failed to write bin link file")?;
|
.context("failed to write bin link file")?;
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::{
|
manifest::{
|
||||||
target::{Target, TargetKind},
|
target::{Target, TargetKind},
|
||||||
DependencyType,
|
Alias, DependencyType,
|
||||||
},
|
},
|
||||||
source::{
|
source::{
|
||||||
ids::{PackageId, VersionId},
|
ids::{PackageId, VersionId},
|
||||||
|
@ -22,10 +22,10 @@ pub type Graph<Node> = BTreeMap<PackageId, Node>;
|
||||||
pub struct DependencyGraphNode {
|
pub struct DependencyGraphNode {
|
||||||
/// The alias, specifier, and original (as in the manifest) type for the dependency, if it is a direct dependency (i.e. used by the current project)
|
/// The alias, specifier, and original (as in the manifest) type for the dependency, if it is a direct dependency (i.e. used by the current project)
|
||||||
#[serde(default, skip_serializing_if = "Option::is_none")]
|
#[serde(default, skip_serializing_if = "Option::is_none")]
|
||||||
pub direct: Option<(String, DependencySpecifiers, DependencyType)>,
|
pub direct: Option<(Alias, DependencySpecifiers, DependencyType)>,
|
||||||
/// The dependencies of the package
|
/// The dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<PackageId, String>,
|
pub dependencies: BTreeMap<PackageId, Alias>,
|
||||||
/// The resolved (transformed, for example Peer -> Standard) type of the dependency
|
/// The resolved (transformed, for example Peer -> Standard) type of the dependency
|
||||||
pub resolved_ty: DependencyType,
|
pub resolved_ty: DependencyType,
|
||||||
/// Whether the resolved type should be Peer if this isn't depended on
|
/// Whether the resolved type should be Peer if this isn't depended on
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
graph::{DependencyGraphNodeWithTarget, DependencyGraphWithTarget},
|
graph::{DependencyGraphNodeWithTarget, DependencyGraphWithTarget},
|
||||||
linking::generator::get_file_types,
|
linking::generator::get_file_types,
|
||||||
manifest::Manifest,
|
manifest::{Alias, Manifest},
|
||||||
scripts::{execute_script, ExecuteScriptHooks, ScriptName},
|
scripts::{execute_script, ExecuteScriptHooks, ScriptName},
|
||||||
source::{
|
source::{
|
||||||
fs::{cas_path, store_in_cas},
|
fs::{cas_path, store_in_cas},
|
||||||
|
@ -169,7 +169,7 @@ impl Project {
|
||||||
relative_container_folder: &Path,
|
relative_container_folder: &Path,
|
||||||
node: &DependencyGraphNodeWithTarget,
|
node: &DependencyGraphNodeWithTarget,
|
||||||
package_id: &PackageId,
|
package_id: &PackageId,
|
||||||
alias: &str,
|
alias: &Alias,
|
||||||
package_types: &Arc<PackageTypes>,
|
package_types: &Arc<PackageTypes>,
|
||||||
manifest: &Arc<Manifest>,
|
manifest: &Arc<Manifest>,
|
||||||
remove: bool,
|
remove: bool,
|
||||||
|
@ -243,7 +243,8 @@ impl Project {
|
||||||
.filter(|s| !s.is_empty() && node.node.direct.is_some() && is_root)
|
.filter(|s| !s.is_empty() && node.node.direct.is_some() && is_root)
|
||||||
{
|
{
|
||||||
let scripts_container = self.package_dir().join(SCRIPTS_LINK_FOLDER);
|
let scripts_container = self.package_dir().join(SCRIPTS_LINK_FOLDER);
|
||||||
let scripts_base = create_and_canonicalize(scripts_container.join(alias)).await?;
|
let scripts_base =
|
||||||
|
create_and_canonicalize(scripts_container.join(alias.as_str())).await?;
|
||||||
|
|
||||||
if remove {
|
if remove {
|
||||||
tasks.spawn(async move {
|
tasks.spawn(async move {
|
||||||
|
|
|
@ -41,7 +41,7 @@ pub mod old {
|
||||||
manifest::{
|
manifest::{
|
||||||
overrides::OverrideKey,
|
overrides::OverrideKey,
|
||||||
target::{Target, TargetKind},
|
target::{Target, TargetKind},
|
||||||
DependencyType,
|
Alias, DependencyType,
|
||||||
},
|
},
|
||||||
names::{PackageName, PackageNames},
|
names::{PackageName, PackageNames},
|
||||||
source::{
|
source::{
|
||||||
|
@ -60,10 +60,10 @@ pub mod old {
|
||||||
pub struct DependencyGraphNodeOld {
|
pub struct DependencyGraphNodeOld {
|
||||||
/// The alias, specifier, and original (as in the manifest) type for the dependency, if it is a direct dependency (i.e. used by the current project)
|
/// The alias, specifier, and original (as in the manifest) type for the dependency, if it is a direct dependency (i.e. used by the current project)
|
||||||
#[serde(default, skip_serializing_if = "Option::is_none")]
|
#[serde(default, skip_serializing_if = "Option::is_none")]
|
||||||
pub direct: Option<(String, DependencySpecifiers, DependencyType)>,
|
pub direct: Option<(Alias, DependencySpecifiers, DependencyType)>,
|
||||||
/// The dependencies of the package
|
/// The dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<PackageNames, (VersionId, String)>,
|
pub dependencies: BTreeMap<PackageNames, (VersionId, Alias)>,
|
||||||
/// The resolved (transformed, for example Peer -> Standard) type of the dependency
|
/// The resolved (transformed, for example Peer -> Standard) type of the dependency
|
||||||
pub resolved_ty: DependencyType,
|
pub resolved_ty: DependencyType,
|
||||||
/// Whether the resolved type should be Peer if this isn't depended on
|
/// Whether the resolved type should be Peer if this isn't depended on
|
||||||
|
|
|
@ -10,7 +10,12 @@ use crate::{
|
||||||
use relative_path::RelativePathBuf;
|
use relative_path::RelativePathBuf;
|
||||||
use semver::{Version, VersionReq};
|
use semver::{Version, VersionReq};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use std::collections::{BTreeMap, HashMap};
|
use serde_with::{DeserializeFromStr, SerializeDisplay};
|
||||||
|
use std::{
|
||||||
|
collections::{BTreeMap, HashMap},
|
||||||
|
fmt::Display,
|
||||||
|
str::FromStr,
|
||||||
|
};
|
||||||
use tracing::instrument;
|
use tracing::instrument;
|
||||||
|
|
||||||
/// Overrides
|
/// Overrides
|
||||||
|
@ -99,19 +104,75 @@ pub struct Manifest {
|
||||||
|
|
||||||
/// The standard dependencies of the package
|
/// The standard dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<String, DependencySpecifiers>,
|
pub dependencies: BTreeMap<Alias, DependencySpecifiers>,
|
||||||
/// The peer dependencies of the package
|
/// The peer dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub peer_dependencies: BTreeMap<String, DependencySpecifiers>,
|
pub peer_dependencies: BTreeMap<Alias, DependencySpecifiers>,
|
||||||
/// The dev dependencies of the package
|
/// The dev dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dev_dependencies: BTreeMap<String, DependencySpecifiers>,
|
pub dev_dependencies: BTreeMap<Alias, DependencySpecifiers>,
|
||||||
/// The user-defined fields of the package
|
/// The user-defined fields of the package
|
||||||
#[cfg_attr(feature = "schema", schemars(skip))]
|
#[cfg_attr(feature = "schema", schemars(skip))]
|
||||||
#[serde(flatten)]
|
#[serde(flatten)]
|
||||||
pub user_defined_fields: HashMap<String, toml::Value>,
|
pub user_defined_fields: HashMap<String, toml::Value>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// An alias of a dependency
|
||||||
|
#[derive(
|
||||||
|
SerializeDisplay, DeserializeFromStr, Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord,
|
||||||
|
)]
|
||||||
|
pub struct Alias(String);
|
||||||
|
|
||||||
|
impl Display for Alias {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.pad(&self.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FromStr for Alias {
|
||||||
|
type Err = errors::AliasFromStr;
|
||||||
|
|
||||||
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||||
|
if s.is_empty() {
|
||||||
|
return Err(errors::AliasFromStr::Empty);
|
||||||
|
}
|
||||||
|
|
||||||
|
if !s
|
||||||
|
.chars()
|
||||||
|
.all(|c| c.is_ascii_alphanumeric() || c == '-' || c == '_')
|
||||||
|
{
|
||||||
|
return Err(errors::AliasFromStr::InvalidCharacters(s.to_string()));
|
||||||
|
}
|
||||||
|
|
||||||
|
if EngineKind::from_str(s).is_ok() {
|
||||||
|
return Err(errors::AliasFromStr::EngineName(s.to_string()));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Self(s.to_string()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "schema")]
|
||||||
|
impl schemars::JsonSchema for Alias {
|
||||||
|
fn schema_name() -> std::borrow::Cow<'static, str> {
|
||||||
|
"Alias".into()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn json_schema(_: &mut schemars::SchemaGenerator) -> schemars::Schema {
|
||||||
|
schemars::json_schema!({
|
||||||
|
"type": "string",
|
||||||
|
"pattern": r#"^[a-zA-Z0-9_-]+$"#,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Alias {
|
||||||
|
/// Get the alias as a string
|
||||||
|
pub fn as_str(&self) -> &str {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// A dependency type
|
/// A dependency type
|
||||||
#[derive(Serialize, Deserialize, Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
#[derive(Serialize, Deserialize, Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
||||||
#[serde(rename_all = "snake_case")]
|
#[serde(rename_all = "snake_case")]
|
||||||
|
@ -129,10 +190,8 @@ impl Manifest {
|
||||||
#[instrument(skip(self), ret(level = "trace"), level = "debug")]
|
#[instrument(skip(self), ret(level = "trace"), level = "debug")]
|
||||||
pub fn all_dependencies(
|
pub fn all_dependencies(
|
||||||
&self,
|
&self,
|
||||||
) -> Result<
|
) -> Result<BTreeMap<Alias, (DependencySpecifiers, DependencyType)>, errors::AllDependenciesError>
|
||||||
BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
{
|
||||||
errors::AllDependenciesError,
|
|
||||||
> {
|
|
||||||
let mut all_deps = BTreeMap::new();
|
let mut all_deps = BTreeMap::new();
|
||||||
|
|
||||||
for (deps, ty) in [
|
for (deps, ty) in [
|
||||||
|
@ -153,14 +212,32 @@ impl Manifest {
|
||||||
|
|
||||||
/// Errors that can occur when interacting with manifests
|
/// Errors that can occur when interacting with manifests
|
||||||
pub mod errors {
|
pub mod errors {
|
||||||
|
use crate::manifest::Alias;
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
|
|
||||||
|
/// Errors that can occur when parsing an alias from a string
|
||||||
|
#[derive(Debug, Error)]
|
||||||
|
#[non_exhaustive]
|
||||||
|
pub enum AliasFromStr {
|
||||||
|
/// The alias is empty
|
||||||
|
#[error("the alias is empty")]
|
||||||
|
Empty,
|
||||||
|
|
||||||
|
/// The alias contains characters outside a-z, A-Z, 0-9, -, and _
|
||||||
|
#[error("alias `{0}` contains characters outside a-z, A-Z, 0-9, -, and _")]
|
||||||
|
InvalidCharacters(String),
|
||||||
|
|
||||||
|
/// The alias is an engine name
|
||||||
|
#[error("alias `{0}` is an engine name")]
|
||||||
|
EngineName(String),
|
||||||
|
}
|
||||||
|
|
||||||
/// Errors that can occur when trying to get all dependencies from a manifest
|
/// Errors that can occur when trying to get all dependencies from a manifest
|
||||||
#[derive(Debug, Error)]
|
#[derive(Debug, Error)]
|
||||||
#[non_exhaustive]
|
#[non_exhaustive]
|
||||||
pub enum AllDependenciesError {
|
pub enum AllDependenciesError {
|
||||||
/// Another specifier is already using the alias
|
/// Another specifier is already using the alias
|
||||||
#[error("another specifier is already using the alias {0}")]
|
#[error("another specifier is already using the alias {0}")]
|
||||||
AliasConflict(String),
|
AliasConflict(Alias),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use crate::source::specifiers::DependencySpecifiers;
|
use crate::{manifest::Alias, source::specifiers::DependencySpecifiers};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use serde_with::{DeserializeFromStr, SerializeDisplay};
|
use serde_with::{DeserializeFromStr, SerializeDisplay};
|
||||||
use std::{
|
use std::{
|
||||||
|
@ -10,7 +10,7 @@ use std::{
|
||||||
#[derive(
|
#[derive(
|
||||||
Debug, DeserializeFromStr, SerializeDisplay, Clone, PartialEq, Eq, Hash, PartialOrd, Ord,
|
Debug, DeserializeFromStr, SerializeDisplay, Clone, PartialEq, Eq, Hash, PartialOrd, Ord,
|
||||||
)]
|
)]
|
||||||
pub struct OverrideKey(pub Vec<Vec<String>>);
|
pub struct OverrideKey(pub Vec<Vec<Alias>>);
|
||||||
|
|
||||||
impl FromStr for OverrideKey {
|
impl FromStr for OverrideKey {
|
||||||
type Err = errors::OverrideKeyFromStr;
|
type Err = errors::OverrideKeyFromStr;
|
||||||
|
@ -18,8 +18,13 @@ impl FromStr for OverrideKey {
|
||||||
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||||
let overrides = s
|
let overrides = s
|
||||||
.split(',')
|
.split(',')
|
||||||
.map(|overrides| overrides.split('>').map(ToString::to_string).collect())
|
.map(|overrides| {
|
||||||
.collect::<Vec<Vec<String>>>();
|
overrides
|
||||||
|
.split('>')
|
||||||
|
.map(Alias::from_str)
|
||||||
|
.collect::<Result<_, _>>()
|
||||||
|
})
|
||||||
|
.collect::<Result<Vec<Vec<Alias>>, _>>()?;
|
||||||
|
|
||||||
if overrides.is_empty() {
|
if overrides.is_empty() {
|
||||||
return Err(errors::OverrideKeyFromStr::Empty);
|
return Err(errors::OverrideKeyFromStr::Empty);
|
||||||
|
@ -38,7 +43,7 @@ impl schemars::JsonSchema for OverrideKey {
|
||||||
fn json_schema(_: &mut schemars::SchemaGenerator) -> schemars::Schema {
|
fn json_schema(_: &mut schemars::SchemaGenerator) -> schemars::Schema {
|
||||||
schemars::json_schema!({
|
schemars::json_schema!({
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"pattern": r#"^([a-zA-Z]+(>[a-zA-Z]+)+)(,([a-zA-Z]+(>[a-zA-Z]+)+))*$"#,
|
"pattern": r#"^(?:[a-zA-Z0-9_-]+>[a-zA-Z0-9_-]+(?:>[a-zA-Z0-9_-]+)*)(?:,(?:[a-zA-Z0-9_-]+>[a-zA-Z0-9_-]+(?:>[a-zA-Z0-9_-]+)*))*$"#,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -53,7 +58,7 @@ impl Display for OverrideKey {
|
||||||
.map(|overrides| {
|
.map(|overrides| {
|
||||||
overrides
|
overrides
|
||||||
.iter()
|
.iter()
|
||||||
.map(String::as_str)
|
.map(Alias::as_str)
|
||||||
.collect::<Vec<_>>()
|
.collect::<Vec<_>>()
|
||||||
.join(">")
|
.join(">")
|
||||||
})
|
})
|
||||||
|
@ -71,7 +76,7 @@ pub enum OverrideSpecifier {
|
||||||
/// A specifier for a dependency
|
/// A specifier for a dependency
|
||||||
Specifier(DependencySpecifiers),
|
Specifier(DependencySpecifiers),
|
||||||
/// An alias for a dependency the current project depends on
|
/// An alias for a dependency the current project depends on
|
||||||
Alias(String),
|
Alias(Alias),
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Errors that can occur when interacting with override keys
|
/// Errors that can occur when interacting with override keys
|
||||||
|
@ -85,5 +90,9 @@ pub mod errors {
|
||||||
/// The override key is empty
|
/// The override key is empty
|
||||||
#[error("empty override key")]
|
#[error("empty override key")]
|
||||||
Empty,
|
Empty,
|
||||||
|
|
||||||
|
/// An alias in the override key is invalid
|
||||||
|
#[error("invalid alias in override key")]
|
||||||
|
InvalidAlias(#[from] crate::manifest::errors::AliasFromStr),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
graph::{DependencyGraph, DependencyGraphNode},
|
graph::{DependencyGraph, DependencyGraphNode},
|
||||||
manifest::{overrides::OverrideSpecifier, DependencyType},
|
manifest::{overrides::OverrideSpecifier, Alias, DependencyType},
|
||||||
source::{
|
source::{
|
||||||
ids::PackageId,
|
ids::PackageId,
|
||||||
pesde::PesdePackageSource,
|
pesde::PesdePackageSource,
|
||||||
|
@ -97,7 +97,7 @@ impl Project {
|
||||||
continue;
|
continue;
|
||||||
};
|
};
|
||||||
|
|
||||||
let span = tracing::info_span!("resolve from old graph", alias);
|
let span = tracing::info_span!("resolve from old graph", alias = alias.as_str());
|
||||||
let _guard = span.enter();
|
let _guard = span.enter();
|
||||||
|
|
||||||
tracing::debug!("resolved {package_id} from old dependency graph");
|
tracing::debug!("resolved {package_id} from old dependency graph");
|
||||||
|
@ -121,6 +121,7 @@ impl Project {
|
||||||
let inner_span =
|
let inner_span =
|
||||||
tracing::info_span!("resolve dependency", path = path.join(">"));
|
tracing::info_span!("resolve dependency", path = path.join(">"));
|
||||||
let _inner_guard = inner_span.enter();
|
let _inner_guard = inner_span.enter();
|
||||||
|
|
||||||
if let Some(dep_node) = previous_graph.get(dep_id) {
|
if let Some(dep_node) = previous_graph.get(dep_id) {
|
||||||
tracing::debug!("resolved sub-dependency {dep_id}");
|
tracing::debug!("resolved sub-dependency {dep_id}");
|
||||||
insert_node(&mut graph, dep_id, dep_node.clone(), false);
|
insert_node(&mut graph, dep_id, dep_node.clone(), false);
|
||||||
|
@ -339,7 +340,7 @@ impl Project {
|
||||||
tracing::debug!(
|
tracing::debug!(
|
||||||
"overridden specifier found for {} ({dependency_spec})",
|
"overridden specifier found for {} ({dependency_spec})",
|
||||||
path.iter()
|
path.iter()
|
||||||
.map(String::as_str)
|
.map(Alias::as_str)
|
||||||
.chain(std::iter::once(dependency_alias.as_str()))
|
.chain(std::iter::once(dependency_alias.as_str()))
|
||||||
.collect::<Vec<_>>()
|
.collect::<Vec<_>>()
|
||||||
.join(">"),
|
.join(">"),
|
||||||
|
@ -368,7 +369,7 @@ impl Project {
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
.instrument(tracing::info_span!("resolve new/changed", path = path.join(">")))
|
.instrument(tracing::info_span!("resolve new/changed", path = path.iter().map(Alias::as_str).collect::<Vec<_>>().join(">")))
|
||||||
.await?;
|
.await?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -388,6 +389,7 @@ impl Project {
|
||||||
|
|
||||||
/// Errors that can occur when resolving dependencies
|
/// Errors that can occur when resolving dependencies
|
||||||
pub mod errors {
|
pub mod errors {
|
||||||
|
use crate::manifest::Alias;
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
|
|
||||||
/// Errors that can occur when creating a dependency graph
|
/// Errors that can occur when creating a dependency graph
|
||||||
|
@ -425,6 +427,6 @@ pub mod errors {
|
||||||
|
|
||||||
/// An alias for an override was not found in the manifest
|
/// An alias for an override was not found in the manifest
|
||||||
#[error("alias `{0}` not found in manifest")]
|
#[error("alias `{0}` not found in manifest")]
|
||||||
AliasNotFound(String),
|
AliasNotFound(Alias),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ use serde::{Deserialize, Serialize};
|
||||||
use std::collections::BTreeMap;
|
use std::collections::BTreeMap;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::DependencyType,
|
manifest::{Alias, DependencyType},
|
||||||
source::{git::GitPackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
source::{git::GitPackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -19,12 +19,12 @@ pub struct GitPackageRef {
|
||||||
pub tree_id: String,
|
pub tree_id: String,
|
||||||
/// The dependencies of the package
|
/// The dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
pub dependencies: BTreeMap<Alias, (DependencySpecifiers, DependencyType)>,
|
||||||
/// Whether this package uses the new structure
|
/// Whether this package uses the new structure
|
||||||
pub new_structure: bool,
|
pub new_structure: bool,
|
||||||
}
|
}
|
||||||
impl PackageRef for GitPackageRef {
|
impl PackageRef for GitPackageRef {
|
||||||
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> {
|
fn dependencies(&self) -> &BTreeMap<Alias, (DependencySpecifiers, DependencyType)> {
|
||||||
&self.dependencies
|
&self.dependencies
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::DependencyType,
|
manifest::{Alias, DependencyType},
|
||||||
source::{path::PathPackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
source::{path::PathPackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
||||||
};
|
};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
@ -12,10 +12,10 @@ pub struct PathPackageRef {
|
||||||
pub path: PathBuf,
|
pub path: PathBuf,
|
||||||
/// The dependencies of the package
|
/// The dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
pub dependencies: BTreeMap<Alias, (DependencySpecifiers, DependencyType)>,
|
||||||
}
|
}
|
||||||
impl PackageRef for PathPackageRef {
|
impl PackageRef for PathPackageRef {
|
||||||
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> {
|
fn dependencies(&self) -> &BTreeMap<Alias, (DependencySpecifiers, DependencyType)> {
|
||||||
&self.dependencies
|
&self.dependencies
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -14,7 +14,7 @@ use specifier::PesdeDependencySpecifier;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
engine::EngineKind,
|
engine::EngineKind,
|
||||||
manifest::{target::Target, DependencyType},
|
manifest::{target::Target, Alias, DependencyType},
|
||||||
names::{PackageName, PackageNames},
|
names::{PackageName, PackageNames},
|
||||||
reporters::{response_to_async_read, DownloadProgressReporter},
|
reporters::{response_to_async_read, DownloadProgressReporter},
|
||||||
source::{
|
source::{
|
||||||
|
@ -498,7 +498,7 @@ pub struct IndexFileEntry {
|
||||||
|
|
||||||
/// The dependencies of this package
|
/// The dependencies of this package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
pub dependencies: BTreeMap<Alias, (DependencySpecifiers, DependencyType)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The package metadata in the index file
|
/// The package metadata in the index file
|
||||||
|
|
|
@ -3,7 +3,7 @@ use std::collections::BTreeMap;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::DependencyType,
|
manifest::{Alias, DependencyType},
|
||||||
source::{pesde::PesdePackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
source::{pesde::PesdePackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -18,10 +18,10 @@ pub struct PesdePackageRef {
|
||||||
pub index_url: gix::Url,
|
pub index_url: gix::Url,
|
||||||
/// The dependencies of the package
|
/// The dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
pub dependencies: BTreeMap<Alias, (DependencySpecifiers, DependencyType)>,
|
||||||
}
|
}
|
||||||
impl PackageRef for PesdePackageRef {
|
impl PackageRef for PesdePackageRef {
|
||||||
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> {
|
fn dependencies(&self) -> &BTreeMap<Alias, (DependencySpecifiers, DependencyType)> {
|
||||||
&self.dependencies
|
&self.dependencies
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::DependencyType,
|
manifest::{Alias, DependencyType},
|
||||||
source::{pesde, specifiers::DependencySpecifiers, traits::PackageRef, PackageSources},
|
source::{pesde, specifiers::DependencySpecifiers, traits::PackageRef, PackageSources},
|
||||||
};
|
};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
@ -35,7 +35,7 @@ impl PackageRefs {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PackageRef for PackageRefs {
|
impl PackageRef for PackageRefs {
|
||||||
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> {
|
fn dependencies(&self) -> &BTreeMap<Alias, (DependencySpecifiers, DependencyType)> {
|
||||||
match self {
|
match self {
|
||||||
PackageRefs::Pesde(pkg_ref) => pkg_ref.dependencies(),
|
PackageRefs::Pesde(pkg_ref) => pkg_ref.dependencies(),
|
||||||
#[cfg(feature = "wally-compat")]
|
#[cfg(feature = "wally-compat")]
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::{
|
manifest::{
|
||||||
target::{Target, TargetKind},
|
target::{Target, TargetKind},
|
||||||
DependencyType,
|
Alias, DependencyType,
|
||||||
},
|
},
|
||||||
reporters::DownloadProgressReporter,
|
reporters::DownloadProgressReporter,
|
||||||
source::{ids::PackageId, DependencySpecifiers, PackageFs, PackageSources, ResolveResult},
|
source::{ids::PackageId, DependencySpecifiers, PackageFs, PackageSources, ResolveResult},
|
||||||
|
@ -21,7 +21,7 @@ pub trait DependencySpecifier: Debug + Display {}
|
||||||
/// A reference to a package
|
/// A reference to a package
|
||||||
pub trait PackageRef: Debug {
|
pub trait PackageRef: Debug {
|
||||||
/// The dependencies of this package
|
/// The dependencies of this package
|
||||||
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)>;
|
fn dependencies(&self) -> &BTreeMap<Alias, (DependencySpecifiers, DependencyType)>;
|
||||||
/// Whether to use the new structure (`packages` folders inside the package's content folder) or the old structure (Wally-style, with linker files in the parent of the folder containing the package's contents)
|
/// Whether to use the new structure (`packages` folders inside the package's content folder) or the old structure (Wally-style, with linker files in the parent of the folder containing the package's contents)
|
||||||
fn use_new_structure(&self) -> bool;
|
fn use_new_structure(&self) -> bool;
|
||||||
/// The source of this package
|
/// The source of this package
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use std::collections::BTreeMap;
|
use std::collections::BTreeMap;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::{errors, DependencyType},
|
manifest::{errors, Alias, DependencyType},
|
||||||
names::wally::WallyPackageName,
|
names::wally::WallyPackageName,
|
||||||
source::{specifiers::DependencySpecifiers, wally::specifier::WallyDependencySpecifier},
|
source::{specifiers::DependencySpecifiers, wally::specifier::WallyDependencySpecifier},
|
||||||
};
|
};
|
||||||
|
@ -28,9 +28,9 @@ pub struct WallyPackage {
|
||||||
|
|
||||||
pub fn deserialize_specifiers<'de, D: Deserializer<'de>>(
|
pub fn deserialize_specifiers<'de, D: Deserializer<'de>>(
|
||||||
deserializer: D,
|
deserializer: D,
|
||||||
) -> Result<BTreeMap<String, WallyDependencySpecifier>, D::Error> {
|
) -> Result<BTreeMap<Alias, WallyDependencySpecifier>, D::Error> {
|
||||||
// specifier is in form of `name@version_req`
|
// specifier is in form of `name@version_req`
|
||||||
BTreeMap::<String, String>::deserialize(deserializer)?
|
BTreeMap::<Alias, String>::deserialize(deserializer)?
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|(k, v)| {
|
.map(|(k, v)| {
|
||||||
let (name, version) = v.split_once('@').ok_or_else(|| {
|
let (name, version) = v.split_once('@').ok_or_else(|| {
|
||||||
|
@ -54,11 +54,11 @@ pub fn deserialize_specifiers<'de, D: Deserializer<'de>>(
|
||||||
pub struct WallyManifest {
|
pub struct WallyManifest {
|
||||||
pub package: WallyPackage,
|
pub package: WallyPackage,
|
||||||
#[serde(default, deserialize_with = "deserialize_specifiers")]
|
#[serde(default, deserialize_with = "deserialize_specifiers")]
|
||||||
pub dependencies: BTreeMap<String, WallyDependencySpecifier>,
|
pub dependencies: BTreeMap<Alias, WallyDependencySpecifier>,
|
||||||
#[serde(default, deserialize_with = "deserialize_specifiers")]
|
#[serde(default, deserialize_with = "deserialize_specifiers")]
|
||||||
pub server_dependencies: BTreeMap<String, WallyDependencySpecifier>,
|
pub server_dependencies: BTreeMap<Alias, WallyDependencySpecifier>,
|
||||||
#[serde(default, deserialize_with = "deserialize_specifiers")]
|
#[serde(default, deserialize_with = "deserialize_specifiers")]
|
||||||
pub dev_dependencies: BTreeMap<String, WallyDependencySpecifier>,
|
pub dev_dependencies: BTreeMap<Alias, WallyDependencySpecifier>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WallyManifest {
|
impl WallyManifest {
|
||||||
|
@ -66,10 +66,8 @@ impl WallyManifest {
|
||||||
#[instrument(skip(self), ret(level = "trace"), level = "debug")]
|
#[instrument(skip(self), ret(level = "trace"), level = "debug")]
|
||||||
pub fn all_dependencies(
|
pub fn all_dependencies(
|
||||||
&self,
|
&self,
|
||||||
) -> Result<
|
) -> Result<BTreeMap<Alias, (DependencySpecifiers, DependencyType)>, errors::AllDependenciesError>
|
||||||
BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
{
|
||||||
errors::AllDependenciesError,
|
|
||||||
> {
|
|
||||||
let mut all_deps = BTreeMap::new();
|
let mut all_deps = BTreeMap::new();
|
||||||
|
|
||||||
for (deps, ty) in [
|
for (deps, ty) in [
|
||||||
|
|
|
@ -3,7 +3,7 @@ use std::collections::BTreeMap;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::DependencyType,
|
manifest::{Alias, DependencyType},
|
||||||
source::{wally::WallyPackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
source::{wally::WallyPackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -18,10 +18,10 @@ pub struct WallyPackageRef {
|
||||||
pub index_url: gix::Url,
|
pub index_url: gix::Url,
|
||||||
/// The dependencies of the package
|
/// The dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
pub dependencies: BTreeMap<Alias, (DependencySpecifiers, DependencyType)>,
|
||||||
}
|
}
|
||||||
impl PackageRef for WallyPackageRef {
|
impl PackageRef for WallyPackageRef {
|
||||||
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> {
|
fn dependencies(&self) -> &BTreeMap<Alias, (DependencySpecifiers, DependencyType)> {
|
||||||
&self.dependencies
|
&self.dependencies
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@ use serde::{Deserialize, Serialize};
|
||||||
use std::collections::BTreeMap;
|
use std::collections::BTreeMap;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
manifest::DependencyType,
|
manifest::{Alias, DependencyType},
|
||||||
source::{workspace::WorkspacePackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
source::{workspace::WorkspacePackageSource, DependencySpecifiers, PackageRef, PackageSources},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -14,10 +14,10 @@ pub struct WorkspacePackageRef {
|
||||||
pub path: RelativePathBuf,
|
pub path: RelativePathBuf,
|
||||||
/// The dependencies of the package
|
/// The dependencies of the package
|
||||||
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
|
||||||
pub dependencies: BTreeMap<String, (DependencySpecifiers, DependencyType)>,
|
pub dependencies: BTreeMap<Alias, (DependencySpecifiers, DependencyType)>,
|
||||||
}
|
}
|
||||||
impl PackageRef for WorkspacePackageRef {
|
impl PackageRef for WorkspacePackageRef {
|
||||||
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> {
|
fn dependencies(&self) -> &BTreeMap<Alias, (DependencySpecifiers, DependencyType)> {
|
||||||
&self.dependencies
|
&self.dependencies
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue