#![forbid(unsafe_code)]
use std::collections::HashMap;
use std::path::PathBuf;
use std::sync::Arc;
#[doc(hidden)]
pub use paste;
use std::cell::RefCell;
use std::ffi::OsStr;
use std::marker::PhantomData;
use std::path::Path;
pub trait ArtifactId: 'static {
#[doc(hidden)]
const GLOBAL_UNIQUE_ID: &'static str;
#[doc(hidden)]
fn i_know_what_im_doing_with_this_manual_impl_instead_of_using_the_declare_artifacts_macro();
}
#[derive(Copy, Clone, Hash, PartialEq, Eq, PartialOrd, Ord)]
pub struct ArtifactHandle<A: ArtifactId>(PhantomData<A>);
impl<A: ArtifactId + std::fmt::Debug> std::fmt::Debug for ArtifactHandle<A> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
std::fmt::Debug::fmt(&self.erase(), f)
}
}
pub struct ResolvedArtifact<A = ()>(Option<PathBuf>, PhantomData<A>);
impl<A> Clone for ResolvedArtifact<A> {
fn clone(&self) -> Self {
Self(self.0.clone(), self.1)
}
}
impl<A> std::fmt::Debug for ResolvedArtifact<A> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_tuple("ResolvedArtifact").field(&self.0).finish()
}
}
impl<A> ResolvedArtifact<A> {
pub fn erase(self) -> ResolvedArtifact {
ResolvedArtifact(self.0, PhantomData)
}
#[track_caller]
pub fn get(&self) -> &Path {
self.0
.as_ref()
.expect("cannot get path in requirements phase")
}
}
impl<A> From<ResolvedArtifact<A>> for PathBuf {
#[track_caller]
fn from(ra: ResolvedArtifact<A>) -> PathBuf {
ra.0.expect("cannot get path in requirements phase")
}
}
impl<A> AsRef<Path> for ResolvedArtifact<A> {
#[track_caller]
fn as_ref(&self) -> &Path {
self.get()
}
}
impl<A> AsRef<OsStr> for ResolvedArtifact<A> {
#[track_caller]
fn as_ref(&self) -> &OsStr {
self.get().as_ref()
}
}
#[derive(Clone, Debug)]
pub struct ResolvedOptionalArtifact<A = ()>(OptionalArtifactState, PhantomData<A>);
#[derive(Clone, Debug)]
enum OptionalArtifactState {
Collecting,
Missing,
Present(PathBuf),
}
impl<A> ResolvedOptionalArtifact<A> {
pub fn erase(self) -> ResolvedOptionalArtifact {
ResolvedOptionalArtifact(self.0, PhantomData)
}
#[track_caller]
pub fn get(&self) -> Option<&Path> {
match self.0 {
OptionalArtifactState::Collecting => panic!("cannot get path in requirements phase"),
OptionalArtifactState::Missing => None,
OptionalArtifactState::Present(ref path) => Some(path),
}
}
}
pub struct ArtifactResolver<'a>(ArtifactResolverInner<'a>);
impl<'a> ArtifactResolver<'a> {
pub fn collector(requirements: &'a mut TestArtifactRequirements) -> Self {
ArtifactResolver(ArtifactResolverInner::Collecting(RefCell::new(
requirements,
)))
}
pub fn resolver(artifacts: &'a TestArtifacts) -> Self {
ArtifactResolver(ArtifactResolverInner::Resolving(artifacts))
}
pub fn require<A: ArtifactId>(&self, handle: ArtifactHandle<A>) -> ResolvedArtifact<A> {
match &self.0 {
ArtifactResolverInner::Collecting(requirements) => {
requirements.borrow_mut().require(handle.erase());
ResolvedArtifact(None, PhantomData)
}
ArtifactResolverInner::Resolving(artifacts) => {
ResolvedArtifact(Some(artifacts.get(handle).to_owned()), PhantomData)
}
}
}
pub fn try_require<A: ArtifactId>(
&self,
handle: ArtifactHandle<A>,
) -> ResolvedOptionalArtifact<A> {
match &self.0 {
ArtifactResolverInner::Collecting(requirements) => {
requirements.borrow_mut().try_require(handle.erase());
ResolvedOptionalArtifact(OptionalArtifactState::Collecting, PhantomData)
}
ArtifactResolverInner::Resolving(artifacts) => ResolvedOptionalArtifact(
artifacts
.try_get(handle)
.map_or(OptionalArtifactState::Missing, |p| {
OptionalArtifactState::Present(p.to_owned())
}),
PhantomData,
),
}
}
}
enum ArtifactResolverInner<'a> {
Collecting(RefCell<&'a mut TestArtifactRequirements>),
Resolving(&'a TestArtifacts),
}
#[derive(Copy, Clone, Hash, PartialEq, Eq, PartialOrd, Ord)]
pub struct ErasedArtifactHandle {
artifact_id_str: &'static str,
}
impl std::fmt::Debug for ErasedArtifactHandle {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"{}",
self.artifact_id_str
.strip_suffix("__ty")
.unwrap_or(self.artifact_id_str)
)
}
}
impl<A: ArtifactId> PartialEq<ErasedArtifactHandle> for ArtifactHandle<A> {
fn eq(&self, other: &ErasedArtifactHandle) -> bool {
&self.erase() == other
}
}
impl<A: ArtifactId> PartialEq<ArtifactHandle<A>> for ErasedArtifactHandle {
fn eq(&self, other: &ArtifactHandle<A>) -> bool {
self == &other.erase()
}
}
impl<A: ArtifactId> ArtifactHandle<A> {
pub const fn new() -> Self {
Self(PhantomData)
}
}
pub trait AsArtifactHandle {
fn erase(&self) -> ErasedArtifactHandle;
}
impl AsArtifactHandle for ErasedArtifactHandle {
fn erase(&self) -> ErasedArtifactHandle {
*self
}
}
impl<A: ArtifactId> AsArtifactHandle for ArtifactHandle<A> {
fn erase(&self) -> ErasedArtifactHandle {
ErasedArtifactHandle {
artifact_id_str: A::GLOBAL_UNIQUE_ID,
}
}
}
#[macro_export]
macro_rules! declare_artifacts {
(
$(
$(#[$doc:meta])*
$name:ident
),*
$(,)?
) => {
$(
$crate::paste::paste! {
$(#[$doc])*
#[expect(non_camel_case_types)]
pub const $name: $crate::ArtifactHandle<$name> = $crate::ArtifactHandle::new();
#[doc = concat!("Type-tag for [`", stringify!($name), "`]")]
#[expect(non_camel_case_types)]
#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq, PartialOrd, Ord)]
pub enum $name {}
#[expect(non_snake_case)]
mod [< $name __ty >] {
impl $crate::ArtifactId for super::$name {
const GLOBAL_UNIQUE_ID: &'static str = module_path!();
fn i_know_what_im_doing_with_this_manual_impl_instead_of_using_the_declare_artifacts_macro() {}
}
}
}
)*
};
}
pub trait ResolveTestArtifact {
fn resolve(&self, id: ErasedArtifactHandle) -> anyhow::Result<PathBuf>;
}
impl<T: ResolveTestArtifact + ?Sized> ResolveTestArtifact for &T {
fn resolve(&self, id: ErasedArtifactHandle) -> anyhow::Result<PathBuf> {
(**self).resolve(id)
}
}
#[derive(Clone)]
pub struct TestArtifactRequirements {
artifacts: Vec<(ErasedArtifactHandle, bool)>,
}
impl TestArtifactRequirements {
pub fn new() -> Self {
TestArtifactRequirements {
artifacts: Vec::new(),
}
}
pub fn require(&mut self, dependency: impl AsArtifactHandle) -> &mut Self {
self.artifacts.push((dependency.erase(), false));
self
}
pub fn try_require(&mut self, dependency: impl AsArtifactHandle) -> &mut Self {
self.artifacts.push((dependency.erase(), true));
self
}
pub fn required_artifacts(&self) -> impl Iterator<Item = ErasedArtifactHandle> + '_ {
self.artifacts
.iter()
.filter_map(|&(a, optional)| (!optional).then_some(a))
}
pub fn optional_artifacts(&self) -> impl Iterator<Item = ErasedArtifactHandle> + '_ {
self.artifacts
.iter()
.filter_map(|&(a, optional)| optional.then_some(a))
}
pub fn resolve(&self, resolver: impl ResolveTestArtifact) -> anyhow::Result<TestArtifacts> {
let mut failed = String::new();
let mut resolved = HashMap::new();
for &(a, optional) in &self.artifacts {
match resolver.resolve(a) {
Ok(p) => {
resolved.insert(a, p);
}
Err(_) if optional => {}
Err(e) => failed.push_str(&format!("{:?} - {:#}\n", a, e)),
}
}
if !failed.is_empty() {
anyhow::bail!("Artifact resolution failed:\n{}", failed);
}
Ok(TestArtifacts {
artifacts: Arc::new(resolved),
})
}
}
#[derive(Clone)]
pub struct TestArtifacts {
artifacts: Arc<HashMap<ErasedArtifactHandle, PathBuf>>,
}
impl TestArtifacts {
#[track_caller]
pub fn try_get(&self, artifact: impl AsArtifactHandle) -> Option<&Path> {
self.artifacts.get(&artifact.erase()).map(|p| p.as_ref())
}
#[track_caller]
pub fn get(&self, artifact: impl AsArtifactHandle) -> &Path {
self.try_get(artifact.erase())
.unwrap_or_else(|| panic!("Artifact not initially required: {:?}", artifact.erase()))
}
}