sequence.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337
  1. /*=============================================================================
  2. Copyright (c) 2001-2011 Joel de Guzman
  3. Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  5. =============================================================================*/
  6. #include <boost/detail/lightweight_test.hpp>
  7. #include <boost/spirit/include/qi_operator.hpp>
  8. #include <boost/spirit/include/qi_char.hpp>
  9. #include <boost/spirit/include/qi_string.hpp>
  10. #include <boost/spirit/include/qi_numeric.hpp>
  11. #include <boost/spirit/include/qi_directive.hpp>
  12. #include <boost/spirit/include/qi_action.hpp>
  13. #include <boost/spirit/include/qi_nonterminal.hpp>
  14. #include <boost/spirit/include/support_argument.hpp>
  15. #include <boost/spirit/include/phoenix_core.hpp>
  16. #include <boost/spirit/include/phoenix_operator.hpp>
  17. #include <string>
  18. #include <iostream>
  19. #include "test.hpp"
  20. int
  21. main()
  22. {
  23. using namespace boost::spirit::ascii;
  24. using boost::spirit::qi::lit;
  25. using boost::spirit::qi::unused;
  26. using boost::spirit::qi::int_;
  27. using boost::spirit::qi::double_;
  28. using boost::spirit::qi::what;
  29. using boost::spirit::qi::rule;
  30. using boost::spirit::qi::_1;
  31. using boost::spirit::qi::_2;
  32. using boost::fusion::vector;
  33. using boost::fusion::at_c;
  34. using spirit_test::test;
  35. using spirit_test::test_attr;
  36. using spirit_test::print_info;
  37. {
  38. BOOST_TEST((test("aa", char_ >> char_)));
  39. BOOST_TEST((test("aaa", char_ >> char_ >> char_('a'))));
  40. BOOST_TEST((test("xi", char_('x') >> char_('i'))));
  41. BOOST_TEST((!test("xi", char_('x') >> char_('o'))));
  42. BOOST_TEST((test("xin", char_('x') >> char_('i') >> char_('n'))));
  43. }
  44. {
  45. BOOST_TEST((test(" a a", char_ >> char_, space)));
  46. BOOST_TEST((test(" x i", char_('x') >> char_('i'), space)));
  47. BOOST_TEST((!test(" x i", char_('x') >> char_('o'), space)));
  48. }
  49. {
  50. BOOST_TEST((test(" Hello, World", lit("Hello") >> ',' >> "World", space)));
  51. }
  52. {
  53. vector<char, char> attr;
  54. BOOST_TEST((test_attr("abcdefg", char_ >> char_ >> "cdefg", attr)));
  55. BOOST_TEST((at_c<0>(attr) == 'a'));
  56. BOOST_TEST((at_c<1>(attr) == 'b'));
  57. }
  58. {
  59. vector<char, char, char> attr;
  60. BOOST_TEST((test_attr(" a\n b\n c", char_ >> char_ >> char_, attr, space)));
  61. BOOST_TEST((at_c<0>(attr) == 'a'));
  62. BOOST_TEST((at_c<1>(attr) == 'b'));
  63. BOOST_TEST((at_c<2>(attr) == 'c'));
  64. }
  65. {
  66. // unused_type means we don't care about the attribute
  67. vector<char, char> attr;
  68. BOOST_TEST((test_attr("abc", char_ >> 'b' >> char_, attr)));
  69. BOOST_TEST((at_c<0>(attr) == 'a'));
  70. BOOST_TEST((at_c<1>(attr) == 'c'));
  71. }
  72. {
  73. // unused_type means we don't care about the attribute, even at the end
  74. vector<char, char> attr;
  75. BOOST_TEST((test_attr("acb", char_ >> char_ >> 'b', attr)));
  76. BOOST_TEST((at_c<0>(attr) == 'a'));
  77. BOOST_TEST((at_c<1>(attr) == 'c'));
  78. }
  79. {
  80. // "hello" has an unused_type. unused attributes are not part of the sequence
  81. vector<char, char> attr;
  82. BOOST_TEST((test_attr("a hello c", char_ >> "hello" >> char_, attr, space)));
  83. BOOST_TEST((at_c<0>(attr) == 'a'));
  84. BOOST_TEST((at_c<1>(attr) == 'c'));
  85. }
  86. {
  87. // a single element
  88. char attr;
  89. BOOST_TEST((test_attr("ab", char_ >> 'b', attr)));
  90. BOOST_TEST((attr == 'a'));
  91. }
  92. {
  93. // a single element fusion sequence
  94. vector<char> attr;
  95. BOOST_TEST((test_attr("ab", char_ >> 'b', attr)));
  96. BOOST_TEST((at_c<0>(attr) == 'a'));
  97. }
  98. {
  99. // make sure single element tuples get passed through if the rhs
  100. // has a single element tuple as its attribute
  101. vector<double, int> fv;
  102. rule<char const*, vector<double, int>()> r;
  103. r %= double_ >> ',' >> int_;
  104. BOOST_TEST((test_attr("test:2.0,1", "test:" >> r, fv) &&
  105. fv == vector<double, int>(2.0, 1)));
  106. }
  107. {
  108. // unused means we don't care about the attribute
  109. BOOST_TEST((test_attr("abc", char_ >> 'b' >> char_, unused)));
  110. }
  111. {
  112. BOOST_TEST((test("aA", no_case[char_('a') >> 'a'])));
  113. BOOST_TEST((test("BEGIN END", no_case[lit("begin") >> "end"], space)));
  114. BOOST_TEST((!test("BEGIN END", no_case[lit("begin") >> "nend"], space)));
  115. }
  116. {
  117. #ifdef SPIRIT_NO_COMPILE_CHECK
  118. char_ >> char_ = char_ >> char_; // disallow this!
  119. #endif
  120. }
  121. { // alternative forms of attributes. Allow sequences to take in
  122. // stl containers.
  123. std::vector<char> v;
  124. BOOST_TEST(test_attr("abc", char_ >> char_ >> char_, v));
  125. BOOST_TEST(v.size() == 3);
  126. BOOST_TEST(v[0] == 'a');
  127. BOOST_TEST(v[1] == 'b');
  128. BOOST_TEST(v[2] == 'c');
  129. }
  130. { // alternative forms of attributes. Allow sequences to take in
  131. // stl containers.
  132. std::vector<char> v;
  133. BOOST_TEST(test_attr("a,b,c", char_ >> *(',' >> char_), v));
  134. BOOST_TEST(v.size() == 3);
  135. BOOST_TEST(v[0] == 'a');
  136. BOOST_TEST(v[1] == 'b');
  137. BOOST_TEST(v[2] == 'c');
  138. }
  139. { // alternative forms of attributes. Allow sequences to take in
  140. // stl containers.
  141. std::vector<char> v;
  142. BOOST_TEST(test_attr("abc", char_ >> *char_, v));
  143. BOOST_TEST(v.size() == 3);
  144. BOOST_TEST(v[0] == 'a');
  145. BOOST_TEST(v[1] == 'b');
  146. BOOST_TEST(v[2] == 'c');
  147. }
  148. { // alternative forms of attributes. Allow sequences to take in
  149. // stl containers.
  150. using boost::spirit::qi::hold;
  151. std::vector<char> v;
  152. BOOST_TEST(test_attr("abc", char_ >> *(char_ >> char_), v));
  153. BOOST_TEST(v.size() == 3);
  154. BOOST_TEST(v[0] == 'a');
  155. BOOST_TEST(v[1] == 'b');
  156. BOOST_TEST(v[2] == 'c');
  157. v.clear();
  158. BOOST_TEST(!test_attr("abcd", char_ >> *(char_ >> char_), v));
  159. v.clear();
  160. BOOST_TEST(test_attr("abcdef", char_ >> *hold[char_ >> char_] >> char_, v));
  161. BOOST_TEST(v.size() == 6);
  162. BOOST_TEST(v[0] == 'a');
  163. BOOST_TEST(v[1] == 'b');
  164. BOOST_TEST(v[2] == 'c');
  165. BOOST_TEST(v[3] == 'd');
  166. BOOST_TEST(v[4] == 'e');
  167. BOOST_TEST(v[5] == 'f');
  168. v.clear();
  169. BOOST_TEST(test_attr("abc", char_ >> +(char_ >> char_), v));
  170. BOOST_TEST(v.size() == 3);
  171. BOOST_TEST(v[0] == 'a');
  172. BOOST_TEST(v[1] == 'b');
  173. BOOST_TEST(v[2] == 'c');
  174. }
  175. { // alternative forms of attributes. Allow sequences to take in
  176. // stl containers.
  177. std::vector<char> v;
  178. BOOST_TEST(test_attr("abc", char_ >> -(+char_), v));
  179. BOOST_TEST(v.size() == 3);
  180. BOOST_TEST(v[0] == 'a');
  181. BOOST_TEST(v[1] == 'b');
  182. BOOST_TEST(v[2] == 'c');
  183. }
  184. { // alternative forms of attributes. Allow sequences to take in
  185. // stl containers.
  186. std::string s;
  187. BOOST_TEST(test_attr("foobar", string("foo") >> string("bar"), s));
  188. BOOST_TEST(s == "foobar");
  189. s.clear();
  190. using boost::spirit::qi::hold;
  191. rule<char const*, std::string()> word = +char_("abc");
  192. BOOST_TEST(test_attr("ab.bc.ca", *hold[word >> string(".")] >> word, s));
  193. BOOST_TEST(s == "ab.bc.ca");
  194. }
  195. // alternative forms of attributes. Allow sequences to take in
  196. // stl containers of stl containers.
  197. {
  198. std::vector<std::string> v;
  199. BOOST_TEST(test_attr("abc1,abc2",
  200. *~char_(',') >> *(',' >> *~char_(',')), v));
  201. BOOST_TEST(v.size() == 2 && v[0] == "abc1" && v[1] == "abc2");
  202. }
  203. {
  204. std::vector<std::string> v;
  205. rule<char const*, std::string()> e = *~char_(',');
  206. rule<char const*, std::vector<std::string>()> l = e >> *(',' >> e);
  207. BOOST_TEST(test_attr("abc1,abc2,abc3", l, v));
  208. BOOST_TEST(v.size() == 3);
  209. BOOST_TEST(v[0] == "abc1");
  210. BOOST_TEST(v[1] == "abc2");
  211. BOOST_TEST(v[2] == "abc3");
  212. }
  213. // do the same with a plain string object
  214. {
  215. std::string s;
  216. BOOST_TEST(test_attr("abc1,abc2",
  217. *~char_(',') >> *(',' >> *~char_(',')), s));
  218. BOOST_TEST(s == "abc1abc2");
  219. }
  220. {
  221. std::string s;
  222. rule<char const*, std::string()> e = *~char_(',');
  223. rule<char const*, std::string()> l = e >> *(',' >> e);
  224. BOOST_TEST(test_attr("abc1,abc2,abc3", l, s));
  225. BOOST_TEST(s == "abc1abc2abc3");
  226. }
  227. {
  228. std::vector<char> v;
  229. BOOST_TEST(test_attr("ab", char_ >> -char_, v));
  230. BOOST_TEST(v.size() == 2 && v[0] == 'a' && v[1] == 'b');
  231. v.clear();
  232. BOOST_TEST(test_attr("a", char_ >> -char_, v));
  233. BOOST_TEST(v.size() == 1 && v[0] == 'a');
  234. v.clear();
  235. BOOST_TEST(test_attr("a", char_, v));
  236. BOOST_TEST(v.size() == 1 && v[0] == 'a');
  237. }
  238. {
  239. std::vector<boost::optional<char> > v;
  240. BOOST_TEST(test_attr("ab", char_ >> -char_, v));
  241. BOOST_TEST(v.size() == 2 && v[0] == 'a' && v[1] == 'b');
  242. v.clear();
  243. BOOST_TEST(test_attr("a", char_ >> -char_, v));
  244. BOOST_TEST(v.size() == 2 && v[0] == 'a' && !v[1]);
  245. v.clear();
  246. BOOST_TEST(test_attr("a", char_, v));
  247. BOOST_TEST(v.size() == 1 && v[0] == 'a');
  248. }
  249. { // test action
  250. using boost::phoenix::ref;
  251. char c = 0;
  252. int n = 0;
  253. BOOST_TEST(test("x123\"a string\"", (char_ >> int_ >> "\"a string\"")
  254. [ref(c) = _1, ref(n) = _2]));
  255. BOOST_TEST(c == 'x');
  256. BOOST_TEST(n == 123);
  257. }
  258. { // test action
  259. using boost::phoenix::ref;
  260. char c = 0;
  261. int n = 0;
  262. BOOST_TEST(test("x 123 \"a string\"", (char_ >> int_ >> "\"a string\"")
  263. [ref(c) = _1, ref(n) = _2], space));
  264. BOOST_TEST(c == 'x');
  265. BOOST_TEST(n == 123);
  266. }
  267. { // testing "what"
  268. print_info(what(alpha | char_('x') >> lit("hello") >> int_));
  269. }
  270. // { // compile check only
  271. // using boost::spirit::qi::rule;
  272. // typedef boost::fusion::vector<int, double> tuple_type;
  273. // typedef std::vector<boost::fusion::vector<int, double> > attr_type;
  274. //
  275. // rule<char const*, tuple_type()> r = int_ >> ',' >> double_;
  276. // rule<char const*, attr_type()> r2 = r >> *(',' >> r);
  277. // //~ rule<char const*, attr_type()> r2 = r % ',';
  278. // }
  279. return boost::report_errors();
  280. }