/*
	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
#pragma once
#include 
#include 
#include 
#include 
#include 
namespace solidity::test::solprotofuzzer
{
/// Random number generator that is seeded with a fuzzer
/// supplied unsigned integer.
struct SolRandomNumGenerator
{
	using RandomEngine = std::minstd_rand;
	explicit SolRandomNumGenerator(unsigned _seed): m_random(RandomEngine(_seed)) {}
	/// @returns a pseudo random unsigned integer
	unsigned operator()()
	{
		return m_random();
	}
	RandomEngine m_random;
};
/* There are two types of tests created by the converter:
 * - library test
 * - contract test
 *
 * The template for library test is the following:
 *
 * // Library generated from fuzzer protobuf specification
 * library L0 {
 *      function f0(uint) public view returns (uint) {
 *          return 31337;
 *      }
 *      function f1(uint) public pure returns (uint) {
 *          return 455;
 *      }
 * }
 * library L1 {
 *      function f0(uint) external view returns (uint) {
 *          return 607;
 *      }
 * }
 *
 * // Test entry point
 * contract C {
 *     // Uses a single pseudo randomly chosen library
 *     // and calls a pseudo randomly chosen function
 *     // returning a non-zero error code on failure or
 *     // a zero uint when test passes.
 *     using L0 for uint;
 *     function test() public pure returns (uint) {
 *          uint x;
 *          if (x.f1() != 455)
 *              return 1;
 *          return 0;
 *     }
 * }
 *
 * The template for contract test is the following
 * // Contracts generated from fuzzer protobuf specification
 * contract C0B {
 *      function f0() public pure virtual returns (uint)
 *      {
 *          return 42;
 *      }
 * }
 * contract C0 is C0B {
 *      function f0() public pure override returns (uint)
 *      {
 *          return 1337;
 *      }
 * }
 *
 * // Test entry point
 * contract C {
 *      // Invokes one or more contract functions returning
 *      // a non-zero error code for failure, a zero uint
 *      // when all tests pass
 *      function test() public pure returns (uint)
 *      {
 *          C0 tc0 = new C0();
 *          if (tc0.f0() != 1337)
 *              return 1;
 *          C0B tc1 = new C0B();
 *          if (tc1.f0() != 42)
 *              return 2;
 *          // Expected return value if all tests pass
 *          return 0;
 *      }
 * }
 */
class ProtoConverter
{
public:
	ProtoConverter() {}
	ProtoConverter(ProtoConverter const&) = delete;
	ProtoConverter(ProtoConverter&&) = delete;
	std::string protoToSolidity(Program const&);
	/// @returns true if test calls a library function, false
	/// otherwise
	bool libraryTest() const;
	/// @returns name of the library under test
	std::string libraryName() const;
private:
	/// Variant type that points to one of contract, interface, library protobuf messages
	using CIL = std::variant;
	/// Protobuf message visitors that accept a const reference to a protobuf message
	/// type and return its solidity translation.
	std::string visit(Program const&);
	std::string visit(TestContract const&);
	std::string visit(ContractType const&);
	std::string visit(Interface const& _interface);
	std::string visit(Library const& _library);
	std::string visit(Contract const& _contract);
	/// @returns a string pair containing a library declaration (relevant for library
	/// tests only) and a solidity test case
	std::pair generateTestCase(TestContract const& _testContract);
	/// @returns name of a program i.e., contract, library or interface
	std::string programName(CIL _program);
	/// @returns a tuple containing the names of the library and function under
	/// test, and its expected output.
	std::tuple pseudoRandomLibraryTest();
	/// Performs bookkeeping for a fuzzer-supplied program
	void openProgramScope(CIL _program);
	/// @returns a deterministic pseudo random unsigned integer
	unsigned randomNumber();
	/// @returns true if fuzzer supplied Library protobuf message
	/// contains zero functions, false otherwise.
	static bool emptyLibrary(Library const& _library)
	{
		return _library.funcdef_size() == 0;
	}
	/// @returns true if there are no valid library test cases, false
	/// otherwise.
	bool emptyLibraryTests()
	{
		return m_libraryTests.empty();
	}
	/// @returns true if there are no valid contract test cases, false
	/// otherwise.
	bool emptyContractTests()
	{
		return m_contractTests.empty();
	}
	/// Numeric suffix that is part of program names e.g., "0" in "C0"
	unsigned m_programNumericSuffix = 0;
	/// Flag that states whether library call is tested (true) or not (false).
	bool m_libraryTest = false;
	/// A smart pointer to fuzzer driven random number generator
	std::shared_ptr m_randomGen;
	/// Maps protobuf program to its string name
	std::map m_programNameMap;
	/// List of tuples containing library name, function and its expected output
	std::vector> m_libraryTests;
	/// Maps contract name to a map of function names and their expected output
	std::map> m_contractTests;
	/// Name of the library under test, relevant if m_libraryTest is set
	std::string m_libraryName;
	/// Maximum number of local variables in test function to avoid stack too deep
	/// errors
	static unsigned constexpr s_maxVars = 15;
};
}