rbt_datasrc_unittest.cc 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361
  1. // Copyright (C) 2010 Internet Systems Consortium, Inc. ("ISC")
  2. //
  3. // Permission to use, copy, modify, and/or distribute this software for any
  4. // purpose with or without fee is hereby granted, provided that the above
  5. // copyright notice and this permission notice appear in all copies.
  6. //
  7. // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
  8. // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
  9. // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
  10. // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  11. // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
  12. // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  13. // PERFORMANCE OF THIS SOFTWARE.
  14. #include <gtest/gtest.h>
  15. #include <dns/name.h>
  16. #include <dns/rrclass.h>
  17. #include <dns/rrset.h>
  18. #include <dns/rrtype.h>
  19. #include <dns/rrttl.h>
  20. #include <auth/rbt_datasrc.h>
  21. #include <dns/tests/unittest_util.h>
  22. using namespace std;
  23. using isc::UnitTestUtil;
  24. using namespace isc::datasrc;
  25. /* The initial structure of rbtree
  26. *
  27. * b
  28. * / \
  29. * a d.e.f
  30. * / | \
  31. * c | g.h
  32. * | |
  33. * w.y i
  34. * / | \
  35. * x | z
  36. * | |
  37. * p j
  38. * / \
  39. * o q
  40. */
  41. namespace {
  42. class RBTreeTest : public::testing::Test {
  43. protected:
  44. RBTreeTest() : rbtree() {
  45. rbtree.insert(Name("a"), &rbtnode);
  46. rbtree.insert(Name("b"), &rbtnode);
  47. rbtree.insert(Name("c"), &rbtnode);
  48. rbtree.insert(Name("x.d.e.f"), &rbtnode);
  49. rbtree.insert(Name("z.d.e.f"), &rbtnode);
  50. rbtree.insert(Name("g.h"), &rbtnode);
  51. rbtree.insert(Name("i.g.h"), &rbtnode);
  52. rbtree.insert(Name("o.w.y.d.e.f"), &rbtnode);
  53. rbtree.insert(Name("j.z.d.e.f"), &rbtnode);
  54. rbtree.insert(Name("p.w.y.d.e.f"), &rbtnode);
  55. rbtree.insert(Name("q.w.y.d.e.f"), &rbtnode);
  56. }
  57. RBTree<int> rbtree;
  58. RBNode<int> *rbtnode;
  59. };
  60. TEST_F(RBTreeTest, getNodeCount) {
  61. EXPECT_EQ(13, rbtree.getNodeCount());
  62. }
  63. TEST_F(RBTreeTest, insertNames) {
  64. // a node is considered to "formally" exist only if it has data
  65. // associated with it
  66. // return 0, since node "d.e.f" doesn't have data
  67. EXPECT_EQ(0, rbtree.insert(Name("d.e.f"), &rbtnode));
  68. EXPECT_EQ(Name("d.e.f"), rbtnode->getName());
  69. EXPECT_EQ(13, rbtree.getNodeCount());
  70. EXPECT_EQ(0, rbtree.insert(Name("."), &rbtnode));
  71. EXPECT_EQ(Name("."), rbtnode->getName());
  72. EXPECT_EQ(14, rbtree.getNodeCount());
  73. EXPECT_EQ(0, rbtree.insert(Name("example.com"), &rbtnode));
  74. EXPECT_EQ(15, rbtree.getNodeCount());
  75. // return 1, since node "example.com" already has data associated with it
  76. int data = 10;
  77. rbtnode->setData(data);
  78. EXPECT_EQ(1, rbtree.insert(Name("example.com"), &rbtnode));
  79. EXPECT_EQ(15, rbtree.getNodeCount());
  80. // split the node "d.e.f"
  81. EXPECT_EQ(0, rbtree.insert(Name("k.e.f"), &rbtnode));
  82. EXPECT_EQ(Name("k"), rbtnode->getName());
  83. EXPECT_EQ(17, rbtree.getNodeCount());
  84. // split the node "g.h"
  85. EXPECT_EQ(0, rbtree.insert(Name("h"), &rbtnode));
  86. EXPECT_EQ(Name("h"), rbtnode->getName());
  87. EXPECT_EQ(18, rbtree.getNodeCount());
  88. // add child domain
  89. EXPECT_EQ(0, rbtree.insert(Name("m.p.w.y.d.e.f"), &rbtnode));
  90. EXPECT_EQ(Name("m"), rbtnode->getName());
  91. EXPECT_EQ(19, rbtree.getNodeCount());
  92. EXPECT_EQ(0, rbtree.insert(Name("n.p.w.y.d.e.f"), &rbtnode));
  93. EXPECT_EQ(Name("n"), rbtnode->getName());
  94. EXPECT_EQ(20, rbtree.getNodeCount());
  95. EXPECT_EQ(0, rbtree.insert(Name("l.a"), &rbtnode));
  96. EXPECT_EQ(Name("l"), rbtnode->getName());
  97. EXPECT_EQ(21, rbtree.getNodeCount());
  98. EXPECT_EQ(0, rbtree.insert(Name("r.d.e.f"), &rbtnode));
  99. EXPECT_EQ(0, rbtree.insert(Name("s.d.e.f"), &rbtnode));
  100. EXPECT_EQ(23, rbtree.getNodeCount());
  101. }
  102. TEST_F(RBTreeTest, findName) {
  103. // exact match
  104. EXPECT_EQ(RBTree<int>::EXACTMATCH, rbtree.find(Name("a"), &rbtnode));
  105. EXPECT_EQ(Name("a"), rbtnode->getName());
  106. // not found
  107. EXPECT_EQ(RBTree<int>::NOTFOUND, rbtree.find(Name("d.e.f"), &rbtnode));
  108. EXPECT_EQ(RBTree<int>::NOTFOUND, rbtree.find(Name("x"), &rbtnode));
  109. EXPECT_EQ(RBTree<int>::NOTFOUND, rbtree.find(Name("m.n"), &rbtnode));
  110. // partial match
  111. EXPECT_EQ(RBTree<int>::PARTIALMATCH, rbtree.find(Name("m.b"), &rbtnode));
  112. EXPECT_EQ(Name("b"), rbtnode->getName());
  113. }
  114. TEST_F(RBTreeTest, successor) {
  115. // traverse the trees
  116. EXPECT_EQ(RBTree<int>::EXACTMATCH, rbtree.find(Name("a"), &rbtnode));
  117. RBNode<int> *successor_node = rbtnode->successor();
  118. EXPECT_EQ(Name("b"), successor_node->getName());
  119. successor_node = successor_node->successor();
  120. EXPECT_EQ(Name("c"), successor_node->getName());
  121. successor_node = successor_node->successor();
  122. EXPECT_EQ(Name("d.e.f"), successor_node->getName());
  123. successor_node = successor_node->successor();
  124. EXPECT_EQ(Name("g.h"), successor_node->getName());
  125. successor_node = successor_node->successor();
  126. EXPECT_EQ(RBTree<int>::EXACTMATCH, rbtree.find(Name("x.d.e.f"), &rbtnode));
  127. EXPECT_EQ(Name("x"), rbtnode->getName());
  128. successor_node = rbtnode->successor();
  129. EXPECT_EQ(Name("w.y"), successor_node->getName());
  130. successor_node = successor_node->successor();
  131. EXPECT_EQ(Name("z"), successor_node->getName());
  132. EXPECT_EQ(RBTree<int>::EXACTMATCH, rbtree.find(Name("o.w.y.d.e.f"), &rbtnode));
  133. EXPECT_EQ(Name("o"), rbtnode->getName());
  134. successor_node = rbtnode->successor();
  135. EXPECT_EQ(Name("p"), successor_node->getName());
  136. successor_node = successor_node->successor();
  137. EXPECT_EQ(Name("q"), successor_node->getName());
  138. }
  139. }
  140. TEST_F(RBTreeTest, eraseName) {
  141. EXPECT_EQ(0, rbtree.insert(Name("k"), &rbtnode));
  142. EXPECT_EQ(0, rbtree.insert(Name("r.d.e.f"), &rbtnode));
  143. EXPECT_EQ(0, rbtree.insert(Name("s.d.e.f"), &rbtnode));
  144. EXPECT_EQ(0, rbtree.insert(Name("y"), &rbtnode));
  145. EXPECT_EQ(17, rbtree.getNodeCount());
  146. /*
  147. * b
  148. * / \
  149. * a d.e.f
  150. * / | \
  151. * c | k
  152. * | / \
  153. * w.y g.h y
  154. * / | \ |
  155. * s | z i
  156. * / \ | |
  157. * r x p j
  158. * / \
  159. * o q
  160. */
  161. EXPECT_EQ(0, rbtree.erase(Name("a")));
  162. EXPECT_EQ(0, rbtree.insert(Name("a"), &rbtnode));
  163. EXPECT_EQ(0, rbtree.erase(Name("k")));
  164. EXPECT_EQ(0, rbtree.erase(Name("y")));
  165. // can't delete non terminal
  166. EXPECT_EQ(1, rbtree.erase(Name("d.e.f")));
  167. EXPECT_EQ(RBTree<int>::NOTFOUND, rbtree.find(Name("w.y.d.e.f"), &rbtnode));
  168. EXPECT_EQ(0, rbtree.erase(Name("p.w.y.d.e.f")));
  169. EXPECT_EQ(14, rbtree.getNodeCount());
  170. EXPECT_EQ(RBTree<int>::NOTFOUND, rbtree.find(Name("p.w.y.d.e.f"), &rbtnode));
  171. EXPECT_EQ(0, rbtree.erase(Name("q.w.y.d.e.f")));
  172. EXPECT_EQ(12, rbtree.getNodeCount());
  173. EXPECT_EQ(RBTree<int>::NOTFOUND, rbtree.find(Name("q.w.y.d.e.f"), &rbtnode));
  174. // o would not be rejoined with w.y if w.y had data
  175. // associated with the key
  176. EXPECT_EQ(RBTree<int>::EXACTMATCH, rbtree.find(Name("o.w.y.d.e.f"), &rbtnode));
  177. EXPECT_EQ(RBTree<int>::NOTFOUND, rbtree.find(Name("w.y.d.e.f"), &rbtnode));
  178. /*
  179. * d.e.f
  180. * / | \
  181. * b | g.h
  182. * / \ | |
  183. * a c o.w.y i
  184. * / \
  185. * s z
  186. * / \ |
  187. * r x j
  188. */
  189. EXPECT_EQ(0, rbtree.erase(Name("o.w.y.d.e.f")));
  190. EXPECT_EQ(11, rbtree.getNodeCount());
  191. EXPECT_EQ(1, rbtree.erase(Name("w.y.d.e.f")));
  192. EXPECT_EQ(11, rbtree.getNodeCount());
  193. EXPECT_EQ(0, rbtree.erase(Name("x.d.e.f")));
  194. EXPECT_EQ(10, rbtree.getNodeCount());
  195. /*
  196. * d.e.f
  197. * / | \
  198. * b | g.h
  199. * / \ | |
  200. * a c s i
  201. * / \
  202. * r z
  203. * |
  204. * j
  205. */
  206. // erase a non-exist node
  207. EXPECT_EQ(1, rbtree.erase(Name("x.d.e.f")));
  208. // delete all the nodes one by one
  209. EXPECT_EQ(0, rbtree.erase(Name("c")));
  210. EXPECT_EQ(9, rbtree.getNodeCount());
  211. EXPECT_EQ(1, rbtree.erase(Name("g.h")));
  212. EXPECT_EQ(9, rbtree.getNodeCount());
  213. EXPECT_EQ(0, rbtree.erase(Name("a")));
  214. EXPECT_EQ(8, rbtree.getNodeCount());
  215. EXPECT_EQ(0, rbtree.erase(Name("b")));
  216. EXPECT_EQ(7, rbtree.getNodeCount());
  217. EXPECT_EQ(0, rbtree.erase(Name("i.g.h")));
  218. EXPECT_EQ(6, rbtree.getNodeCount());
  219. /*
  220. * d.e.f
  221. * | \
  222. * | g.h
  223. * |
  224. * s
  225. * / \
  226. * r z
  227. * |
  228. * j
  229. */
  230. // can't delete non-terminal node
  231. EXPECT_EQ(1, rbtree.erase(Name("z.d.e.f")));
  232. EXPECT_EQ(6, rbtree.getNodeCount());
  233. EXPECT_EQ(0, rbtree.erase(Name("j.z.d.e.f")));
  234. EXPECT_EQ(5, rbtree.getNodeCount());
  235. EXPECT_EQ(0, rbtree.erase(Name("r.d.e.f")));
  236. EXPECT_EQ(4, rbtree.getNodeCount());
  237. // z will rejoin with d.e.f since d.e.f has no data
  238. EXPECT_EQ(0, rbtree.erase(Name("s.d.e.f")));
  239. EXPECT_EQ(2, rbtree.getNodeCount());
  240. /*
  241. * z.d.e.f
  242. * \
  243. * g.h
  244. */
  245. EXPECT_EQ(0, rbtree.erase(Name("g.h")));
  246. EXPECT_EQ(1, rbtree.getNodeCount());
  247. // rebuild rbtree to cover different execution paths
  248. EXPECT_EQ(0, rbtree.insert(Name("a"), &rbtnode));
  249. EXPECT_EQ(0, rbtree.insert(Name("g"), &rbtnode));
  250. EXPECT_EQ(0, rbtree.insert(Name("b"), &rbtnode));
  251. EXPECT_EQ(0, rbtree.insert(Name("d"), &rbtnode));
  252. EXPECT_EQ(0, rbtree.insert(Name("c"), &rbtnode));
  253. EXPECT_EQ(0, rbtree.insert(Name("e"), &rbtnode));
  254. /*
  255. * z.d.e.f
  256. * / \
  257. * b g
  258. * / \
  259. * a d
  260. * / \
  261. * c e
  262. */
  263. EXPECT_EQ(0, rbtree.erase(Name("g")));
  264. EXPECT_EQ(0, rbtree.erase(Name("z.d.e.f")));
  265. EXPECT_EQ(5, rbtree.getNodeCount());
  266. EXPECT_EQ(0, rbtree.insert(Name("da"), &rbtnode));
  267. EXPECT_EQ(0, rbtree.insert(Name("aa"), &rbtnode));
  268. EXPECT_EQ(0, rbtree.insert(Name("ba"), &rbtnode));
  269. EXPECT_EQ(0, rbtree.insert(Name("ca"), &rbtnode));
  270. EXPECT_EQ(0, rbtree.insert(Name("m"), &rbtnode));
  271. EXPECT_EQ(0, rbtree.insert(Name("nm"), &rbtnode));
  272. EXPECT_EQ(0, rbtree.insert(Name("om"), &rbtnode));
  273. EXPECT_EQ(1, rbtree.insert(Name("da"), &rbtnode));
  274. EXPECT_EQ(0, rbtree.insert(Name("k"), &rbtnode));
  275. EXPECT_EQ(0, rbtree.insert(Name("l"), &rbtnode));
  276. EXPECT_EQ(0, rbtree.insert(Name("fe"), &rbtnode));
  277. EXPECT_EQ(0, rbtree.insert(Name("ge"), &rbtnode));
  278. EXPECT_EQ(0, rbtree.insert(Name("i"), &rbtnode));
  279. EXPECT_EQ(0, rbtree.insert(Name("ae"), &rbtnode));
  280. EXPECT_EQ(0, rbtree.insert(Name("n"), &rbtnode));
  281. EXPECT_EQ(19, rbtree.getNodeCount());
  282. /*
  283. * d
  284. * / \
  285. * b c
  286. * / \ / \
  287. * aa c e nm
  288. * / \ / \ / \ /\
  289. * a ae ba ca da ge m om
  290. * / \ \
  291. * fe k n
  292. * /
  293. * i
  294. */
  295. // delete rbtree node one by one
  296. EXPECT_EQ(0, rbtree.erase(Name("nm")));
  297. EXPECT_EQ(0, rbtree.erase(Name("n")));
  298. EXPECT_EQ(0, rbtree.erase(Name("a")));
  299. EXPECT_EQ(0, rbtree.erase(Name("ae")));
  300. EXPECT_EQ(0, rbtree.erase(Name("i")));
  301. EXPECT_EQ(0, rbtree.erase(Name("aa")));
  302. EXPECT_EQ(0, rbtree.erase(Name("e")));
  303. EXPECT_EQ(0, rbtree.erase(Name("ge")));
  304. EXPECT_EQ(0, rbtree.erase(Name("k")));
  305. EXPECT_EQ(0, rbtree.erase(Name("m")));
  306. EXPECT_EQ(9, rbtree.getNodeCount());
  307. /*
  308. * d
  309. * / \
  310. * c c
  311. * / \ / \
  312. * b ca fe om
  313. * \ /
  314. * ba da
  315. */
  316. EXPECT_EQ(1, rbtree.erase(Name("am")));
  317. EXPECT_EQ(0, rbtree.erase(Name("fe")));
  318. EXPECT_EQ(0, rbtree.erase(Name("da")));
  319. EXPECT_EQ(0, rbtree.erase(Name("om")));
  320. EXPECT_EQ(0, rbtree.erase(Name("d")));
  321. EXPECT_EQ(0, rbtree.erase(Name("b")));
  322. EXPECT_EQ(0, rbtree.erase(Name("ba")));
  323. EXPECT_EQ(0, rbtree.erase(Name("ca")));
  324. EXPECT_EQ(0, rbtree.erase(Name("c")));
  325. EXPECT_EQ(0, rbtree.erase(Name("l")));
  326. EXPECT_EQ(0, rbtree.getNodeCount());
  327. }