/*
    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 <http://www.gnu.org/licenses/>.
*/
/**
 * @date 2017
 * Common functions the iulia tests.
 */

#include <test/libjulia/Common.h>

#include <libjulia/optimiser/Disambiguator.h>

#include <libsolidity/parsing/Scanner.h>

#include <libsolidity/inlineasm/AsmParser.h>
#include <libsolidity/inlineasm/AsmAnalysis.h>
#include <libsolidity/inlineasm/AsmPrinter.h>

#include <libsolidity/interface/SourceReferenceFormatter.h>
#include <libsolidity/interface/ErrorReporter.h>

#include <boost/test/unit_test.hpp>

using namespace std;
using namespace dev::julia;
using namespace dev::solidity;

void dev::julia::test::printErrors(ErrorList const& _errors, Scanner const& _scanner)
{
	for (auto const& error: _errors)
		SourceReferenceFormatter::printExceptionInformation(
			cout,
			*error,
			(error->type() == Error::Type::Warning) ? "Warning" : "Error",
			[&](std::string const&) -> Scanner const& { return _scanner; }
		);
}


pair<shared_ptr<Block>, shared_ptr<assembly::AsmAnalysisInfo>> dev::julia::test::parse(string const& _source, bool _julia)
{
	auto flavour = _julia ? assembly::AsmFlavour::IULIA : assembly::AsmFlavour::Strict;
	ErrorList errors;
	ErrorReporter errorReporter(errors);
	auto scanner = make_shared<Scanner>(CharStream(_source), "");
	auto parserResult = assembly::Parser(errorReporter, flavour).parse(scanner);
	if (parserResult)
	{
		BOOST_REQUIRE(errorReporter.errors().empty());
		auto analysisInfo = make_shared<assembly::AsmAnalysisInfo>();
		assembly::AsmAnalyzer analyzer(*analysisInfo, errorReporter, flavour);
		if (analyzer.analyze(*parserResult))
		{
			BOOST_REQUIRE(errorReporter.errors().empty());
			return make_pair(parserResult, analysisInfo);
		}
	}
	printErrors(errors, *scanner);
	BOOST_FAIL("Invalid source.");

	// Unreachable.
	return {};
}

assembly::Block dev::julia::test::disambiguate(string const& _source, bool _julia)
{
	auto result = parse(_source, _julia);
	return boost::get<Block>(Disambiguator(*result.second)(*result.first));
}

string dev::julia::test::format(string const& _source, bool _julia)
{
	return assembly::AsmPrinter(_julia)(*parse(_source, _julia).first);
}