use darling::FromMeta;
use once_cell::sync::Lazy;
use proc_macro::TokenStream;
use proc_macro2::Span;
use quaint_test_setup::{ConnectorDefinition, Tags, CONNECTORS};
use quote::quote;
use std::str::FromStr;
use syn::{parse_macro_input, spanned::Spanned, AttributeArgs, Ident, ItemFn};
static TAGS_FILTER: Lazy<Tags> = Lazy::new(|| {
    let tags_str = std::env::var("TEST_EACH_CONNECTOR_TAGS").ok();
    let mut tags = Tags::empty();
    if let Some(tags_str) = tags_str {
        for tag_str in tags_str.split(',') {
            let tag = Tags::from_str(tag_str).unwrap();
            tags |= tag;
        }
    }
    tags
});
#[derive(Debug, FromMeta)]
struct TestEachConnectorArgs {
    #[darling(default)]
    tags: TagsWrapper,
    #[darling(default)]
    ignore: TagsWrapper,
}
impl TestEachConnectorArgs {
    fn connectors_to_test(&self) -> impl Iterator<Item = &ConnectorDefinition> {
        CONNECTORS
            .all()
            .filter(move |connector| TAGS_FILTER.is_empty() || connector.tags.contains(*TAGS_FILTER))
            .filter(move |connector| self.tags.0.is_empty() || connector.tags.intersects(self.tags.0))
            .filter(move |connector| !connector.tags.intersects(self.ignore.0))
    }
}
#[derive(Debug)]
struct TagsWrapper(Tags);
impl Default for TagsWrapper {
    fn default() -> Self {
        TagsWrapper(Tags::empty())
    }
}
impl darling::FromMeta for TagsWrapper {
    fn from_list(items: &[syn::NestedMeta]) -> Result<Self, darling::Error> {
        let mut tags = Tags::empty();
        for item in items {
            match item {
                syn::NestedMeta::Lit(syn::Lit::Str(s)) => {
                    let s = s.value();
                    let tag = Tags::from_str(&s)
                        .map_err(|err| darling::Error::unknown_value(&err.to_string()).with_span(&item.span()))?;
                    tags.insert(tag);
                }
                syn::NestedMeta::Lit(other) => {
                    return Err(darling::Error::unexpected_lit_type(other).with_span(&other.span()))
                }
                syn::NestedMeta::Meta(meta) => {
                    return Err(darling::Error::unsupported_shape("Expected string literal").with_span(&meta.span()))
                }
            }
        }
        Ok(TagsWrapper(tags))
    }
}
#[allow(clippy::needless_borrow)]
pub fn test_each_connector_impl(attr: TokenStream, input: TokenStream) -> TokenStream {
    let attributes_meta: syn::AttributeArgs = parse_macro_input!(attr as AttributeArgs);
    let args = TestEachConnectorArgs::from_list(&attributes_meta);
    let mut test_function = parse_macro_input!(input as ItemFn);
    super::strip_test_attribute(&mut test_function);
    let tests = match args {
        Ok(args) => test_each_connector_async_wrapper_functions(&args, &test_function),
        Err(err) => return err.write_errors().into(),
    };
    let output = quote! {
        #(#tests)*
        #test_function
    };
    output.into()
}
#[allow(clippy::needless_borrow)]
fn test_each_connector_async_wrapper_functions(
    args: &TestEachConnectorArgs,
    test_function: &ItemFn,
) -> Vec<proc_macro2::TokenStream> {
    let test_fn_name = &test_function.sig.ident;
    let mut tests = Vec::with_capacity(CONNECTORS.len());
    let optional_unwrap = if super::function_returns_result(&test_function) {
        Some(quote!(.unwrap()))
    } else {
        None
    };
    for connector in args.connectors_to_test() {
        let connector_name = connector.name();
        let feature_name = connector.feature_name();
        let connector_test_fn_name = Ident::new(&format!("{}_on_{}", test_fn_name, connector_name), Span::call_site());
        let conn_api_factory = Ident::new(connector.test_api(), Span::call_site());
        let test = quote! {
            #[test]
            #[cfg(feature = #feature_name)]
            fn #connector_test_fn_name() {
                let fut = async {
                    let mut api = #conn_api_factory().await#optional_unwrap;
                    #test_fn_name(&mut api).await#optional_unwrap
                };
                quaint_test_setup::run_with_tokio(fut)
            }
        };
        tests.push(test);
    }
    tests
}