Package com.orientechnologies.orient.core.storage.impl.local.paginated.wal

Source Code of com.orientechnologies.orient.core.storage.impl.local.paginated.wal.WriteAheadLogConcurrencyTest$ConcurrentWriter

package com.orientechnologies.orient.core.storage.impl.local.paginated.wal;

import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;

import java.io.File;
import java.util.ArrayList;
import java.util.List;
import java.util.NavigableMap;
import java.util.Random;
import java.util.concurrent.Callable;
import java.util.concurrent.ConcurrentSkipListMap;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicReference;

import org.testng.Assert;
import org.testng.annotations.AfterClass;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.Test;

import com.orientechnologies.orient.core.storage.impl.local.paginated.OLocalPaginatedStorage;

/**
* @author Andrey Lomakin
* @since 16.05.13
*/
@Test(enabled = false)
public class WriteAheadLogConcurrencyTest {
  private ODiskWriteAheadLog writeAheadLog;
  private File                                                           testDir;
  private NavigableMap<OLogSequenceNumber, WriteAheadLogTest.TestRecord> recordConcurrentMap = new ConcurrentSkipListMap<OLogSequenceNumber, WriteAheadLogTest.TestRecord>();
  private ExecutorService                                                writerExecutor;
  private AtomicReference<OLogSequenceNumber>                            lastCheckpoint      = new AtomicReference<OLogSequenceNumber>();

  @BeforeClass(enabled = false)
  public void beforeClass() throws Exception {
    OWALRecordsFactory.INSTANCE.registerNewRecord((byte) 128, WriteAheadLogTest.TestRecord.class);

    String buildDirectory = System.getProperty("buildDirectory");
    if (buildDirectory == null || buildDirectory.isEmpty())
      buildDirectory = ".";

    testDir = new File(buildDirectory, "WriteAheadLogConcurrencyTest");
    if (!testDir.exists())
      testDir.mkdir();

    OLocalPaginatedStorage localPaginatedStorage = mock(OLocalPaginatedStorage.class);
    when(localPaginatedStorage.getStoragePath()).thenReturn(testDir.getAbsolutePath());
    when(localPaginatedStorage.getName()).thenReturn("WriteAheadLogConcurrencyTest");

    writeAheadLog = new ODiskWriteAheadLog(200, 500, OWALPage.PAGE_SIZE * 800, 100L * 1024L * 1024L * 1024L, localPaginatedStorage);

    writerExecutor = Executors.newCachedThreadPool();
  }

  public void testConcurrentWrites() throws Exception {
    CountDownLatch startLatch = new CountDownLatch(1);

    List<Future> futures = new ArrayList<Future>();
    Random random = new Random();
    List<Long> seeds = new ArrayList<Long>();
    for (int i = 0; i < 8; i++)
      seeds.add(random.nextLong());

    System.out.println("Seeds");
    for (long seed : seeds)
      System.out.println(seed);

    for (int i = 0; i < 8; i++)
      futures.add(writerExecutor.submit(new ConcurrentWriter(seeds.get(i), startLatch, writeAheadLog, recordConcurrentMap,
          lastCheckpoint)));

    startLatch.countDown();

    for (Future future : futures)
      future.get();

    OLogSequenceNumber lsn = writeAheadLog.begin();
    int recordsCount = 0;
    while (lsn != null) {
      WriteAheadLogTest.TestRecord testRecord = (WriteAheadLogTest.TestRecord) writeAheadLog.read(lsn);
      Assert.assertEquals(testRecord, recordConcurrentMap.get(lsn));
      lsn = writeAheadLog.next(lsn);
      recordsCount++;
    }

    Assert.assertEquals(recordsCount, recordConcurrentMap.size());
    Assert.assertEquals(lastCheckpoint.get(), writeAheadLog.getLastCheckpoint());
  }

  @AfterClass(enabled = false)
  public void afterClass() throws Exception {
    writeAheadLog.delete();

    if (testDir.exists())
      testDir.delete();
  }

  private static final class ConcurrentWriter implements Callable<Void> {
    private final CountDownLatch                                                 startLatch;
    private final ODiskWriteAheadLog writeAheadLog;
    private final NavigableMap<OLogSequenceNumber, WriteAheadLogTest.TestRecord> recordConcurrentMap;
    private final Random                                                         random;
    private final AtomicReference<OLogSequenceNumber>                            lastCheckpoint;

    private ConcurrentWriter(long seed, CountDownLatch startLatch, ODiskWriteAheadLog writeAheadLog,
        NavigableMap<OLogSequenceNumber, WriteAheadLogTest.TestRecord> recordConcurrentMap,
        AtomicReference<OLogSequenceNumber> lastCheckpoint) {
      this.lastCheckpoint = lastCheckpoint;
      random = new Random(seed);
      this.startLatch = startLatch;
      this.writeAheadLog = writeAheadLog;
      this.recordConcurrentMap = recordConcurrentMap;
    }

    @Override
    public Void call() throws Exception {
      startLatch.await();

      try {
        while (writeAheadLog.size() < 3072L * 1024 * 1024) {
          int recordSize = random.nextInt(OWALPage.PAGE_SIZE / 2 - 128) + 128;
          WriteAheadLogTest.TestRecord testRecord = new WriteAheadLogTest.TestRecord(recordSize, random.nextBoolean());

          OLogSequenceNumber lsn = writeAheadLog.log(testRecord);
          if (testRecord.isUpdateMasterRecord()) {
            OLogSequenceNumber checkpoint = lastCheckpoint.get();
            while (checkpoint == null || checkpoint.compareTo(testRecord.getLsn()) < 0) {
              if (lastCheckpoint.compareAndSet(checkpoint, testRecord.getLsn()))
                break;

              checkpoint = lastCheckpoint.get();
            }
          }

          Assert.assertNull(recordConcurrentMap.put(lsn, testRecord));
        }

        return null;
      } catch (Exception e) {
        e.printStackTrace();
        throw e;
      }
    }
  }

}
TOP

Related Classes of com.orientechnologies.orient.core.storage.impl.local.paginated.wal.WriteAheadLogConcurrencyTest$ConcurrentWriter

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.