FPruningTest.m 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  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 <XCTest/XCTest.h>
  17. #import "FirebaseDatabase/Sources/Persistence/FLevelDBStorageEngine.h"
  18. #import "FirebaseDatabase/Sources/Persistence/FPruneForest.h"
  19. #import "FirebaseDatabase/Sources/Snapshot/FEmptyNode.h"
  20. #import "FirebaseDatabase/Tests/Helpers/FMockStorageEngine.h"
  21. #import "FirebaseDatabase/Tests/Helpers/FTestHelpers.h"
  22. @interface FPruningTest : XCTestCase
  23. @end
  24. static id<FNode> ABC_NODE = nil;
  25. static id<FNode> DEF_NODE = nil;
  26. static id<FNode> A_NODE = nil;
  27. static id<FNode> D_NODE = nil;
  28. static id<FNode> BC_NODE = nil;
  29. static id<FNode> LARGE_NODE = nil;
  30. @implementation FPruningTest
  31. + (void)initStatics {
  32. static dispatch_once_t onceToken;
  33. dispatch_once(&onceToken, ^{
  34. ABC_NODE = NODE((@{@"a" : @{@"aa" : @1.1, @"ab" : @1.2}, @"b" : @2, @"c" : @3}));
  35. DEF_NODE = NODE((@{@"d" : @4, @"e" : @5, @"f" : @6}));
  36. A_NODE = NODE((@{@"a" : @{@"aa" : @1.1, @"ab" : @1.2}}));
  37. D_NODE = NODE(@{@"d" : @4});
  38. LARGE_NODE = [FTestHelpers leafNodeOfSize:5 * 1024 * 1024];
  39. BC_NODE = [ABC_NODE updateImmediateChild:@"a" withNewChild:[FEmptyNode emptyNode]];
  40. });
  41. }
  42. - (void)runWithDb:(void (^)(id<FStorageEngine> engine))block {
  43. [FPruningTest initStatics];
  44. {
  45. // Run with level DB implementation
  46. FLevelDBStorageEngine *engine = [[FLevelDBStorageEngine alloc]
  47. initWithPath:[NSTemporaryDirectory() stringByAppendingPathComponent:@"purge-tests"]];
  48. block(engine);
  49. [engine purgeEverything];
  50. [engine close];
  51. }
  52. {
  53. // Run with mock implementation
  54. FMockStorageEngine *engine = [[FMockStorageEngine alloc] init];
  55. block(engine);
  56. [engine close];
  57. }
  58. }
  59. - (FPruneForest *)prune:(NSString *)pathStr {
  60. return [[FPruneForest empty] prunePath:PATH(pathStr)];
  61. }
  62. - (FPruneForest *)prune:(NSString *)path exceptRelative:(NSArray *)except {
  63. __block FPruneForest *pruneForest = [FPruneForest empty];
  64. pruneForest = [pruneForest prunePath:PATH(path)];
  65. [except enumerateObjectsUsingBlock:^(NSString *keepPath, NSUInteger idx, BOOL *stop) {
  66. pruneForest = [pruneForest keepPath:[PATH(path) childFromString:keepPath]];
  67. }];
  68. return pruneForest;
  69. }
  70. // Write document at root, prune it.
  71. - (void)test010 {
  72. [self runWithDb:^(id<FStorageEngine> engine) {
  73. [engine updateServerCache:ABC_NODE atPath:PATH(@"") merge:NO];
  74. [engine pruneCache:[self prune:@""] atPath:PATH(@"")];
  75. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"")], [FEmptyNode emptyNode]);
  76. }];
  77. }
  78. // Write document at /x, prune it via PruneForest for /x, at root.
  79. - (void)test020 {
  80. [self runWithDb:^(id<FStorageEngine> engine) {
  81. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  82. [engine pruneCache:[self prune:@"x"] atPath:PATH(@"")];
  83. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"")], [FEmptyNode emptyNode]);
  84. }];
  85. }
  86. // Write document at /x, prune it via PruneForest for root, at /x.
  87. - (void)test030 {
  88. [self runWithDb:^(id<FStorageEngine> engine) {
  89. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  90. [engine pruneCache:[self prune:@""] atPath:PATH(@"x")];
  91. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")], [FEmptyNode emptyNode]);
  92. }];
  93. }
  94. // Write document at /x, prune it via PruneForest for root, at root
  95. - (void)test040 {
  96. [self runWithDb:^(id<FStorageEngine> engine) {
  97. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  98. [engine pruneCache:[self prune:@""] atPath:PATH(@"")];
  99. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")], [FEmptyNode emptyNode]);
  100. }];
  101. }
  102. // Write document at /x/y, prune it via PruneForest for /y, at /x
  103. - (void)test050 {
  104. [self runWithDb:^(id<FStorageEngine> engine) {
  105. [engine updateServerCache:ABC_NODE atPath:PATH(@"x/y") merge:NO];
  106. [engine pruneCache:[self prune:@"y"] atPath:PATH(@"x")];
  107. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x/y")], [FEmptyNode emptyNode]);
  108. }];
  109. }
  110. // Write abc at /x/y, prune /x/y except b,c via PruneForest for /x/y -b,c, at root
  111. - (void)test060 {
  112. [self runWithDb:^(id<FStorageEngine> engine) {
  113. [engine updateServerCache:ABC_NODE atPath:PATH(@"x/y") merge:NO];
  114. [engine pruneCache:[self prune:@"x/y" exceptRelative:@[ @"b", @"c" ]] atPath:PATH(@"")];
  115. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x/y")], BC_NODE);
  116. }];
  117. }
  118. // Write abc at /x/y, prune /x/y except b,c via PruneForest for /y -b,c, at /x
  119. - (void)test070 {
  120. [self runWithDb:^(id<FStorageEngine> engine) {
  121. [engine updateServerCache:ABC_NODE atPath:PATH(@"x/y") merge:NO];
  122. [engine pruneCache:[self prune:@"y" exceptRelative:@[ @"b", @"c" ]] atPath:PATH(@"x")];
  123. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x/y")], BC_NODE);
  124. }];
  125. }
  126. // Write abc at /x/y, prune /x/y except not-there via PruneForest for /x/y -d, at root
  127. - (void)test080 {
  128. [self runWithDb:^(id<FStorageEngine> engine) {
  129. [engine updateServerCache:ABC_NODE atPath:PATH(@"x/y") merge:NO];
  130. [engine pruneCache:[self prune:@"x/y" exceptRelative:@[ @"not-there" ]] atPath:PATH(@"")];
  131. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x/y")], [FEmptyNode emptyNode]);
  132. }];
  133. }
  134. // Write abc at / and def at /a, prune all via PruneForest for / at root
  135. - (void)test090 {
  136. [self runWithDb:^(id<FStorageEngine> engine) {
  137. [engine updateServerCache:ABC_NODE atPath:PATH(@"") merge:NO];
  138. [engine updateServerCache:DEF_NODE atPath:PATH(@"a") merge:NO];
  139. [engine pruneCache:[self prune:@""] atPath:PATH(@"")];
  140. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"")], [FEmptyNode emptyNode]);
  141. }];
  142. }
  143. // Write abc at / and def at /a, prune all except b,c via PruneForest for root -b,c, at root
  144. - (void)test100 {
  145. [self runWithDb:^(id<FStorageEngine> engine) {
  146. [engine updateServerCache:ABC_NODE atPath:PATH(@"") merge:NO];
  147. [engine updateServerCache:DEF_NODE atPath:PATH(@"a") merge:NO];
  148. [engine pruneCache:[self prune:@"" exceptRelative:@[ @"b", @"c" ]] atPath:PATH(@"")];
  149. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"")], BC_NODE);
  150. }];
  151. }
  152. // Write abc at /x and def at /x/a, prune /x except b,c via PruneForest for /x -b,c, at root
  153. - (void)test110 {
  154. [self runWithDb:^(id<FStorageEngine> engine) {
  155. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  156. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a") merge:NO];
  157. [engine pruneCache:[self prune:@"x" exceptRelative:@[ @"b", @"c" ]] atPath:PATH(@"")];
  158. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")], BC_NODE);
  159. }];
  160. }
  161. // Write abc at /x and def at /x/a, prune /x except b,c via PruneForest for root -b,c, at /x
  162. - (void)test120 {
  163. [self runWithDb:^(id<FStorageEngine> engine) {
  164. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  165. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a") merge:NO];
  166. [engine pruneCache:[self prune:@"" exceptRelative:@[ @"b", @"c" ]] atPath:PATH(@"x")];
  167. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")], BC_NODE);
  168. }];
  169. }
  170. // Write abc at /x and def at /x/a, prune /x except a via PruneForest for /x -a, at root
  171. - (void)test130 {
  172. [self runWithDb:^(id<FStorageEngine> engine) {
  173. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  174. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a") merge:NO];
  175. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  176. [ABC_NODE updateImmediateChild:@"a" withNewChild:DEF_NODE]);
  177. [engine pruneCache:[self prune:@"x" exceptRelative:@[ @"a" ]] atPath:PATH(@"")];
  178. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  179. [[FEmptyNode emptyNode] updateImmediateChild:@"a" withNewChild:DEF_NODE]);
  180. }];
  181. }
  182. // Write abc at /x and def at /x/a, prune /x except a via PruneForest for root -a, at /x
  183. - (void)test140 {
  184. [self runWithDb:^(id<FStorageEngine> engine) {
  185. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  186. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a") merge:NO];
  187. [engine pruneCache:[self prune:@"" exceptRelative:@[ @"a" ]] atPath:PATH(@"x")];
  188. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  189. [[FEmptyNode emptyNode] updateImmediateChild:@"a" withNewChild:DEF_NODE]);
  190. }];
  191. }
  192. // Write abc at /x and def at /x/a, prune /x except a/d via PruneForest for /x -a/d, at root
  193. - (void)test150 {
  194. [self runWithDb:^(id<FStorageEngine> engine) {
  195. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  196. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a") merge:NO];
  197. [engine pruneCache:[self prune:@"x" exceptRelative:@[ @"a/d" ]] atPath:PATH(@"")];
  198. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  199. [[FEmptyNode emptyNode] updateImmediateChild:@"a" withNewChild:D_NODE]);
  200. }];
  201. }
  202. // Write abc at /x and def at /x/a, prune /x except a/d via PruneForest for / -a/d, at /x
  203. - (void)test160 {
  204. [self runWithDb:^(id<FStorageEngine> engine) {
  205. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  206. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a") merge:NO];
  207. [engine pruneCache:[self prune:@"" exceptRelative:@[ @"a/d" ]] atPath:PATH(@"x")];
  208. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  209. [[FEmptyNode emptyNode] updateImmediateChild:@"a" withNewChild:D_NODE]);
  210. }];
  211. }
  212. // Write abc at /x and def at /x/a/aa, prune /x except a via PruneForest for /x -a, at root
  213. - (void)test170 {
  214. [self runWithDb:^(id<FStorageEngine> engine) {
  215. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  216. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a/aa") merge:NO];
  217. [engine pruneCache:[self prune:@"x" exceptRelative:@[ @"a" ]] atPath:PATH(@"")];
  218. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")], [A_NODE updateChild:PATH(@"a/aa")
  219. withNewChild:DEF_NODE]);
  220. }];
  221. }
  222. // Write abc at /x and def at /x/a/aa, prune /x except a via PruneForest for / -a, at /x
  223. - (void)test180 {
  224. [self runWithDb:^(id<FStorageEngine> engine) {
  225. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  226. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a/aa") merge:NO];
  227. [engine pruneCache:[self prune:@"" exceptRelative:@[ @"a/aa" ]] atPath:PATH(@"x")];
  228. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  229. [[FEmptyNode emptyNode] updateChild:PATH(@"a/aa") withNewChild:DEF_NODE]);
  230. }];
  231. }
  232. // Write abc at /x and def at /x/a/aa, prune /x except a/aa via PruneForest for /x -a/aa, at root
  233. - (void)test190 {
  234. [self runWithDb:^(id<FStorageEngine> engine) {
  235. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  236. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a/aa") merge:NO];
  237. [engine pruneCache:[self prune:@"x" exceptRelative:@[ @"a/aa" ]] atPath:PATH(@"")];
  238. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  239. [[FEmptyNode emptyNode] updateChild:PATH(@"a/aa") withNewChild:DEF_NODE]);
  240. }];
  241. }
  242. // Write abc at /x and def at /x/a/aa, prune /x except a/aa via PruneForest for / -a/aa, at /x
  243. - (void)test200 {
  244. [self runWithDb:^(id<FStorageEngine> engine) {
  245. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  246. [engine updateServerCache:DEF_NODE atPath:PATH(@"x/a/aa") merge:NO];
  247. [engine pruneCache:[self prune:@"" exceptRelative:@[ @"a/aa" ]] atPath:PATH(@"x")];
  248. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  249. [[FEmptyNode emptyNode] updateChild:PATH(@"a/aa") withNewChild:DEF_NODE]);
  250. }];
  251. }
  252. // Write large node at /x, prune x via PruneForest for x at root
  253. - (void)test210 {
  254. [self runWithDb:^(id<FStorageEngine> engine) {
  255. [engine updateServerCache:LARGE_NODE atPath:PATH(@"x") merge:NO];
  256. [engine pruneCache:[self prune:@"x"] atPath:PATH(@"")];
  257. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")], [FEmptyNode emptyNode]);
  258. }];
  259. }
  260. // Write abc at x and large node at /x/a, prune x except a via PruneForest for / -a, at x
  261. - (void)test220 {
  262. [self runWithDb:^(id<FStorageEngine> engine) {
  263. [engine updateServerCache:ABC_NODE atPath:PATH(@"x") merge:NO];
  264. [engine updateServerCache:LARGE_NODE atPath:PATH(@"x/a") merge:NO];
  265. [engine pruneCache:[self prune:@"" exceptRelative:@[ @"a" ]] atPath:PATH(@"x")];
  266. XCTAssertEqualObjects([engine serverCacheAtPath:PATH(@"x")],
  267. [[FEmptyNode emptyNode] updateImmediateChild:@"a"
  268. withNewChild:LARGE_NODE]);
  269. }];
  270. }
  271. @end