fork of https://github.com/oxigraph/rocksdb and https://github.com/facebook/rocksdb for nextgraph and oxigraph
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
215 lines
6.3 KiB
215 lines
6.3 KiB
// Copyright (c) 2014, Facebook, Inc. All rights reserved.
|
|
// This source code is licensed under the BSD-style license found in the
|
|
// LICENSE file in the root directory of this source tree. An additional grant
|
|
// of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
package org.rocksdb.test;
|
|
|
|
import org.rocksdb.*;
|
|
|
|
import java.io.IOException;
|
|
import java.nio.file.*;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.Random;
|
|
|
|
import static org.assertj.core.api.Assertions.assertThat;
|
|
import static org.rocksdb.test.Types.byteToInt;
|
|
import static org.rocksdb.test.Types.intToByte;
|
|
|
|
/**
|
|
* Abstract tests for both Comparator and DirectComparator
|
|
*/
|
|
public abstract class AbstractComparatorTest {
|
|
|
|
/**
|
|
* Get a comparator which will expect Integer keys
|
|
* and determine an ascending order
|
|
*
|
|
* @return An integer ascending order key comparator
|
|
*/
|
|
public abstract AbstractComparator getAscendingIntKeyComparator();
|
|
|
|
/**
|
|
* Test which stores random keys into the database
|
|
* using an @see getAscendingIntKeyComparator
|
|
* it then checks that these keys are read back in
|
|
* ascending order
|
|
*
|
|
* @param db_path A path where we can store database
|
|
* files temporarily
|
|
*
|
|
* @throws java.io.IOException if IO error happens.
|
|
*/
|
|
public void testRoundtrip(final Path db_path) throws IOException, RocksDBException {
|
|
|
|
Options opt = null;
|
|
RocksDB db = null;
|
|
|
|
try {
|
|
opt = new Options();
|
|
opt.setCreateIfMissing(true);
|
|
opt.setComparator(getAscendingIntKeyComparator());
|
|
|
|
// store 10,000 random integer keys
|
|
final int ITERATIONS = 10000;
|
|
|
|
db = RocksDB.open(opt, db_path.toString());
|
|
final Random random = new Random();
|
|
for (int i = 0; i < ITERATIONS; i++) {
|
|
final byte key[] = intToByte(random.nextInt());
|
|
if (i > 0 && db.get(key) != null) { // does key already exist (avoid duplicates)
|
|
i--; // generate a different key
|
|
} else {
|
|
db.put(key, "value".getBytes());
|
|
}
|
|
}
|
|
db.close();
|
|
|
|
// re-open db and read from start to end
|
|
// integer keys should be in ascending
|
|
// order as defined by SimpleIntComparator
|
|
db = RocksDB.open(opt, db_path.toString());
|
|
final RocksIterator it = db.newIterator();
|
|
it.seekToFirst();
|
|
int lastKey = Integer.MIN_VALUE;
|
|
int count = 0;
|
|
for (it.seekToFirst(); it.isValid(); it.next()) {
|
|
final int thisKey = byteToInt(it.key());
|
|
assertThat(thisKey).isGreaterThan(lastKey);
|
|
lastKey = thisKey;
|
|
count++;
|
|
}
|
|
db.close();
|
|
|
|
assertThat(count).isEqualTo(ITERATIONS);
|
|
|
|
} finally {
|
|
if (db != null) {
|
|
db.close();
|
|
}
|
|
|
|
if (opt != null) {
|
|
opt.dispose();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Test which stores random keys into a column family
|
|
* in the database
|
|
* using an @see getAscendingIntKeyComparator
|
|
* it then checks that these keys are read back in
|
|
* ascending order
|
|
*
|
|
* @param db_path A path where we can store database
|
|
* files temporarily
|
|
*
|
|
* @throws java.io.IOException if IO error happens.
|
|
*/
|
|
public void testRoundtripCf(final Path db_path) throws IOException,
|
|
RocksDBException {
|
|
|
|
DBOptions opt = null;
|
|
RocksDB db = null;
|
|
List<ColumnFamilyDescriptor> cfDescriptors =
|
|
new ArrayList<>();
|
|
cfDescriptors.add(new ColumnFamilyDescriptor(
|
|
RocksDB.DEFAULT_COLUMN_FAMILY));
|
|
cfDescriptors.add(new ColumnFamilyDescriptor("new_cf",
|
|
new ColumnFamilyOptions().setComparator(
|
|
getAscendingIntKeyComparator())));
|
|
List<ColumnFamilyHandle> cfHandles = new ArrayList<>();
|
|
try {
|
|
opt = new DBOptions().
|
|
setCreateIfMissing(true).
|
|
setCreateMissingColumnFamilies(true);
|
|
|
|
// store 10,000 random integer keys
|
|
final int ITERATIONS = 10000;
|
|
|
|
db = RocksDB.open(opt, db_path.toString(), cfDescriptors, cfHandles);
|
|
assertThat(cfDescriptors.size()).isEqualTo(2);
|
|
assertThat(cfHandles.size()).isEqualTo(2);
|
|
|
|
final Random random = new Random();
|
|
for (int i = 0; i < ITERATIONS; i++) {
|
|
final byte key[] = intToByte(random.nextInt());
|
|
if (i > 0 && db.get(cfHandles.get(1), key) != null) { // does key already exist (avoid duplicates)
|
|
i--; // generate a different key
|
|
} else {
|
|
db.put(cfHandles.get(1), key, "value".getBytes());
|
|
}
|
|
}
|
|
for (ColumnFamilyHandle handle : cfHandles) {
|
|
handle.dispose();
|
|
}
|
|
cfHandles.clear();
|
|
db.close();
|
|
|
|
// re-open db and read from start to end
|
|
// integer keys should be in ascending
|
|
// order as defined by SimpleIntComparator
|
|
db = RocksDB.open(opt, db_path.toString(), cfDescriptors, cfHandles);
|
|
assertThat(cfDescriptors.size()).isEqualTo(2);
|
|
assertThat(cfHandles.size()).isEqualTo(2);
|
|
final RocksIterator it = db.newIterator(cfHandles.get(1));
|
|
it.seekToFirst();
|
|
int lastKey = Integer.MIN_VALUE;
|
|
int count = 0;
|
|
for (it.seekToFirst(); it.isValid(); it.next()) {
|
|
final int thisKey = byteToInt(it.key());
|
|
assertThat(thisKey).isGreaterThan(lastKey);
|
|
lastKey = thisKey;
|
|
count++;
|
|
}
|
|
for (ColumnFamilyHandle handle : cfHandles) {
|
|
handle.dispose();
|
|
}
|
|
cfHandles.clear();
|
|
db.close();
|
|
assertThat(count).isEqualTo(ITERATIONS);
|
|
|
|
} finally {
|
|
for (ColumnFamilyHandle handle : cfHandles) {
|
|
handle.dispose();
|
|
}
|
|
|
|
if (db != null) {
|
|
db.close();
|
|
}
|
|
|
|
if (opt != null) {
|
|
opt.dispose();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Compares integer keys
|
|
* so that they are in ascending order
|
|
*
|
|
* @param a 4-bytes representing an integer key
|
|
* @param b 4-bytes representing an integer key
|
|
*
|
|
* @return negative if a < b, 0 if a == b, positive otherwise
|
|
*/
|
|
protected final int compareIntKeys(final byte[] a, final byte[] b) {
|
|
|
|
final int iA = byteToInt(a);
|
|
final int iB = byteToInt(b);
|
|
|
|
// protect against int key calculation overflow
|
|
final double diff = (double)iA - iB;
|
|
final int result;
|
|
if (diff < Integer.MIN_VALUE) {
|
|
result = Integer.MIN_VALUE;
|
|
} else if(diff > Integer.MAX_VALUE) {
|
|
result = Integer.MAX_VALUE;
|
|
} else {
|
|
result = (int)diff;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
}
|
|
|