All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.ddth.commons.rocksdb.RocksDbWrapper Maven / Gradle / Ivy

There is a newer version: 1.1.0
Show newest version
package com.github.ddth.commons.rocksdb;

import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.apache.commons.io.FileUtils;
import org.rocksdb.ColumnFamilyDescriptor;
import org.rocksdb.ColumnFamilyHandle;
import org.rocksdb.DBOptions;
import org.rocksdb.ReadOptions;
import org.rocksdb.RocksDB;
import org.rocksdb.RocksDBException;
import org.rocksdb.RocksIterator;
import org.rocksdb.WriteBatch;
import org.rocksdb.WriteBatchWithIndex;
import org.rocksdb.WriteOptions;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Wrapper around {@link RocksDB} instance.
 *
 * @author Thanh Nguyen 
 * @see https://github.com/facebook/rocksdb/wiki/RocksJava-Basics
 * @since 0.8.0
 */
public class RocksDbWrapper implements AutoCloseable {
    static {
        RocksDB.loadLibrary();
    }

    public final static String DEFAULT_COLUMN_FAMILY = new String(RocksDB.DEFAULT_COLUMN_FAMILY,
            StandardCharsets.UTF_8);

    /**
     * Open a {@link RocksDB} with default options in read-only mode.
     * 
     * @param directory
     *            existing {@link RocksDB} data directory
     * @return
     * @throws RocksDBException
     * @throws IOException
     */
    public static RocksDbWrapper openReadOnly(File directory) throws RocksDbException, IOException {
        RocksDbWrapper rocksDbWrapper = new RocksDbWrapper(directory, true);
        rocksDbWrapper.init();
        return rocksDbWrapper;
    }

    /**
     * Open a {@link RocksDB} with default options in read-only mode.
     * 
     * @param dirPath
     *            existing {@link RocksDB} data directory
     * @return
     * @throws RocksDbException
     * @throws IOException
     */
    public static RocksDbWrapper openReadOnly(String dirPath) throws RocksDbException, IOException {
        RocksDbWrapper rocksDbWrapper = new RocksDbWrapper(dirPath, true);
        rocksDbWrapper.init();
        return rocksDbWrapper;
    }

    /**
     * Open a {@link RocksDB}, specifying options, in read-only mode.
     * 
     * @param directory
     *            existing {@link RocksDB} data directory
     * @param dbOptions
     * @param readOptions
     * @return
     * @throws RocksDBException
     * @throws IOException
     */
    public static RocksDbWrapper openReadOnly(File directory, DBOptions dbOptions,
            ReadOptions readOptions) throws RocksDbException, IOException {
        RocksDbWrapper rocksDbWrapper = new RocksDbWrapper(directory, true);
        rocksDbWrapper.setDbOptions(dbOptions).setReadOptions(readOptions);
        rocksDbWrapper.init();
        return rocksDbWrapper;
    }

    /**
     * Open a {@link RocksDB}, specifying options, in read-only mode.
     * 
     * @param dirPath
     *            existing {@link RocksDB} data directory
     * @param dbOptions
     * @param readOptions
     * @return
     * @throws RocksDBException
     * @throws IOException
     */
    public static RocksDbWrapper openReadOnly(String dirPath, DBOptions dbOptions,
            ReadOptions readOptions) throws RocksDbException, IOException {
        RocksDbWrapper rocksDbWrapper = new RocksDbWrapper(dirPath, true);
        rocksDbWrapper.setDbOptions(dbOptions).setReadOptions(readOptions);
        rocksDbWrapper.init();
        return rocksDbWrapper;
    }

    /**
     * Open a {@link RocksDB} with default options in read-write mode.
     * 
     * @param directory
     *            directory to store {@link RocksDB} data
     * @param columnFamilies
     *            list of column families to store key/value (the column family
     *            "default" will be automatically added)
     * @return
     * @throws RocksDbException
     * @throws IOException
     */
    public static RocksDbWrapper openReadWrite(File directory, String... columnFamilies)
            throws RocksDbException, IOException {
        RocksDbWrapper rocksDbWrapper = new RocksDbWrapper(directory, false);
        rocksDbWrapper.setColumnFamilies(RocksDbUtils.buildColumnFamilyDescriptors(columnFamilies));
        rocksDbWrapper.init();
        return rocksDbWrapper;
    }

    /**
     * Open a {@link RocksDB} with default options in read-write mode.
     * 
     * @param dirPath
     *            directory to store {@link RocksDB} data
     * @param columnFamilies
     *            list of column families to store key/value (the column family
     *            "default" will be automatically added)
     * @return
     * @throws RocksDbException
     * @throws IOException
     */
    public static RocksDbWrapper openReadWrite(String dirPath, String... columnFamilies)
            throws RocksDbException, IOException {
        RocksDbWrapper rocksDbWrapper = new RocksDbWrapper(dirPath, false);
        rocksDbWrapper.setColumnFamilies(RocksDbUtils.buildColumnFamilyDescriptors(columnFamilies));
        rocksDbWrapper.init();
        return rocksDbWrapper;
    }

    /**
     * Open a {@link RocksDB}, specifying options, in read-write mode.
     * 
     * @param directory
     *            directory to store {@link RocksDB} data
     * @param dbOptions
     * @param readOptions
     * @param writeOptions
     * @param columnFamilies
     *            list of column families to store key/value (the column family
     *            "default" will be automatically added)
     * @return
     * @throws RocksDbException
     * @throws IOException
     */
    public static RocksDbWrapper openReadWrite(File directory, DBOptions dbOptions,
            ReadOptions readOptions, WriteOptions writeOptions, String... columnFamilies)
            throws RocksDbException, IOException {
        RocksDbWrapper rocksDbWrapper = new RocksDbWrapper(directory, false);
        rocksDbWrapper.setDbOptions(dbOptions).setReadOptions(readOptions)
                .setWriteOptions(writeOptions);
        rocksDbWrapper.setColumnFamilies(RocksDbUtils.buildColumnFamilyDescriptors(columnFamilies));
        rocksDbWrapper.init();
        return rocksDbWrapper;
    }

    /**
     * Open a {@link RocksDB}, specifying options, in read-write mode.
     * 
     * @param dirPath
     *            directory to store {@link RocksDB} data
     * @param dbOptions
     * @param readOptions
     * @param writeOptions
     * @param columnFamilies
     *            list of column families to store key/value (the column family
     *            "default" will be automatically added)
     * @return
     * @throws RocksDbException
     * @throws IOException
     */
    public static RocksDbWrapper openReadWrite(String dirPath, DBOptions dbOptions,
            ReadOptions readOptions, WriteOptions writeOptions, String... columnFamilies)
            throws RocksDbException, IOException {
        RocksDbWrapper rocksDbWrapper = new RocksDbWrapper(dirPath, false);
        rocksDbWrapper.setDbOptions(dbOptions).setReadOptions(readOptions)
                .setWriteOptions(writeOptions);
        rocksDbWrapper.setColumnFamilies(RocksDbUtils.buildColumnFamilyDescriptors(columnFamilies));
        rocksDbWrapper.init();
        return rocksDbWrapper;
    }
    /*----------------------------------------------------------------------*/

    private final Logger LOGGER = LoggerFactory.getLogger(RocksDbWrapper.class);

    private File directory;
    private final boolean readOnly;
    private RocksDB rocksDb;

    private DBOptions dbOptions;
    private boolean myOwnDbOptions = true;

    private WriteOptions writeOptions;
    private boolean myOwnWriteOptions = true;

    private ReadOptions readOptions;
    private boolean myOwnReadOptions = true;

    private Set columnFamilies = new HashSet<>();
    private Set columnFamilyNames = new HashSet<>();
    private Map columnFamilyHandles = new HashMap<>();
    private Map iterators = new HashMap<>();

    /**
     * Construct a new {@link RocksDbWrapper} object.
     * 
     * @param dirPath
     *            directory to store {@link RocksDB} data
     * @param readOnly
     */
    public RocksDbWrapper(String dirPath, boolean readOnly) {
        this(new File(dirPath), readOnly);
    }

    /**
     * Construct a new {@link RocksDbWrapper} object.
     * 
     * @param directory
     *            directory to store {@link RocksDB} data
     * @param readOnly
     */
    public RocksDbWrapper(File directory, boolean readOnly) {
        this.directory = directory;
        this.readOnly = readOnly;
    }

    synchronized public RocksDbWrapper setColumnFamilies(
            Collection columnFamilies) {
        this.columnFamilies.clear();
        this.columnFamilyNames.clear();
        RocksDbUtils.closeRocksObjects(
                this.columnFamilyHandles.values().toArray(new ColumnFamilyHandle[0]));
        this.columnFamilyHandles.clear();

        if (columnFamilies != null) {
            columnFamilies.forEach(cf -> {
                this.columnFamilies.add(cf);
                this.columnFamilyNames.add(new String(cf.getName(), StandardCharsets.UTF_8));
            });
        }
        return this;
    }

    public Collection getColumnFamilies() {
        return Collections.unmodifiableSet(columnFamilies);
    }

    public Collection getColumnFamilyNames() {
        return Collections.unmodifiableSet(columnFamilyNames);
    }

    synchronized public RocksDbWrapper setDbOptions(DBOptions dbOptions) {
        if (this.dbOptions != null && myOwnDbOptions) {
            RocksDbUtils.closeRocksObjects(this.dbOptions);
        }
        this.dbOptions = dbOptions;
        myOwnDbOptions = false;
        return this;
    }

    public DBOptions getDbOptions() {
        return dbOptions;
    }

    synchronized public RocksDbWrapper setReadOptions(ReadOptions readOptions) {
        if (this.readOptions != null && myOwnReadOptions) {
            RocksDbUtils.closeRocksObjects(this.readOptions);
        }
        this.readOptions = readOptions;
        myOwnReadOptions = false;
        return this;
    }

    public ReadOptions getReadOptions() {
        return readOptions;
    }

    synchronized public RocksDbWrapper setWriteOptions(WriteOptions writeOptions) {
        if (this.writeOptions != null && myOwnWriteOptions) {
            RocksDbUtils.closeRocksObjects(this.writeOptions);
        }
        this.writeOptions = writeOptions;
        myOwnWriteOptions = false;
        return this;
    }

    public WriteOptions getWriteOptions() {
        return writeOptions;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public void close() {
        destroy();
    }

    public void destroy() {
        try {
            RocksDbUtils.closeRocksObjects(iterators.values().toArray(new RocksIterator[0]));
        } catch (Exception e) {
            LOGGER.warn(e.getMessage(), e);
        }

        if (myOwnReadOptions) {
            RocksDbUtils.closeRocksObjects(readOptions);
        }

        if (myOwnWriteOptions) {
            RocksDbUtils.closeRocksObjects(writeOptions);
        }

        if (myOwnDbOptions) {
            RocksDbUtils.closeRocksObjects(dbOptions);
        }

        try {
            RocksDbUtils.closeRocksObjects(
                    columnFamilyHandles.values().toArray(new ColumnFamilyHandle[0]));
        } catch (Exception e) {
            LOGGER.warn(e.getMessage(), e);
        }

        RocksDbUtils.closeRocksObjects(rocksDb);
    }

    private boolean inited = false;

    synchronized public RocksDbWrapper init() throws IOException, RocksDbException {
        if (inited) {
            LOGGER.warn("This wrapper has been already initialized");
            return this;
        }

        if (!directory.exists() && !readOnly) {
            FileUtils.forceMkdir(directory);
        }
        if (!directory.isDirectory() || !directory.exists()) {
            throw new IllegalArgumentException(directory + " does not exist, or not a directory!");
        }

        prepareColumnFamilyDescriptors();

        if (dbOptions == null) {
            dbOptions = RocksDbUtils.defaultDbOptions();
            myOwnDbOptions = true;
        } else {
            myOwnDbOptions = false;
        }

        String path = directory.getAbsolutePath();
        List cfdList = new ArrayList<>(columnFamilies);
        List cfhList = new ArrayList<>();
        try {
            if (readOnly) {
                if (readOptions == null) {
                    readOptions = RocksDbUtils.defaultReadOptions();
                    myOwnReadOptions = true;
                } else {
                    myOwnReadOptions = false;
                }
                rocksDb = RocksDB.openReadOnly(dbOptions, path, cfdList, cfhList);
            } else {
                if (readOptions == null) {
                    readOptions = RocksDbUtils.defaultReadOptions();
                    myOwnReadOptions = true;
                } else {
                    myOwnReadOptions = false;
                }
                if (writeOptions == null) {
                    writeOptions = RocksDbUtils.defaultWriteOptions();
                    myOwnWriteOptions = true;
                } else {
                    myOwnWriteOptions = false;
                }
                rocksDb = RocksDB.open(dbOptions, path, cfdList, cfhList);
            }
        } catch (Exception e) {
            throw e instanceof RocksDbException ? (RocksDbException) e : new RocksDbException(e);
        }
        cfhList.forEach(cfh -> {
            try {
                columnFamilyHandles.put(new String(cfh.getName(), StandardCharsets.UTF_8), cfh);
            } catch (Exception e) {
                throw e instanceof RocksDbException ? (RocksDbException) e
                        : new RocksDbException(e);
            }
        });
        inited = true;
        return this;
    }

    /**
     * If column families are not specified, they will be fetched from existing data directory.
     * 
     * @throws RocksDbException
     */
    protected void prepareColumnFamilyDescriptors() throws RocksDbException {
        if (columnFamilies == null || columnFamilies.size() == 0) {
            if (columnFamilies == null) {
                columnFamilies = new HashSet<>();
            }
            try {
                String[] cfList = RocksDbUtils.getColumnFamilyList(directory.getAbsolutePath());
                for (String cf : cfList) {
                    columnFamilies.add(RocksDbUtils.buildColumnFamilyDescriptor(cf));
                }
            } catch (Exception e) {
                throw e instanceof RocksDbException ? (RocksDbException) e
                        : new RocksDbException(e);
            }
        }

        boolean hasDefaultCf = false;
        for (ColumnFamilyDescriptor cfd : columnFamilies) {
            if (Arrays.equals(cfd.getName(), RocksDB.DEFAULT_COLUMN_FAMILY)) {
                hasDefaultCf = true;
                break;
            }
        }
        if (!hasDefaultCf) {
            columnFamilies.add(new ColumnFamilyDescriptor(RocksDB.DEFAULT_COLUMN_FAMILY));
        }
    }

    /*----------------------------------------------------------------------*/
    public ColumnFamilyHandle getColumnFamilyHandle(String cfName) {
        return columnFamilyHandles.get(cfName);
    }

    /**
     * See {@link RocksDB#compactRange()}.
     * 
     * @throws RocksDbException
     */
    public void compactRange() throws RocksDbException {
        try {
            rocksDb.compactRange();
        } catch (Exception e) {
            throw e instanceof RocksDbException ? (RocksDbException) e : new RocksDbException(e);
        }
    }

    /**
     * See {@link RocksDB#getProperty(ColumnFamilyHandle, String)}.
     * 
     * @param cfName
     * @param name
     * @return
     * @throws RocksDbException
     */
    public String getProperty(String cfName, String name) throws RocksDbException {
        ColumnFamilyHandle cfh = getColumnFamilyHandle(cfName);
        if (cfh == null) {
            throw new RocksDbException.ColumnFamilyNotExists(cfName);
        }
        try {
            return rocksDb.getProperty(cfh, name);
        } catch (Exception e) {
            throw e instanceof RocksDbException ? (RocksDbException) e : new RocksDbException(e);
        }
    }

    /**
     * Get estimated number of keys for a column family.
     * 
     * @param cfName
     * @return
     * @throws RocksDbException
     */
    public long getEstimateNumKeys(String cfName) throws RocksDbException {
        String prop = getProperty(cfName, "rocksdb.estimate-num-keys");
        return prop != null ? Long.parseLong(prop) : 0;
    }

    /**
     * Obtain an iterator for a column family.
     * 
     * 

* Iterators will be automatically closed by this wrapper. *

* * @param cfName * @return */ public RocksIterator getIterator(String cfName) { synchronized (iterators) { RocksIterator it = iterators.get(cfName); if (it == null) { ColumnFamilyHandle cfh = getColumnFamilyHandle(cfName); if (cfh == null) { return null; } it = rocksDb.newIterator(cfh, readOptions); iterators.put(cfName, it); } return it; } } /*----------------------------------------------------------------------*/ /** * Delete a key from the default family. * * @param key * @throws RocksDBException */ public void delete(String key) throws RocksDbException { delete(DEFAULT_COLUMN_FAMILY, key); } /** * Delete a key from the default family, specifying write options.F * * @param writeOptions * @param key * @throws RocksDBException */ public void delete(WriteOptions writeOptions, String key) throws RocksDbException { delete(DEFAULT_COLUMN_FAMILY, writeOptions, key); } /** * Delete a key from a column family. * * @param cfName * @param key * @throws RocksDBException */ public void delete(String cfName, String key) throws RocksDbException { delete(cfName, writeOptions, key); } /** * Delete a key from a column family, specifying write options. * * @param cfName * @param writeOptions * @param key * @throws RocksDBException */ public void delete(String cfName, WriteOptions writeOptions, String key) throws RocksDbException { if (cfName == null) { cfName = DEFAULT_COLUMN_FAMILY; } try { delete(getColumnFamilyHandle(cfName), writeOptions, key.getBytes(StandardCharsets.UTF_8)); } catch (RocksDbException.ColumnFamilyNotExists e) { throw new RocksDbException.ColumnFamilyNotExists(cfName); } } /** * Delete a key. * * @param cfh * @param writeOptions * @param key * @throws RocksDBException */ protected void delete(ColumnFamilyHandle cfh, WriteOptions writeOptions, byte[] key) throws RocksDbException { if (readOnly) { throw new RocksDbException.ReadOnlyException("delete"); } if (cfh == null) { throw new RocksDbException.ColumnFamilyNotExists(); } try { rocksDb.delete(cfh, writeOptions != null ? writeOptions : this.writeOptions, key); } catch (Exception e) { throw e instanceof RocksDbException ? (RocksDbException) e : new RocksDbException(e); } } /*----------------------------------------------------------------------*/ /** * Put a key/value to the default column family. * * @param key * @param value * @throws RocksDbException */ public void put(String key, String value) throws RocksDbException { put(DEFAULT_COLUMN_FAMILY, key, value); } /** * Put a key/value to the default column family, specifying write options. * * @param writeOptions * @param key * @param value * @throws RocksDbException */ public void put(WriteOptions writeOptions, String key, String value) throws RocksDbException { put(DEFAULT_COLUMN_FAMILY, writeOptions, key, value); } /** * Put a key/value to the default column family. * * @param key * @param value * @throws RocksDbException */ public void put(String key, byte[] value) throws RocksDbException { put(DEFAULT_COLUMN_FAMILY, key, value); } /** * Put a key/value to the default column family, specifying write options. * * @param writeOptions * @param key * @param value * @throws RocksDbException */ public void put(WriteOptions writeOptions, String key, byte[] value) throws RocksDbException { put(DEFAULT_COLUMN_FAMILY, writeOptions, key, value); } /** * Put a key/value to a column family. * * @param cfName * @param key * @param value * @throws RocksDbException */ public void put(String cfName, String key, String value) throws RocksDbException { put(cfName, key, value != null ? value.getBytes(StandardCharsets.UTF_8) : null); } /** * Put a key/value to a column family, specifying write options. * * @param cfName * @param writeOptions * @param key * @param value * @throws RocksDbException */ public void put(String cfName, WriteOptions writeOptions, String key, String value) throws RocksDbException { put(cfName, writeOptions, key, value != null ? value.getBytes(StandardCharsets.UTF_8) : null); } /** * Put a key/value to a column family. * * @param cfName * @param key * @param value * @throws RocksDbException */ public void put(String cfName, String key, byte[] value) throws RocksDbException { put(cfName, writeOptions, key, value); } /** * Put a key/value to a column family, specifying write options. * * @param cfName * @param writeOptions * @param key * @param value * @throws RocksDbException */ public void put(String cfName, WriteOptions writeOptions, String key, byte[] value) throws RocksDbException { if (cfName == null) { cfName = DEFAULT_COLUMN_FAMILY; } try { put(columnFamilyHandles.get(cfName), writeOptions, key.getBytes(StandardCharsets.UTF_8), value); } catch (RocksDbException.ColumnFamilyNotExists e) { throw new RocksDbException.ColumnFamilyNotExists(cfName); } } /** * Put a key/value. * * @param cfh * @param writeOptions * @param key * @param value * @throws RocksDbException */ protected void put(ColumnFamilyHandle cfh, WriteOptions writeOptions, byte[] key, byte[] value) throws RocksDbException { if (readOnly) { throw new RocksDbException.ReadOnlyException("put"); } if (cfh == null) { throw new RocksDbException.ColumnFamilyNotExists(); } if (value == null) { delete(cfh, writeOptions, key); } else { try { rocksDb.put(cfh, writeOptions != null ? writeOptions : this.writeOptions, key, value); } catch (Exception e) { throw e instanceof RocksDbException ? (RocksDbException) e : new RocksDbException(e); } } } /*----------------------------------------------------------------------*/ /** * Get a value from the default column family. * * @param key * @return * @throws RocksDbException */ public byte[] get(String key) throws RocksDbException { return get(DEFAULT_COLUMN_FAMILY, key); } /** * Get a value from the default column family, specifying read options. * * @param readOPtions * @param key * @return * @throws RocksDbException */ public byte[] get(ReadOptions readOPtions, String key) throws RocksDbException { return get(DEFAULT_COLUMN_FAMILY, readOPtions, key); } /** * Get a value from a column family. * * @param cfName * @param key * @return * @throws RocksDbException */ public byte[] get(String cfName, String key) throws RocksDbException { return get(cfName, readOptions, key); } /** * Get a value from a column family, specifying read options. * * @param cfName * @param readOptions * @param key * @return * @throws RocksDbException */ public byte[] get(String cfName, ReadOptions readOptions, String key) throws RocksDbException { if (cfName == null) { cfName = DEFAULT_COLUMN_FAMILY; } ColumnFamilyHandle cfh = columnFamilyHandles.get(cfName); if (cfh == null) { throw new RocksDbException.ColumnFamilyNotExists(cfName); } return get(cfh, readOptions, key.getBytes(StandardCharsets.UTF_8)); } /** * Get a value. * * @param cfh * @param readOptions * @param key * @return * @throws RocksDbException */ protected byte[] get(ColumnFamilyHandle cfh, ReadOptions readOptions, byte[] key) throws RocksDbException { try { return rocksDb.get(cfh, readOptions != null ? readOptions : this.readOptions, key); } catch (Exception e) { throw e instanceof RocksDbException ? (RocksDbException) e : new RocksDbException(e); } } /*----------------------------------------------------------------------*/ /** * See {@link RocksDB#write(WriteOptions, WriteBatch)}. * * @param batch * @throws RocksDbException */ public void write(WriteBatch batch) throws RocksDbException { write(writeOptions, batch); } /** * See {@link RocksDB#write(WriteOptions, WriteBatch)}. * * @param writeOptions * @param batch * @throws RocksDbException */ public void write(WriteOptions writeOptions, WriteBatch batch) throws RocksDbException { try { rocksDb.write(writeOptions != null ? writeOptions : this.writeOptions, batch); } catch (Exception e) { throw e instanceof RocksDbException ? (RocksDbException) e : new RocksDbException(e); } } /** * See {@link RocksDB#write(WriteOptions, WriteBatchWithIndex)}. * * @param batch * @throws RocksDbException */ public void write(WriteBatchWithIndex batch) throws RocksDbException { write(writeOptions, batch); } /** * See {@link RocksDB#write(WriteOptions, WriteBatchWithIndex)}. * * @param writeOptions * @param batch * @throws RocksDbException */ public void write(WriteOptions writeOptions, WriteBatchWithIndex batch) throws RocksDbException { try { rocksDb.write(writeOptions != null ? writeOptions : this.writeOptions, batch); } catch (Exception e) { throw e instanceof RocksDbException ? (RocksDbException) e : new RocksDbException(e); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy