use std::fmt::{Display, Formatter};
use std::iter;
use std::sync::Arc;
use uv_distribution_types::IncompatibleDist;
use uv_pep440::{Version, VersionSpecifiers};
use uv_platform_tags::{AbiTag, Tags};
use crate::resolver::{MetadataUnavailable, VersionFork};
#[derive(Debug, Clone, Eq, PartialEq)]
pub enum UnavailableReason {
Package(UnavailablePackage),
Version(UnavailableVersion),
}
impl Display for UnavailableReason {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
match self {
Self::Version(version) => Display::fmt(version, f),
Self::Package(package) => Display::fmt(package, f),
}
}
}
#[derive(Debug, Clone, Eq, PartialEq)]
pub enum UnavailableVersion {
IncompatibleDist(IncompatibleDist),
InvalidMetadata,
InconsistentMetadata,
InvalidStructure,
Offline,
RequiresPython(VersionSpecifiers),
}
impl UnavailableVersion {
pub(crate) fn message(&self) -> String {
match self {
Self::IncompatibleDist(invalid_dist) => format!("{invalid_dist}"),
Self::InvalidMetadata => "invalid metadata".into(),
Self::InconsistentMetadata => "inconsistent metadata".into(),
Self::InvalidStructure => "an invalid package format".into(),
Self::Offline => "to be downloaded from a registry".into(),
Self::RequiresPython(requires_python) => {
format!("Python {requires_python}")
}
}
}
pub(crate) fn singular_message(&self) -> String {
match self {
Self::IncompatibleDist(invalid_dist) => invalid_dist.singular_message(),
Self::InvalidMetadata => format!("has {self}"),
Self::InconsistentMetadata => format!("has {self}"),
Self::InvalidStructure => format!("has {self}"),
Self::Offline => format!("needs {self}"),
Self::RequiresPython(..) => format!("requires {self}"),
}
}
pub(crate) fn plural_message(&self) -> String {
match self {
Self::IncompatibleDist(invalid_dist) => invalid_dist.plural_message(),
Self::InvalidMetadata => format!("have {self}"),
Self::InconsistentMetadata => format!("have {self}"),
Self::InvalidStructure => format!("have {self}"),
Self::Offline => format!("need {self}"),
Self::RequiresPython(..) => format!("require {self}"),
}
}
pub(crate) fn context_message(
&self,
tags: Option<&Tags>,
requires_python: Option<AbiTag>,
) -> Option<String> {
match self {
Self::IncompatibleDist(invalid_dist) => {
invalid_dist.context_message(tags, requires_python)
}
Self::InvalidMetadata => None,
Self::InconsistentMetadata => None,
Self::InvalidStructure => None,
Self::Offline => None,
Self::RequiresPython(..) => None,
}
}
}
impl Display for UnavailableVersion {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
f.write_str(&self.message())
}
}
impl From<&MetadataUnavailable> for UnavailableVersion {
fn from(reason: &MetadataUnavailable) -> Self {
match reason {
MetadataUnavailable::Offline => Self::Offline,
MetadataUnavailable::InvalidMetadata(_) => Self::InvalidMetadata,
MetadataUnavailable::InconsistentMetadata(_) => Self::InconsistentMetadata,
MetadataUnavailable::InvalidStructure(_) => Self::InvalidStructure,
MetadataUnavailable::RequiresPython(requires_python, _python_version) => {
Self::RequiresPython(requires_python.clone())
}
}
}
}
#[derive(Debug, Clone)]
pub struct UnavailableErrorChain(Arc<dyn std::error::Error + Send + Sync + 'static>);
impl Display for UnavailableErrorChain {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
for source in iter::successors(Some(&self.0 as &dyn std::error::Error), |&err| err.source())
{
writeln!(f, "Caused by: {}", source.to_string().trim())?;
}
Ok(())
}
}
impl PartialEq for UnavailableErrorChain {
fn eq(&self, other: &Self) -> bool {
self.to_string() == other.to_string()
}
}
impl Eq for UnavailableErrorChain {}
#[derive(Debug, Clone, Eq, PartialEq)]
pub enum UnavailablePackage {
NoIndex,
Offline,
NotFound,
InvalidMetadata(UnavailableErrorChain),
InvalidStructure(UnavailableErrorChain),
}
impl UnavailablePackage {
pub(crate) fn message(&self) -> &'static str {
match self {
Self::NoIndex => "not found in the provided package locations",
Self::Offline => "not found in the cache",
Self::NotFound => "not found in the package registry",
Self::InvalidMetadata(_) => "invalid metadata",
Self::InvalidStructure(_) => "an invalid package format",
}
}
pub(crate) fn singular_message(&self) -> String {
match self {
Self::NoIndex => format!("was {self}"),
Self::Offline => format!("was {self}"),
Self::NotFound => format!("was {self}"),
Self::InvalidMetadata(_) => format!("has {self}"),
Self::InvalidStructure(_) => format!("has {self}"),
}
}
}
impl Display for UnavailablePackage {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
f.write_str(self.message())
}
}
impl From<&MetadataUnavailable> for UnavailablePackage {
fn from(reason: &MetadataUnavailable) -> Self {
match reason {
MetadataUnavailable::Offline => Self::Offline,
MetadataUnavailable::InvalidMetadata(err) => {
Self::InvalidMetadata(UnavailableErrorChain(err.clone()))
}
MetadataUnavailable::InconsistentMetadata(err) => {
Self::InvalidMetadata(UnavailableErrorChain(err.clone()))
}
MetadataUnavailable::InvalidStructure(err) => {
Self::InvalidStructure(UnavailableErrorChain(err.clone()))
}
MetadataUnavailable::RequiresPython(..) => {
unreachable!("`requires-python` is only known upfront for registry distributions")
}
}
}
}
#[derive(Debug, Clone)]
pub(crate) enum ResolverVersion {
Unavailable(Version, UnavailableVersion),
Unforked(Version),
Forked(Vec<VersionFork>),
}