/*
	This file is part of solidity.
	solidity is free software: you can redistribute it and/or modify
	it under the terms of the GNU General Public License as published by
	the Free Software Foundation, either version 3 of the License, or
	(at your option) any later version.
	solidity is distributed in the hope that it will be useful,
	but WITHOUT ANY WARRANTY; without even the implied warranty of
	MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
	GNU General Public License for more details.
	You should have received a copy of the GNU General Public License
	along with solidity.  If not, see .
*/
// SPDX-License-Identifier: GPL-3.0
/**
 * Unit tests for the mini moustache class.
 */
#include 
#include 
#include 
namespace solidity::util::test
{
BOOST_AUTO_TEST_SUITE(WhiskersTest, *boost::unit_test::label("nooptions"))
BOOST_AUTO_TEST_CASE(no_templates)
{
	std::string templ = "this text does not contain templates";
	BOOST_CHECK_EQUAL(Whiskers(templ).render(), templ);
}
BOOST_AUTO_TEST_CASE(basic_replacement)
{
	std::string templ = "a  x  -> .";
	std::string result = Whiskers(templ)
		("b", "BE")
		("c", "CE")
		("d", "DE")
		.render();
	BOOST_CHECK_EQUAL(result, "a BE x CE -> DE.");
}
BOOST_AUTO_TEST_CASE(tag_unavailable)
{
	std::string templ = "";
	Whiskers m(templ);
	BOOST_CHECK_THROW(m.render(), WhiskersError);
}
BOOST_AUTO_TEST_CASE(list_unavailable)
{
	std::string templ = "<#b>";
	Whiskers m(templ);
	BOOST_CHECK_THROW(m.render(), WhiskersError);
}
BOOST_AUTO_TEST_CASE(name_type_collision)
{
	std::string templ = "<#b>";
	Whiskers m(templ);
	m("b", "x");
	BOOST_CHECK_THROW(m("b", std::vector>{}), WhiskersError);
}
BOOST_AUTO_TEST_CASE(conditional)
{
	std::string templ = "X";
	BOOST_CHECK_EQUAL(Whiskers(templ)("b", true).render(), "X");
	BOOST_CHECK_EQUAL(Whiskers(templ)("b", false).render(), "");
}
BOOST_AUTO_TEST_CASE(conditional_with_else)
{
	std::string templ = "XY";
	BOOST_CHECK_EQUAL(Whiskers(templ)("b", true).render(), "X");
	BOOST_CHECK_EQUAL(Whiskers(templ)("b", false).render(), "Y");
}
BOOST_AUTO_TEST_CASE(broken_conditional_with_else)
{
	std::string templ = "X";
	BOOST_CHECK_THROW(Whiskers{templ}, WhiskersError);
	templ = "Y";
	BOOST_CHECK_THROW(Whiskers{templ}, WhiskersError);
	templ = "XYX";
	BOOST_CHECK_THROW(Whiskers{templ}, WhiskersError);
	templ = "+bXY+b>";
	BOOST_CHECK_THROW(Whiskers{templ}, WhiskersError);
	templ = "+b>XY+b";
	BOOST_CHECK_THROW(Whiskers{templ}, WhiskersError);
}
BOOST_AUTO_TEST_CASE(broken_list_parameter)
{
	std::string templ = "<#b>";
	BOOST_CHECK_THROW(Whiskers{templ}, WhiskersError);
}
BOOST_AUTO_TEST_CASE(conditional_plus_params)
{
	std::string templ = " - _^ - ";
	Whiskers m1(templ);
	m1("b", true);
	m1("r", "R");
	m1("t", "T");
	BOOST_CHECK_EQUAL(m1.render(), " - _R - ");
	Whiskers m2(templ);
	m2("b", false);
	m2("r", "R");
	m2("t", "T");
	BOOST_CHECK_EQUAL(m2.render(), " - ^T - ");
}
BOOST_AUTO_TEST_CASE(conditional_plus_list)
{
	std::string templ = " - _<#l><#l> - ";
	Whiskers m(templ);
	m("b", false);
	std::vector> list(2);
	list[0]["x"] = "1";
	list[0]["y"] = "a";
	list[1]["x"] = "2";
	list[1]["y"] = "b";
	m("l", list);
	BOOST_CHECK_EQUAL(m.render(), " - ab - ");
}
BOOST_AUTO_TEST_CASE(string_as_conditional)
{
	std::string templ = "+b>+-+b>";
	BOOST_CHECK_EQUAL(Whiskers(templ)("b", "abc").render(), "+abc");
	BOOST_CHECK_EQUAL(Whiskers(templ)("b", "").render(), "-");
}
BOOST_AUTO_TEST_CASE(string_as_conditional_wrong)
{
	std::string templ = "+b>+";
	BOOST_CHECK_EQUAL(Whiskers(templ)("b", "abc").render(), "+b>+abc");
}
BOOST_AUTO_TEST_CASE(complicated_replacement)
{
	std::string templ = "a  x  \n >.";
	std::string result = Whiskers(templ)
		("b", "BE")
		("complicated", "COPL")
		("nested", "NEST")
		.render();
	BOOST_CHECK_EQUAL(result, "a BE x COPL \n NEST>.");
}
BOOST_AUTO_TEST_CASE(non_existing_list)
{
	std::string templ = "a <#b>";
	Whiskers m(templ);
	BOOST_CHECK_THROW(m.render(), WhiskersError);
}
BOOST_AUTO_TEST_CASE(empty_list)
{
	std::string templ = "a <#b>x";
	std::string result = Whiskers(templ)("b", std::vector{}).render();
	BOOST_CHECK_EQUAL(result, "a x");
}
BOOST_AUTO_TEST_CASE(list)
{
	std::string templ = "a<#b>(  -  )x";
	std::vector> list(2);
	list[0]["g"] = "GE";
	list[0]["h"] = "H";
	list[1]["g"] = "2GE";
	list[1]["h"] = "2H";
	std::string result = Whiskers(templ)("b", list).render();
	BOOST_CHECK_EQUAL(result, "a( GE - H )( 2GE - 2H )x");
}
BOOST_AUTO_TEST_CASE(recursive_list)
{
	// Check that templates resulting from lists are not expanded again
	std::string templ = "a<#b> 13 ";
	std::vector> list(1);
	list[0]["g"] = "";
	std::string result = Whiskers(templ)("x", "X")("b", list).render();
	BOOST_CHECK_EQUAL(result, "a 13 X");
}
BOOST_AUTO_TEST_CASE(list_can_access_upper)
{
	std::string templ = "<#b>()";
	std::vector> list(2);
	Whiskers m(templ);
	std::string result = m("a", "A")("b", list).render();
	BOOST_CHECK_EQUAL(result, "(A)(A)");
}
BOOST_AUTO_TEST_CASE(parameter_collision)
{
	std::string templ = "a <#b>";
	std::vector> list(1);
	list[0]["a"] = "x";
	Whiskers m(templ);
	BOOST_CHECK_THROW(m("a", "X")("b", list), WhiskersError);
	BOOST_CHECK_THROW(m.render(), WhiskersError);
}
BOOST_AUTO_TEST_CASE(invalid_param)
{
	std::string templ = "a ";
	Whiskers m(templ);
	BOOST_CHECK_THROW(m("b ", "X"), WhiskersError);
}
BOOST_AUTO_TEST_CASE(invalid_param_rendered)
{
	std::string templ = "a ";
	Whiskers m(templ);
	BOOST_CHECK_EQUAL(m.render(), templ);
}
BOOST_AUTO_TEST_SUITE_END()
}