aboutsummaryrefslogblamecommitdiffstatshomepage
path: root/test/unit_tests/parser.cpp
blob: 37d9e0a955d24db7389037e07960feaea7bb7ed6 (plain) (tree)
1
2
3
4
5
6
7
8
9
10




                                                               

                                   
 

                                          

                                   


                      
 















                                                                               



                                                                               








                                   



              
















































                                                                 

     

                           
// Copyright (c) 2019 Egor Tensin <Egor.Tensin@gmail.com>
// This file is part of the "math-server" project.
// For details, see https://github.com/egor-tensin/math-server.
// Distributed under the MIT License.

#include <server/parser/error.hpp>
#include <server/parser/parser.hpp>

#include <boost/test/data/test_case.hpp>
#include <boost/test/data/monomorphic.hpp>
#include <boost/test/unit_test.hpp>

#include <string>
#include <string_view>
#include <vector>

BOOST_AUTO_TEST_SUITE(parser_tests)

namespace bdata = boost::unit_test::data;
using math::server::Parser;
using math::server::ParserError;

namespace {
namespace exec::valid {

const std::vector<std::string_view> input{
    "1",
    " 1 + 2 ",
    " 2 * 1 + 3 ",
    " 2 * (1 + 3) ",
    " 2 * (1 + 3 * (1 - -3)) ",
    " -2 * ---- (3 + -100e-1)  ", // Looks weird, but also works in e.g. Python
    "2 ^ 3 ^ 3",
    "(2 ^ 3) ^ 3", // Power operator is right-associative
    "(.5 ^ -1) ^ 4",
    ".5 ^ -1 ^ 4", // Power operator has higher precedence than the unary minus
};

const std::vector<double> expected{
    1,
    3,
    5,
    8,
    26,
    14,
    134217728,
    512,
    16,
    2,
};

}

namespace exec::invalid {

const std::vector<std::string_view> input{
    "",
    " 1 + ",
    " 2 * (1 + 3 ",
    " 2 * (1 + 3) )",
};

const std::vector<std::string> error_msg{
    "server error: parser error: expected '-', '(' or a number",
    "server error: parser error: expected '-', '(' or a number",
    "server error: parser error: missing closing ')'",
    "server error: parser error: expected a binary operator",
};

}
}

BOOST_DATA_TEST_CASE(
    test_exec_valid,
    bdata::make(exec::valid::input) ^ exec::valid::expected,
    input,
    expected) {

    Parser parser{input};
    BOOST_TEST(parser.exec() == expected);
}

BOOST_DATA_TEST_CASE(
    test_exec_invalid,
    bdata::make(exec::invalid::input) ^ exec::invalid::error_msg,
    input,
    error_msg) {

    BOOST_REQUIRE_THROW(do {
        Parser parser{input};
        parser.exec();
    } while (0), ParserError);

    try {
        Parser parser{input};
        parser.exec();
    } catch (const ParserError& e) {
        BOOST_TEST(error_msg == e.what());
    }
}

BOOST_AUTO_TEST_SUITE_END()