stream_test.cxx 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. //
  2. // stream_td.hpp
  3. // stream
  4. //
  5. // Created by Sam Jaffe on 1/28/17.
  6. //
  7. #include "xcode_gtest_helper.h"
  8. #include <map>
  9. #include <string>
  10. #include <vector>
  11. #include "stream/streams.hpp"
  12. using ::testing::ElementsAre;
  13. using ::testing::ElementsAreArray;
  14. using ::testing::Eq;
  15. namespace views = stream::views;
  16. namespace ranges = stream::ranges;
  17. // Workaround for OSX and pointer-to-member-functions
  18. template class std::basic_string<char>;
  19. TEST(StreamTest, IteratorPreservesElements) {
  20. std::vector<int> input{1, 2, 3, 4, 5};
  21. auto s = views::all(input);
  22. std::vector<int> out{s.begin(), s.end()};
  23. EXPECT_THAT(out, Eq(input));
  24. }
  25. TEST(MapStreamTest, IteratorPreservesElements) {
  26. std::map<int, int> input{{1, 1}, {2, 2}};
  27. auto s = views::all(input);
  28. std::map<int, int> out{s.begin(), s.end()};
  29. EXPECT_THAT(out, Eq(input));
  30. }
  31. TEST(StreamTest, CollectPreservesElements) {
  32. std::vector<int> input{1, 2, 3, 4, 5};
  33. std::vector<int> out = views::all(input) | ranges::to_vector();
  34. EXPECT_THAT(out, Eq(input));
  35. }
  36. TEST(StreamTest, CollectToObjectPreservesElements) {
  37. std::vector<int> input{1, 2, 3, 4, 5};
  38. auto s = views::all(input);
  39. std::set<int> out = s | ranges::to_set();
  40. EXPECT_THAT(out, ElementsAreArray(input));
  41. }
  42. TEST(StreamTest, MapToSelfIsSelfs) {
  43. std::vector<int> input{1, 2, 3, 4, 5};
  44. auto identity = [](int i) { return i; };
  45. auto out = input | views::transform(identity) | ranges::to_vector();
  46. EXPECT_THAT(out, Eq(input));
  47. }
  48. TEST(StreamTest, MapCanAlterValues) {
  49. std::vector<int> input{1, 2, 3, 4, 5};
  50. std::vector<int> expected{3, 5, 7, 9, 11};
  51. auto fmap = [](int i) { return 2 * i + 1; };
  52. auto out = input | views::transform(fmap) | ranges::to_vector();
  53. EXPECT_THAT(out, Eq(expected));
  54. }
  55. template <typename T> struct nocopy {
  56. T value;
  57. nocopy(T const & val) : value(val) {}
  58. nocopy(nocopy const &) = delete;
  59. nocopy & operator=(nocopy const &) = delete;
  60. nocopy(nocopy &&) = default;
  61. nocopy & operator=(nocopy &&) = default;
  62. operator T() const { return value; }
  63. };
  64. TEST(MapStreamTest, MapToValue) {
  65. auto const input = []() {
  66. std::map<int, nocopy<int>> tmp;
  67. tmp.emplace(0, 1);
  68. tmp.emplace(2, 2);
  69. return tmp;
  70. }();
  71. auto fmap = [](auto & pair) -> auto & { return pair.second; };
  72. auto s = input | views::transform(fmap);
  73. std::vector<int> out(s.begin(), s.end());
  74. std::vector<int> const expected{1, 2};
  75. EXPECT_THAT(out, Eq(expected));
  76. }
  77. TEST(StreamTest, CanBuildFromSingleElement) {
  78. int value = 11;
  79. auto even = [](int i) { return i % 2 == 0; };
  80. auto s = views::single(value) | views::filter(even);
  81. EXPECT_THAT(ranges::size(s), 0);
  82. }
  83. TEST(StreamTest, CanBuildFromIterators) {
  84. std::vector<int> input{1, 2, 3, 4, 5};
  85. std::vector<int> expected{5, 7};
  86. auto fmap = [](int i) { return 2 * i + 1; };
  87. auto out = ranges::ref_view(input.begin() + 1, input.begin() + 3) |
  88. views::transform(fmap) | ranges::to_vector();
  89. EXPECT_THAT(out, Eq(expected));
  90. }
  91. TEST(StreamTest, NoOpFilterReturnOriginal) {
  92. std::vector<int> input{1, 2, 3, 4, 5};
  93. auto pass = [](int) { return true; };
  94. auto out = input | views::filter(pass) | ranges::to_vector();
  95. EXPECT_THAT(out, Eq(input));
  96. }
  97. TEST(StreamTest, CanFilterOutElements) {
  98. std::vector<int> input{1, 2, 3, 4, 5};
  99. std::vector<int> expected{2, 4};
  100. auto even = [](int i) { return i % 2 == 0; };
  101. auto out = input | views::filter(even) | ranges::to_vector();
  102. EXPECT_THAT(out, Eq(expected));
  103. }
  104. TEST(StreamTest, CanTransformFilteredElements) {
  105. std::vector<int> input{1, 2, 3, 4, 5};
  106. auto range = input | views::filter([](int i) { return i % 2 == 0; }) |
  107. views::transform([](int i) { return i / 2; });
  108. auto out = range | ranges::to_vector();
  109. EXPECT_THAT(out, ElementsAre(1, 2));
  110. ranges::for_each(range, [&](int i) { out.push_back(i); });
  111. EXPECT_THAT(out, ElementsAre(1, 2, 1, 2));
  112. }
  113. TEST(StreamTest, AccumulateDefaultsToAdd) {
  114. std::vector<int> input{1, 2, 3, 4, 5};
  115. auto even = [](int i) { return i % 2 == 0; };
  116. auto s = input | views::filter(even) | ranges::to_vector();
  117. EXPECT_THAT(ranges::fold_left(s, 0, std::plus<>()), Eq(6));
  118. }
  119. TEST(StreamTest, AccumulateCanTakeCustomAccumulator) {
  120. std::vector<int> input{1, 2, 3, 4, 5};
  121. auto even = [](int i) { return i % 2 == 0; };
  122. auto prod = [](int lhs, int rhs) { return lhs * rhs; };
  123. auto s = input | views::filter(even) | ranges::to_vector();
  124. EXPECT_THAT(ranges::fold_left(s, 0, prod), Eq(0));
  125. EXPECT_THAT(ranges::fold_left(s, 1, prod), Eq(8));
  126. }
  127. TEST(StreamTest, FlatmapJoinsIterableOutputs) {
  128. std::vector<int> vv{1, 2, 3, 4, 5};
  129. auto next3 = [](int i) { return std::vector<int>{i, i + 1, i + 2}; };
  130. std::vector<int> expected{1, 2, 3, 2, 3, 4, 3, 4, 5, 4, 5, 6, 5, 6, 7};
  131. auto out = vv | views::transform(next3) | views::join() | ranges::to_vector();
  132. EXPECT_THAT(out, Eq(expected));
  133. }
  134. TEST(StreamTest, CanDereferenceElements) {
  135. int val = 5;
  136. std::vector<int *> input{&val};
  137. auto data = input | views::deref() | ranges::to_vector();
  138. EXPECT_THAT(data.front(), Eq(val));
  139. }
  140. TEST(StreamTest, CanForEachConsume) {
  141. int hits = 0;
  142. std::vector<int> input{1, 2, 3, 4, 5};
  143. ranges::for_each(views::all(input), [&hits](int) { ++hits; });
  144. EXPECT_THAT(hits, Eq(5));
  145. }
  146. TEST(StreamTest, CanFetchMemPtr) {
  147. struct test {
  148. int val;
  149. };
  150. std::vector<test> input{{1}, {3}, {2}};
  151. std::vector<int> expected{1, 3, 2};
  152. auto out = input | views::transform(&test::val) | ranges::to_vector();
  153. EXPECT_THAT(out, Eq(expected));
  154. }
  155. TEST(StreamTest, CanMapToMemFn) {
  156. std::vector<std::string> input{"hello", "goodbye"};
  157. std::vector<std::string::size_type> expected{5, 7};
  158. auto out = input | views::transform(&std::string::size) | ranges::to_vector();
  159. EXPECT_THAT(out, Eq(expected));
  160. }
  161. //
  162. // TEST(StreamTest, CastStreamToParentType) {
  163. // struct base {
  164. // char cat[4] = "cat";
  165. // };
  166. // struct test : base {
  167. // test(int v) : val(v) {}
  168. // int val;
  169. // };
  170. // std::vector<test> input{{1}, {3}, {2}};
  171. //
  172. // auto addressof_void = [](auto const & p) { return (void *)&p; };
  173. //
  174. // auto strm = stream::of(input).cast<base>();
  175. // auto first = stream::of(input).map(addressof_void).collect();
  176. // auto second = strm.map(addressof_void).collect();
  177. //
  178. // EXPECT_THAT(first, second);
  179. //}
  180. TEST(StreamTest, Iota) {
  181. auto out = views::iota(0, 4) |
  182. views::transform([](size_t i) { return std::vector(i, i); }) |
  183. views::join() | ranges::to_vector();
  184. std::vector<size_t> expected{1, 2, 2, 3, 3, 3};
  185. EXPECT_THAT(out, expected);
  186. }