TestHashTable.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. /*
  2. * Copyright (c) 2021, thislooksfun <tlf@thislooks.fun>
  3. * Copyright (c) 2023, Jelle Raaijmakers <jelle@gmta.nl>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <LibTest/TestCase.h>
  8. #include <AK/DeprecatedString.h>
  9. #include <AK/HashTable.h>
  10. #include <AK/NonnullOwnPtr.h>
  11. #include <AK/Vector.h>
  12. TEST_CASE(construct)
  13. {
  14. using IntTable = HashTable<int>;
  15. EXPECT(IntTable().is_empty());
  16. EXPECT_EQ(IntTable().size(), 0u);
  17. }
  18. TEST_CASE(basic_move)
  19. {
  20. HashTable<int> foo;
  21. foo.set(1);
  22. EXPECT_EQ(foo.size(), 1u);
  23. auto bar = move(foo);
  24. EXPECT_EQ(bar.size(), 1u);
  25. EXPECT_EQ(foo.size(), 0u);
  26. foo = move(bar);
  27. EXPECT_EQ(bar.size(), 0u);
  28. EXPECT_EQ(foo.size(), 1u);
  29. }
  30. TEST_CASE(move_is_not_swap)
  31. {
  32. HashTable<int> foo;
  33. foo.set(1);
  34. HashTable<int> bar;
  35. bar.set(2);
  36. foo = move(bar);
  37. EXPECT(foo.contains(2));
  38. EXPECT(!bar.contains(1));
  39. EXPECT_EQ(bar.size(), 0u);
  40. }
  41. TEST_CASE(populate)
  42. {
  43. HashTable<DeprecatedString> strings;
  44. strings.set("One");
  45. strings.set("Two");
  46. strings.set("Three");
  47. EXPECT_EQ(strings.is_empty(), false);
  48. EXPECT_EQ(strings.size(), 3u);
  49. }
  50. TEST_CASE(range_loop)
  51. {
  52. HashTable<DeprecatedString> strings;
  53. EXPECT_EQ(strings.set("One"), AK::HashSetResult::InsertedNewEntry);
  54. EXPECT_EQ(strings.set("Two"), AK::HashSetResult::InsertedNewEntry);
  55. EXPECT_EQ(strings.set("Three"), AK::HashSetResult::InsertedNewEntry);
  56. int loop_counter = 0;
  57. for (auto& it : strings) {
  58. EXPECT_EQ(it.is_empty(), false);
  59. ++loop_counter;
  60. }
  61. EXPECT_EQ(loop_counter, 3);
  62. }
  63. TEST_CASE(range_loop_reverse)
  64. {
  65. Array strings = { "One"sv, "Two"sv, "Three"sv };
  66. OrderedHashTable<DeprecatedString> table;
  67. EXPECT_EQ(table.set(strings[0]), AK::HashSetResult::InsertedNewEntry);
  68. EXPECT_EQ(table.set(strings[1]), AK::HashSetResult::InsertedNewEntry);
  69. EXPECT_EQ(table.set(strings[2]), AK::HashSetResult::InsertedNewEntry);
  70. int loop_counter = 0;
  71. int index = strings.size() - 1;
  72. for (auto& it : table.in_reverse()) {
  73. EXPECT_EQ(it, strings[index]);
  74. ++loop_counter;
  75. --index;
  76. }
  77. EXPECT_EQ(loop_counter, 3);
  78. }
  79. TEST_CASE(table_remove)
  80. {
  81. HashTable<DeprecatedString> strings;
  82. EXPECT_EQ(strings.set("One"), AK::HashSetResult::InsertedNewEntry);
  83. EXPECT_EQ(strings.set("Two"), AK::HashSetResult::InsertedNewEntry);
  84. EXPECT_EQ(strings.set("Three"), AK::HashSetResult::InsertedNewEntry);
  85. EXPECT_EQ(strings.remove("One"), true);
  86. EXPECT_EQ(strings.size(), 2u);
  87. EXPECT(strings.find("One") == strings.end());
  88. EXPECT_EQ(strings.remove("Three"), true);
  89. EXPECT_EQ(strings.size(), 1u);
  90. EXPECT(strings.find("Three") == strings.end());
  91. EXPECT(strings.find("Two") != strings.end());
  92. }
  93. TEST_CASE(remove_all_matching)
  94. {
  95. HashTable<int> ints;
  96. ints.set(1);
  97. ints.set(2);
  98. ints.set(3);
  99. ints.set(4);
  100. EXPECT_EQ(ints.size(), 4u);
  101. EXPECT_EQ(ints.remove_all_matching([&](int value) { return value > 2; }), true);
  102. EXPECT_EQ(ints.remove_all_matching([&](int) { return false; }), false);
  103. EXPECT_EQ(ints.size(), 2u);
  104. EXPECT(ints.contains(1));
  105. EXPECT(ints.contains(2));
  106. EXPECT_EQ(ints.remove_all_matching([&](int) { return true; }), true);
  107. EXPECT(ints.is_empty());
  108. EXPECT_EQ(ints.remove_all_matching([&](int) { return true; }), false);
  109. }
  110. TEST_CASE(case_insensitive)
  111. {
  112. HashTable<DeprecatedString, CaseInsensitiveStringTraits> casetable;
  113. EXPECT_EQ(DeprecatedString("nickserv").to_lowercase(), DeprecatedString("NickServ").to_lowercase());
  114. EXPECT_EQ(casetable.set("nickserv"), AK::HashSetResult::InsertedNewEntry);
  115. EXPECT_EQ(casetable.set("NickServ"), AK::HashSetResult::ReplacedExistingEntry);
  116. EXPECT_EQ(casetable.size(), 1u);
  117. }
  118. TEST_CASE(many_strings)
  119. {
  120. HashTable<DeprecatedString> strings;
  121. for (int i = 0; i < 999; ++i) {
  122. EXPECT_EQ(strings.set(DeprecatedString::number(i)), AK::HashSetResult::InsertedNewEntry);
  123. }
  124. EXPECT_EQ(strings.size(), 999u);
  125. for (int i = 0; i < 999; ++i) {
  126. EXPECT_EQ(strings.remove(DeprecatedString::number(i)), true);
  127. }
  128. EXPECT_EQ(strings.is_empty(), true);
  129. }
  130. TEST_CASE(many_collisions)
  131. {
  132. struct StringCollisionTraits : public GenericTraits<DeprecatedString> {
  133. static unsigned hash(DeprecatedString const&) { return 0; }
  134. };
  135. HashTable<DeprecatedString, StringCollisionTraits> strings;
  136. for (int i = 0; i < 999; ++i) {
  137. EXPECT_EQ(strings.set(DeprecatedString::number(i)), AK::HashSetResult::InsertedNewEntry);
  138. }
  139. EXPECT_EQ(strings.set("foo"), AK::HashSetResult::InsertedNewEntry);
  140. EXPECT_EQ(strings.size(), 1000u);
  141. for (int i = 0; i < 999; ++i) {
  142. EXPECT_EQ(strings.remove(DeprecatedString::number(i)), true);
  143. }
  144. EXPECT(strings.find("foo") != strings.end());
  145. }
  146. TEST_CASE(space_reuse)
  147. {
  148. struct StringCollisionTraits : public GenericTraits<DeprecatedString> {
  149. static unsigned hash(DeprecatedString const&) { return 0; }
  150. };
  151. HashTable<DeprecatedString, StringCollisionTraits> strings;
  152. // Add a few items to allow it to do initial resizing.
  153. EXPECT_EQ(strings.set("0"), AK::HashSetResult::InsertedNewEntry);
  154. for (int i = 1; i < 5; ++i) {
  155. EXPECT_EQ(strings.set(DeprecatedString::number(i)), AK::HashSetResult::InsertedNewEntry);
  156. EXPECT_EQ(strings.remove(DeprecatedString::number(i - 1)), true);
  157. }
  158. auto capacity = strings.capacity();
  159. for (int i = 5; i < 999; ++i) {
  160. EXPECT_EQ(strings.set(DeprecatedString::number(i)), AK::HashSetResult::InsertedNewEntry);
  161. EXPECT_EQ(strings.remove(DeprecatedString::number(i - 1)), true);
  162. }
  163. EXPECT_EQ(strings.capacity(), capacity);
  164. }
  165. TEST_CASE(basic_remove)
  166. {
  167. HashTable<int> table;
  168. table.set(1);
  169. table.set(2);
  170. table.set(3);
  171. EXPECT_EQ(table.remove(3), true);
  172. EXPECT_EQ(table.remove(3), false);
  173. EXPECT_EQ(table.size(), 2u);
  174. EXPECT_EQ(table.remove(1), true);
  175. EXPECT_EQ(table.remove(1), false);
  176. EXPECT_EQ(table.size(), 1u);
  177. EXPECT_EQ(table.remove(2), true);
  178. EXPECT_EQ(table.remove(2), false);
  179. EXPECT_EQ(table.size(), 0u);
  180. }
  181. TEST_CASE(basic_contains)
  182. {
  183. HashTable<int> table;
  184. table.set(1);
  185. table.set(2);
  186. table.set(3);
  187. EXPECT_EQ(table.contains(1), true);
  188. EXPECT_EQ(table.contains(2), true);
  189. EXPECT_EQ(table.contains(3), true);
  190. EXPECT_EQ(table.contains(4), false);
  191. EXPECT_EQ(table.remove(3), true);
  192. EXPECT_EQ(table.contains(3), false);
  193. EXPECT_EQ(table.contains(1), true);
  194. EXPECT_EQ(table.contains(2), true);
  195. EXPECT_EQ(table.remove(2), true);
  196. EXPECT_EQ(table.contains(2), false);
  197. EXPECT_EQ(table.contains(3), false);
  198. EXPECT_EQ(table.contains(1), true);
  199. EXPECT_EQ(table.remove(1), true);
  200. EXPECT_EQ(table.contains(1), false);
  201. }
  202. TEST_CASE(capacity_leak)
  203. {
  204. HashTable<int> table;
  205. for (size_t i = 0; i < 10000; ++i) {
  206. table.set(i);
  207. table.remove(i);
  208. }
  209. EXPECT(table.capacity() < 100u);
  210. }
  211. TEST_CASE(non_trivial_type_table)
  212. {
  213. HashTable<NonnullOwnPtr<int>> table;
  214. table.set(make<int>(3));
  215. table.set(make<int>(11));
  216. for (int i = 0; i < 1'000; ++i) {
  217. table.set(make<int>(-i));
  218. }
  219. for (int i = 0; i < 10'000; ++i) {
  220. table.set(make<int>(i));
  221. table.remove(make<int>(i));
  222. }
  223. EXPECT_EQ(table.remove_all_matching([&](auto&) { return true; }), true);
  224. EXPECT(table.is_empty());
  225. EXPECT_EQ(table.remove_all_matching([&](auto&) { return true; }), false);
  226. }
  227. TEST_CASE(floats)
  228. {
  229. HashTable<float> table;
  230. table.set(0);
  231. table.set(1.0f);
  232. table.set(2.0f);
  233. EXPECT_EQ(table.size(), 3u);
  234. EXPECT(table.contains(0));
  235. EXPECT(table.contains(1.0f));
  236. EXPECT(table.contains(2.0f));
  237. }
  238. TEST_CASE(doubles)
  239. {
  240. HashTable<double> table;
  241. table.set(0);
  242. table.set(1.0);
  243. table.set(2.0);
  244. EXPECT_EQ(table.size(), 3u);
  245. EXPECT(table.contains(0));
  246. EXPECT(table.contains(1.0));
  247. EXPECT(table.contains(2.0));
  248. }
  249. TEST_CASE(reinsertion)
  250. {
  251. OrderedHashTable<DeprecatedString> map;
  252. map.set("ytidb::LAST_RESULT_ENTRY_KEY");
  253. map.set("__sak");
  254. map.remove("__sak");
  255. map.set("__sak");
  256. }
  257. TEST_CASE(clear_with_capacity_when_empty)
  258. {
  259. HashTable<int> map;
  260. map.clear_with_capacity();
  261. map.set(0);
  262. map.set(1);
  263. VERIFY(map.size() == 2);
  264. }
  265. TEST_CASE(iterator_removal)
  266. {
  267. HashTable<int> map;
  268. map.set(0);
  269. map.set(1);
  270. auto it = map.begin();
  271. map.remove(it);
  272. EXPECT_EQ(it, map.end());
  273. EXPECT_EQ(map.size(), 1u);
  274. }
  275. TEST_CASE(ordered_insertion_and_deletion)
  276. {
  277. OrderedHashTable<int> table;
  278. EXPECT_EQ(table.set(0), HashSetResult::InsertedNewEntry);
  279. EXPECT_EQ(table.set(1), HashSetResult::InsertedNewEntry);
  280. EXPECT_EQ(table.set(2), HashSetResult::InsertedNewEntry);
  281. EXPECT_EQ(table.set(3), HashSetResult::InsertedNewEntry);
  282. EXPECT_EQ(table.size(), 4u);
  283. auto expect_table = [](OrderedHashTable<int>& table, Span<int> values) {
  284. auto index = 0u;
  285. for (auto it = table.begin(); it != table.end(); ++it, ++index) {
  286. EXPECT_EQ(*it, values[index]);
  287. EXPECT(table.contains(values[index]));
  288. }
  289. index = table.size() - 1;
  290. for (auto it = table.rbegin(); it != table.rend(); ++it, --index) {
  291. EXPECT_EQ(*it, values[index]);
  292. EXPECT(table.contains(values[index]));
  293. }
  294. };
  295. expect_table(table, Array<int, 4> { 0, 1, 2, 3 });
  296. EXPECT(table.remove(0));
  297. EXPECT(table.remove(2));
  298. EXPECT(!table.remove(4));
  299. EXPECT_EQ(table.size(), 2u);
  300. expect_table(table, Array<int, 2> { 1, 3 });
  301. }
  302. TEST_CASE(ordered_deletion_and_reinsertion)
  303. {
  304. OrderedHashTable<int> table;
  305. table.set(1);
  306. table.set(3);
  307. table.remove(1);
  308. EXPECT_EQ(table.size(), 1u);
  309. // By adding 1 again but this time in a different position, we
  310. // test whether the bucket's neighbors are reset properly.
  311. table.set(1);
  312. EXPECT_EQ(table.size(), 2u);
  313. auto it = table.begin();
  314. EXPECT_EQ(*it, 3);
  315. ++it;
  316. EXPECT_EQ(*it, 1);
  317. ++it;
  318. EXPECT_EQ(it, table.end());
  319. auto rit = table.rbegin();
  320. EXPECT_EQ(*rit, 1);
  321. ++rit;
  322. EXPECT_EQ(*rit, 3);
  323. ++rit;
  324. EXPECT_EQ(rit, table.rend());
  325. }
  326. TEST_CASE(ordered_take_last)
  327. {
  328. OrderedHashTable<int> table;
  329. table.set(1);
  330. table.set(2);
  331. table.set(3);
  332. EXPECT_EQ(table.take_last(), 3);
  333. EXPECT_EQ(table.take_last(), 2);
  334. EXPECT_EQ(table.take_last(), 1);
  335. EXPECT(table.is_empty());
  336. }
  337. TEST_CASE(ordered_iterator_removal)
  338. {
  339. OrderedHashTable<int> map;
  340. map.set(0);
  341. map.set(1);
  342. auto it = map.begin();
  343. map.remove(it);
  344. EXPECT_EQ(it, map.end());
  345. EXPECT_EQ(map.size(), 1u);
  346. }
  347. TEST_CASE(ordered_remove_from_head)
  348. {
  349. OrderedHashTable<int> map;
  350. map.set(1);
  351. map.set(2);
  352. map.set(3);
  353. map.set(4);
  354. map.set(5);
  355. map.set(6);
  356. EXPECT_EQ(map.size(), 6u);
  357. auto it = map.begin();
  358. map.remove(it);
  359. it = map.begin();
  360. map.remove(it);
  361. it = map.begin();
  362. map.remove(it);
  363. it = map.begin();
  364. map.remove(it);
  365. it = map.begin();
  366. map.remove(it);
  367. it = map.begin();
  368. map.remove(it);
  369. EXPECT_EQ(map.size(), 0u);
  370. }
  371. TEST_CASE(ordered_infinite_loop_clang_regression)
  372. {
  373. OrderedHashTable<DeprecatedString> map;
  374. map.set("");
  375. map.set("1");
  376. map.set("_cb");
  377. map.set("2");
  378. map.set("3");
  379. map.set("_cb_svref");
  380. map.set("_cb_svref_expires");
  381. map.remove("_cb_svref");
  382. map.remove("_cb_svref_expires");
  383. map.set("_cb_svref");
  384. size_t iterations = 0;
  385. auto size = map.size();
  386. for (auto it = map.begin(); it != map.end(); ++it) {
  387. if (++iterations > size) {
  388. VERIFY(false);
  389. break;
  390. }
  391. }
  392. }
  393. TEST_CASE(values)
  394. {
  395. OrderedHashTable<int> table;
  396. table.set(10);
  397. table.set(30);
  398. table.set(20);
  399. Vector<int> values = table.values();
  400. EXPECT_EQ(values.size(), table.size());
  401. EXPECT_EQ(values[0], 10);
  402. EXPECT_EQ(values[1], 30);
  403. EXPECT_EQ(values[2], 20);
  404. }