/*
	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
/**
 * @author Christian 
 * @date 2016
 * Utilities to handle semantic versioning.
 */
#pragma once
#include 
#include 
#include 
#include 
#include 
#include 
namespace solidity::langutil
{
class SemVerError: util::Exception
{
};
#undef major
#undef minor
struct SemVerVersion
{
	unsigned numbers[3];
	std::string prerelease;
	std::string build;
	unsigned major() const { return numbers[0]; }
	unsigned minor() const { return numbers[1]; }
	unsigned patch() const { return numbers[2]; }
	bool isPrerelease() const { return !prerelease.empty(); }
	explicit SemVerVersion(std::string const& _versionString = "0.0.0");
};
struct SemVerMatchExpression
{
	bool matches(SemVerVersion const& _version) const;
	bool isValid() const { return !m_disjunction.empty(); }
	struct MatchComponent
	{
		/// Prefix from < > <= >= ~ ^
		Token prefix = Token::Illegal;
		/// Version, where unsigned(-1) in major, minor or patch denotes '*', 'x' or 'X'
		SemVerVersion version;
		/// Whether we have 1, 1.2 or 1.2.4
		unsigned levelsPresent = 1;
		bool matches(SemVerVersion const& _version) const;
	};
	struct Conjunction
	{
		std::vector components;
		bool matches(SemVerVersion const& _version) const;
	};
	std::vector m_disjunction;
};
class SemVerMatchExpressionParser
{
public:
	SemVerMatchExpressionParser(std::vector _tokens, std::vector _literals):
		m_tokens(std::move(_tokens)), m_literals(std::move(_literals))
	{
		solAssert(m_tokens.size() == m_literals.size(), "");
	}
	/// Returns an expression if it was parseable, or nothing otherwise.
	std::optional parse();
private:
	void reset();
	void parseMatchExpression();
	SemVerMatchExpression::MatchComponent parseMatchComponent();
	unsigned parseVersionPart();
	char currentChar() const;
	char nextChar();
	Token currentToken() const;
	void nextToken();
	std::vector m_tokens;
	std::vector m_literals;
	unsigned m_pos = 0;
	unsigned m_posInside = 0;
	SemVerMatchExpression m_expression;
};
}