diff --git a/tests/test.hpp b/tests/test.hpp
index 41dc218..028da62 100644
--- a/tests/test.hpp
+++ b/tests/test.hpp
@@ -46,6 +46,15 @@ inline bool test_xpath_number(const pugi::xml_node& node, const char* query, dou
return fabs(q.evaluate_number(node) - expected) < 1e-8f;
}
+inline bool test_xpath_number_nan(const pugi::xml_node& node, const char* query)
+{
+ pugi::xpath_query q(query);
+
+ double r = q.evaluate_number(node);
+
+ return r != r;
+}
+
inline bool test_xpath_fail_compile(const char* query)
{
try
@@ -127,6 +136,7 @@ struct dummy_fixture {};
#define CHECK_XPATH_STRING(node, query, expected) CHECK_TEXT(test_xpath_string(node, query, expected), #query " does not evaluate to " #expected " in context " #node)
#define CHECK_XPATH_BOOLEAN(node, query, expected) CHECK_TEXT(test_xpath_boolean(node, query, expected), #query " does not evaluate to " #expected " in context " #node)
#define CHECK_XPATH_NUMBER(node, query, expected) CHECK_TEXT(test_xpath_number(node, query, expected), #query " does not evaluate to " #expected " in context " #node)
+#define CHECK_XPATH_NUMBER_NAN(node, query) CHECK_TEXT(test_xpath_number_nan(node, query), #query " does not evaluate to NaN in context " #node)
#define CHECK_XPATH_FAIL(query) CHECK_TEXT(test_xpath_fail_compile(query), #query " should not compile")
#endif
diff --git a/tests/test_xpath_functions.cpp b/tests/test_xpath_functions.cpp
new file mode 100644
index 0000000..e4430f3
--- /dev/null
+++ b/tests/test_xpath_functions.cpp
@@ -0,0 +1,189 @@
+#include "common.hpp"
+
+TEST_XML(xpath_number_number, "123")
+{
+ xml_node c;
+ xml_node n = doc.child("node").first_child();
+
+ // number with 0 arguments
+ CHECK_XPATH_NUMBER_NAN(c, "number()");
+ CHECK_XPATH_NUMBER(n, "number()", 123);
+
+ // number with 1 string argument
+ CHECK_XPATH_NUMBER(c, "number(' -123.456 ')", -123.456);
+ CHECK_XPATH_NUMBER(c, "number(' -123.')", -123);
+ CHECK_XPATH_NUMBER(c, "number('123.')", 123);
+ CHECK_XPATH_NUMBER(c, "number('.56')", 0.56);
+ CHECK_XPATH_NUMBER_NAN(c, "number('foobar')");
+ CHECK_XPATH_NUMBER_NAN(c, "number('f1')");
+ CHECK_XPATH_NUMBER_NAN(c, "number('1f')");
+ CHECK_XPATH_NUMBER_NAN(c, "number('1.f')");
+ CHECK_XPATH_NUMBER_NAN(c, "number('1.0f')");
+
+ // number with 1 bool argument
+ CHECK_XPATH_NUMBER(c, "number(true())", 1);
+ CHECK_XPATH_NUMBER(c, "number(false())", 0);
+
+ // number with 1 node set argument
+ CHECK_XPATH_NUMBER(n, "number(.)", 123);
+
+ // number with 1 number argument
+ CHECK_XPATH_NUMBER(c, "number(1)", 1);
+
+ // number with 2 arguments
+ CHECK_XPATH_FAIL("number(1, 2)");
+}
+
+TEST_XML(xpath_number_sum, "123789")
+{
+ xml_node c;
+ xml_node n = doc.child("node");
+
+ // sum with 0 arguments
+ CHECK_XPATH_FAIL("sum()");
+
+ // sum with 1 argument
+ CHECK_XPATH_NUMBER(c, "sum(.)", 0);
+ CHECK_XPATH_NUMBER(n, "sum(.)", 123789); // 123 .. 789
+
+ CHECK_XPATH_NUMBER(n, "sum(./descendant-or-self::node())", 125490); // node + 123 + child + 789 = 123789 + 123 + 789 + 789 = 125490
+ CHECK_XPATH_NUMBER_NAN(doc.last_child(), "sum(.)");
+
+ // sum with 2 arguments
+ CHECK_XPATH_FAIL("sum(1, 2)");
+
+ // sum with 1 non-node-set argument
+ CHECK_XPATH_FAIL("sum(1)");
+}
+
+TEST(xpath_number_floor)
+{
+ xml_node c;
+
+ // floor with 0 arguments
+ CHECK_XPATH_FAIL("floor()");
+
+ // floor with 1 argument
+ CHECK_XPATH_NUMBER(c, "floor(1.2)", 1);
+ CHECK_XPATH_NUMBER(c, "floor(1)", 1);
+ CHECK_XPATH_NUMBER(c, "floor(-1.2)", -2);
+ CHECK_XPATH_NUMBER_NAN(c, "floor(string('nan'))");
+ CHECK_XPATH_STRING(c, "string(floor(1 div 0))", "Infinity");
+ CHECK_XPATH_STRING(c, "string(floor(-1 div 0))", "-Infinity");
+
+ // floor with 2 arguments
+ CHECK_XPATH_FAIL("floor(1, 2)");
+}
+
+TEST(xpath_number_ceiling)
+{
+ xml_node c;
+
+ // ceiling with 0 arguments
+ CHECK_XPATH_FAIL("ceiling()");
+
+ // ceiling with 1 argument
+ CHECK_XPATH_NUMBER(c, "ceiling(1.2)", 2);
+ CHECK_XPATH_NUMBER(c, "ceiling(1)", 1);
+ CHECK_XPATH_NUMBER(c, "ceiling(-1.2)", -1);
+ CHECK_XPATH_NUMBER_NAN(c, "ceiling(string('nan'))");
+ CHECK_XPATH_STRING(c, "string(ceiling(1 div 0))", "Infinity");
+ CHECK_XPATH_STRING(c, "string(ceiling(-1 div 0))", "-Infinity");
+
+ // ceiling with 2 arguments
+ CHECK_XPATH_FAIL("ceiling(1, 2)");
+}
+
+TEST(xpath_number_round)
+{
+ xml_node c;
+
+ // round with 0 arguments
+ CHECK_XPATH_FAIL("round()");
+
+ // round with 1 argument
+ CHECK_XPATH_NUMBER(c, "round(1.2)", 1);
+ CHECK_XPATH_NUMBER(c, "round(1.5)", 2);
+ CHECK_XPATH_NUMBER(c, "round(1.8)", 2);
+ CHECK_XPATH_NUMBER(c, "round(1)", 1);
+ CHECK_XPATH_NUMBER(c, "round(-1.2)", -1);
+ CHECK_XPATH_NUMBER(c, "round(-1.5)", -1);
+ CHECK_XPATH_NUMBER(c, "round(-1.6)", -2);
+ CHECK_XPATH_NUMBER_NAN(c, "round(string('nan'))");
+ CHECK_XPATH_STRING(c, "string(round(1 div 0))", "Infinity");
+ CHECK_XPATH_STRING(c, "string(round(-1 div 0))", "-Infinity");
+
+ // round with 2 arguments
+ CHECK_XPATH_FAIL("round(1, 2)");
+
+ // round with negative zero results
+ // $$ CHECK_XPATH_NUMBER(c, "round(-0.3)", -0)
+ // $$ CHECK_XPATH_NUMBER(c, "round(-0)", -0)
+}
+
+TEST_XML(xpath_boolean_boolean, "")
+{
+ xml_node c;
+
+ // boolean with 0 arguments
+ CHECK_XPATH_FAIL("boolean()");
+
+ // boolean with 1 number argument
+ CHECK_XPATH_BOOLEAN(c, "boolean(0)", false);
+ CHECK_XPATH_BOOLEAN(c, "boolean(1)", true);
+ CHECK_XPATH_BOOLEAN(c, "boolean(-1)", true);
+ CHECK_XPATH_BOOLEAN(c, "boolean(0.1)", true);
+ CHECK_XPATH_BOOLEAN(c, "boolean(number('nan'))", false);
+
+ // boolean with 1 string argument
+ CHECK_XPATH_BOOLEAN(c, "boolean('x')", true);
+ CHECK_XPATH_BOOLEAN(c, "boolean('')", false);
+
+ // boolean with 1 node set argument
+ CHECK_XPATH_BOOLEAN(c, "boolean(.)", false);
+ CHECK_XPATH_BOOLEAN(doc, "boolean(.)", true);
+ CHECK_XPATH_BOOLEAN(doc, "boolean(foo)", false);
+
+ // boolean with 2 arguments
+ CHECK_XPATH_FAIL("boolean(1, 2)");
+}
+
+TEST(xpath_boolean_not)
+{
+ xml_node c;
+
+ // not with 0 arguments
+ CHECK_XPATH_FAIL("not()");
+
+ // not with 1 argument
+ CHECK_XPATH_BOOLEAN(c, "not(true())", false);
+ CHECK_XPATH_BOOLEAN(c, "not(false())", true);
+
+ // boolean with 2 arguments
+ CHECK_XPATH_FAIL("not(1, 2)");
+}
+
+TEST(xpath_boolean_true)
+{
+ xml_node c;
+
+ // true with 0 arguments
+ CHECK_XPATH_BOOLEAN(c, "true()", true);
+
+ // true with 1 argument
+ CHECK_XPATH_FAIL("true(1)");
+}
+
+TEST(xpath_boolean_false)
+{
+ xml_node c;
+
+ // false with 0 arguments
+ CHECK_XPATH_BOOLEAN(c, "false()", false);
+
+ // false with 1 argument
+ CHECK_XPATH_FAIL("false(1)");
+}
+
+// $$$: string value of 123789100 should be 123789100 (?)
+// $$$: type -> node set conversion seems to be guarded via assert instead of exception