FRangeMergeTest.m 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337
  1. /*
  2. * Copyright 2017 Google
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #import <Foundation/Foundation.h>
  17. #import <XCTest/XCTest.h>
  18. #import "FirebaseDatabase/Sources/Core/FRangeMerge.h"
  19. #import "FirebaseDatabase/Sources/Snapshot/FEmptyNode.h"
  20. #import "FirebaseDatabase/Sources/Snapshot/FNode.h"
  21. #import "FirebaseDatabase/Tests/Helpers/FTestHelpers.h"
  22. @interface FRangeMergeTest : XCTestCase
  23. @end
  24. @implementation FRangeMergeTest
  25. - (void)testSmokeTest {
  26. id<FNode> node = NODE((@{
  27. @"bar" : @"bar-value",
  28. @"foo" : @{@"a" : @{@"deep-a-1" : @1, @"deep-a-2" : @2}, @"b" : @"b", @"c" : @"c", @"d" : @"d"},
  29. @"quu" : @"quu-value"
  30. }));
  31. id<FNode> updates = NODE((@{
  32. @"foo" :
  33. @{@"a" : @{@"deep-a-2" : @"new-a-2", @"deep-a-3" : @3}, @"b-2" : @"new-b", @"c" : @"new-c"}
  34. }));
  35. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"foo/a/deep-a-1")
  36. end:PATH(@"foo/c")
  37. updates:updates];
  38. id<FNode> expected = NODE((@{
  39. @"bar" : @"bar-value",
  40. @"foo" : @{
  41. @"a" : @{@"deep-a-1" : @1, @"deep-a-2" : @"new-a-2", @"deep-a-3" : @3},
  42. @"b-2" : @"new-b",
  43. @"c" : @"new-c",
  44. @"d" : @"d"
  45. },
  46. @"quu" : @"quu-value"
  47. }));
  48. id<FNode> actual = [merge applyToNode:node];
  49. XCTAssertEqualObjects(actual, expected);
  50. }
  51. - (void)testStartIsExclusive {
  52. id<FNode> node = NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value", @"quu" : @"quu-value"}));
  53. id<FNode> updates = NODE((@{@"foo" : @"new-foo-value"}));
  54. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"bar")
  55. end:PATH(@"foo")
  56. updates:updates];
  57. id<FNode> expected =
  58. NODE((@{@"bar" : @"bar-value", @"foo" : @"new-foo-value", @"quu" : @"quu-value"}));
  59. id<FNode> actual = [merge applyToNode:node];
  60. XCTAssertEqualObjects(actual, expected);
  61. }
  62. - (void)testStartIsExclusiveButIncludesChildren {
  63. id<FNode> node = NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value", @"quu" : @"quu-value"}));
  64. id<FNode> updates =
  65. NODE((@{@"bar" : @{@"bar-child" : @"bar-child-value"}, @"foo" : @"new-foo-value"}));
  66. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"bar")
  67. end:PATH(@"foo")
  68. updates:updates];
  69. id<FNode> expected = NODE((@{
  70. @"bar" : @{@"bar-child" : @"bar-child-value"},
  71. @"foo" : @"new-foo-value",
  72. @"quu" : @"quu-value"
  73. }));
  74. id<FNode> actual = [merge applyToNode:node];
  75. XCTAssertEqualObjects(actual, expected);
  76. }
  77. - (void)testEndIsInclusive {
  78. id<FNode> node = NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value", @"quu" : @"quu-value"}));
  79. id<FNode> updates = NODE((@{@"baz" : @"baz-value"}));
  80. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"bar")
  81. end:PATH(@"foo")
  82. updates:updates]; // foo should be deleted
  83. id<FNode> expected =
  84. NODE((@{@"bar" : @"bar-value", @"baz" : @"baz-value", @"quu" : @"quu-value"}));
  85. id<FNode> actual = [merge applyToNode:node];
  86. XCTAssertEqualObjects(actual, expected);
  87. }
  88. - (void)testEndIsInclusiveButExcludesChildren {
  89. id<FNode> node = NODE((@{
  90. @"bar" : @"bar-value",
  91. @"foo" : @{@"foo-child" : @"foo-child-value"},
  92. @"quu" : @"quu-value"
  93. }));
  94. id<FNode> updates = NODE((@{@"baz" : @"baz-value"}));
  95. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"bar")
  96. end:PATH(@"foo")
  97. updates:updates]; // foo should be deleted
  98. id<FNode> expected = NODE((@{
  99. @"bar" : @"bar-value",
  100. @"baz" : @"baz-value",
  101. @"foo" : @{@"foo-child" : @"foo-child-value"},
  102. @"quu" : @"quu-value"
  103. }));
  104. id<FNode> actual = [merge applyToNode:node];
  105. XCTAssertEqualObjects(actual, expected);
  106. }
  107. - (void)testCanUpdateLeafNode {
  108. id<FNode> node = NODE(@"leaf-value");
  109. id<FNode> updates = NODE((@{@"bar" : @"bar-value"}));
  110. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil end:PATH(@"foo") updates:updates];
  111. id<FNode> expected = NODE((@{@"bar" : @"bar-value"}));
  112. id<FNode> actual = [merge applyToNode:node];
  113. XCTAssertEqualObjects(actual, expected);
  114. }
  115. - (void)testCanReplaceLeafNodeWithLeafNode {
  116. id<FNode> node = NODE(@"leaf-value");
  117. id<FNode> updates = NODE(@"new-leaf-value");
  118. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil end:PATH(@"") updates:updates];
  119. id<FNode> expected = NODE(@"new-leaf-value");
  120. id<FNode> actual = [merge applyToNode:node];
  121. XCTAssertEqualObjects(actual, expected);
  122. }
  123. - (void)testLeavesAreUpdatedWhenRangesIncludeDeeperPath {
  124. id<FNode> node = NODE((@{@"foo" : @{@"bar" : @"bar-value"}}));
  125. id<FNode> updates = NODE((@{@"foo" : @{@"bar" : @"new-bar-value"}}));
  126. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"foo")
  127. end:PATH(@"foo/bar/deep")
  128. updates:updates];
  129. id<FNode> expected = NODE((@{@"foo" : @{@"bar" : @"new-bar-value"}}));
  130. id<FNode> actual = [merge applyToNode:node];
  131. XCTAssertEqualObjects(actual, expected);
  132. }
  133. - (void)testLeavesAreNotUpdatedWhenRangesIncludeDeeperPaths {
  134. id<FNode> node = NODE((@{@"foo" : @{@"bar" : @"bar-value"}}));
  135. id<FNode> updates = NODE((@{@"foo" : @{@"bar" : @"new-bar-value"}}));
  136. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"foo/bar")
  137. end:PATH(@"foo/bar/deep")
  138. updates:updates];
  139. id<FNode> expected = NODE((@{@"foo" : @{@"bar" : @"bar-value"}}));
  140. id<FNode> actual = [merge applyToNode:node];
  141. XCTAssertEqualObjects(actual, expected);
  142. }
  143. - (void)testUpdatingEntireRangeUpdatesEverything {
  144. id<FNode> node = [FEmptyNode emptyNode];
  145. id<FNode> updates = NODE((@{@"foo" : @"foo-value", @"bar" : @{@"child" : @"bar-child-value"}}));
  146. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil end:nil updates:updates];
  147. id<FNode> expected = NODE((@{@"foo" : @"foo-value", @"bar" : @{@"child" : @"bar-child-value"}}));
  148. id<FNode> actual = [merge applyToNode:node];
  149. XCTAssertEqualObjects(actual, expected);
  150. }
  151. - (void)testUpdatingRangeWithUnboundedLeftPostWorks {
  152. id<FNode> node = NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value"}));
  153. id<FNode> updates = NODE((@{@"bar" : @"new-bar"}));
  154. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil end:PATH(@"bar") updates:updates];
  155. id<FNode> expected = NODE((@{@"bar" : @"new-bar", @"foo" : @"foo-value"}));
  156. id<FNode> actual = [merge applyToNode:node];
  157. XCTAssertEqualObjects(actual, expected);
  158. }
  159. - (void)testUpdatingRangeWithRightPostChildOfLeftPostWorks {
  160. id<FNode> node =
  161. NODE((@{@"foo" : @{@"a" : @"a", @"b" : @{@"1" : @"1", @"2" : @"2"}, @"c" : @"c"}}));
  162. id<FNode> updates = NODE((@{@"foo" : @{@"a" : @"new-a", @"b" : @{@"1" : @"new-1"}}}));
  163. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"foo")
  164. end:PATH(@"foo/b/1")
  165. updates:updates];
  166. id<FNode> expected =
  167. NODE((@{@"foo" : @{@"a" : @"new-a", @"b" : @{@"1" : @"new-1", @"2" : @"2"}, @"c" : @"c"}}));
  168. id<FNode> actual = [merge applyToNode:node];
  169. XCTAssertEqualObjects(actual, expected);
  170. }
  171. - (void)testUpdatingRangeWithRightPostChildOfLeftPostWorksWithIntegerKeys {
  172. id<FNode> node = NODE(
  173. (@{@"foo" : @{@"a" : @"a", @"b" : @{@"1" : @"1", @"2" : @"2", @"10" : @"10"}, @"c" : @"c"}}));
  174. id<FNode> updates = NODE((@{@"foo" : @{@"a" : @"new-a", @"b" : @{@"1" : @"new-1"}}}));
  175. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"foo")
  176. end:PATH(@"foo/b/2")
  177. updates:updates];
  178. id<FNode> expected =
  179. NODE((@{@"foo" : @{@"a" : @"new-a", @"b" : @{@"1" : @"new-1", @"10" : @"10"}, @"c" : @"c"}}));
  180. id<FNode> actual = [merge applyToNode:node];
  181. XCTAssertEqualObjects(actual, expected);
  182. }
  183. - (void)testUpdatingLeafIncludesPriority {
  184. id<FNode> node = NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value", @"quu" : @"quu-value"}));
  185. id<FNode> updates = NODE((@{@"foo" : @{@".value" : @"new-foo", @".priority" : @"prio"}}));
  186. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"bar")
  187. end:PATH(@"foo")
  188. updates:updates];
  189. id<FNode> expected = NODE((@{
  190. @"bar" : @"bar-value",
  191. @"foo" : @{@".value" : @"new-foo", @".priority" : @"prio"},
  192. @"quu" : @"quu-value"
  193. }));
  194. id<FNode> actual = [merge applyToNode:node];
  195. XCTAssertEqualObjects(actual, expected);
  196. }
  197. - (void)testUpdatingPriorityInChildrenNodeWorks {
  198. id<FNode> node = NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value"}));
  199. id<FNode> updates = NODE((@{@"bar" : @"new-bar", @".priority" : @"prio"}));
  200. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil end:PATH(@"bar") updates:updates];
  201. id<FNode> expected =
  202. NODE((@{@"bar" : @"new-bar", @"foo" : @"foo-value", @".priority" : @"prio"}));
  203. id<FNode> actual = [merge applyToNode:node];
  204. XCTAssertEqualObjects(actual, expected);
  205. }
  206. // TODO: this test should actuall;y work, but priorities on empty nodes are ignored :(
  207. - (void)updatingPriorityInChildrenNodeWorksAlone {
  208. id<FNode> node = NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value"}));
  209. id<FNode> updates = NODE((@{@".priority" : @"prio"}));
  210. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil
  211. end:PATH(@".priority")
  212. updates:updates];
  213. id<FNode> expected =
  214. NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value", @".priority" : @"prio"}));
  215. id<FNode> actual = [merge applyToNode:node];
  216. XCTAssertEqualObjects(actual, expected);
  217. }
  218. - (void)testUpdatingPriorityOnInitiallyEmptyNodeDoesNotBreak {
  219. id<FNode> node = NODE((@{}));
  220. id<FNode> updates = NODE((@{@".priority" : @"prio", @"foo" : @"foo-value"}));
  221. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil end:PATH(@"foo") updates:updates];
  222. id<FNode> expected = NODE((@{@"foo" : @"foo-value", @".priority" : @"prio"}));
  223. id<FNode> actual = [merge applyToNode:node];
  224. XCTAssertEqualObjects(actual, expected);
  225. }
  226. - (void)testPriorityIsDeletedWhenIncludedInChildrenRange {
  227. id<FNode> node = NODE((@{@"bar" : @"bar-value", @"foo" : @"foo-value", @".priority" : @"prio"}));
  228. id<FNode> updates = NODE((@{@"bar" : @"new-bar"}));
  229. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil
  230. end:PATH(@"bar")
  231. updates:updates]; // deletes priority
  232. id<FNode> expected = NODE((@{@"bar" : @"new-bar", @"foo" : @"foo-value"}));
  233. id<FNode> actual = [merge applyToNode:node];
  234. XCTAssertEqualObjects(actual, expected);
  235. }
  236. - (void)testPriorityIsIncludedInOpenStart {
  237. id<FNode> node = NODE((@{@"foo" : @{@"bar" : @"bar-value"}}));
  238. id<FNode> updates = NODE((@{@".priority" : @"prio", @"baz" : @"baz"}));
  239. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:nil end:PATH(@"foo/bar") updates:updates];
  240. id<FNode> expected = NODE((@{@"baz" : @"baz", @".priority" : @"prio"}));
  241. id<FNode> actual = [merge applyToNode:node];
  242. XCTAssertEqualObjects(actual, expected);
  243. }
  244. - (void)testPriorityIsIncludedInOpenEnd {
  245. id<FNode> node = NODE(@"leaf-node");
  246. id<FNode> updates = NODE((@{@".priority" : @"prio", @"foo" : @"bar"}));
  247. FRangeMerge *merge = [[FRangeMerge alloc] initWithStart:PATH(@"/") end:nil updates:updates];
  248. id<FNode> expected = NODE((@{@"foo" : @"bar", @".priority" : @"prio"}));
  249. id<FNode> actual = [merge applyToNode:node];
  250. XCTAssertEqualObjects(actual, expected);
  251. }
  252. @end