FPruningTest.m 13 KB

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