lexical_cast_stream_detection_test.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. // Unit test for boost::lexical_cast.
  2. //
  3. // See http://www.boost.org for most recent version, including documentation.
  4. //
  5. // Copyright Antony Polukhin, 2011-2019.
  6. //
  7. // Distributed under the Boost
  8. // Software License, Version 1.0. (See accompanying file
  9. // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt).
  10. #include <boost/config.hpp>
  11. #include <boost/test/unit_test.hpp>
  12. #include <boost/lexical_cast.hpp>
  13. #include <iostream>
  14. ///////////////////////// char streamable classes ///////////////////////////////////////////
  15. struct streamable_easy { enum ENU {value = 0}; };
  16. std::ostream& operator << (std::ostream& ostr, const streamable_easy&) {
  17. return ostr << streamable_easy::value;
  18. }
  19. std::istream& operator >> (std::istream& istr, const streamable_easy&) {
  20. int i; istr >> i; BOOST_CHECK_EQUAL(i, streamable_easy::value);
  21. return istr;
  22. }
  23. struct streamable_medium { enum ENU {value = 1}; };
  24. template <class CharT>
  25. typename boost::enable_if<boost::is_same<CharT, char>, std::basic_ostream<CharT>&>::type
  26. operator << (std::basic_ostream<CharT>& ostr, const streamable_medium&) {
  27. return ostr << streamable_medium::value;
  28. }
  29. template <class CharT>
  30. typename boost::enable_if<boost::is_same<CharT, char>, std::basic_istream<CharT>&>::type
  31. operator >> (std::basic_istream<CharT>& istr, const streamable_medium&) {
  32. int i; istr >> i; BOOST_CHECK_EQUAL(i, streamable_medium::value);
  33. return istr;
  34. }
  35. struct streamable_hard { enum ENU {value = 2}; };
  36. template <class CharT, class TraitsT>
  37. typename boost::enable_if<boost::is_same<CharT, char>, std::basic_ostream<CharT, TraitsT>&>::type
  38. operator << (std::basic_ostream<CharT, TraitsT>& ostr, const streamable_hard&) {
  39. return ostr << streamable_hard::value;
  40. }
  41. template <class CharT, class TraitsT>
  42. typename boost::enable_if<boost::is_same<CharT, char>, std::basic_istream<CharT, TraitsT>&>::type
  43. operator >> (std::basic_istream<CharT, TraitsT>& istr, const streamable_hard&) {
  44. int i; istr >> i; BOOST_CHECK_EQUAL(i, streamable_hard::value);
  45. return istr;
  46. }
  47. struct streamable_hard2 { enum ENU {value = 3}; };
  48. template <class TraitsT>
  49. std::basic_ostream<char, TraitsT>& operator << (std::basic_ostream<char, TraitsT>& ostr, const streamable_hard2&) {
  50. return ostr << streamable_hard2::value;
  51. }
  52. template <class TraitsT>
  53. std::basic_istream<char, TraitsT>& operator >> (std::basic_istream<char, TraitsT>& istr, const streamable_hard2&) {
  54. int i; istr >> i; BOOST_CHECK_EQUAL(i, streamable_hard2::value);
  55. return istr;
  56. }
  57. ///////////////////////// wchar_t streamable classes ///////////////////////////////////////////
  58. struct wstreamable_easy { enum ENU {value = 4}; };
  59. std::wostream& operator << (std::wostream& ostr, const wstreamable_easy&) {
  60. return ostr << wstreamable_easy::value;
  61. }
  62. std::wistream& operator >> (std::wistream& istr, const wstreamable_easy&) {
  63. int i; istr >> i; BOOST_CHECK_EQUAL(i, wstreamable_easy::value);
  64. return istr;
  65. }
  66. struct wstreamable_medium { enum ENU {value = 5}; };
  67. template <class CharT>
  68. typename boost::enable_if<boost::is_same<CharT, wchar_t>, std::basic_ostream<CharT>& >::type
  69. operator << (std::basic_ostream<CharT>& ostr, const wstreamable_medium&) {
  70. return ostr << wstreamable_medium::value;
  71. }
  72. template <class CharT>
  73. typename boost::enable_if<boost::is_same<CharT, wchar_t>, std::basic_istream<CharT>& >::type
  74. operator >> (std::basic_istream<CharT>& istr, const wstreamable_medium&) {
  75. int i; istr >> i; BOOST_CHECK_EQUAL(i, wstreamable_medium::value);
  76. return istr;
  77. }
  78. struct wstreamable_hard { enum ENU {value = 6}; };
  79. template <class CharT, class TraitsT>
  80. typename boost::enable_if<boost::is_same<CharT, wchar_t>, std::basic_ostream<CharT, TraitsT>&>::type
  81. operator << (std::basic_ostream<CharT, TraitsT>& ostr, const wstreamable_hard&) {
  82. return ostr << wstreamable_hard::value;
  83. }
  84. template <class CharT, class TraitsT>
  85. typename boost::enable_if<boost::is_same<CharT, wchar_t>, std::basic_istream<CharT, TraitsT>&>::type
  86. operator >> (std::basic_istream<CharT, TraitsT>& istr, const wstreamable_hard&) {
  87. int i; istr >> i; BOOST_CHECK_EQUAL(i, wstreamable_hard::value);
  88. return istr;
  89. }
  90. struct wstreamable_hard2 { enum ENU {value = 7}; };
  91. template <class TraitsT>
  92. std::basic_ostream<wchar_t, TraitsT>& operator << (std::basic_ostream<wchar_t, TraitsT>& ostr, const wstreamable_hard2&) {
  93. return ostr << wstreamable_hard2::value;
  94. }
  95. template <class TraitsT>
  96. std::basic_istream<wchar_t, TraitsT>& operator >> (std::basic_istream<wchar_t, TraitsT>& istr, const wstreamable_hard2&) {
  97. int i; istr >> i; BOOST_CHECK_EQUAL(i, wstreamable_hard2::value);
  98. return istr;
  99. }
  100. ///////////////////////// char and wchar_t streamable classes ///////////////////////////////////////////
  101. struct bistreamable_easy { enum ENU {value = 8}; };
  102. std::ostream& operator << (std::ostream& ostr, const bistreamable_easy&) {
  103. return ostr << bistreamable_easy::value;
  104. }
  105. std::istream& operator >> (std::istream& istr, const bistreamable_easy&) {
  106. int i; istr >> i; BOOST_CHECK_EQUAL(i, bistreamable_easy::value);
  107. return istr;
  108. }
  109. std::wostream& operator << (std::wostream& ostr, const bistreamable_easy&) {
  110. return ostr << bistreamable_easy::value + 100;
  111. }
  112. std::wistream& operator >> (std::wistream& istr, const bistreamable_easy&) {
  113. int i; istr >> i; BOOST_CHECK_EQUAL(i, bistreamable_easy::value + 100);
  114. return istr;
  115. }
  116. struct bistreamable_medium { enum ENU {value = 9}; };
  117. template <class CharT>
  118. std::basic_ostream<CharT>& operator << (std::basic_ostream<CharT>& ostr, const bistreamable_medium&) {
  119. return ostr << bistreamable_medium::value + (sizeof(CharT) == 1 ? 0 : 100);
  120. }
  121. template <class CharT>
  122. std::basic_istream<CharT>& operator >> (std::basic_istream<CharT>& istr, const bistreamable_medium&) {
  123. int i; istr >> i; BOOST_CHECK_EQUAL(i, bistreamable_medium::value + (sizeof(CharT) == 1 ? 0 : 100));
  124. return istr;
  125. }
  126. struct bistreamable_hard { enum ENU {value = 10}; };
  127. template <class CharT, class TraitsT>
  128. std::basic_ostream<CharT, TraitsT>& operator << (std::basic_ostream<CharT, TraitsT>& ostr, const bistreamable_hard&) {
  129. return ostr << bistreamable_hard::value + (sizeof(CharT) == 1 ? 0 : 100);
  130. }
  131. template <class CharT, class TraitsT>
  132. std::basic_istream<CharT, TraitsT>& operator >> (std::basic_istream<CharT, TraitsT>& istr, const bistreamable_hard&) {
  133. int i; istr >> i; BOOST_CHECK_EQUAL(i, bistreamable_hard::value + (sizeof(CharT) == 1 ? 0 : 100));
  134. return istr;
  135. }
  136. struct bistreamable_hard2 { enum ENU {value = 11}; };
  137. template <class TraitsT>
  138. std::basic_ostream<char, TraitsT>& operator << (std::basic_ostream<char, TraitsT>& ostr, const bistreamable_hard2&) {
  139. return ostr << bistreamable_hard2::value;
  140. }
  141. template <class TraitsT>
  142. std::basic_istream<char, TraitsT>& operator >> (std::basic_istream<char, TraitsT>& istr, const bistreamable_hard2&) {
  143. int i; istr >> i; BOOST_CHECK_EQUAL(i, bistreamable_hard2::value);
  144. return istr;
  145. }
  146. template <class TraitsT>
  147. std::basic_ostream<wchar_t, TraitsT>& operator << (std::basic_ostream<wchar_t, TraitsT>& ostr, const bistreamable_hard2&) {
  148. return ostr << bistreamable_hard2::value + 100;
  149. }
  150. template <class TraitsT>
  151. std::basic_istream<wchar_t, TraitsT>& operator >> (std::basic_istream<wchar_t, TraitsT>& istr, const bistreamable_hard2&) {
  152. int i; istr >> i; BOOST_CHECK_EQUAL(i, bistreamable_hard2::value + 100);
  153. return istr;
  154. }
  155. void test_ostream_character_detection();
  156. void test_istream_character_detection();
  157. void test_mixed_stream_character_detection();
  158. boost::unit_test::test_suite *init_unit_test_suite(int, char *[])
  159. {
  160. boost::unit_test::test_suite *suite =
  161. BOOST_TEST_SUITE("lexical_cast stream character detection");
  162. suite->add(BOOST_TEST_CASE(&test_ostream_character_detection));
  163. suite->add(BOOST_TEST_CASE(&test_istream_character_detection));
  164. suite->add(BOOST_TEST_CASE(&test_mixed_stream_character_detection));
  165. return suite;
  166. }
  167. template <class T>
  168. static void test_ostr_impl() {
  169. T streamable;
  170. BOOST_CHECK_EQUAL(T::value, boost::lexical_cast<int>(streamable));
  171. BOOST_CHECK_EQUAL(boost::lexical_cast<std::string>(T::value), boost::lexical_cast<std::string>(streamable));
  172. }
  173. template <class T>
  174. static void test_wostr_impl() {
  175. T streamable;
  176. BOOST_CHECK_EQUAL(T::value, boost::lexical_cast<int>(streamable));
  177. // BOOST_CHECK_EQUAL(boost::lexical_cast<std::string>(T::value), boost::lexical_cast<std::string>(streamable)); // Shall not compile???
  178. BOOST_CHECK(boost::lexical_cast<std::wstring>(T::value) == boost::lexical_cast<std::wstring>(streamable));
  179. }
  180. template <class T>
  181. static void test_bistr_impl() {
  182. T streamable;
  183. BOOST_CHECK_EQUAL(T::value, boost::lexical_cast<int>(streamable));
  184. BOOST_CHECK_EQUAL(boost::lexical_cast<std::string>(T::value), boost::lexical_cast<std::string>(streamable));
  185. BOOST_CHECK(boost::lexical_cast<std::wstring>(T::value + 100) == boost::lexical_cast<std::wstring>(streamable));
  186. }
  187. void test_ostream_character_detection() {
  188. test_ostr_impl<streamable_easy>();
  189. test_ostr_impl<streamable_medium>();
  190. test_ostr_impl<streamable_hard>();
  191. test_ostr_impl<streamable_hard2>();
  192. test_wostr_impl<wstreamable_easy>();
  193. test_wostr_impl<wstreamable_medium>();
  194. test_wostr_impl<wstreamable_hard>();
  195. test_wostr_impl<wstreamable_hard2>();
  196. test_bistr_impl<bistreamable_easy>();
  197. test_bistr_impl<bistreamable_medium>();
  198. test_bistr_impl<bistreamable_hard>();
  199. test_bistr_impl<bistreamable_hard2>();
  200. }
  201. template <class T>
  202. static void test_istr_impl() {
  203. boost::lexical_cast<T>(T::value);
  204. boost::lexical_cast<T>(boost::lexical_cast<std::string>(T::value));
  205. }
  206. template <class T>
  207. static void test_wistr_impl() {
  208. boost::lexical_cast<T>(T::value);
  209. //boost::lexical_cast<T>(boost::lexical_cast<std::string>(T::value)); // Shall not compile???
  210. boost::lexical_cast<T>(boost::lexical_cast<std::wstring>(T::value));
  211. }
  212. template <class T>
  213. static void test_bistr_instr_impl() {
  214. boost::lexical_cast<T>(T::value);
  215. boost::lexical_cast<T>(boost::lexical_cast<std::string>(T::value));
  216. boost::lexical_cast<T>(boost::lexical_cast<std::wstring>(T::value + 100));
  217. }
  218. void test_istream_character_detection() {
  219. test_istr_impl<streamable_easy>();
  220. test_istr_impl<streamable_medium>();
  221. test_istr_impl<streamable_hard>();
  222. test_istr_impl<streamable_hard2>();
  223. test_wistr_impl<wstreamable_easy>();
  224. test_wistr_impl<wstreamable_medium>();
  225. test_wistr_impl<wstreamable_hard>();
  226. test_wistr_impl<wstreamable_hard2>();
  227. test_bistr_instr_impl<bistreamable_easy>();
  228. test_bistr_instr_impl<bistreamable_medium>();
  229. test_bistr_instr_impl<bistreamable_hard>();
  230. test_bistr_instr_impl<bistreamable_hard2>();
  231. }
  232. struct wistreamble_ostreamable { enum ENU {value = 200}; };
  233. std::ostream& operator << (std::ostream& ostr, const wistreamble_ostreamable&) {
  234. return ostr << wistreamble_ostreamable::value;
  235. }
  236. std::wistream& operator >> (std::wistream& istr, const wistreamble_ostreamable&) {
  237. int i; istr >> i; BOOST_CHECK_EQUAL(i, wistreamble_ostreamable::value);
  238. return istr;
  239. }
  240. struct istreamble_wostreamable { enum ENU {value = 201}; };
  241. std::wostream& operator << (std::wostream& ostr, const istreamble_wostreamable&) {
  242. return ostr << istreamble_wostreamable::value;
  243. }
  244. std::istream& operator >> (std::istream& istr, const istreamble_wostreamable&) {
  245. int i; istr >> i; BOOST_CHECK_EQUAL(i, istreamble_wostreamable::value);
  246. return istr;
  247. }
  248. void test_mixed_stream_character_detection() {
  249. //boost::lexical_cast<std::wstring>(std::string("qwe")); // TODO: ALLOW IT AS EXTENSION!
  250. boost::lexical_cast<wistreamble_ostreamable>(wistreamble_ostreamable::value);
  251. BOOST_CHECK_EQUAL(boost::lexical_cast<int>(wistreamble_ostreamable()), wistreamble_ostreamable::value);
  252. boost::lexical_cast<istreamble_wostreamable>(istreamble_wostreamable::value);
  253. BOOST_CHECK_EQUAL(boost::lexical_cast<int>(istreamble_wostreamable()), istreamble_wostreamable::value);
  254. }