use syrette::ptr::TransientPtr;
use syrette::{declare_default_factory, AsyncDIContainer};

use crate::animals::cat::Cat;
use crate::animals::dog::Dog;
use crate::animals::human::Human;
use crate::food::Food;
use crate::interfaces::cat::ICat;
use crate::interfaces::dog::IDog;
use crate::interfaces::food::{IFood, IFoodFactory};
use crate::interfaces::human::IHuman;

declare_default_factory!(dyn ICat, threadsafe = true);

pub async fn bootstrap() -> Result<AsyncDIContainer, anyhow::Error>
{
    let mut di_container = AsyncDIContainer::new();

    di_container
        .bind::<dyn IDog>()
        .to::<Dog>()
        .await?
        .in_singleton_scope()
        .await?;

    di_container
        .bind::<dyn ICat>()
        .to_default_factory(&|_| {
            Box::new(|| {
                let cat: TransientPtr<dyn ICat> = TransientPtr::new(Cat::new());

                cat
            })
        })
        .await?;

    di_container.bind::<dyn IHuman>().to::<Human>().await?;

    di_container
        .bind::<IFoodFactory>()
        .to_factory(&|_| {
            Box::new(|| {
                let food: Box<dyn IFood> = Box::new(Food::new());

                food
            })
        })
        .await?;

    Ok(di_container)
}