switch_tests_single.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. /*=============================================================================
  2. Copyright (c) 2003 Hartmut Kaiser
  3. http://spirit.sourceforge.net/
  4. Use, modification and distribution is subject to the Boost Software
  5. License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  6. http://www.boost.org/LICENSE_1_0.txt)
  7. =============================================================================*/
  8. #include <iostream>
  9. #include <boost/detail/lightweight_test.hpp>
  10. #define BOOST_SPIRIT_SWITCH_CASE_LIMIT 6
  11. #define BOOST_SPIRIT_SELECT_LIMIT 6
  12. #define PHOENIX_LIMIT 6
  13. //#define BOOST_SPIRIT_DEBUG
  14. #include <boost/mpl/list.hpp>
  15. #include <boost/mpl/for_each.hpp>
  16. #include <boost/spirit/include/classic_primitives.hpp>
  17. #include <boost/spirit/include/classic_numerics.hpp>
  18. #include <boost/spirit/include/classic_actions.hpp>
  19. #include <boost/spirit/include/classic_operators.hpp>
  20. #include <boost/spirit/include/classic_rule.hpp>
  21. #include <boost/spirit/include/classic_grammar.hpp>
  22. #include <boost/spirit/include/classic_switch.hpp>
  23. #include <boost/spirit/include/classic_select.hpp>
  24. #include <boost/spirit/include/classic_closure.hpp>
  25. using namespace BOOST_SPIRIT_CLASSIC_NS;
  26. namespace test_grammars {
  27. ///////////////////////////////////////////////////////////////////////////////
  28. // Test the direct switch_p usage
  29. struct switch_grammar_direct_single
  30. : public grammar<switch_grammar_direct_single>
  31. {
  32. template <typename ScannerT>
  33. struct definition
  34. {
  35. definition(switch_grammar_direct_single const& /*self*/)
  36. {
  37. r = switch_p [
  38. case_p<'a'>(int_p)
  39. ];
  40. }
  41. rule<ScannerT> r;
  42. rule<ScannerT> const& start() const { return r; }
  43. };
  44. };
  45. struct switch_grammar_direct_default_single1
  46. : public grammar<switch_grammar_direct_default_single1>
  47. {
  48. template <typename ScannerT>
  49. struct definition
  50. {
  51. definition(switch_grammar_direct_default_single1 const& /*self*/)
  52. {
  53. r = switch_p [
  54. default_p(str_p("default"))
  55. ];
  56. }
  57. rule<ScannerT> r;
  58. rule<ScannerT> const& start() const { return r; }
  59. };
  60. };
  61. struct switch_grammar_direct_default_single2
  62. : public grammar<switch_grammar_direct_default_single2>
  63. {
  64. template <typename ScannerT>
  65. struct definition
  66. {
  67. definition(switch_grammar_direct_default_single2 const& /*self*/)
  68. {
  69. r = switch_p [
  70. default_p
  71. ];
  72. }
  73. rule<ScannerT> r;
  74. rule<ScannerT> const& start() const { return r; }
  75. };
  76. };
  77. ///////////////////////////////////////////////////////////////////////////////
  78. // Test the switch_p usage given a parser as the switch condition
  79. struct switch_grammar_parser_single
  80. : public grammar<switch_grammar_parser_single>
  81. {
  82. template <typename ScannerT>
  83. struct definition
  84. {
  85. definition(switch_grammar_parser_single const& /*self*/)
  86. {
  87. r = switch_p(anychar_p) [
  88. case_p<'a'>(int_p)
  89. ];
  90. }
  91. rule<ScannerT> r;
  92. rule<ScannerT> const& start() const { return r; }
  93. };
  94. };
  95. struct switch_grammar_parser_default_single1
  96. : public grammar<switch_grammar_parser_default_single1>
  97. {
  98. template <typename ScannerT>
  99. struct definition
  100. {
  101. definition(switch_grammar_parser_default_single1 const& /*self*/)
  102. {
  103. r = switch_p(anychar_p) [
  104. default_p(str_p("default"))
  105. ];
  106. }
  107. rule<ScannerT> r;
  108. rule<ScannerT> const& start() const { return r; }
  109. };
  110. };
  111. struct switch_grammar_parser_default_single2
  112. : public grammar<switch_grammar_parser_default_single2>
  113. {
  114. template <typename ScannerT>
  115. struct definition
  116. {
  117. definition(switch_grammar_parser_default_single2 const& /*self*/)
  118. {
  119. r = switch_p(anychar_p) [
  120. default_p
  121. ];
  122. }
  123. rule<ScannerT> r;
  124. rule<ScannerT> const& start() const { return r; }
  125. };
  126. };
  127. ///////////////////////////////////////////////////////////////////////////////
  128. // Test the switch_p usage given an actor as the switch condition
  129. struct select_result : public BOOST_SPIRIT_CLASSIC_NS::closure<select_result, int>
  130. {
  131. member1 val;
  132. };
  133. struct switch_grammar_actor_single
  134. : public grammar<switch_grammar_actor_single>
  135. {
  136. template <typename ScannerT>
  137. struct definition
  138. {
  139. definition(switch_grammar_actor_single const& /*self*/)
  140. {
  141. using phoenix::arg1;
  142. r = select_p('a')[r.val = arg1] >>
  143. switch_p(r.val) [
  144. case_p<0>(int_p)
  145. ];
  146. }
  147. rule<ScannerT, select_result::context_t> r;
  148. rule<ScannerT, select_result::context_t> const&
  149. start() const { return r; }
  150. };
  151. };
  152. struct switch_grammar_actor_default_single1
  153. : public grammar<switch_grammar_actor_default_single1>
  154. {
  155. template <typename ScannerT>
  156. struct definition
  157. {
  158. definition(switch_grammar_actor_default_single1 const& /*self*/)
  159. {
  160. using phoenix::arg1;
  161. r = select_p('d')[r.val = arg1] >>
  162. switch_p(r.val) [
  163. default_p(str_p("default"))
  164. ];
  165. }
  166. rule<ScannerT, select_result::context_t> r;
  167. rule<ScannerT, select_result::context_t> const&
  168. start() const { return r; }
  169. };
  170. };
  171. struct switch_grammar_actor_default_single2
  172. : public grammar<switch_grammar_actor_default_single2>
  173. {
  174. template <typename ScannerT>
  175. struct definition
  176. {
  177. definition(switch_grammar_actor_default_single2 const& /*self*/)
  178. {
  179. using phoenix::arg1;
  180. r = select_p('d')[r.val = arg1] >>
  181. switch_p(r.val) [
  182. default_p
  183. ];
  184. }
  185. rule<ScannerT, select_result::context_t> r;
  186. rule<ScannerT, select_result::context_t> const&
  187. start() const { return r; }
  188. };
  189. };
  190. } // namespace test_grammars
  191. ///////////////////////////////////////////////////////////////////////////////
  192. namespace tests {
  193. // Tests for known (to the grammars) sequences
  194. struct check_grammar_unknown {
  195. template <typename GrammarT>
  196. void operator()(GrammarT)
  197. {
  198. GrammarT g;
  199. BOOST_TEST(!parse("a1", g).hit);
  200. BOOST_TEST(!parse("a,", g).hit);
  201. BOOST_TEST(!parse("abcd", g).hit);
  202. BOOST_TEST(!parse("a 1", g, space_p).hit);
  203. BOOST_TEST(!parse("a ,", g, space_p).hit);
  204. BOOST_TEST(!parse("a bcd", g, space_p).hit);
  205. BOOST_TEST(!parse("b1", g).hit);
  206. BOOST_TEST(!parse("b,", g).hit);
  207. BOOST_TEST(!parse("bbcd", g).hit);
  208. BOOST_TEST(!parse("b 1", g, space_p).hit);
  209. BOOST_TEST(!parse("b ,", g, space_p).hit);
  210. BOOST_TEST(!parse("b bcd", g, space_p).hit);
  211. BOOST_TEST(!parse("c1", g).hit);
  212. BOOST_TEST(!parse("c,", g).hit);
  213. BOOST_TEST(!parse("cbcd", g).hit);
  214. BOOST_TEST(!parse("c 1", g, space_p).hit);
  215. BOOST_TEST(!parse("c ,", g, space_p).hit);
  216. BOOST_TEST(!parse("c bcd", g, space_p).hit);
  217. }
  218. };
  219. // Tests for the default branches (with parsers) of the grammars
  220. struct check_grammar_default {
  221. template <typename GrammarT>
  222. void operator()(GrammarT)
  223. {
  224. GrammarT g;
  225. BOOST_TEST(parse("ddefault", g).full);
  226. BOOST_TEST(parse("d default", g, space_p).full);
  227. }
  228. };
  229. // Tests for the default branches (without parsers) of the grammars
  230. struct check_grammar_default_plain {
  231. template <typename GrammarT>
  232. void operator()(GrammarT)
  233. {
  234. GrammarT g;
  235. BOOST_TEST(parse("d", g).full);
  236. BOOST_TEST(parse(" d", g, space_p).full); // JDG 10-18-2005 removed trailing ' ' to
  237. // avoid post skip problems
  238. }
  239. };
  240. // Tests grammars with a single case_p branch
  241. struct check_grammar_single {
  242. template <typename GrammarT>
  243. void operator()(GrammarT)
  244. {
  245. GrammarT g;
  246. BOOST_TEST(parse("a1", g).full);
  247. BOOST_TEST(!parse("a,", g).hit);
  248. BOOST_TEST(!parse("abcd", g).hit);
  249. BOOST_TEST(parse("a 1", g, space_p).full);
  250. BOOST_TEST(!parse("a ,", g, space_p).hit);
  251. BOOST_TEST(!parse("a bcd", g, space_p).hit);
  252. BOOST_TEST(!parse("b1", g).hit);
  253. BOOST_TEST(!parse("b,", g).hit);
  254. BOOST_TEST(!parse("bbcd", g).hit);
  255. BOOST_TEST(!parse("b 1", g, space_p).hit);
  256. BOOST_TEST(!parse("b ,", g, space_p).hit);
  257. BOOST_TEST(!parse("b bcd", g, space_p).hit);
  258. BOOST_TEST(!parse("c1", g).hit);
  259. BOOST_TEST(!parse("c,", g).hit);
  260. BOOST_TEST(!parse("cbcd", g).hit);
  261. BOOST_TEST(!parse("c 1", g, space_p).hit);
  262. BOOST_TEST(!parse("c ,", g, space_p).hit);
  263. BOOST_TEST(!parse("c bcd", g, space_p).hit);
  264. }
  265. };
  266. } // namespace tests
  267. int
  268. main()
  269. {
  270. // Test switch_p with a single case_p branch
  271. typedef boost::mpl::list<
  272. test_grammars::switch_grammar_direct_single,
  273. test_grammars::switch_grammar_parser_single,
  274. test_grammars::switch_grammar_actor_single
  275. > single_list_t;
  276. boost::mpl::for_each<single_list_t>(tests::check_grammar_single());
  277. typedef boost::mpl::list<
  278. test_grammars::switch_grammar_direct_default_single1,
  279. test_grammars::switch_grammar_parser_default_single1,
  280. test_grammars::switch_grammar_actor_default_single1
  281. > default_single_t;
  282. boost::mpl::for_each<default_single_t>(tests::check_grammar_default());
  283. boost::mpl::for_each<default_single_t>(tests::check_grammar_unknown());
  284. typedef boost::mpl::list<
  285. test_grammars::switch_grammar_direct_default_single2,
  286. test_grammars::switch_grammar_parser_default_single2,
  287. test_grammars::switch_grammar_actor_default_single2
  288. > default_plain_single_t;
  289. boost::mpl::for_each<default_plain_single_t>(
  290. tests::check_grammar_default_plain());
  291. return boost::report_errors();
  292. }