mirror of
https://github.com/Laupetin/OpenAssetTools.git
synced 2025-04-19 15:52:53 +00:00
842 lines
30 KiB
C++
842 lines
30 KiB
C++
#include <catch2/catch.hpp>
|
|
|
|
#include <sstream>
|
|
|
|
#include "Parsing/Impl/AbstractParser.h"
|
|
#include "Parsing/Impl/ParserSingleInputStream.h"
|
|
#include "Utils/ClassUtils.h"
|
|
#include "Parsing/Mock/MockLexer.h"
|
|
#include "Parsing/Simple/SimpleParserValue.h"
|
|
#include "Parsing/Simple/Expression/ISimpleExpression.h"
|
|
#include "Parsing/Simple/Expression/SimpleExpressionMatchers.h"
|
|
#include "Parsing/Simple/Matcher/SimpleMatcherFactory.h"
|
|
|
|
namespace test::parsing::simple::expression
|
|
{
|
|
class SimpleExpressionTestState
|
|
{
|
|
public:
|
|
std::unique_ptr<ISimpleExpression> m_expression;
|
|
};
|
|
|
|
class SimpleExpressionSequence final : public AbstractSequence<SimpleParserValue, SimpleExpressionTestState>
|
|
{
|
|
const SimpleExpressionMatchers m_expression_matchers;
|
|
|
|
public:
|
|
SimpleExpressionSequence()
|
|
: m_expression_matchers(true, true, true, true, true)
|
|
{
|
|
const SimpleMatcherFactory create(this);
|
|
|
|
AddLabeledMatchers(m_expression_matchers.Expression(this), SimpleExpressionMatchers::LABEL_EXPRESSION);
|
|
AddMatchers(create.Label(SimpleExpressionMatchers::LABEL_EXPRESSION));
|
|
}
|
|
|
|
protected:
|
|
void ProcessMatch(SimpleExpressionTestState* state, SequenceResult<SimpleParserValue>& result) const override
|
|
{
|
|
if (state->m_expression)
|
|
throw ParsingException(TokenPos(), "Expression already set");
|
|
|
|
state->m_expression = m_expression_matchers.ProcessExpression(result);
|
|
}
|
|
};
|
|
|
|
class SimpleExpressionParser final : public AbstractParser<SimpleParserValue, SimpleExpressionTestState>
|
|
{
|
|
public:
|
|
explicit SimpleExpressionParser(ILexer<SimpleParserValue>* lexer)
|
|
: AbstractParser(lexer, std::make_unique<SimpleExpressionTestState>())
|
|
{
|
|
}
|
|
|
|
_NODISCARD SimpleExpressionTestState* GetState() const
|
|
{
|
|
return m_state.get();
|
|
}
|
|
|
|
protected:
|
|
const std::vector<sequence_t*>& GetTestsForState() override
|
|
{
|
|
static std::vector<sequence_t*> tests({
|
|
new SimpleExpressionSequence()
|
|
});
|
|
|
|
return tests;
|
|
}
|
|
};
|
|
|
|
class SimpleExpressionTestsHelper
|
|
{
|
|
public:
|
|
SimpleExpressionTestState* m_state;
|
|
std::unique_ptr<SimpleExpressionTestState> m_state_holder;
|
|
std::string m_str;
|
|
std::istringstream m_ss;
|
|
std::unique_ptr<IParserLineStream> m_stream;
|
|
std::unique_ptr<ILexer<SimpleParserValue>> m_lexer;
|
|
std::unique_ptr<SimpleExpressionParser> m_parser;
|
|
|
|
std::unique_ptr<SimpleExpressionSequence> m_sequence;
|
|
|
|
unsigned m_consumed_token_count;
|
|
|
|
SimpleExpressionTestsHelper()
|
|
: m_state(nullptr),
|
|
m_sequence(std::make_unique<SimpleExpressionSequence>()),
|
|
m_consumed_token_count(0u)
|
|
{
|
|
}
|
|
|
|
void Tokens(std::initializer_list<Movable<SimpleParserValue>> tokens)
|
|
{
|
|
m_state_holder = std::make_unique<SimpleExpressionTestState>();
|
|
m_state = m_state_holder.get();
|
|
m_lexer = std::make_unique<MockLexer<SimpleParserValue>>(tokens, SimpleParserValue::EndOfFile(TokenPos()));
|
|
}
|
|
|
|
void Tokens(std::vector<SimpleParserValue> tokens)
|
|
{
|
|
m_state_holder = std::make_unique<SimpleExpressionTestState>();
|
|
m_state = m_state_holder.get();
|
|
m_lexer = std::make_unique<MockLexer<SimpleParserValue>>(std::move(tokens), SimpleParserValue::EndOfFile(TokenPos()));
|
|
}
|
|
|
|
void String(std::string str)
|
|
{
|
|
m_str = std::move(str);
|
|
m_ss = std::istringstream(m_str);
|
|
m_stream = std::make_unique<ParserSingleInputStream>(m_ss, "InputString");
|
|
|
|
SimpleLexer::Config lexerConfig;
|
|
lexerConfig.m_read_strings = true;
|
|
lexerConfig.m_read_numbers = true;
|
|
lexerConfig.m_emit_new_line_tokens = false;
|
|
SimpleExpressionMatchers(true, true, true, true, true).ApplyTokensToLexerConfig(lexerConfig);
|
|
m_lexer = std::make_unique<SimpleLexer>(m_stream.get(), std::move(lexerConfig));
|
|
m_parser = std::make_unique<SimpleExpressionParser>(m_lexer.get());
|
|
m_state = m_parser->GetState();
|
|
}
|
|
|
|
bool PerformTest()
|
|
{
|
|
REQUIRE(m_lexer);
|
|
|
|
m_consumed_token_count = 0;
|
|
return m_sequence->MatchSequence(m_lexer.get(), m_state, m_consumed_token_count);
|
|
}
|
|
|
|
_NODISCARD bool PerformIntegrationTest() const
|
|
{
|
|
REQUIRE(m_lexer);
|
|
REQUIRE(m_parser);
|
|
|
|
return m_parser->Parse();
|
|
}
|
|
};
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with add operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 1336),
|
|
SimpleParserValue::Character(pos, '+'),
|
|
SimpleParserValue::Integer(pos, 1),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1337);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with subtract operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 428),
|
|
SimpleParserValue::Character(pos, '-'),
|
|
SimpleParserValue::Integer(pos, 8),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with multiply operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 105),
|
|
SimpleParserValue::Character(pos, '*'),
|
|
SimpleParserValue::Integer(pos, 4),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with divide operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 561540),
|
|
SimpleParserValue::Character(pos, '/'),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1337);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with remainder operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 92673),
|
|
SimpleParserValue::Character(pos, '%'),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with bitwise and operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 0x7FFFFFF0),
|
|
SimpleParserValue::Character(pos, '&'),
|
|
SimpleParserValue::Integer(pos, 0x2AAAAAAA),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 0x2AAAAAA0);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with bitwise or operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 5),
|
|
SimpleParserValue::Character(pos, '|'),
|
|
SimpleParserValue::Integer(pos, 3),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 7);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with shift left operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 105),
|
|
SimpleParserValue::MultiCharacter(pos, SimpleExpressionMatchers::MULTI_TOKEN_OFFSET_BINARY + static_cast<int>(SimpleBinaryOperationId::SHIFT_LEFT)),
|
|
SimpleParserValue::Integer(pos, 2),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with shift right operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 42784),
|
|
SimpleParserValue::MultiCharacter(pos, SimpleExpressionMatchers::MULTI_TOKEN_OFFSET_BINARY + static_cast<int>(SimpleBinaryOperationId::SHIFT_RIGHT)),
|
|
SimpleParserValue::Integer(pos, 5),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1337);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with greater operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::Character(pos, '>'),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with greater equal operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::MultiCharacter(pos, SimpleExpressionMatchers::MULTI_TOKEN_OFFSET_BINARY + static_cast<int>(SimpleBinaryOperationId::GREATER_EQUAL_THAN)),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with less operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::Character(pos, '<'),
|
|
SimpleParserValue::Integer(pos, 421),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with less equal operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::MultiCharacter(pos, SimpleExpressionMatchers::MULTI_TOKEN_OFFSET_BINARY + static_cast<int>(SimpleBinaryOperationId::LESS_EQUAL_THAN)),
|
|
SimpleParserValue::Integer(pos, 421),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with equals operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::MultiCharacter(pos, SimpleExpressionMatchers::MULTI_TOKEN_OFFSET_BINARY + static_cast<int>(SimpleBinaryOperationId::EQUALS)),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with not equal operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::MultiCharacter(pos, SimpleExpressionMatchers::MULTI_TOKEN_OFFSET_BINARY + static_cast<int>(SimpleBinaryOperationId::NOT_EQUAL)),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 0);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with and operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::MultiCharacter(pos, SimpleExpressionMatchers::MULTI_TOKEN_OFFSET_BINARY + static_cast<int>(SimpleBinaryOperationId::AND)),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Can parse expression with or operation", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 0),
|
|
SimpleParserValue::MultiCharacter(pos, SimpleExpressionMatchers::MULTI_TOKEN_OFFSET_BINARY + static_cast<int>(SimpleBinaryOperationId::OR)),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 3);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1337);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Operator precedence is applied", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 100),
|
|
SimpleParserValue::Character(pos, '+'),
|
|
SimpleParserValue::Integer(pos, 4),
|
|
SimpleParserValue::Character(pos, '*'),
|
|
SimpleParserValue::Integer(pos, 25),
|
|
SimpleParserValue::Character(pos, '+'),
|
|
SimpleParserValue::Integer(pos, 220),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 7);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Parenthesis have highest precedence", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Character(pos, '('),
|
|
SimpleParserValue::Integer(pos, 40),
|
|
SimpleParserValue::Character(pos, '+'),
|
|
SimpleParserValue::Integer(pos, 2),
|
|
SimpleParserValue::Character(pos, ')'),
|
|
SimpleParserValue::Character(pos, '*'),
|
|
SimpleParserValue::Integer(pos, 10),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 7);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Simple conditional operator can be used with true value", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 1),
|
|
SimpleParserValue::Character(pos, '?'),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::Character(pos, ':'),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 5);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Simple conditional operator can be used with false value", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 0),
|
|
SimpleParserValue::Character(pos, '?'),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::Character(pos, ':'),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 5);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1337);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Simple conditional operator can be used within parenthesis with true value", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Character(pos, '('),
|
|
SimpleParserValue::Integer(pos, 1),
|
|
SimpleParserValue::Character(pos, '?'),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::Character(pos, ':'),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::Character(pos, ')'),
|
|
SimpleParserValue::Character(pos, '+'),
|
|
SimpleParserValue::Integer(pos, 1),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 9);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 421);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Simple conditional operator can be used within parenthesis with false value", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Character(pos, '('),
|
|
SimpleParserValue::Integer(pos, 0),
|
|
SimpleParserValue::Character(pos, '?'),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::Character(pos, ':'),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::Character(pos, ')'),
|
|
SimpleParserValue::Character(pos, '+'),
|
|
SimpleParserValue::Integer(pos, 1),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 9);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1338);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Simple conditional operator can have an expression as condition", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Character(pos, '('),
|
|
SimpleParserValue::Integer(pos, -1),
|
|
SimpleParserValue::Character(pos, '+'),
|
|
SimpleParserValue::Integer(pos, 2),
|
|
SimpleParserValue::Character(pos, ')'),
|
|
SimpleParserValue::Character(pos, '?'),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::Character(pos, ':'),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 9);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Simple conditional operator can have an expression as true value", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 1),
|
|
SimpleParserValue::Character(pos, '?'),
|
|
SimpleParserValue::Character(pos, '('),
|
|
SimpleParserValue::Integer(pos, 210),
|
|
SimpleParserValue::Character(pos, '*'),
|
|
SimpleParserValue::Integer(pos, 2),
|
|
SimpleParserValue::Character(pos, ')'),
|
|
SimpleParserValue::Character(pos, ':'),
|
|
SimpleParserValue::Integer(pos, 1337),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 9);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 420);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressions: Simple conditional operator can have an expression as false value", "[parsing][simple][expression]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
const TokenPos pos;
|
|
helper.Tokens({
|
|
SimpleParserValue::Integer(pos, 0),
|
|
SimpleParserValue::Character(pos, '?'),
|
|
SimpleParserValue::Integer(pos, 420),
|
|
SimpleParserValue::Character(pos, ':'),
|
|
SimpleParserValue::Character(pos, '('),
|
|
SimpleParserValue::Integer(pos, 1336),
|
|
SimpleParserValue::Character(pos, '+'),
|
|
SimpleParserValue::Integer(pos, 1),
|
|
SimpleParserValue::Character(pos, ')'),
|
|
SimpleParserValue::EndOfFile(pos)
|
|
});
|
|
|
|
const auto result = helper.PerformTest();
|
|
|
|
REQUIRE(result);
|
|
REQUIRE(helper.m_consumed_token_count == 9);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1337);
|
|
}
|
|
|
|
namespace it
|
|
{
|
|
TEST_CASE("SimpleExpressionsIT: Can parse subtraction without space", "[parsing][simple][expression][it]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
helper.String("6-5");
|
|
|
|
const auto result = helper.PerformIntegrationTest();
|
|
|
|
REQUIRE(result);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 1);
|
|
}
|
|
|
|
TEST_CASE("SimpleExpressionsIT: Can parse addition without space", "[parsing][simple][expression][it]")
|
|
{
|
|
SimpleExpressionTestsHelper helper;
|
|
helper.String("6+5");
|
|
|
|
const auto result = helper.PerformIntegrationTest();
|
|
|
|
REQUIRE(result);
|
|
|
|
const auto& expression = helper.m_state->m_expression;
|
|
REQUIRE(expression->IsStatic());
|
|
|
|
const auto value = expression->Evaluate();
|
|
REQUIRE(value.m_type == SimpleExpressionValue::Type::INT);
|
|
REQUIRE(value.m_int_value == 11);
|
|
}
|
|
}
|
|
}
|