/**
 * @file sfinae_test.cpp
 * @author Kirill Mishchenko
 *
 * Test file for SFINAE utilities.
 *
 * mlpack is free software; you may redistribute it and/or modify it under the
 * terms of the 3-clause BSD license.  You should have received a copy of the
 * 3-clause BSD license along with mlpack.  If not, see
 * http://www.opensource.org/licenses/BSD-3-Clause for more information.
 */

#include <mlpack/core.hpp>
#include <mlpack/core/util/sfinae_utility.hpp>

#include <boost/test/unit_test.hpp>

BOOST_AUTO_TEST_SUITE(SFINAETest);

class A
{
 public:
  void M(const arma::mat&, const arma::Row<size_t>&, double);

  void M(const arma::mat&, const arma::Row<size_t>&, double, double);

  void M(const arma::vec&, size_t, double);
};

class B
{
 public:
  void M(const arma::mat&, const arma::rowvec&);

  template<typename MatType>
  void M(const MatType&, const arma::rowvec&, double);

  template<typename MatType, typename PredictionsType>
  void M(const MatType&, const PredictionsType&, const arma::rowvec&, int, int);
};

class C
{
 public:
  double M(const arma::mat&, const arma::rowvec&, double);
};

class WithFunctionModelA
{
 public:
  size_t Model()
  {
    return 0;
  }
};

class WithFunctionModelB
{
 public:
  double Model(int a)
  {
    return (double) a;
  }
};

class WithOutFunctionModel
{
 public:
  size_t Model;
};

class WithInheritedModelA : public WithFunctionModelA
{
  // Nothing here.
};

template<typename Class, typename...T>
using MForm1 = void(Class::*)(const arma::mat&, const arma::Row<size_t>&, T...);

template<typename Class, typename...T>
using MForm2 = void(Class::*)(const arma::mat&, const arma::rowvec&, T...);

template<typename Class, typename...T>
using MForm3 = void(Class::*)(const arma::mat&, const arma::rowvec&,
    const arma::rowvec&, T...);

template<typename Class, typename RT, typename...T>
using MForm4 = RT(Class::*)(const arma::mat&, const arma::rowvec&, T...);

HAS_METHOD_FORM(M, HasM);
HAS_METHOD_FORM(template M<arma::mat>, HasTemplatedM);
HAS_METHOD_FORM(SINGLE_ARG(template M<arma::mat, arma::rowvec>),
    HasVeryTemplatedM);
HAS_ANY_METHOD_FORM(Model, HasModel);

/*
 * Test at compile time the presence of methods of the specified forms with the
 * stated number of additional arguments.
 */
BOOST_AUTO_TEST_CASE(HasMethodFormWithNAdditionalArgsTest)
{
  static_assert(!HasM<A, MForm1>::WithNAdditionalArgs<0>::value,
      "value should be false");
  static_assert(HasM<A, MForm1>::WithNAdditionalArgs<1>::value,
      "value should be true");
  static_assert(HasM<A, MForm1>::WithNAdditionalArgs<2>::value,
      "value should be true");

  static_assert(!HasM<B, MForm1>::WithNAdditionalArgs<0>::value,
      "value should be false");
  static_assert(!HasM<B, MForm1>::WithNAdditionalArgs<1>::value,
      "value should be false");
  static_assert(!HasM<B, MForm1>::WithNAdditionalArgs<2>::value,
      "value should be false");

  static_assert(!HasM<A, MForm2>::WithNAdditionalArgs<0>::value,
      "value should be false");
  static_assert(!HasM<A, MForm2>::WithNAdditionalArgs<1>::value,
      "value should be false");
  static_assert(!HasM<A, MForm2>::WithNAdditionalArgs<2>::value,
      "value should be false");

  static_assert(HasM<B, MForm2>::WithNAdditionalArgs<0>::value,
      "value should be true");
  static_assert(!HasM<B, MForm2>::WithNAdditionalArgs<1>::value,
      "value should be false");
  static_assert(!HasM<B, MForm2>::WithNAdditionalArgs<2>::value,
      "value should be false");

  static_assert(!HasTemplatedM<B, MForm2>::WithNAdditionalArgs<0>::value,
      "value should be false");
  static_assert(HasTemplatedM<B, MForm2>::WithNAdditionalArgs<1>::value,
      "value should be true");
  static_assert(!HasTemplatedM<B, MForm2>::WithNAdditionalArgs<2>::value,
      "value should be false");

  static_assert(!HasVeryTemplatedM<B, MForm3>::WithNAdditionalArgs<0>::value,
      "value should be false");
  static_assert(!HasVeryTemplatedM<B, MForm3>::WithNAdditionalArgs<1>::value,
      "value should be false");
  static_assert(HasVeryTemplatedM<B, MForm3>::WithNAdditionalArgs<2>::value,
      "value should be true");
}

/*
 * Test at compile time the presence of methods of the specified forms.
 */
BOOST_AUTO_TEST_CASE(HasMethodFormTest)
{
  static_assert(HasM<A, MForm1>::value, "value should be true");

  static_assert(!HasM<B, MForm1>::value, "value should be false");

  static_assert(!HasM<A, MForm2>::value, "value should be false");

  static_assert(HasM<B, MForm2>::value, "value should be true");

  static_assert(HasTemplatedM<B, MForm2>::value, "value should be true");

  static_assert(HasVeryTemplatedM<B, MForm3>::value, "value should be true");

  static_assert(HasM<C, MForm4, 1>::value, "value should be true");

  static_assert(HasTemplatedM<B, MForm4, 1>::value, "value should be true");
}

/*
 * Test at compile time, for the presence/absence of a specific member
 * function in a class.
 */
BOOST_AUTO_TEST_CASE(HasMethodNameTest)
{
  static_assert(!HasModel<WithOutFunctionModel>::value,
                "value should be false");
  static_assert(HasModel<WithFunctionModelA>::value, "value should be true");
  static_assert(HasModel<WithFunctionModelB>::value, "value should be true");
  static_assert(HasModel<WithInheritedModelA>::value, "value should be true");
}

BOOST_AUTO_TEST_SUITE_END();
