2017-08-22 22:34:41 +00:00
|
|
|
/*
|
2017-11-18 21:49:52 +00:00
|
|
|
__________
|
|
|
|
_____ __ __\______ \_____ _______ ______ ____ _______
|
2017-08-22 22:34:41 +00:00
|
|
|
/ \ | | \| ___/\__ \ \_ __ \/ ___/_/ __ \\_ __ \
|
|
|
|
| Y Y \| | /| | / __ \_| | \/\___ \ \ ___/ | | \/
|
2017-11-18 21:49:52 +00:00
|
|
|
|__|_| /|____/ |____| (____ /|__| /____ > \___ >|__|
|
|
|
|
\/ \/ \/ \/
|
2017-08-22 22:34:41 +00:00
|
|
|
Copyright (C) 2013 Ingo Berg
|
|
|
|
|
2017-11-18 21:49:52 +00:00
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy of this
|
2017-08-22 22:34:41 +00:00
|
|
|
software and associated documentation files (the "Software"), to deal in the Software
|
2017-11-18 21:49:52 +00:00
|
|
|
without restriction, including without limitation the rights to use, copy, modify,
|
|
|
|
merge, publish, distribute, sublicense, and/or sell copies of the Software, and to
|
2017-08-22 22:34:41 +00:00
|
|
|
permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
|
|
|
|
2017-11-18 21:49:52 +00:00
|
|
|
The above copyright notice and this permission notice shall be included in all copies or
|
2017-08-22 22:34:41 +00:00
|
|
|
substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT
|
2017-11-18 21:49:52 +00:00
|
|
|
NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
|
|
DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
2017-08-22 22:34:41 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef MU_PARSER_TEST_H
|
|
|
|
#define MU_PARSER_TEST_H
|
|
|
|
|
|
|
|
#include <cstdlib>
|
2017-11-18 21:49:52 +00:00
|
|
|
#include <numeric> // for accumulate
|
|
|
|
#include <string>
|
2017-08-22 22:34:41 +00:00
|
|
|
#include "muParser.h"
|
|
|
|
#include "muParserInt.h"
|
|
|
|
|
|
|
|
/** \file
|
|
|
|
\brief This file contains the parser test class.
|
|
|
|
*/
|
|
|
|
|
2017-11-18 21:49:52 +00:00
|
|
|
namespace mu {
|
|
|
|
/** \brief Namespace for test cases. */
|
|
|
|
namespace Test {
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
/** \brief Test cases for unit testing.
|
|
|
|
|
|
|
|
(C) 2004-2011 Ingo Berg
|
|
|
|
*/
|
|
|
|
class ParserTester // final
|
2017-08-22 22:34:41 +00:00
|
|
|
{
|
2017-11-18 21:49:52 +00:00
|
|
|
private:
|
|
|
|
static int c_iCount;
|
|
|
|
|
|
|
|
// Multiarg callbacks
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f1of1(value_type v) { return v; };
|
2017-11-18 21:49:52 +00:00
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f1of2(value_type v, value_type) { return v; };
|
|
|
|
static ValueOrError f2of2(value_type, value_type v) { return v; };
|
2017-11-18 21:49:52 +00:00
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f1of3(value_type v, value_type, value_type) { return v; };
|
|
|
|
static ValueOrError f2of3(value_type, value_type v, value_type) { return v; };
|
|
|
|
static ValueOrError f3of3(value_type, value_type, value_type v) { return v; };
|
2017-11-18 21:49:52 +00:00
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f1of4(value_type v, value_type, value_type, value_type) { return v; }
|
|
|
|
static ValueOrError f2of4(value_type, value_type v, value_type, value_type) { return v; }
|
|
|
|
static ValueOrError f3of4(value_type, value_type, value_type v, value_type) { return v; }
|
|
|
|
static ValueOrError f4of4(value_type, value_type, value_type, value_type v) { return v; }
|
2017-11-18 21:49:52 +00:00
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f1of5(value_type v, value_type, value_type, value_type, value_type) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return v;
|
|
|
|
}
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f2of5(value_type, value_type v, value_type, value_type, value_type) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return v;
|
|
|
|
}
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f3of5(value_type, value_type, value_type v, value_type, value_type) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return v;
|
|
|
|
}
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f4of5(value_type, value_type, value_type, value_type v, value_type) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return v;
|
|
|
|
}
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError f5of5(value_type, value_type, value_type, value_type, value_type v) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError Min(value_type a_fVal1, value_type a_fVal2) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return (a_fVal1 < a_fVal2) ? a_fVal1 : a_fVal2;
|
|
|
|
}
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError Max(value_type a_fVal1, value_type a_fVal2) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return (a_fVal1 > a_fVal2) ? a_fVal1 : a_fVal2;
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError plus2(value_type v1) { return v1 + 2; }
|
|
|
|
static ValueOrError times3(value_type v1) { return v1 * 3; }
|
|
|
|
static ValueOrError sqr(value_type v1) { return v1 * v1; }
|
|
|
|
static ValueOrError sign(value_type v) { return -v; }
|
|
|
|
static ValueOrError add(value_type v1, value_type v2) { return v1 + v2; }
|
|
|
|
static ValueOrError land(value_type v1, value_type v2) { return (int)v1 & (int)v2; }
|
2017-11-18 21:49:52 +00:00
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError FirstArg(const value_type* a_afArg, int a_iArgc) {
|
2017-11-18 21:49:52 +00:00
|
|
|
if (!a_iArgc)
|
|
|
|
throw mu::Parser::exception_type(_T("too few arguments for function FirstArg."));
|
|
|
|
|
|
|
|
return a_afArg[0];
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError LastArg(const value_type* a_afArg, int a_iArgc) {
|
2017-11-18 21:49:52 +00:00
|
|
|
if (!a_iArgc)
|
|
|
|
throw mu::Parser::exception_type(_T("too few arguments for function LastArg."));
|
|
|
|
|
|
|
|
return a_afArg[a_iArgc - 1];
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError Sum(const value_type* a_afArg, int a_iArgc) {
|
2017-11-18 21:49:52 +00:00
|
|
|
if (!a_iArgc) throw mu::Parser::exception_type(_T("too few arguments for function sum."));
|
|
|
|
|
|
|
|
value_type fRes = 0;
|
|
|
|
for (int i = 0; i < a_iArgc; ++i) fRes += a_afArg[i];
|
|
|
|
return fRes;
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError Rnd(value_type v) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return (value_type)(1 + (v * std::rand() / (RAND_MAX + 1.0)));
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError RndWithString(const char_type*) {
|
2017-11-18 21:49:52 +00:00
|
|
|
return (value_type)(1 + (1000.0f * std::rand() / (RAND_MAX + 1.0)));
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError Ping() { return 10; }
|
2017-11-18 21:49:52 +00:00
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError ValueOf(const char_type*) { return 123; }
|
2017-11-18 21:49:52 +00:00
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError StrFun1(const char_type* v1) {
|
2017-11-18 21:49:52 +00:00
|
|
|
int val(0);
|
|
|
|
stringstream_type(v1) >> val;
|
|
|
|
return (value_type)val;
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError StrFun2(const char_type* v1, value_type v2) {
|
2017-11-18 21:49:52 +00:00
|
|
|
int val(0);
|
|
|
|
stringstream_type(v1) >> val;
|
|
|
|
return (value_type)(val + v2);
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError StrFun3(const char_type* v1, value_type v2, value_type v3) {
|
2017-11-18 21:49:52 +00:00
|
|
|
int val(0);
|
|
|
|
stringstream_type(v1) >> val;
|
|
|
|
return val + v2 + v3;
|
|
|
|
}
|
|
|
|
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError StrToFloat(const char_type* a_szMsg) {
|
2017-11-18 21:49:52 +00:00
|
|
|
value_type val(0);
|
|
|
|
stringstream_type(a_szMsg) >> val;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
// postfix operator callback
|
2017-11-18 22:36:14 +00:00
|
|
|
static ValueOrError Mega(value_type a_fVal) { return a_fVal * (value_type)1e6; }
|
|
|
|
static ValueOrError Micro(value_type a_fVal) { return a_fVal * (value_type)1e-6; }
|
|
|
|
static ValueOrError Milli(value_type a_fVal) { return a_fVal / (value_type)1e3; }
|
2017-11-18 21:49:52 +00:00
|
|
|
|
|
|
|
// Custom value recognition
|
|
|
|
static int IsHexVal(const char_type* a_szExpr, int* a_iPos, value_type* a_fVal);
|
|
|
|
|
|
|
|
int TestNames();
|
|
|
|
int TestSyntax();
|
|
|
|
int TestMultiArg();
|
|
|
|
int TestPostFix();
|
|
|
|
int TestExpression();
|
|
|
|
int TestInfixOprt();
|
|
|
|
int TestBinOprt();
|
|
|
|
int TestInterface();
|
|
|
|
int TestException();
|
|
|
|
int TestStrArg();
|
|
|
|
int TestIfThenElse();
|
|
|
|
|
|
|
|
void Abort() const;
|
|
|
|
|
|
|
|
public:
|
|
|
|
typedef int (ParserTester::*testfun_type)();
|
|
|
|
|
|
|
|
ParserTester();
|
|
|
|
void Run();
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<testfun_type> m_vTestFun;
|
|
|
|
void AddTest(testfun_type a_pFun);
|
|
|
|
|
|
|
|
// Test Double Parser
|
|
|
|
int EqnTest(const string_type& a_str, double a_fRes, bool a_fPass);
|
|
|
|
int EqnTestWithVarChange(const string_type& a_str, double a_fRes1, double a_fVar1,
|
|
|
|
double a_fRes2, double a_fVar2);
|
|
|
|
int ThrowTest(const string_type& a_str, int a_iErrc, bool a_bFail = true);
|
|
|
|
|
|
|
|
// Test Int Parser
|
|
|
|
int EqnTestInt(const string_type& a_str, double a_fRes, bool a_fPass);
|
|
|
|
};
|
|
|
|
} // namespace Test
|
|
|
|
} // namespace mu
|
2017-08-22 22:34:41 +00:00
|
|
|
|
|
|
|
#endif
|