|
26 | 26 | import java.util.UUID;
|
27 | 27 | import java.util.concurrent.CompletableFuture;
|
28 | 28 | import java.util.concurrent.ConcurrentHashMap;
|
| 29 | +import java.util.concurrent.CountDownLatch; |
29 | 30 | import java.util.concurrent.TimeUnit;
|
30 | 31 | import java.util.stream.Collectors;
|
31 | 32 |
|
| 33 | +import org.apache.bookkeeper.client.BKException; |
32 | 34 | import org.apache.bookkeeper.client.api.ReadHandle;
|
| 35 | +import org.apache.bookkeeper.mledger.AsyncCallbacks; |
| 36 | +import org.apache.bookkeeper.mledger.Entry; |
33 | 37 | import org.apache.bookkeeper.mledger.LedgerOffloader;
|
34 | 38 | import org.apache.bookkeeper.mledger.ManagedCursor;
|
35 | 39 | import org.apache.bookkeeper.mledger.ManagedLedgerConfig;
|
| 40 | +import org.apache.bookkeeper.mledger.ManagedLedgerException; |
| 41 | +import org.apache.bookkeeper.mledger.Position; |
| 42 | +import org.apache.bookkeeper.mledger.PositionFactory; |
36 | 43 | import org.apache.bookkeeper.mledger.proto.MLDataFormats;
|
37 | 44 | import org.apache.bookkeeper.mledger.util.MockClock;
|
38 | 45 | import org.apache.bookkeeper.test.MockedBookKeeperTestCase;
|
39 | 46 |
|
40 | 47 | import org.apache.pulsar.common.policies.data.OffloadPoliciesImpl;
|
| 48 | +import org.apache.pulsar.common.policies.data.OffloadedReadPriority; |
| 49 | +import org.awaitility.Awaitility; |
41 | 50 | import org.mockito.Mockito;
|
42 | 51 | import org.slf4j.Logger;
|
43 | 52 | import org.slf4j.LoggerFactory;
|
@@ -207,6 +216,125 @@ public void testLaggedDelete() throws Exception {
|
207 | 216 | assertEventuallyTrue(() -> offloader.deletedOffloads().contains(firstLedgerId));
|
208 | 217 | }
|
209 | 218 |
|
| 219 | + @Test |
| 220 | + public void testGetReadLedgerHandleAfterTrimOffloadedLedgers() throws Exception { |
| 221 | + // Create managed ledger. |
| 222 | + final long offloadThresholdSeconds = 5; |
| 223 | + final long offloadDeletionLagInSeconds = 1; |
| 224 | + OffloadPrefixTest.MockLedgerOffloader offloader = new OffloadPrefixTest.MockLedgerOffloader(); |
| 225 | + ManagedLedgerConfig config = new ManagedLedgerConfig(); |
| 226 | + config.setMaxEntriesPerLedger(10); |
| 227 | + config.setMinimumRolloverTime(0, TimeUnit.SECONDS); |
| 228 | + config.setRetentionTime(10, TimeUnit.MINUTES); |
| 229 | + config.setRetentionSizeInMB(10); |
| 230 | + offloader.getOffloadPolicies().setManagedLedgerOffloadDeletionLagInMillis(offloadDeletionLagInSeconds * 1000); |
| 231 | + offloader.getOffloadPolicies().setManagedLedgerOffloadThresholdInSeconds(offloadThresholdSeconds); |
| 232 | + offloader.getOffloadPolicies().setManagedLedgerOffloadedReadPriority(OffloadedReadPriority.BOOKKEEPER_FIRST); |
| 233 | + config.setLedgerOffloader(offloader); |
| 234 | + ManagedLedgerImpl ml = |
| 235 | + (ManagedLedgerImpl)factory.open("testGetReadLedgerHandleAfterTrimOffloadedLedgers", config); |
| 236 | + ml.openCursor("c1"); |
| 237 | + |
| 238 | + // Write entries. |
| 239 | + int i = 0; |
| 240 | + for (; i < 35; i++) { |
| 241 | + String content = "entry-" + i; |
| 242 | + ml.addEntry(content.getBytes()); |
| 243 | + } |
| 244 | + Assert.assertEquals(ml.getLedgersInfoAsList().size(), 4); |
| 245 | + long ledger1 = ml.getLedgersInfoAsList().get(0).getLedgerId(); |
| 246 | + long ledger2 = ml.getLedgersInfoAsList().get(1).getLedgerId(); |
| 247 | + long ledger3 = ml.getLedgersInfoAsList().get(2).getLedgerId(); |
| 248 | + long ledger4 = ml.getLedgersInfoAsList().get(3).getLedgerId(); |
| 249 | + |
| 250 | + // Offload ledgers. |
| 251 | + Thread.sleep(offloadThresholdSeconds * 2 * 1000); |
| 252 | + CompletableFuture<Position> offloadFuture = new CompletableFuture<Position>(); |
| 253 | + ml.maybeOffloadInBackground(offloadFuture); |
| 254 | + offloadFuture.join(); |
| 255 | + |
| 256 | + // Cache ledger handle. |
| 257 | + CountDownLatch readCountDownLatch = new CountDownLatch(4); |
| 258 | + AsyncCallbacks.ReadEntryCallback readCb = new AsyncCallbacks.ReadEntryCallback(){ |
| 259 | + |
| 260 | + @Override |
| 261 | + public void readEntryComplete(Entry entry, Object ctx) { |
| 262 | + readCountDownLatch.countDown(); |
| 263 | + } |
| 264 | + |
| 265 | + @Override |
| 266 | + public void readEntryFailed(ManagedLedgerException exception, Object ctx) { |
| 267 | + readCountDownLatch.countDown(); |
| 268 | + } |
| 269 | + }; |
| 270 | + ml.asyncReadEntry(PositionFactory.create(ledger1, 0), readCb, null); |
| 271 | + ml.asyncReadEntry(PositionFactory.create(ledger2, 0), readCb, null); |
| 272 | + ml.asyncReadEntry(PositionFactory.create(ledger3, 0), readCb, null); |
| 273 | + ml.asyncReadEntry(PositionFactory.create(ledger4, 0), readCb, null); |
| 274 | + readCountDownLatch.await(); |
| 275 | + ReadHandle originalReadHandle4 = ml.getLedgerHandle(ledger4).join(); |
| 276 | + |
| 277 | + // Trim offloaded BK ledger handles. |
| 278 | + Thread.sleep(offloadDeletionLagInSeconds * 2 * 1000); |
| 279 | + CompletableFuture<Position> trimLedgerFuture = new CompletableFuture<Position>(); |
| 280 | + ml.internalTrimLedgers(false, trimLedgerFuture); |
| 281 | + trimLedgerFuture.join(); |
| 282 | + MLDataFormats.ManagedLedgerInfo.LedgerInfo ledgerInfo1 = ml.getLedgerInfo(ledger1).get(); |
| 283 | + MLDataFormats.ManagedLedgerInfo.LedgerInfo ledgerInfo2 = ml.getLedgerInfo(ledger2).get(); |
| 284 | + MLDataFormats.ManagedLedgerInfo.LedgerInfo ledgerInfo3 = ml.getLedgerInfo(ledger3).get(); |
| 285 | + MLDataFormats.ManagedLedgerInfo.LedgerInfo ledgerInfo4 = ml.getLedgerInfo(ledger4).get(); |
| 286 | + Assert.assertTrue(ledgerInfo1.hasOffloadContext() && ledgerInfo1.getOffloadContext().getBookkeeperDeleted()); |
| 287 | + Assert.assertTrue(ledgerInfo2.hasOffloadContext() && ledgerInfo2.getOffloadContext().getBookkeeperDeleted()); |
| 288 | + Assert.assertTrue(ledgerInfo3.hasOffloadContext() && ledgerInfo3.getOffloadContext().getBookkeeperDeleted()); |
| 289 | + Assert.assertFalse(ledgerInfo4.hasOffloadContext() || ledgerInfo4.getOffloadContext().getBookkeeperDeleted()); |
| 290 | + |
| 291 | + Awaitility.await().untilAsserted(() -> { |
| 292 | + try { |
| 293 | + factory.getBookKeeper().get().openLedger(ledger3, ml.digestType, ml.config.getPassword()); |
| 294 | + Assert.fail("Should fail: the ledger has been deleted"); |
| 295 | + } catch (BKException.BKNoSuchLedgerExistsException ex) { |
| 296 | + // Expected. |
| 297 | + } |
| 298 | + try { |
| 299 | + factory.getBookKeeper().get().openLedger(ledger2, ml.digestType, ml.config.getPassword()); |
| 300 | + Assert.fail("Should fail: the ledger has been deleted"); |
| 301 | + } catch (BKException.BKNoSuchLedgerExistsException ex) { |
| 302 | + // Expected. |
| 303 | + } |
| 304 | + try { |
| 305 | + factory.getBookKeeper().get().openLedger(ledger1, ml.digestType, ml.config.getPassword()); |
| 306 | + Assert.fail("Should fail: the ledger has been deleted"); |
| 307 | + } catch (BKException.BKNoSuchLedgerExistsException ex) { |
| 308 | + // Expected. |
| 309 | + } |
| 310 | + }); |
| 311 | + |
| 312 | + // Verify: "ml.getLedgerHandle" returns a correct ledger handle. |
| 313 | + ReadHandle currentReadHandle4 = ml.getLedgerHandle(ledger4).join(); |
| 314 | + Assert.assertEquals(currentReadHandle4, originalReadHandle4); |
| 315 | + try { |
| 316 | + ml.getLedgerHandle(ledger3).join(); |
| 317 | + Assert.fail("should get a failure: MockLedgerOffloader does not support read"); |
| 318 | + } catch (Exception ex) { |
| 319 | + Assert.assertTrue(ex.getCause().getCause().getMessage() |
| 320 | + .contains("MockLedgerOffloader does not support read")); |
| 321 | + } |
| 322 | + try { |
| 323 | + ml.getLedgerHandle(ledger2).join(); |
| 324 | + Assert.fail("should get a failure: MockLedgerOffloader does not support read"); |
| 325 | + } catch (Exception ex) { |
| 326 | + Assert.assertTrue(ex.getCause().getCause().getMessage() |
| 327 | + .contains("MockLedgerOffloader does not support read")); |
| 328 | + } |
| 329 | + try { |
| 330 | + ml.getLedgerHandle(ledger1).join(); |
| 331 | + Assert.fail("should get a failure: MockLedgerOffloader does not support read"); |
| 332 | + } catch (Exception ex) { |
| 333 | + Assert.assertTrue(ex.getCause().getCause().getMessage() |
| 334 | + .contains("MockLedgerOffloader does not support read")); |
| 335 | + } |
| 336 | + } |
| 337 | + |
210 | 338 | @Test(timeOut = 5000)
|
211 | 339 | public void testFileSystemOffloadDeletePath() throws Exception {
|
212 | 340 | MockFileSystemLedgerOffloader offloader = new MockFileSystemLedgerOffloader();
|
|
0 commit comments