use tracing::{Span, Value};
use crate::utils::{FnWrapper, KV};
pub trait EnrichSpan<T> {
    fn enrich_span(&self, span: &Span, t: &T);
}
impl<T, F> EnrichSpan<T> for FnWrapper<F>
where
    F: Fn(&Span, &T),
{
    fn enrich_span(&self, span: &Span, t: &T) {
        (self.0)(span, t);
    }
}
#[must_use]
pub fn enrich_span_fn<T, F>(f: F) -> FnWrapper<F>
where
    F: Fn(&Span, &T),
{
    FnWrapper(f)
}
impl<T> EnrichSpan<T> for () {
    fn enrich_span(&self, _span: &Span, _t: &T) {}
}
impl<V, T> EnrichSpan<T> for KV<V>
where
    V: Value,
{
    fn enrich_span(&self, span: &Span, _t: &T) {
        span.record(self.0, &self.1);
    }
}
macro_rules! impl_for_tuple {
    ($($T:ident),+) => {
        impl<T, $($T),+> EnrichSpan<T> for ($($T,)+)
        where
            $($T: EnrichSpan<T>),+
        {
            fn enrich_span(&self, span: &Span, t: &T) {
                #[allow(non_snake_case)]
                let ($(ref $T,)+) = *self;
                $(
                    $T.enrich_span(span, t);
                )+
            }
        }
    };
}
impl_for_tuple!(T1);
impl_for_tuple!(T1, T2);
impl_for_tuple!(T1, T2, T3);
impl_for_tuple!(T1, T2, T3, T4);
impl_for_tuple!(T1, T2, T3, T4, T5);
impl_for_tuple!(T1, T2, T3, T4, T5, T6);
impl_for_tuple!(T1, T2, T3, T4, T5, T6, T7);
impl_for_tuple!(T1, T2, T3, T4, T5, T6, T7, T8);
impl<T, R> EnrichSpan<R> for Option<T>
where
    T: EnrichSpan<R>,
{
    fn enrich_span(&self, span: &Span, request: &R) {
        if let Some(ref t) = *self {
            t.enrich_span(span, request);
        }
    }
}
impl<T, R, const N: usize> EnrichSpan<R> for [T; N]
where
    T: EnrichSpan<R>,
{
    fn enrich_span(&self, span: &Span, request: &R) {
        for t in self {
            t.enrich_span(span, request);
        }
    }
}
impl<T, R> EnrichSpan<R> for Vec<T>
where
    T: EnrichSpan<R>,
{
    fn enrich_span(&self, span: &Span, request: &R) {
        for t in self {
            t.enrich_span(span, request);
        }
    }
}