795 lines
30 KiB
C++
795 lines
30 KiB
C++
#ifndef PUGIXML_NO_XPATH
|
|
|
|
#include "common.hpp"
|
|
|
|
TEST_XML(xpath_number_number, "<node>123</node>")
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node")).first_child();
|
|
|
|
// number with 0 arguments
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number()"));
|
|
CHECK_XPATH_NUMBER(n, STR("number()"), 123);
|
|
|
|
// number with 1 string argument
|
|
CHECK_XPATH_NUMBER(c, STR("number(' -123.456 ')"), -123.456);
|
|
CHECK_XPATH_NUMBER(c, STR("number(' -123.')"), -123);
|
|
CHECK_XPATH_NUMBER(c, STR("number('123.')"), 123);
|
|
CHECK_XPATH_NUMBER(c, STR("number('.56')"), 0.56);
|
|
CHECK_XPATH_NUMBER(c, STR("number('123 ')"), 123);
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number('foobar')"));
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number('f1')"));
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number('1f')"));
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number('1.f')"));
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number('1.0f')"));
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number('123 f')"));
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number('')"));
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("number('.')"));
|
|
|
|
// number with 1 bool argument
|
|
CHECK_XPATH_NUMBER(c, STR("number(true())"), 1);
|
|
CHECK_XPATH_NUMBER(c, STR("number(false())"), 0);
|
|
|
|
// number with 1 node set argument
|
|
CHECK_XPATH_NUMBER(n, STR("number(.)"), 123);
|
|
|
|
// number with 1 number argument
|
|
CHECK_XPATH_NUMBER(c, STR("number(1)"), 1);
|
|
|
|
// number with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("number(1, 2)"));
|
|
}
|
|
|
|
TEST_XML(xpath_number_sum, "<node>123<child>789</child></node><node/>")
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// sum with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("sum()"));
|
|
|
|
// sum with 1 argument
|
|
CHECK_XPATH_NUMBER(c, STR("sum(.)"), 0);
|
|
CHECK_XPATH_NUMBER(n, STR("sum(.)"), 123789); // 123 .. 789
|
|
|
|
CHECK_XPATH_NUMBER(n, STR("sum(./descendant-or-self::node())"), 125490); // node + 123 + child + 789 = 123789 + 123 + 789 + 789 = 125490
|
|
CHECK_XPATH_NUMBER(n, STR("sum(.//node())"), 1701); // 123 + child + 789 = 123 + 789 + 789
|
|
CHECK_XPATH_NUMBER_NAN(doc.last_child(), STR("sum(.)"));
|
|
|
|
// sum with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("sum(1, 2)"));
|
|
|
|
// sum with 1 non-node-set argument
|
|
CHECK_XPATH_FAIL(STR("sum(1)"));
|
|
}
|
|
|
|
TEST(xpath_number_floor)
|
|
{
|
|
xml_node c;
|
|
|
|
// floor with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("floor()"));
|
|
|
|
// floor with 1 argument
|
|
CHECK_XPATH_NUMBER(c, STR("floor(0)"), 0);
|
|
CHECK_XPATH_NUMBER(c, STR("floor(1.2)"), 1);
|
|
CHECK_XPATH_NUMBER(c, STR("floor(1)"), 1);
|
|
CHECK_XPATH_NUMBER(c, STR("floor(-1.2)"), -2);
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("floor(string('nan'))"));
|
|
CHECK_XPATH_STRING(c, STR("string(floor(1 div 0))"), STR("Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(floor(-1 div 0))"), STR("-Infinity"));
|
|
|
|
// floor with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("floor(1, 2)"));
|
|
|
|
// floor with argument 0 should return 0
|
|
CHECK_XPATH_STRING(c, STR("string(1 div floor(0))"), STR("Infinity"));
|
|
|
|
// floor with argument -0 should return -0
|
|
#if !(defined(__APPLE__) && defined(__MACH__)) // MacOS X gcc 4.0.1 implements floor incorrectly (floor never returns -0)
|
|
CHECK_XPATH_STRING(c, STR("string(1 div floor(-0))"), STR("-Infinity"));
|
|
#endif
|
|
}
|
|
|
|
TEST(xpath_number_ceiling)
|
|
{
|
|
xml_node c;
|
|
|
|
// ceiling with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("ceiling()"));
|
|
|
|
// ceiling with 1 argument
|
|
CHECK_XPATH_NUMBER(c, STR("ceiling(0)"), 0);
|
|
CHECK_XPATH_NUMBER(c, STR("ceiling(1.2)"), 2);
|
|
CHECK_XPATH_NUMBER(c, STR("ceiling(1)"), 1);
|
|
CHECK_XPATH_NUMBER(c, STR("ceiling(-1.2)"), -1);
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("ceiling(string('nan'))"));
|
|
CHECK_XPATH_STRING(c, STR("string(ceiling(1 div 0))"), STR("Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(ceiling(-1 div 0))"), STR("-Infinity"));
|
|
|
|
// ceiling with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("ceiling(1, 2)"));
|
|
|
|
// ceiling with argument 0 should return 0
|
|
CHECK_XPATH_STRING(c, STR("string(1 div ceiling(0))"), STR("Infinity"));
|
|
|
|
// ceiling with argument in range (-1, -0] should result in minus zero
|
|
#if !(defined(__APPLE__) && defined(__MACH__)) && !defined(__CLR_VER) // MacOS X gcc 4.0.1 and x64 CLR implement ceil incorrectly (ceil never returns -0)
|
|
CHECK_XPATH_STRING(c, STR("string(1 div ceiling(-0))"), STR("-Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(1 div ceiling(-0.1))"), STR("-Infinity"));
|
|
#endif
|
|
}
|
|
|
|
TEST(xpath_number_round)
|
|
{
|
|
xml_node c;
|
|
|
|
// round with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("round()"));
|
|
|
|
// round with 1 argument
|
|
CHECK_XPATH_NUMBER(c, STR("round(1.2)"), 1);
|
|
CHECK_XPATH_NUMBER(c, STR("round(1.5)"), 2);
|
|
CHECK_XPATH_NUMBER(c, STR("round(1.8)"), 2);
|
|
CHECK_XPATH_NUMBER(c, STR("round(1)"), 1);
|
|
CHECK_XPATH_NUMBER(c, STR("round(-1.2)"), -1);
|
|
CHECK_XPATH_NUMBER(c, STR("round(-1.5)"), -1);
|
|
CHECK_XPATH_NUMBER(c, STR("round(-1.6)"), -2);
|
|
CHECK_XPATH_NUMBER_NAN(c, STR("round(string('nan'))"));
|
|
CHECK_XPATH_STRING(c, STR("string(round(1 div 0))"), STR("Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(round(-1 div 0))"), STR("-Infinity"));
|
|
|
|
// round with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("round(1, 2)"));
|
|
|
|
// round with argument in range [-0.5, -0] should result in minus zero
|
|
CHECK_XPATH_STRING(c, STR("string(1 div round(0))"), STR("Infinity"));
|
|
|
|
#if !(defined(__APPLE__) && defined(__MACH__)) && !defined(__CLR_VER) // MacOS X gcc 4.0.1 and x64 CLR implement ceil incorrectly (ceil never returns -0)
|
|
CHECK_XPATH_STRING(c, STR("string(1 div round(-0.5))"), STR("-Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(1 div round(-0))"), STR("-Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(1 div round(-0.1))"), STR("-Infinity"));
|
|
#endif
|
|
}
|
|
|
|
TEST_XML(xpath_boolean_boolean, "<node />")
|
|
{
|
|
xml_node c;
|
|
|
|
// boolean with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("boolean()"));
|
|
|
|
// boolean with 1 number argument
|
|
CHECK_XPATH_BOOLEAN(c, STR("boolean(0)"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("boolean(1)"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("boolean(-1)"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("boolean(0.1)"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("boolean(number('nan'))"), false);
|
|
|
|
// boolean with 1 string argument
|
|
CHECK_XPATH_BOOLEAN(c, STR("boolean('x')"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("boolean('')"), false);
|
|
|
|
// boolean with 1 node set argument
|
|
CHECK_XPATH_BOOLEAN(c, STR("boolean(.)"), false);
|
|
CHECK_XPATH_BOOLEAN(doc, STR("boolean(.)"), true);
|
|
CHECK_XPATH_BOOLEAN(doc, STR("boolean(foo)"), false);
|
|
|
|
// boolean with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("boolean(1, 2)"));
|
|
}
|
|
|
|
TEST(xpath_boolean_not)
|
|
{
|
|
xml_node c;
|
|
|
|
// not with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("not()"));
|
|
|
|
// not with 1 argument
|
|
CHECK_XPATH_BOOLEAN(c, STR("not(true())"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("not(false())"), true);
|
|
|
|
// boolean with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("not(1, 2)"));
|
|
}
|
|
|
|
TEST(xpath_boolean_true)
|
|
{
|
|
xml_node c;
|
|
|
|
// true with 0 arguments
|
|
CHECK_XPATH_BOOLEAN(c, STR("true()"), true);
|
|
|
|
// true with 1 argument
|
|
CHECK_XPATH_FAIL(STR("true(1)"));
|
|
}
|
|
|
|
TEST(xpath_boolean_false)
|
|
{
|
|
xml_node c;
|
|
|
|
// false with 0 arguments
|
|
CHECK_XPATH_BOOLEAN(c, STR("false()"), false);
|
|
|
|
// false with 1 argument
|
|
CHECK_XPATH_FAIL(STR("false(1)"));
|
|
}
|
|
|
|
TEST_XML(xpath_boolean_lang, "<node xml:lang='en'><child xml:lang='zh-UK'><subchild attr=''/></child></node><foo><bar/></foo>")
|
|
{
|
|
xml_node c;
|
|
|
|
// lang with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("lang()"));
|
|
|
|
// lang with 1 argument, no language
|
|
CHECK_XPATH_BOOLEAN(c, STR("lang('en')"), false);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("foo")), STR("lang('en')"), false);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("foo")), STR("lang('')"), false);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("foo")).child(STR("bar")), STR("lang('en')"), false);
|
|
|
|
// lang with 1 argument, same language/prefix
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")), STR("lang('en')"), true);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")).child(STR("child")), STR("lang('zh-uk')"), true);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")).child(STR("child")), STR("lang('zh')"), true);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")).child(STR("child")).child(STR("subchild")), STR("lang('zh')"), true);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")).child(STR("child")).child(STR("subchild")), STR("lang('ZH')"), true);
|
|
|
|
// lang with 1 argument, different language/prefix
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")), STR("lang('')"), false);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")), STR("lang('e')"), false);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")).child(STR("child")), STR("lang('en')"), false);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")).child(STR("child")), STR("lang('zh-gb')"), false);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")).child(STR("child")), STR("lang('r')"), false);
|
|
CHECK_XPATH_BOOLEAN(doc.child(STR("node")).child(STR("child")).child(STR("subchild")), STR("lang('en')"), false);
|
|
|
|
// lang with 1 attribute argument
|
|
CHECK_XPATH_NODESET(doc, STR("//@*[lang('en')]"));
|
|
|
|
// lang with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("lang(1, 2)"));
|
|
}
|
|
|
|
TEST_XML(xpath_string_string, "<node>123<child id='1'>789</child><child><subchild><![CDATA[200]]></subchild></child>100</node>")
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// string with 0 arguments
|
|
CHECK_XPATH_STRING(c, STR("string()"), STR(""));
|
|
CHECK_XPATH_STRING(n.child(STR("child")), STR("string()"), STR("789"));
|
|
|
|
// string with 1 node-set argument
|
|
CHECK_XPATH_STRING(n, STR("string(child)"), STR("789"));
|
|
CHECK_XPATH_STRING(n, STR("string(child/@id)"), STR("1"));
|
|
CHECK_XPATH_STRING(n, STR("string(.)"), STR("123789200100"));
|
|
|
|
// string with 1 number argument
|
|
CHECK_XPATH_STRING(c, STR("string(0 div 0)"), STR("NaN"));
|
|
CHECK_XPATH_STRING(c, STR("string(0)"), STR("0"));
|
|
CHECK_XPATH_STRING(c, STR("string(-0)"), STR("0"));
|
|
CHECK_XPATH_STRING(c, STR("string(1 div 0)"), STR("Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(-1 div -0)"), STR("Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(-1 div 0)"), STR("-Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(1 div -0)"), STR("-Infinity"));
|
|
CHECK_XPATH_STRING(c, STR("string(1234567)"), STR("1234567"));
|
|
CHECK_XPATH_STRING(c, STR("string(-1234567)"), STR("-1234567"));
|
|
CHECK_XPATH_STRING(c, STR("string(1234.5678)"), STR("1234.5678"));
|
|
CHECK_XPATH_STRING(c, STR("string(-1234.5678)"), STR("-1234.5678"));
|
|
CHECK_XPATH_STRING(c, STR("string(0.5678)"), STR("0.5678"));
|
|
CHECK_XPATH_STRING(c, STR("string(-0.5678)"), STR("-0.5678"));
|
|
CHECK_XPATH_STRING(c, STR("string(0.0)"), STR("0"));
|
|
CHECK_XPATH_STRING(c, STR("string(-0.0)"), STR("0"));
|
|
|
|
// string with 1 boolean argument
|
|
CHECK_XPATH_STRING(c, STR("string(true())"), STR("true"));
|
|
CHECK_XPATH_STRING(c, STR("string(false())"), STR("false"));
|
|
|
|
// string with 1 string argument
|
|
CHECK_XPATH_STRING(c, STR("string('abc')"), STR("abc"));
|
|
|
|
// string with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("string(1, 2)"));
|
|
}
|
|
|
|
TEST(xpath_string_concat)
|
|
{
|
|
xml_node c;
|
|
|
|
// concat with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("concat()"));
|
|
|
|
// concat with 1 argument
|
|
CHECK_XPATH_FAIL(STR("concat('')"));
|
|
|
|
// concat with exactly 2 arguments
|
|
CHECK_XPATH_STRING(c, STR("concat('prev','next')"), STR("prevnext"));
|
|
CHECK_XPATH_STRING(c, STR("concat('','next')"), STR("next"));
|
|
CHECK_XPATH_STRING(c, STR("concat('prev','')"), STR("prev"));
|
|
|
|
// concat with 3 or more arguments
|
|
CHECK_XPATH_STRING(c, STR("concat('a', 'b', 'c')"), STR("abc"));
|
|
CHECK_XPATH_STRING(c, STR("concat('a', 'b', 'c', 'd')"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("concat('a', 'b', 'c', 'd', 'e')"), STR("abcde"));
|
|
CHECK_XPATH_STRING(c, STR("concat('a', 'b', 'c', 'd', 'e', 'f')"), STR("abcdef"));
|
|
CHECK_XPATH_STRING(c, STR("concat('a', 'b', 'c', 'd', 'e', 'f', 'g')"), STR("abcdefg"));
|
|
CHECK_XPATH_STRING(c, STR("concat(1, 2, 3, 4, 5, 6, 7, 8)"), STR("12345678"));
|
|
}
|
|
|
|
TEST(xpath_string_starts_with)
|
|
{
|
|
xml_node c;
|
|
|
|
// starts-with with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("starts-with()"));
|
|
|
|
// starts-with with 1 argument
|
|
CHECK_XPATH_FAIL(STR("starts-with('a')"));
|
|
|
|
// starts-with with 2 arguments
|
|
CHECK_XPATH_BOOLEAN(c, STR("starts-with('abc', '')"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("starts-with('abc', 'a')"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("starts-with('abc', 'abc')"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("starts-with('abc', 'abcd')"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("starts-with('bc', 'c')"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("starts-with('', 'c')"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("starts-with('', '')"), true);
|
|
|
|
// starts-with with 3 arguments
|
|
CHECK_XPATH_FAIL(STR("starts-with('a', 'b', 'c')"));
|
|
}
|
|
|
|
TEST(xpath_string_contains)
|
|
{
|
|
xml_node c;
|
|
|
|
// contains with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("contains()"));
|
|
|
|
// contains with 1 argument
|
|
CHECK_XPATH_FAIL(STR("contains('a')"));
|
|
|
|
// contains with 2 arguments
|
|
CHECK_XPATH_BOOLEAN(c, STR("contains('abc', '')"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("contains('abc', 'a')"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("contains('abc', 'abc')"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("contains('abcd', 'bc')"), true);
|
|
CHECK_XPATH_BOOLEAN(c, STR("contains('abc', 'abcd')"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("contains('b', 'bc')"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("contains('', 'c')"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("contains('', '')"), true);
|
|
|
|
// contains with 3 arguments
|
|
CHECK_XPATH_FAIL(STR("contains('a', 'b', 'c')"));
|
|
}
|
|
|
|
TEST(xpath_string_substring_before)
|
|
{
|
|
xml_node c;
|
|
|
|
// substring-before with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("substring-before()"));
|
|
|
|
// substring-before with 1 argument
|
|
CHECK_XPATH_FAIL(STR("substring-before('a')"));
|
|
|
|
// substring-before with 2 arguments
|
|
CHECK_XPATH_STRING(c, STR("substring-before('abc', 'abc')"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring-before('abc', 'a')"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring-before('abc', 'cd')"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring-before('abc', 'b')"), STR("a"));
|
|
CHECK_XPATH_STRING(c, STR("substring-before('abc', 'c')"), STR("ab"));
|
|
CHECK_XPATH_STRING(c, STR("substring-before('abc', '')"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring-before('', '')"), STR(""));
|
|
|
|
// substring-before with 2 arguments, from W3C standard
|
|
CHECK_XPATH_STRING(c, STR("substring-before(\"1999/04/01\",\"/\")"), STR("1999"));
|
|
|
|
// substring-before with 3 arguments
|
|
CHECK_XPATH_FAIL(STR("substring-before('a', 'b', 'c')"));
|
|
}
|
|
|
|
TEST(xpath_string_substring_after)
|
|
{
|
|
xml_node c;
|
|
|
|
// substring-after with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("substring-after()"));
|
|
|
|
// substring-after with 1 argument
|
|
CHECK_XPATH_FAIL(STR("substring-after('a')"));
|
|
|
|
// substring-after with 2 arguments
|
|
CHECK_XPATH_STRING(c, STR("substring-after('abc', 'abc')"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring-after('abc', 'a')"), STR("bc"));
|
|
CHECK_XPATH_STRING(c, STR("substring-after('abc', 'cd')"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring-after('abc', 'b')"), STR("c"));
|
|
CHECK_XPATH_STRING(c, STR("substring-after('abc', 'c')"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring-after('abc', '')"), STR("abc"));
|
|
CHECK_XPATH_STRING(c, STR("substring-after('', '')"), STR(""));
|
|
|
|
// substring-before with 2 arguments, from W3C standard
|
|
CHECK_XPATH_STRING(c, STR("substring-after(\"1999/04/01\",\"/\")"), STR("04/01"));
|
|
CHECK_XPATH_STRING(c, STR("substring-after(\"1999/04/01\",\"19\")"), STR("99/04/01"));
|
|
|
|
// substring-after with 3 arguments
|
|
CHECK_XPATH_FAIL(STR("substring-after('a', 'b', 'c')"));
|
|
}
|
|
|
|
TEST_XML(xpath_string_substring_after_heap, "<node>foo<child/>bar</node>")
|
|
{
|
|
CHECK_XPATH_STRING(doc, STR("substring-after(node, 'fo')"), STR("obar"));
|
|
CHECK_XPATH_STRING(doc, STR("substring-after(node, 'fooba')"), STR("r"));
|
|
CHECK_XPATH_STRING(doc, STR("substring-after(node, 'foobar')"), STR(""));
|
|
}
|
|
|
|
TEST(xpath_string_substring)
|
|
{
|
|
xml_node c;
|
|
|
|
// substring with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("substring()"));
|
|
|
|
// substring with 1 argument
|
|
CHECK_XPATH_FAIL(STR("substring('')"));
|
|
|
|
// substring with 2 arguments
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 2)"), STR("bcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1)"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1.1)"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1.5)"), STR("bcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1.8)"), STR("bcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 10)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 0)"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100)"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -1 div 0)"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1 div 0)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 0 div 0)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('', 1)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('', 0)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring(substring('internalexternalcorrect substring',9),9)"), STR("correct substring"));
|
|
|
|
// substring with 3 arguments
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 2, 1)"), STR("b"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 2, 2)"), STR("bc"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1, 0)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1, 0.4)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1, 0.5)"), STR("a"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 10, -5)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 0, -1)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100, 100)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100, 101)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100, 102)"), STR("a"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100, 103)"), STR("ab"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100, 104)"), STR("abc"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100, 105)"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100, 106)"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -100, 1 div 0)"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', -1 div 0, 4)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 1 div 0, 0 div 0)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('abcd', 0 div 0, 1)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('', 1, 2)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('', 0, 0)"), STR(""));
|
|
|
|
// substring with 3 arguments, from W3C standard
|
|
CHECK_XPATH_STRING(c, STR("substring('12345', 1.5, 2.6)"), STR("234"));
|
|
CHECK_XPATH_STRING(c, STR("substring('12345', 0, 3)"), STR("12"));
|
|
CHECK_XPATH_STRING(c, STR("substring('12345', 0 div 0, 3)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('12345', 1, 0 div 0)"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("substring('12345', -42, 1 div 0)"), STR("12345"));
|
|
CHECK_XPATH_STRING(c, STR("substring('12345', -1 div 0, 1 div 0)"), STR(""));
|
|
|
|
// substring with 4 arguments
|
|
CHECK_XPATH_FAIL(STR("substring('', 1, 2, 3)"));
|
|
}
|
|
|
|
TEST_XML(xpath_string_substring_heap, "<node>foo<child/>bar</node>")
|
|
{
|
|
CHECK_XPATH_STRING(doc, STR("substring(node, 3)"), STR("obar"));
|
|
CHECK_XPATH_STRING(doc, STR("substring(node, 6)"), STR("r"));
|
|
CHECK_XPATH_STRING(doc, STR("substring(node, 7)"), STR(""));
|
|
}
|
|
|
|
TEST_XML(xpath_string_string_length, "<node>123</node>")
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// string-length with 0 arguments
|
|
CHECK_XPATH_NUMBER(c, STR("string-length()"), 0);
|
|
CHECK_XPATH_NUMBER(n, STR("string-length()"), 3);
|
|
|
|
// string-length with 1 argument
|
|
CHECK_XPATH_NUMBER(c, STR("string-length('')"), 0);
|
|
CHECK_XPATH_NUMBER(c, STR("string-length('a')"), 1);
|
|
CHECK_XPATH_NUMBER(c, STR("string-length('abcdef')"), 6);
|
|
|
|
// string-length with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("string-length(1, 2)"));
|
|
}
|
|
|
|
TEST_XML_FLAGS(xpath_string_normalize_space, "<node> \t\r\rval1 \rval2\r\nval3\nval4\r\r</node>", parse_minimal)
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// normalize-space with 0 arguments
|
|
CHECK_XPATH_STRING(c, STR("normalize-space()"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("normalize-space()"), STR("val1 val2 val3 val4"));
|
|
|
|
// normalize-space with 1 argument
|
|
CHECK_XPATH_STRING(c, STR("normalize-space('')"), STR(""));
|
|
CHECK_XPATH_STRING(c, STR("normalize-space('abcd')"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("normalize-space(' \r\nabcd')"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("normalize-space('abcd \n\r')"), STR("abcd"));
|
|
CHECK_XPATH_STRING(c, STR("normalize-space('ab\r\n\tcd')"), STR("ab cd"));
|
|
CHECK_XPATH_STRING(c, STR("normalize-space('ab cd')"), STR("ab cd"));
|
|
CHECK_XPATH_STRING(c, STR("normalize-space('\07')"), STR("\07"));
|
|
|
|
// normalize-space with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("normalize-space(1, 2)"));
|
|
}
|
|
|
|
TEST(xpath_string_translate)
|
|
{
|
|
xml_node c;
|
|
|
|
// translate with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("translate()"));
|
|
|
|
// translate with 1 argument
|
|
CHECK_XPATH_FAIL(STR("translate('a')"));
|
|
|
|
// translate with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("translate('a', 'b')"));
|
|
|
|
// translate with 3 arguments
|
|
CHECK_XPATH_STRING(c, STR("translate('abc', '', '')"), STR("abc"));
|
|
CHECK_XPATH_STRING(c, STR("translate('abc', '', 'foo')"), STR("abc"));
|
|
CHECK_XPATH_STRING(c, STR("translate('abc', 'ab', 'ba')"), STR("bac"));
|
|
CHECK_XPATH_STRING(c, STR("translate('abc', 'ab', 'f')"), STR("fc"));
|
|
CHECK_XPATH_STRING(c, STR("translate('abc', 'aabb', '1234')"), STR("13c"));
|
|
CHECK_XPATH_STRING(c, STR("translate('', 'abc', 'bac')"), STR(""));
|
|
|
|
// translate with 3 arguments, from W3C standard
|
|
CHECK_XPATH_STRING(c, STR("translate('bar','abc','ABC')"), STR("BAr"));
|
|
CHECK_XPATH_STRING(c, STR("translate('--aaa--','abc-','ABC')"), STR("AAA"));
|
|
|
|
// translate with 4 arguments
|
|
CHECK_XPATH_FAIL(STR("translate('a', 'b', 'c', 'd')"));
|
|
}
|
|
|
|
TEST(xpath_string_translate_table)
|
|
{
|
|
xml_node c;
|
|
|
|
CHECK_XPATH_STRING(c, STR("translate('abcd\xe9 ', 'abc', 'ABC')"), STR("ABCd\xe9 "));
|
|
CHECK_XPATH_STRING(c, STR("translate('abcd\xe9 ', 'abc\xe9', 'ABC!')"), STR("ABCd! "));
|
|
CHECK_XPATH_STRING(c, STR("translate('abcde', concat('abc', 'd'), 'ABCD')"), STR("ABCDe"));
|
|
CHECK_XPATH_STRING(c, STR("translate('abcde', 'abcd', concat('ABC', 'D'))"), STR("ABCDe"));
|
|
}
|
|
|
|
TEST_XML(xpath_nodeset_last, "<node><c1/><c1/><c2/><c3/><c3/><c3/><c3/></node>")
|
|
{
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// last with 0 arguments
|
|
CHECK_XPATH_NUMBER(n, STR("last()"), 1);
|
|
CHECK_XPATH_NODESET(n, STR("c1[last() = 1]"));
|
|
CHECK_XPATH_NODESET(n, STR("c1[last() = 2]")) % 3 % 4; // c1, c1
|
|
CHECK_XPATH_NODESET(n, STR("c2/preceding-sibling::node()[last() = 2]")) % 4 % 3; // c1, c1
|
|
|
|
// last with 1 argument
|
|
CHECK_XPATH_FAIL(STR("last(c)"));
|
|
}
|
|
|
|
TEST_XML(xpath_nodeset_position, "<node><c1/><c1/><c2/><c3/><c3/><c3/><c3/></node>")
|
|
{
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// position with 0 arguments
|
|
CHECK_XPATH_NUMBER(n, STR("position()"), 1);
|
|
CHECK_XPATH_NODESET(n, STR("c1[position() = 0]"));
|
|
CHECK_XPATH_NODESET(n, STR("c1[position() = 1]")) % 3;
|
|
CHECK_XPATH_NODESET(n, STR("c1[position() = 2]")) % 4;
|
|
CHECK_XPATH_NODESET(n, STR("c1[position() = 3]"));
|
|
CHECK_XPATH_NODESET(n, STR("c2/preceding-sibling::node()[position() = 1]")) % 4;
|
|
CHECK_XPATH_NODESET(n, STR("c2/preceding-sibling::node()[position() = 2]")) % 3;
|
|
|
|
// position with 1 argument
|
|
CHECK_XPATH_FAIL(STR("position(c)"));
|
|
}
|
|
|
|
TEST_XML(xpath_nodeset_count, "<node><c1/><c1/><c2/><c3/><c3/><c3/><c3/></node>")
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// count with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("count()"));
|
|
|
|
// count with 1 non-node-set argument
|
|
CHECK_XPATH_FAIL(STR("count(1)"));
|
|
CHECK_XPATH_FAIL(STR("count(true())"));
|
|
CHECK_XPATH_FAIL(STR("count('')"));
|
|
|
|
// count with 1 node-set argument
|
|
CHECK_XPATH_NUMBER(c, STR("count(.)"), 0);
|
|
CHECK_XPATH_NUMBER(n, STR("count(.)"), 1);
|
|
CHECK_XPATH_NUMBER(n, STR("count(c1)"), 2);
|
|
CHECK_XPATH_NUMBER(n, STR("count(c2)"), 1);
|
|
CHECK_XPATH_NUMBER(n, STR("count(c3)"), 4);
|
|
CHECK_XPATH_NUMBER(n, STR("count(c4)"), 0);
|
|
|
|
// count with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("count(x, y)"));
|
|
}
|
|
|
|
TEST_XML(xpath_nodeset_id, "<node id='foo'/>")
|
|
{
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// id with 0 arguments
|
|
CHECK_XPATH_FAIL(STR("id()"));
|
|
|
|
// id with 1 argument - no DTD => no id
|
|
CHECK_XPATH_NODESET(n, STR("id('foo')"));
|
|
|
|
// id with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("id(1, 2)"));
|
|
}
|
|
|
|
TEST_XML_FLAGS(xpath_nodeset_local_name, "<node xmlns:foo='http://foo'><c1>text</c1><c2 xmlns:foo='http://foo2' foo:attr='value'><foo:child/></c2><c3 xmlns='http://def' attr='value'><child/></c3><c4><?target stuff?></c4></node>", parse_default | parse_pi)
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// local-name with 0 arguments
|
|
CHECK_XPATH_STRING(c, STR("local-name()"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("local-name()"), STR("node"));
|
|
|
|
// local-name with 1 non-node-set argument
|
|
CHECK_XPATH_FAIL(STR("local-name(1)"));
|
|
|
|
// local-name with 1 node-set argument
|
|
CHECK_XPATH_STRING(n, STR("local-name(c1)"), STR("c1"));
|
|
CHECK_XPATH_STRING(n, STR("local-name(c2/node())"), STR("child"));
|
|
CHECK_XPATH_STRING(n, STR("local-name(c2/attribute::node())"), STR("attr"));
|
|
CHECK_XPATH_STRING(n, STR("local-name(c1/node())"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("local-name(c4/node())"), STR("target"));
|
|
CHECK_XPATH_STRING(n, STR("local-name(c1/following-sibling::node())"), STR("c2"));
|
|
CHECK_XPATH_STRING(n, STR("local-name(c4/preceding-sibling::node())"), STR("c1"));
|
|
|
|
// local-name with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("local-name(c1, c2)"));
|
|
}
|
|
|
|
TEST_XML_FLAGS(xpath_nodeset_namespace_uri, "<node xmlns:foo='http://foo'><c1>text</c1><c2 xmlns:foo='http://foo2' foo:attr='value'><foo:child/></c2><c3 xmlns='http://def' attr='value'><child/></c3><c4><?target stuff?></c4><c5><foo:child/></c5><c6 bar:attr=''/><c7><node foo:attr=''/></c7></node>", parse_default | parse_pi)
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// namespace-uri with 0 arguments
|
|
CHECK_XPATH_STRING(c, STR("namespace-uri()"), STR(""));
|
|
CHECK_XPATH_STRING(n.child(STR("c2")).child(STR("foo:child")), STR("namespace-uri()"), STR("http://foo2"));
|
|
|
|
// namespace-uri with 1 non-node-set argument
|
|
CHECK_XPATH_FAIL(STR("namespace-uri(1)"));
|
|
|
|
// namespace-uri with 1 node-set argument
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c1)"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c5/child::node())"), STR("http://foo"));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c2/attribute::node())"), STR("http://foo2"));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c2/child::node())"), STR("http://foo2"));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c1/child::node())"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c4/child::node())"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c3)"), STR("http://def"));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c3/@attr)"), STR("")); // the namespace name for an unprefixed attribute name always has no value (Namespaces in XML 1.0)
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c3/child::node())"), STR("http://def"));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c6/@bar:attr)"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("namespace-uri(c7/node/@foo:attr)"), STR("http://foo"));
|
|
|
|
// namespace-uri with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("namespace-uri(c1, c2)"));
|
|
}
|
|
|
|
TEST_XML_FLAGS(xpath_nodeset_name, "<node xmlns:foo='http://foo'><c1>text</c1><c2 xmlns:foo='http://foo2' foo:attr='value'><foo:child/></c2><c3 xmlns='http://def' attr='value'><child/></c3><c4><?target stuff?></c4></node>", parse_default | parse_pi)
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
// name with 0 arguments
|
|
CHECK_XPATH_STRING(c, STR("name()"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("name()"), STR("node"));
|
|
|
|
// name with 1 non-node-set argument
|
|
CHECK_XPATH_FAIL(STR("name(1)"));
|
|
|
|
// name with 1 node-set argument
|
|
CHECK_XPATH_STRING(n, STR("name(c1)"), STR("c1"));
|
|
CHECK_XPATH_STRING(n, STR("name(c2/node())"), STR("foo:child"));
|
|
CHECK_XPATH_STRING(n, STR("name(c2/attribute::node())"), STR("foo:attr"));
|
|
CHECK_XPATH_STRING(n, STR("name(c1/node())"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("name(c4/node())"), STR("target"));
|
|
CHECK_XPATH_STRING(n, STR("name(c1/following-sibling::node())"), STR("c2"));
|
|
CHECK_XPATH_STRING(n, STR("name(c4/preceding-sibling::node())"), STR("c1"));
|
|
|
|
// name with 2 arguments
|
|
CHECK_XPATH_FAIL(STR("name(c1, c2)"));
|
|
}
|
|
|
|
TEST(xpath_function_arguments)
|
|
{
|
|
xml_node c;
|
|
|
|
// conversion to string
|
|
CHECK_XPATH_NUMBER(c, STR("string-length(12)"), 2);
|
|
|
|
// conversion to number
|
|
CHECK_XPATH_NUMBER(c, STR("round('1.2')"), 1);
|
|
CHECK_XPATH_NUMBER(c, STR("round('1.7')"), 2);
|
|
|
|
// conversion to boolean
|
|
CHECK_XPATH_BOOLEAN(c, STR("not('1')"), false);
|
|
CHECK_XPATH_BOOLEAN(c, STR("not('')"), true);
|
|
|
|
// conversion to node set
|
|
CHECK_XPATH_FAIL(STR("sum(1)"));
|
|
|
|
// expression evaluation
|
|
CHECK_XPATH_NUMBER(c, STR("round((2 + 2 * 2) div 4)"), 2);
|
|
|
|
// empty expressions
|
|
CHECK_XPATH_FAIL(STR("round(,)"));
|
|
CHECK_XPATH_FAIL(STR("substring(,)"));
|
|
CHECK_XPATH_FAIL(STR("substring('a',)"));
|
|
CHECK_XPATH_FAIL(STR("substring(,'a')"));
|
|
|
|
// extra commas
|
|
CHECK_XPATH_FAIL(STR("round(,1)"));
|
|
CHECK_XPATH_FAIL(STR("round(1,)"));
|
|
|
|
// lack of commas
|
|
CHECK_XPATH_FAIL(STR("substring(1 2)"));
|
|
|
|
// whitespace after function name
|
|
CHECK_XPATH_BOOLEAN(c, STR("true ()"), true);
|
|
|
|
// too many arguments
|
|
CHECK_XPATH_FAIL(STR("round(1, 2, 3, 4, 5, 6)"));
|
|
}
|
|
|
|
TEST_XML_FLAGS(xpath_string_value, "<node><c1>pcdata</c1><c2><child/></c2><c3 attr='avalue'/><c4><?target pivalue?></c4><c5><!--comment--></c5><c6><![CDATA[cdata]]></c6></node>", parse_default | parse_pi | parse_comments)
|
|
{
|
|
xml_node c;
|
|
xml_node n = doc.child(STR("node"));
|
|
|
|
CHECK_XPATH_STRING(c, STR("string()"), STR(""));
|
|
CHECK_XPATH_STRING(doc, STR("string()"), STR("pcdatacdata"));
|
|
CHECK_XPATH_STRING(n, STR("string()"), STR("pcdatacdata"));
|
|
CHECK_XPATH_STRING(n, STR("string(c1/node())"), STR("pcdata"));
|
|
CHECK_XPATH_STRING(n, STR("string(c2/node())"), STR(""));
|
|
CHECK_XPATH_STRING(n, STR("string(c3/@attr)"), STR("avalue"));
|
|
CHECK_XPATH_STRING(n, STR("string(c4/node())"), STR("pivalue"));
|
|
CHECK_XPATH_STRING(n, STR("string(c5/node())"), STR("comment"));
|
|
CHECK_XPATH_STRING(n, STR("string(c6/node())"), STR("cdata"));
|
|
}
|
|
|
|
TEST(xpath_string_value_empty)
|
|
{
|
|
xml_document doc;
|
|
doc.append_child(node_pcdata).set_value(STR("head"));
|
|
doc.append_child(node_pcdata);
|
|
doc.append_child(node_pcdata).set_value(STR("tail"));
|
|
|
|
CHECK_XPATH_STRING(doc, STR("string()"), STR("headtail"));
|
|
}
|
|
|
|
TEST_XML(xpath_string_concat_translate, "<node>foobar</node>")
|
|
{
|
|
CHECK_XPATH_STRING(doc, STR("concat('a', 'b', 'c', translate(node, 'o', 'a'), 'd')"), STR("abcfaabard"));
|
|
}
|
|
|
|
#endif
|