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