use crate::r1cs::{errors::SynthesisError, ConstraintSystem, Index, LinearCombination, Variable};
use snarkvm_fields::Field;
pub struct TestConstraintChecker<F: Field> {
    public_variables: Vec<F>,
    private_variables: Vec<F>,
    found_unsatisfactory_constraint: bool,
    num_constraints: usize,
    segments: Vec<String>,
    first_unsatisfied_constraint: Option<String>,
}
impl<F: Field> Default for TestConstraintChecker<F> {
    fn default() -> Self {
        Self {
            public_variables: vec![F::one()],
            private_variables: vec![],
            found_unsatisfactory_constraint: false,
            num_constraints: 0,
            segments: vec![],
            first_unsatisfied_constraint: None,
        }
    }
}
impl<F: Field> TestConstraintChecker<F> {
    pub fn new() -> Self {
        Self::default()
    }
    pub fn which_is_unsatisfied(&self) -> Option<String> {
        self.first_unsatisfied_constraint.clone()
    }
    #[inline]
    pub fn is_satisfied(&self) -> bool {
        !self.found_unsatisfactory_constraint
    }
    #[inline]
    pub fn num_constraints(&self) -> usize {
        self.num_constraints
    }
    #[inline]
    pub fn public_inputs(&self) -> Vec<F> {
        self.public_variables[1..].to_vec()
    }
}
impl<F: Field> ConstraintSystem<F> for TestConstraintChecker<F> {
    type Root = Self;
    fn alloc<Fn, A, AR>(&mut self, _annotation: A, f: Fn) -> Result<Variable, SynthesisError>
    where
        Fn: FnOnce() -> Result<F, SynthesisError>,
        A: FnOnce() -> AR,
        AR: AsRef<str>,
    {
        let index = self.private_variables.len();
        self.private_variables.push(f()?);
        let var = Variable::new_unchecked(Index::Private(index));
        Ok(var)
    }
    fn alloc_input<Fn, A, AR>(&mut self, _annotation: A, f: Fn) -> Result<Variable, SynthesisError>
    where
        Fn: FnOnce() -> Result<F, SynthesisError>,
        A: FnOnce() -> AR,
        AR: AsRef<str>,
    {
        let index = self.public_variables.len();
        self.public_variables.push(f()?);
        let var = Variable::new_unchecked(Index::Public(index));
        Ok(var)
    }
    fn enforce<A, AR, LA, LB, LC>(&mut self, annotation: A, a: LA, b: LB, c: LC)
    where
        A: FnOnce() -> AR,
        AR: AsRef<str>,
        LA: FnOnce(LinearCombination<F>) -> LinearCombination<F>,
        LB: FnOnce(LinearCombination<F>) -> LinearCombination<F>,
        LC: FnOnce(LinearCombination<F>) -> LinearCombination<F>,
    {
        self.num_constraints += 1;
        let eval_lc = |lc: Vec<(Variable, F)>| -> F {
            lc.into_iter()
                .map(|(var, coeff)| {
                    let value = match var.get_unchecked() {
                        Index::Public(index) => self.public_variables[index],
                        Index::Private(index) => self.private_variables[index],
                    };
                    value * coeff
                })
                .sum::<F>()
        };
        let a = eval_lc(a(LinearCombination::zero()).0);
        let b = eval_lc(b(LinearCombination::zero()).0);
        let c = eval_lc(c(LinearCombination::zero()).0);
        if a * b != c && self.first_unsatisfied_constraint.is_none() {
            self.found_unsatisfactory_constraint = true;
            let new = annotation().as_ref().to_string();
            assert!(!new.contains('/'), "'/' is not allowed in names");
            let mut path = self.segments.clone();
            path.push(new);
            self.first_unsatisfied_constraint = Some(path.join("/"));
        }
    }
    fn push_namespace<NR: AsRef<str>, N: FnOnce() -> NR>(&mut self, name_fn: N) {
        let new = name_fn().as_ref().to_string();
        assert!(!new.contains('/'), "'/' is not allowed in names");
        self.segments.push(new)
    }
    fn pop_namespace(&mut self) {
        self.segments.pop();
    }
    #[inline]
    fn get_root(&mut self) -> &mut Self::Root {
        self
    }
    #[inline]
    fn num_constraints(&self) -> usize {
        self.num_constraints()
    }
    #[inline]
    fn num_public_variables(&self) -> usize {
        self.public_variables.len()
    }
    #[inline]
    fn num_private_variables(&self) -> usize {
        self.private_variables.len()
    }
    #[inline]
    fn is_in_setup_mode(&self) -> bool {
        false
    }
}