ysaveload_ut.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  1. #include "ysaveload.h"
  2. #include <library/cpp/testing/unittest/registar.h>
  3. #include <util/memory/pool.h>
  4. #include <util/stream/buffer.h>
  5. #include <util/memory/blob.h>
  6. #include <util/generic/list.h>
  7. #include <util/generic/map.h>
  8. #include <util/generic/hash_multi_map.h>
  9. #include <util/generic/deque.h>
  10. #include <util/generic/string.h>
  11. #include <util/generic/vector.h>
  12. #include <util/generic/buffer.h>
  13. #include <util/generic/hash_set.h>
  14. #include <util/generic/maybe.h>
  15. static inline char* AllocateFromPool(TMemoryPool& pool, size_t len) {
  16. return (char*)pool.Allocate(len);
  17. }
  18. class TSaveLoadTest: public TTestBase {
  19. UNIT_TEST_SUITE(TSaveLoadTest);
  20. UNIT_TEST(TestSaveLoad)
  21. UNIT_TEST(TestNewStyle)
  22. UNIT_TEST(TestNewNewStyle)
  23. UNIT_TEST(TestList)
  24. UNIT_TEST(TestTuple)
  25. UNIT_TEST(TestVariant)
  26. UNIT_TEST(TestInheritNonVirtualClass)
  27. UNIT_TEST(TestInheritVirtualClass)
  28. UNIT_TEST_SUITE_END();
  29. struct TSaveHelper {
  30. inline void Save(IOutputStream* o) const {
  31. o->Write("qwerty", 7);
  32. }
  33. inline void Load(IInputStream* i) {
  34. char buf[7];
  35. UNIT_ASSERT_EQUAL(i->Load(buf, 7), 7);
  36. UNIT_ASSERT_EQUAL(strcmp(buf, "qwerty"), 0);
  37. }
  38. };
  39. struct TNewStyleSaveHelper {
  40. template <class S>
  41. inline void SaveLoad(S* s) {
  42. ::SaveLoad(s, Str);
  43. }
  44. TString Str;
  45. };
  46. struct TNewNewStyleHelper {
  47. TString Str;
  48. ui32 Int;
  49. Y_SAVELOAD_DEFINE(Str, Int)
  50. };
  51. private:
  52. inline void TestNewNewStyle() {
  53. TString ss;
  54. {
  55. TNewNewStyleHelper h;
  56. h.Str = "qw";
  57. h.Int = 42;
  58. TStringOutput so(ss);
  59. ::Save(&so, h);
  60. }
  61. {
  62. TNewNewStyleHelper h;
  63. TStringInput si(ss);
  64. ::Load(&si, h);
  65. UNIT_ASSERT_EQUAL(h.Str, "qw");
  66. UNIT_ASSERT_EQUAL(h.Int, 42);
  67. }
  68. }
  69. inline void TestNewStyle() {
  70. TString ss;
  71. {
  72. TNewStyleSaveHelper sh;
  73. sh.Str = "qwerty";
  74. TStringOutput so(ss);
  75. SaveLoad(&so, sh);
  76. }
  77. {
  78. TNewStyleSaveHelper sh;
  79. TStringInput si(ss);
  80. SaveLoad(&si, sh);
  81. UNIT_ASSERT_EQUAL(sh.Str, "qwerty");
  82. }
  83. }
  84. inline void TestSaveLoad() {
  85. TBufferStream S_;
  86. //save part
  87. {
  88. Save(&S_, (ui8)1);
  89. Save(&S_, (ui16)2);
  90. Save(&S_, (ui32)3);
  91. Save(&S_, (ui64)4);
  92. }
  93. {
  94. TVector<ui16> vec;
  95. vec.push_back((ui16)1);
  96. vec.push_back((ui16)2);
  97. vec.push_back((ui16)4);
  98. Save(&S_, vec);
  99. }
  100. {
  101. TMap<ui16, ui32> map;
  102. map[(ui16)1] = 2;
  103. map[(ui16)2] = 3;
  104. map[(ui16)3] = 4;
  105. Save(&S_, map);
  106. }
  107. {
  108. TMultiMap<ui16, ui32> multimap;
  109. multimap.emplace((ui16)1, 2);
  110. multimap.emplace((ui16)2, 3);
  111. multimap.emplace((ui16)2, 4);
  112. multimap.emplace((ui16)2, 5);
  113. multimap.emplace((ui16)3, 6);
  114. Save(&S_, multimap);
  115. }
  116. {
  117. TSaveHelper helper;
  118. Save(&S_, helper);
  119. }
  120. {
  121. TString val("123456");
  122. Save(&S_, val);
  123. }
  124. {
  125. TBuffer buf;
  126. buf.Append("asdf", 4);
  127. Save(&S_, buf);
  128. }
  129. {
  130. TVector<const char*> vec;
  131. vec.push_back("1");
  132. vec.push_back("123");
  133. vec.push_back("4567");
  134. Save(&S_, vec);
  135. }
  136. {
  137. TDeque<ui16> deq;
  138. deq.push_back(1);
  139. deq.push_back(2);
  140. deq.push_back(4);
  141. deq.push_back(5);
  142. Save(&S_, deq);
  143. }
  144. {
  145. TMaybe<size_t> h(10);
  146. Save(&S_, h);
  147. }
  148. {
  149. TMaybe<size_t> h(20);
  150. Save(&S_, h);
  151. }
  152. {
  153. TMaybe<size_t> h;
  154. Save(&S_, h);
  155. }
  156. {
  157. TMaybe<size_t> h;
  158. Save(&S_, h);
  159. }
  160. {
  161. THashMultiMap<TString, int> mm;
  162. mm.insert({"one", 1});
  163. mm.insert({"two", 2});
  164. mm.insert({"two", 22});
  165. Save(&S_, mm);
  166. }
  167. //load part
  168. {
  169. ui8 val;
  170. Load(&S_, val);
  171. UNIT_ASSERT_EQUAL(val, 1);
  172. }
  173. {
  174. ui16 val;
  175. Load(&S_, val);
  176. UNIT_ASSERT_EQUAL(val, 2);
  177. }
  178. {
  179. ui32 val;
  180. Load(&S_, val);
  181. UNIT_ASSERT_EQUAL(val, 3);
  182. }
  183. {
  184. ui64 val;
  185. Load(&S_, val);
  186. UNIT_ASSERT_EQUAL(val, 4);
  187. }
  188. {
  189. TVector<ui16> vec;
  190. Load(&S_, vec);
  191. UNIT_ASSERT_EQUAL(vec.size(), 3);
  192. UNIT_ASSERT_EQUAL(vec[0], 1);
  193. UNIT_ASSERT_EQUAL(vec[1], 2);
  194. UNIT_ASSERT_EQUAL(vec[2], 4);
  195. }
  196. {
  197. TMap<ui16, ui32> map;
  198. Load(&S_, map);
  199. UNIT_ASSERT_EQUAL(map.size(), 3);
  200. UNIT_ASSERT_EQUAL(map[(ui16)1], 2);
  201. UNIT_ASSERT_EQUAL(map[(ui16)2], 3);
  202. UNIT_ASSERT_EQUAL(map[(ui16)3], 4);
  203. }
  204. {
  205. TMultiMap<ui16, ui32> multimap;
  206. Load(&S_, multimap);
  207. UNIT_ASSERT_EQUAL(multimap.size(), 5);
  208. UNIT_ASSERT_EQUAL(multimap.find((ui16)1)->second, 2);
  209. UNIT_ASSERT_EQUAL(multimap.find((ui16)3)->second, 6);
  210. THashSet<ui32> values;
  211. auto range = multimap.equal_range((ui16)2);
  212. for (auto i = range.first; i != range.second; ++i) {
  213. values.insert(i->second);
  214. }
  215. UNIT_ASSERT_EQUAL(values.size(), 3);
  216. UNIT_ASSERT_EQUAL(values.contains(3), true);
  217. UNIT_ASSERT_EQUAL(values.contains(4), true);
  218. UNIT_ASSERT_EQUAL(values.contains(5), true);
  219. }
  220. {
  221. TSaveHelper helper;
  222. Load(&S_, helper);
  223. }
  224. {
  225. TString val;
  226. Load(&S_, val);
  227. UNIT_ASSERT_EQUAL(val, "123456");
  228. }
  229. {
  230. TBuffer buf;
  231. Load(&S_, buf);
  232. UNIT_ASSERT_EQUAL(buf.size(), 4);
  233. UNIT_ASSERT_EQUAL(memcmp(buf.data(), "asdf", 4), 0);
  234. }
  235. {
  236. TVector<const char*> vec;
  237. TMemoryPool pool(1024);
  238. Load(&S_, vec, pool);
  239. UNIT_ASSERT_EQUAL(vec.size(), 3);
  240. UNIT_ASSERT_EQUAL(vec[0], TString("1"));
  241. UNIT_ASSERT_EQUAL(vec[1], TString("123"));
  242. UNIT_ASSERT_EQUAL(vec[2], TString("4567"));
  243. }
  244. {
  245. TDeque<ui16> deq;
  246. Load(&S_, deq);
  247. UNIT_ASSERT_EQUAL(deq.size(), 4);
  248. UNIT_ASSERT_EQUAL(deq[0], 1);
  249. UNIT_ASSERT_EQUAL(deq[1], 2);
  250. UNIT_ASSERT_EQUAL(deq[2], 4);
  251. UNIT_ASSERT_EQUAL(deq[3], 5);
  252. }
  253. {
  254. TMaybe<size_t> h(5);
  255. Load(&S_, h);
  256. UNIT_ASSERT_EQUAL(*h, 10);
  257. }
  258. {
  259. TMaybe<size_t> h;
  260. Load(&S_, h);
  261. UNIT_ASSERT_EQUAL(*h, 20);
  262. }
  263. {
  264. TMaybe<size_t> h;
  265. UNIT_ASSERT(!h);
  266. Load(&S_, h);
  267. UNIT_ASSERT(!h);
  268. }
  269. {
  270. TMaybe<size_t> h(7);
  271. UNIT_ASSERT(!!h);
  272. Load(&S_, h);
  273. UNIT_ASSERT(!h);
  274. }
  275. {
  276. THashMultiMap<TString, int> mm;
  277. Load(&S_, mm);
  278. UNIT_ASSERT_EQUAL(mm.size(), 3);
  279. UNIT_ASSERT_EQUAL(mm.count("one"), 1);
  280. auto oneIter = mm.equal_range("one").first;
  281. UNIT_ASSERT_EQUAL(oneIter->second, 1);
  282. UNIT_ASSERT_EQUAL(mm.count("two"), 2);
  283. auto twoIter = mm.equal_range("two").first;
  284. UNIT_ASSERT_EQUAL(twoIter->second, 2);
  285. UNIT_ASSERT_EQUAL((++twoIter)->second, 22);
  286. }
  287. }
  288. void TestList() {
  289. TBufferStream s;
  290. TList<int> list = {0, 1, 10};
  291. Save(&s, list);
  292. list.clear();
  293. Load(&s, list);
  294. UNIT_ASSERT_VALUES_EQUAL(list.size(), 3);
  295. UNIT_ASSERT_VALUES_EQUAL(*std::next(list.begin(), 0), 0);
  296. UNIT_ASSERT_VALUES_EQUAL(*std::next(list.begin(), 1), 1);
  297. UNIT_ASSERT_VALUES_EQUAL(*std::next(list.begin(), 2), 10);
  298. }
  299. void TestTuple() {
  300. TBufferStream s;
  301. using TTuple = std::tuple<int, TString, unsigned int>;
  302. const TTuple toSave{-10, "qwerty", 15};
  303. Save(&s, toSave);
  304. TTuple toLoad;
  305. Load(&s, toLoad);
  306. UNIT_ASSERT_VALUES_EQUAL(std::get<0>(toLoad), std::get<0>(toSave));
  307. UNIT_ASSERT_VALUES_EQUAL(std::get<1>(toLoad), std::get<1>(toSave));
  308. UNIT_ASSERT_VALUES_EQUAL(std::get<2>(toLoad), std::get<2>(toSave));
  309. }
  310. template <class TVariant, class T>
  311. void TestVariantImpl(TVariant& v, const T& expected) {
  312. v = expected;
  313. TBufferStream s;
  314. ::Save(&s, v);
  315. ::Load(&s, v);
  316. UNIT_ASSERT_VALUES_EQUAL(std::get<T>(v), expected);
  317. }
  318. void TestVariant() {
  319. std::variant<int, bool, TString, TVector<char>> v(1);
  320. TestVariantImpl(v, 42);
  321. TestVariantImpl(v, true);
  322. TestVariantImpl(v, TString("foo"));
  323. TestVariantImpl(v, TVector<char>{'b', 'a', 'r'});
  324. v = TString("baz");
  325. TBufferStream s;
  326. ::Save(&s, v);
  327. std::variant<char, bool> v2 = false;
  328. UNIT_ASSERT_EXCEPTION(::Load(&s, v2), TLoadEOF);
  329. }
  330. // tests serialization of class with three public string members
  331. template <class TDerived, class TInterface = TDerived>
  332. void TestInheritClassImpl() {
  333. TBufferStream s;
  334. {
  335. TDerived v1;
  336. v1.Str1 = "One";
  337. v1.Str2 = "Two";
  338. v1.Str3 = "Three";
  339. ::Save(&s, static_cast<const TInterface&>(v1));
  340. }
  341. {
  342. TDerived v2;
  343. ::Load(&s, static_cast<TInterface&>(v2));
  344. UNIT_ASSERT_VALUES_EQUAL_C(v2.Str1, "One", TypeName<TDerived>() << " via " << TypeName<TInterface>());
  345. UNIT_ASSERT_VALUES_EQUAL_C(v2.Str2, "Two", TypeName<TDerived>() << " via " << TypeName<TInterface>());
  346. UNIT_ASSERT_VALUES_EQUAL_C(v2.Str3, "Three", TypeName<TDerived>() << " via " << TypeName<TInterface>());
  347. }
  348. }
  349. void TestInheritNonVirtualClass() {
  350. struct TBaseNonVirtual {
  351. TString Str1;
  352. Y_SAVELOAD_DEFINE(Str1);
  353. };
  354. struct TDerivedNonVirtual: TBaseNonVirtual {
  355. TString Str2;
  356. TString Str3;
  357. Y_SAVELOAD_DEFINE(TNonVirtualSaver<TBaseNonVirtual>{this}, Str2, Str3);
  358. };
  359. TestInheritClassImpl<TDerivedNonVirtual>();
  360. }
  361. void TestInheritVirtualClass() {
  362. struct IInterface {
  363. virtual void Save(IOutputStream* out) const = 0;
  364. virtual void Load(IInputStream* in) = 0;
  365. };
  366. struct TBaseVirtual: IInterface {
  367. TString Str1;
  368. Y_SAVELOAD_DEFINE_OVERRIDE(Str1);
  369. };
  370. struct TDerivedVirtual: TBaseVirtual {
  371. TString Str2;
  372. TString Str3;
  373. Y_SAVELOAD_DEFINE_OVERRIDE(TNonVirtualSaver<TBaseVirtual>{this}, Str2, Str3);
  374. };
  375. TestInheritClassImpl<TDerivedVirtual>();
  376. TestInheritClassImpl<TDerivedVirtual, TBaseVirtual>();
  377. TestInheritClassImpl<TDerivedVirtual, IInterface>();
  378. }
  379. };
  380. UNIT_TEST_SUITE_REGISTRATION(TSaveLoadTest);