mirror of https://github.com/apache/nifi.git
NIFI-4775: FlowFile Repository implementation based on RocksDB
+l from markobean. This closes #3638. Signed-off-by: Brandon <devriesb@apache.org>
This commit is contained in:
parent
2baafcc2e6
commit
7d77b464cc
|
@ -0,0 +1,44 @@
|
||||||
|
<?xml version="1.0"?>
|
||||||
|
<!--
|
||||||
|
Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
contributor license agreements. See the NOTICE file distributed with
|
||||||
|
this work for additional information regarding copyright ownership.
|
||||||
|
The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
(the "License"); you may not use this file except in compliance with
|
||||||
|
the License. You may obtain a copy of the License at
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
-->
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
<parent>
|
||||||
|
<groupId>org.apache.nifi</groupId>
|
||||||
|
<artifactId>nifi-commons</artifactId>
|
||||||
|
<version>1.10.0-SNAPSHOT</version>
|
||||||
|
</parent>
|
||||||
|
<artifactId>nifi-rocksdb-utils</artifactId>
|
||||||
|
<version>1.10.0-SNAPSHOT</version>
|
||||||
|
<packaging>jar</packaging>
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.nifi</groupId>
|
||||||
|
<artifactId>nifi-api</artifactId>
|
||||||
|
<version>1.10.0-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.nifi</groupId>
|
||||||
|
<artifactId>nifi-properties</artifactId>
|
||||||
|
<version>1.10.0-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.rocksdb</groupId>
|
||||||
|
<artifactId>rocksdbjni</artifactId>
|
||||||
|
<version>6.0.1</version>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
</project>
|
|
@ -0,0 +1,891 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.apache.nifi.rocksdb;
|
||||||
|
|
||||||
|
import org.apache.nifi.processor.DataUnit;
|
||||||
|
import org.apache.nifi.util.StringUtils;
|
||||||
|
import org.rocksdb.AccessHint;
|
||||||
|
import org.rocksdb.ColumnFamilyDescriptor;
|
||||||
|
import org.rocksdb.ColumnFamilyHandle;
|
||||||
|
import org.rocksdb.ColumnFamilyOptions;
|
||||||
|
import org.rocksdb.CompressionType;
|
||||||
|
import org.rocksdb.DBOptions;
|
||||||
|
import org.rocksdb.InfoLogLevel;
|
||||||
|
import org.rocksdb.Options;
|
||||||
|
import org.rocksdb.RocksDB;
|
||||||
|
import org.rocksdb.RocksDBException;
|
||||||
|
import org.rocksdb.RocksIterator;
|
||||||
|
import org.rocksdb.WriteOptions;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.io.Closeable;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
import java.util.concurrent.ScheduledExecutorService;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
import java.util.concurrent.locks.Condition;
|
||||||
|
import java.util.concurrent.locks.ReentrantLock;
|
||||||
|
import java.util.concurrent.locks.ReentrantReadWriteLock;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This RocksDB helper class, instead of forcing to disk every time it's given a record,
|
||||||
|
* persists all waiting records on a regular interval (using a ScheduledExecutorService).
|
||||||
|
* Like when a metronome ticks.
|
||||||
|
*/
|
||||||
|
|
||||||
|
public class RocksDBMetronome implements Closeable {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(RocksDBMetronome.class);
|
||||||
|
|
||||||
|
static final String CONFIGURATION_FAMILY = "configuration.column.family";
|
||||||
|
static final String DEFAULT_FAMILY = "default";
|
||||||
|
|
||||||
|
private final AtomicLong lastSyncWarningNanos = new AtomicLong(0L);
|
||||||
|
private final int parallelThreads;
|
||||||
|
private final int maxWriteBufferNumber;
|
||||||
|
private final int minWriteBufferNumberToMerge;
|
||||||
|
private final long writeBufferSize;
|
||||||
|
private final long maxTotalWalSize;
|
||||||
|
private final long delayedWriteRate;
|
||||||
|
private final int level0SlowdownWritesTrigger;
|
||||||
|
private final int level0StopWritesTrigger;
|
||||||
|
private final int maxBackgroundFlushes;
|
||||||
|
private final int maxBackgroundCompactions;
|
||||||
|
private final int statDumpSeconds;
|
||||||
|
private final long syncMillis;
|
||||||
|
private final long syncWarningNanos;
|
||||||
|
private final Path storagePath;
|
||||||
|
private final boolean adviseRandomOnOpen;
|
||||||
|
private final boolean createIfMissing;
|
||||||
|
private final boolean createMissingColumnFamilies;
|
||||||
|
private final boolean useFsync;
|
||||||
|
|
||||||
|
private final Set<byte[]> columnFamilyNames;
|
||||||
|
private final Map<String, ColumnFamilyHandle> columnFamilyHandles;
|
||||||
|
|
||||||
|
private final boolean periodicSyncEnabled;
|
||||||
|
private final ScheduledExecutorService syncExecutor;
|
||||||
|
private final ReentrantLock syncLock = new ReentrantLock();
|
||||||
|
private final Condition syncCondition = syncLock.newCondition();
|
||||||
|
private final AtomicInteger syncCounter = new AtomicInteger(0);
|
||||||
|
|
||||||
|
private volatile RocksDB rocksDB = null;
|
||||||
|
private final ReentrantReadWriteLock dbReadWriteLock = new ReentrantReadWriteLock();
|
||||||
|
private final ReentrantReadWriteLock.ReadLock dbReadLock = dbReadWriteLock.readLock();
|
||||||
|
private final ReentrantReadWriteLock.WriteLock dbWriteLock = dbReadWriteLock.writeLock();
|
||||||
|
private volatile boolean closed = false;
|
||||||
|
|
||||||
|
private ColumnFamilyHandle configurationColumnFamilyHandle;
|
||||||
|
private ColumnFamilyHandle defaultColumnFamilyHandle;
|
||||||
|
private WriteOptions forceSyncWriteOptions;
|
||||||
|
private WriteOptions noSyncWriteOptions;
|
||||||
|
|
||||||
|
private RocksDBMetronome(Builder builder) {
|
||||||
|
statDumpSeconds = builder.statDumpSeconds;
|
||||||
|
parallelThreads = builder.parallelThreads;
|
||||||
|
maxWriteBufferNumber = builder.maxWriteBufferNumber;
|
||||||
|
minWriteBufferNumberToMerge = builder.minWriteBufferNumberToMerge;
|
||||||
|
writeBufferSize = builder.writeBufferSize;
|
||||||
|
maxTotalWalSize = builder.getMaxTotalWalSize();
|
||||||
|
delayedWriteRate = builder.delayedWriteRate;
|
||||||
|
level0SlowdownWritesTrigger = builder.level0SlowdownWritesTrigger;
|
||||||
|
level0StopWritesTrigger = builder.level0StopWritesTrigger;
|
||||||
|
maxBackgroundFlushes = builder.maxBackgroundFlushes;
|
||||||
|
maxBackgroundCompactions = builder.maxBackgroundCompactions;
|
||||||
|
syncMillis = builder.syncMillis;
|
||||||
|
syncWarningNanos = builder.syncWarningNanos;
|
||||||
|
storagePath = builder.storagePath;
|
||||||
|
adviseRandomOnOpen = builder.adviseRandomOnOpen;
|
||||||
|
createIfMissing = builder.createIfMissing;
|
||||||
|
createMissingColumnFamilies = builder.createMissingColumnFamilies;
|
||||||
|
useFsync = builder.useFsync;
|
||||||
|
columnFamilyNames = builder.columnFamilyNames;
|
||||||
|
columnFamilyHandles = new HashMap<>(columnFamilyNames.size());
|
||||||
|
|
||||||
|
periodicSyncEnabled = builder.periodicSyncEnabled;
|
||||||
|
syncExecutor = Executors.newSingleThreadScheduledExecutor(r -> {
|
||||||
|
Thread thread = Executors.defaultThreadFactory().newThread(r);
|
||||||
|
thread.setDaemon(true);
|
||||||
|
return thread;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initialize the metronome
|
||||||
|
*
|
||||||
|
* @throws IOException if there is an issue with the underlying database
|
||||||
|
*/
|
||||||
|
public void initialize() throws IOException {
|
||||||
|
|
||||||
|
final String rocksSharedLibDir = System.getenv("ROCKSDB_SHAREDLIB_DIR");
|
||||||
|
final String javaTmpDir = System.getProperty("java.io.tmpdir");
|
||||||
|
final String libDir = !StringUtils.isBlank(rocksSharedLibDir) ? rocksSharedLibDir : javaTmpDir;
|
||||||
|
try {
|
||||||
|
Files.createDirectories(Paths.get(libDir));
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new IOException("Unable to load the RocksDB shared library into directory: " + libDir, e);
|
||||||
|
}
|
||||||
|
|
||||||
|
// delete any previous librocksdbjni.so files
|
||||||
|
final File[] rocksSos = Paths.get(libDir).toFile().listFiles((dir, name) -> name.startsWith("librocksdbjni") && name.endsWith(".so"));
|
||||||
|
if (rocksSos != null) {
|
||||||
|
for (File rocksSo : rocksSos) {
|
||||||
|
if (!rocksSo.delete()) {
|
||||||
|
logger.warn("Could not delete existing librocksdbjni*.so file {}", rocksSo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
RocksDB.loadLibrary();
|
||||||
|
} catch (Throwable t) {
|
||||||
|
if (System.getProperty("os.name").startsWith("Windows")) {
|
||||||
|
logger.error("The RocksDBMetronome will only work on Windows if you have Visual C++ runtime libraries for Visual Studio 2015 installed. " +
|
||||||
|
"If the DLLs required to support RocksDB cannot be found, then NiFi will not start!");
|
||||||
|
}
|
||||||
|
throw t;
|
||||||
|
}
|
||||||
|
|
||||||
|
Files.createDirectories(storagePath);
|
||||||
|
|
||||||
|
forceSyncWriteOptions = new WriteOptions()
|
||||||
|
.setDisableWAL(false)
|
||||||
|
.setSync(true);
|
||||||
|
|
||||||
|
noSyncWriteOptions = new WriteOptions()
|
||||||
|
.setDisableWAL(false)
|
||||||
|
.setSync(false);
|
||||||
|
|
||||||
|
|
||||||
|
dbWriteLock.lock();
|
||||||
|
try (final DBOptions dbOptions = new DBOptions()
|
||||||
|
.setAccessHintOnCompactionStart(AccessHint.SEQUENTIAL)
|
||||||
|
.setAdviseRandomOnOpen(adviseRandomOnOpen)
|
||||||
|
.setAllowMmapWrites(false) // required to be false for RocksDB.syncWal() to work
|
||||||
|
.setCreateIfMissing(createIfMissing)
|
||||||
|
.setCreateMissingColumnFamilies(createMissingColumnFamilies)
|
||||||
|
.setDelayedWriteRate(delayedWriteRate)
|
||||||
|
.setIncreaseParallelism(parallelThreads)
|
||||||
|
.setLogger(getRocksLogger())
|
||||||
|
.setMaxBackgroundCompactions(maxBackgroundCompactions)
|
||||||
|
.setMaxBackgroundFlushes(maxBackgroundFlushes)
|
||||||
|
.setMaxTotalWalSize(maxTotalWalSize)
|
||||||
|
.setStatsDumpPeriodSec(statDumpSeconds)
|
||||||
|
.setUseFsync(useFsync)
|
||||||
|
;
|
||||||
|
|
||||||
|
final ColumnFamilyOptions cfOptions = new ColumnFamilyOptions()
|
||||||
|
.setCompressionType(CompressionType.LZ4_COMPRESSION)
|
||||||
|
.setLevel0SlowdownWritesTrigger(level0SlowdownWritesTrigger)
|
||||||
|
.setLevel0StopWritesTrigger(level0StopWritesTrigger)
|
||||||
|
.setMaxWriteBufferNumber(maxWriteBufferNumber)
|
||||||
|
.setMinWriteBufferNumberToMerge(minWriteBufferNumberToMerge)
|
||||||
|
.setWriteBufferSize(writeBufferSize)
|
||||||
|
) {
|
||||||
|
|
||||||
|
// create family descriptors
|
||||||
|
List<ColumnFamilyDescriptor> familyDescriptors = new ArrayList<>(columnFamilyNames.size());
|
||||||
|
for (byte[] name : columnFamilyNames) {
|
||||||
|
familyDescriptors.add(new ColumnFamilyDescriptor(name, cfOptions));
|
||||||
|
}
|
||||||
|
|
||||||
|
List<ColumnFamilyHandle> columnFamilyList = new ArrayList<>(columnFamilyNames.size());
|
||||||
|
|
||||||
|
rocksDB = RocksDB.open(dbOptions, storagePath.toString(), familyDescriptors, columnFamilyList);
|
||||||
|
|
||||||
|
// create the map of names to handles
|
||||||
|
columnFamilyHandles.put(DEFAULT_FAMILY, rocksDB.getDefaultColumnFamily());
|
||||||
|
for (ColumnFamilyHandle cf : columnFamilyList) {
|
||||||
|
columnFamilyHandles.put(new String(cf.getName(), StandardCharsets.UTF_8), cf);
|
||||||
|
}
|
||||||
|
|
||||||
|
// set specific special handles
|
||||||
|
defaultColumnFamilyHandle = rocksDB.getDefaultColumnFamily();
|
||||||
|
configurationColumnFamilyHandle = columnFamilyHandles.get(CONFIGURATION_FAMILY);
|
||||||
|
|
||||||
|
} catch (RocksDBException e) {
|
||||||
|
throw new IOException(e);
|
||||||
|
} finally {
|
||||||
|
dbWriteLock.unlock();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (periodicSyncEnabled) {
|
||||||
|
syncExecutor.scheduleWithFixedDelay(this::doSync, syncMillis, syncMillis, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.info("Initialized RocksDB Repository at {}", storagePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method checks the state of the database to ensure it is available for use.
|
||||||
|
* <p>
|
||||||
|
* NOTE: This *must* be called holding the dbReadLock
|
||||||
|
*
|
||||||
|
* @throws IllegalStateException if the database is closed or not yet initialized
|
||||||
|
*/
|
||||||
|
private void checkDbState() throws IllegalStateException {
|
||||||
|
if (rocksDB == null) {
|
||||||
|
if (closed) {
|
||||||
|
throw new IllegalStateException("RocksDBMetronome is closed");
|
||||||
|
}
|
||||||
|
throw new IllegalStateException("RocksDBMetronome has not been initialized");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return an iterator over the specified column family. The iterator is initially invalid (caller must call one of the Seek methods on the iterator before using it).
|
||||||
|
* <p>
|
||||||
|
* Caller should close the iterator when it is no longer needed. The returned iterator should be closed before this db is closed.
|
||||||
|
*
|
||||||
|
* @param columnFamilyHandle specifies the column family for the iterator
|
||||||
|
* @return an iterator over the specified column family
|
||||||
|
*/
|
||||||
|
public RocksIterator getIterator(final ColumnFamilyHandle columnFamilyHandle) {
|
||||||
|
dbReadLock.lock();
|
||||||
|
try {
|
||||||
|
checkDbState();
|
||||||
|
return rocksDB.newIterator(columnFamilyHandle);
|
||||||
|
} finally {
|
||||||
|
dbReadLock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the value for the provided key in the specified column family
|
||||||
|
*
|
||||||
|
* @param columnFamilyHandle the column family from which to get the value
|
||||||
|
* @param key the key of the value to retrieve
|
||||||
|
* @return the value for the specified key
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public byte[] get(final ColumnFamilyHandle columnFamilyHandle, final byte[] key) throws RocksDBException {
|
||||||
|
dbReadLock.lock();
|
||||||
|
try {
|
||||||
|
checkDbState();
|
||||||
|
return rocksDB.get(columnFamilyHandle, key);
|
||||||
|
} finally {
|
||||||
|
dbReadLock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Put the key / value pair into the database in the specified column family
|
||||||
|
*
|
||||||
|
* @param columnFamilyHandle the column family in to which to put the value
|
||||||
|
* @param writeOptions specification of options for write operations
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @param value the value to be associated with the specified key
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void put(final ColumnFamilyHandle columnFamilyHandle, WriteOptions writeOptions, final byte[] key, final byte[] value) throws RocksDBException {
|
||||||
|
dbReadLock.lock();
|
||||||
|
try {
|
||||||
|
checkDbState();
|
||||||
|
rocksDB.put(columnFamilyHandle, writeOptions, key, value);
|
||||||
|
} finally {
|
||||||
|
dbReadLock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete the key / value pair from the specified column family
|
||||||
|
*
|
||||||
|
* @param columnFamilyHandle the column family in to which to put the value
|
||||||
|
* @param writeOptions specification of options for write operations
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void delete(final ColumnFamilyHandle columnFamilyHandle, final byte[] key, final WriteOptions writeOptions) throws RocksDBException {
|
||||||
|
dbReadLock.lock();
|
||||||
|
try {
|
||||||
|
checkDbState();
|
||||||
|
rocksDB.delete(columnFamilyHandle, writeOptions, key);
|
||||||
|
} finally {
|
||||||
|
dbReadLock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Flushes the WAL and syncs to disk
|
||||||
|
*
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void forceSync() throws RocksDBException {
|
||||||
|
dbReadLock.lock();
|
||||||
|
try {
|
||||||
|
checkDbState();
|
||||||
|
rocksDB.syncWal();
|
||||||
|
} finally {
|
||||||
|
dbReadLock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the handle for the specified column family
|
||||||
|
*
|
||||||
|
* @param familyName the name of the column family
|
||||||
|
* @return the handle
|
||||||
|
*/
|
||||||
|
public ColumnFamilyHandle getColumnFamilyHandle(String familyName) {
|
||||||
|
return columnFamilyHandles.get(familyName);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Put the key / value pair into the configuration column family and sync the wal
|
||||||
|
*
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @param value the value to be associated with the specified key
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void putConfiguration(final byte[] key, final byte[] value) throws RocksDBException {
|
||||||
|
put(configurationColumnFamilyHandle, forceSyncWriteOptions, key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Put the key / value pair into the database in the specified column family without syncing the wal
|
||||||
|
*
|
||||||
|
* @param columnFamilyHandle the column family in to which to put the value
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @param value the value to be associated with the specified key
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void put(final ColumnFamilyHandle columnFamilyHandle, final byte[] key, final byte[] value) throws RocksDBException {
|
||||||
|
put(columnFamilyHandle, noSyncWriteOptions, key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Put the key / value pair into the database in the specified column family, optionally syncing the wal
|
||||||
|
*
|
||||||
|
* @param columnFamilyHandle the column family in to which to put the value
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @param value the value to be associated with the specified key
|
||||||
|
* @param forceSync if true, sync the wal
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void put(final ColumnFamilyHandle columnFamilyHandle, final byte[] key, final byte[] value, final boolean forceSync) throws RocksDBException {
|
||||||
|
put(columnFamilyHandle, getWriteOptions(forceSync), key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Put the key / value pair into the database in the default column family, optionally syncing the wal
|
||||||
|
*
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @param value the value to be associated with the specified key
|
||||||
|
* @param forceSync if true, sync the wal
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void put(final byte[] key, final byte[] value, final boolean forceSync) throws RocksDBException {
|
||||||
|
put(defaultColumnFamilyHandle, getWriteOptions(forceSync), key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Put the key / value pair into the database in the default column family, without syncing the wal
|
||||||
|
*
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @param value the value to be associated with the specified key
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void put(final byte[] key, final byte[] value) throws RocksDBException {
|
||||||
|
put(defaultColumnFamilyHandle, noSyncWriteOptions, key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the value for the provided key in the default column family
|
||||||
|
*
|
||||||
|
* @param key the key of the value to retrieve
|
||||||
|
* @return the value for the specified key
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public byte[] get(final byte[] key) throws RocksDBException {
|
||||||
|
return get(defaultColumnFamilyHandle, key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the value for the provided key in the configuration column family
|
||||||
|
*
|
||||||
|
* @param key the key of the value to retrieve
|
||||||
|
* @return the value for the specified key
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public byte[] getConfiguration(final byte[] key) throws RocksDBException {
|
||||||
|
return get(configurationColumnFamilyHandle, key);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete the key / value pair from the default column family without syncing the wal
|
||||||
|
*
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void delete(byte[] key) throws RocksDBException {
|
||||||
|
delete(defaultColumnFamilyHandle, key, noSyncWriteOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete the key / value pair from the default column family, optionally syncing the wal
|
||||||
|
*
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @param forceSync if true, sync the wal
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void delete(byte[] key, final boolean forceSync) throws RocksDBException {
|
||||||
|
delete(defaultColumnFamilyHandle, key, getWriteOptions(forceSync));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete the key / value pair from the default column family without syncing the wal
|
||||||
|
*
|
||||||
|
* @param columnFamilyHandle the column family in to which to put the value
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void delete(final ColumnFamilyHandle columnFamilyHandle, byte[] key) throws RocksDBException {
|
||||||
|
delete(columnFamilyHandle, key, noSyncWriteOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete the key / value pair from the specified column family, optionally syncing the wal
|
||||||
|
*
|
||||||
|
* @param columnFamilyHandle the column family in to which to put the value
|
||||||
|
* @param key the key to be inserted
|
||||||
|
* @param forceSync if true, sync the wal
|
||||||
|
* @throws RocksDBException thrown if there is an error in the underlying library.
|
||||||
|
*/
|
||||||
|
public void delete(final ColumnFamilyHandle columnFamilyHandle, final byte[] key, final boolean forceSync) throws RocksDBException {
|
||||||
|
delete(columnFamilyHandle, key, getWriteOptions(forceSync));
|
||||||
|
}
|
||||||
|
|
||||||
|
private WriteOptions getWriteOptions(boolean forceSync) {
|
||||||
|
return forceSync ? forceSyncWriteOptions : noSyncWriteOptions;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return an iterator over the default column family. The iterator is initially invalid (caller must call one of the Seek methods on the iterator before using it).
|
||||||
|
* <p>
|
||||||
|
* Caller should close the iterator when it is no longer needed. The returned iterator should be closed before this db is closed.
|
||||||
|
*
|
||||||
|
* @return an iterator over the default column family
|
||||||
|
*/
|
||||||
|
public RocksIterator getIterator() {
|
||||||
|
return getIterator(defaultColumnFamilyHandle);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the current value of the sync counter. This can be used with waitForSync() to verify that operations have been synced to disk
|
||||||
|
*
|
||||||
|
* @return the current value of the sync counter
|
||||||
|
*/
|
||||||
|
public int getSyncCounterValue() {
|
||||||
|
return syncCounter.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Close the Metronome and the RocksDB Objects
|
||||||
|
*
|
||||||
|
* @throws IOException if there are issues in closing any of the underlying RocksDB objects
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void close() throws IOException {
|
||||||
|
|
||||||
|
logger.info("Closing RocksDBMetronome");
|
||||||
|
|
||||||
|
dbWriteLock.lock();
|
||||||
|
try {
|
||||||
|
logger.info("Shutting down RocksDBMetronome sync executor");
|
||||||
|
syncExecutor.shutdownNow();
|
||||||
|
|
||||||
|
try {
|
||||||
|
logger.info("Pausing RocksDB background work");
|
||||||
|
rocksDB.pauseBackgroundWork();
|
||||||
|
} catch (RocksDBException e) {
|
||||||
|
logger.warn("Unable to pause background work before close.", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
final AtomicReference<Exception> exceptionReference = new AtomicReference<>();
|
||||||
|
|
||||||
|
logger.info("Closing RocksDB configurations");
|
||||||
|
|
||||||
|
safeClose(forceSyncWriteOptions, exceptionReference);
|
||||||
|
safeClose(noSyncWriteOptions, exceptionReference);
|
||||||
|
|
||||||
|
// close the column family handles first, then the db last
|
||||||
|
for (ColumnFamilyHandle cfh : columnFamilyHandles.values()) {
|
||||||
|
safeClose(cfh, exceptionReference);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.info("Closing RocksDB database");
|
||||||
|
safeClose(rocksDB, exceptionReference);
|
||||||
|
rocksDB = null;
|
||||||
|
closed = true;
|
||||||
|
|
||||||
|
if (exceptionReference.get() != null) {
|
||||||
|
throw new IOException(exceptionReference.get());
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
dbWriteLock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param autoCloseable An {@link AutoCloseable} to be closed
|
||||||
|
* @param exceptionReference A reference to contain any encountered {@link Exception}
|
||||||
|
*/
|
||||||
|
private void safeClose(final AutoCloseable autoCloseable, final AtomicReference<Exception> exceptionReference) {
|
||||||
|
if (autoCloseable != null) {
|
||||||
|
try {
|
||||||
|
autoCloseable.close();
|
||||||
|
} catch (Exception e) {
|
||||||
|
exceptionReference.set(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return The capacity of the store
|
||||||
|
* @throws IOException if encountered
|
||||||
|
*/
|
||||||
|
public long getStorageCapacity() throws IOException {
|
||||||
|
return Files.getFileStore(storagePath).getTotalSpace();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return The usable space of the store
|
||||||
|
* @throws IOException if encountered
|
||||||
|
*/
|
||||||
|
public long getUsableStorageSpace() throws IOException {
|
||||||
|
return Files.getFileStore(storagePath).getUsableSpace();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method is scheduled by the syncExecutor. It runs at the specified interval, forcing the RocksDB WAL to disk.
|
||||||
|
* <p>
|
||||||
|
* If the sync is successful, it notifies threads that had been waiting for their records to be persisted using
|
||||||
|
* syncCondition and the syncCounter, which is incremented to indicate success
|
||||||
|
*/
|
||||||
|
void doSync() {
|
||||||
|
syncLock.lock();
|
||||||
|
try {
|
||||||
|
// if we're interrupted, return
|
||||||
|
if (Thread.currentThread().isInterrupted()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
forceSync();
|
||||||
|
syncCounter.incrementAndGet(); // its ok if it rolls over... we're just going to check that the value changed
|
||||||
|
syncCondition.signalAll();
|
||||||
|
} catch (final IllegalArgumentException e) {
|
||||||
|
logger.error("Unable to sync, likely because the repository is out of space.", e);
|
||||||
|
} catch (final Throwable t) {
|
||||||
|
logger.error("Unable to sync", t);
|
||||||
|
} finally {
|
||||||
|
syncLock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method blocks until the next time the WAL is forced to disk, ensuring that all records written before this point have been persisted.
|
||||||
|
*/
|
||||||
|
public void waitForSync() throws InterruptedException {
|
||||||
|
final int counterValue = syncCounter.get();
|
||||||
|
waitForSync(counterValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method blocks until the WAL has been forced to disk, ensuring that all records written before the point specified by the counterValue have been persisted.
|
||||||
|
*
|
||||||
|
* @param counterValue The value of the counter at the time of a write we must persist
|
||||||
|
* @throws InterruptedException if the thread is interrupted
|
||||||
|
*/
|
||||||
|
public void waitForSync(final int counterValue) throws InterruptedException {
|
||||||
|
if (counterValue != syncCounter.get()) {
|
||||||
|
return; // if the counter has already changed, we don't need to wait (or grab the lock) because the records we're concerned with have already been persisted
|
||||||
|
}
|
||||||
|
long waitTimeRemaining = syncWarningNanos;
|
||||||
|
syncLock.lock();
|
||||||
|
try {
|
||||||
|
while (counterValue == syncCounter.get()) { // wait until the counter changes (indicating sync occurred)
|
||||||
|
waitTimeRemaining = syncCondition.awaitNanos(waitTimeRemaining);
|
||||||
|
if (waitTimeRemaining <= 0L) { // this means the wait timed out
|
||||||
|
|
||||||
|
// only log a warning every syncWarningNanos... don't spam the logs
|
||||||
|
final long now = System.nanoTime();
|
||||||
|
final long lastWarning = lastSyncWarningNanos.get();
|
||||||
|
if (now - lastWarning > syncWarningNanos
|
||||||
|
&& lastSyncWarningNanos.compareAndSet(lastWarning, now)) {
|
||||||
|
logger.warn("Failed to sync within {} seconds... system configuration may need to be adjusted", TimeUnit.NANOSECONDS.toSeconds(syncWarningNanos));
|
||||||
|
}
|
||||||
|
|
||||||
|
// reset waiting time
|
||||||
|
waitTimeRemaining = syncWarningNanos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
syncLock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a representation of a long as a byte array
|
||||||
|
*
|
||||||
|
* @param value a long to convert
|
||||||
|
* @return a byte[] representation
|
||||||
|
*/
|
||||||
|
public static byte[] getBytes(long value) {
|
||||||
|
byte[] bytes = new byte[8];
|
||||||
|
writeLong(value, bytes);
|
||||||
|
return bytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes a representation of a long to the specified byte array
|
||||||
|
*
|
||||||
|
* @param l a long to convert
|
||||||
|
* @param bytes an array to store the byte representation
|
||||||
|
*/
|
||||||
|
public static void writeLong(long l, byte[] bytes) {
|
||||||
|
bytes[0] = (byte) (l >>> 56);
|
||||||
|
bytes[1] = (byte) (l >>> 48);
|
||||||
|
bytes[2] = (byte) (l >>> 40);
|
||||||
|
bytes[3] = (byte) (l >>> 32);
|
||||||
|
bytes[4] = (byte) (l >>> 24);
|
||||||
|
bytes[5] = (byte) (l >>> 16);
|
||||||
|
bytes[6] = (byte) (l >>> 8);
|
||||||
|
bytes[7] = (byte) (l);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a long from it's byte array representation
|
||||||
|
* @param bytes to convert to a long
|
||||||
|
* @return a long
|
||||||
|
* @throws IOException if the given byte array is of the wrong size
|
||||||
|
*/
|
||||||
|
public static long readLong(final byte[] bytes) throws IOException {
|
||||||
|
if (bytes.length != 8) {
|
||||||
|
throw new IOException("wrong number of bytes to convert to long (must be 8)");
|
||||||
|
}
|
||||||
|
return (((long) (bytes[0]) << 56) +
|
||||||
|
((long) (bytes[1] & 255) << 48) +
|
||||||
|
((long) (bytes[2] & 255) << 40) +
|
||||||
|
((long) (bytes[3] & 255) << 32) +
|
||||||
|
((long) (bytes[4] & 255) << 24) +
|
||||||
|
((long) (bytes[5] & 255) << 16) +
|
||||||
|
((long) (bytes[6] & 255) << 8) +
|
||||||
|
((long) (bytes[7] & 255)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the storage path of the db
|
||||||
|
*/
|
||||||
|
public Path getStoragePath() {
|
||||||
|
return storagePath;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return A RocksDB logger capturing all logging output from RocksDB
|
||||||
|
*/
|
||||||
|
private org.rocksdb.Logger getRocksLogger() {
|
||||||
|
try (Options options = new Options()
|
||||||
|
// make RocksDB give us everything, and we'll decide what we want to log in our wrapper
|
||||||
|
.setInfoLogLevel(InfoLogLevel.DEBUG_LEVEL)) {
|
||||||
|
return new LogWrapper(options);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An Extension of org.rocksdb.Logger that wraps the slf4j Logger
|
||||||
|
*/
|
||||||
|
private class LogWrapper extends org.rocksdb.Logger {
|
||||||
|
|
||||||
|
LogWrapper(Options options) {
|
||||||
|
super(options);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void log(final InfoLogLevel infoLogLevel, final String logMsg) {
|
||||||
|
switch (infoLogLevel) {
|
||||||
|
case ERROR_LEVEL:
|
||||||
|
case FATAL_LEVEL:
|
||||||
|
logger.error(logMsg);
|
||||||
|
break;
|
||||||
|
case WARN_LEVEL:
|
||||||
|
logger.warn(logMsg);
|
||||||
|
break;
|
||||||
|
case DEBUG_LEVEL:
|
||||||
|
logger.debug(logMsg);
|
||||||
|
break;
|
||||||
|
case INFO_LEVEL:
|
||||||
|
case HEADER_LEVEL:
|
||||||
|
default:
|
||||||
|
logger.info(logMsg);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class Builder {
|
||||||
|
|
||||||
|
int parallelThreads = 8;
|
||||||
|
int maxWriteBufferNumber = 4;
|
||||||
|
int minWriteBufferNumberToMerge = 1;
|
||||||
|
long writeBufferSize = (long) DataUnit.MB.toB(256);
|
||||||
|
long delayedWriteRate = (long) DataUnit.MB.toB(16);
|
||||||
|
int level0SlowdownWritesTrigger = 20;
|
||||||
|
int level0StopWritesTrigger = 40;
|
||||||
|
int maxBackgroundFlushes = 1;
|
||||||
|
int maxBackgroundCompactions = 1;
|
||||||
|
int statDumpSeconds = 600;
|
||||||
|
long syncMillis = 10;
|
||||||
|
long syncWarningNanos = TimeUnit.SECONDS.toNanos(30);
|
||||||
|
Path storagePath;
|
||||||
|
boolean adviseRandomOnOpen = false;
|
||||||
|
boolean createIfMissing = true;
|
||||||
|
boolean createMissingColumnFamilies = true;
|
||||||
|
boolean useFsync = true;
|
||||||
|
boolean periodicSyncEnabled = true;
|
||||||
|
final Set<byte[]> columnFamilyNames = new HashSet<>();
|
||||||
|
|
||||||
|
public RocksDBMetronome build() {
|
||||||
|
if (storagePath == null) {
|
||||||
|
throw new IllegalStateException("Cannot create RocksDBMetronome because storagePath is not set");
|
||||||
|
}
|
||||||
|
|
||||||
|
// add default column families
|
||||||
|
columnFamilyNames.add(RocksDB.DEFAULT_COLUMN_FAMILY);
|
||||||
|
columnFamilyNames.add(CONFIGURATION_FAMILY.getBytes(StandardCharsets.UTF_8));
|
||||||
|
|
||||||
|
return new RocksDBMetronome(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder addColumnFamily(String name) {
|
||||||
|
this.columnFamilyNames.add(name.getBytes(StandardCharsets.UTF_8));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setStoragePath(Path storagePath) {
|
||||||
|
this.storagePath = storagePath;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setParallelThreads(int parallelThreads) {
|
||||||
|
this.parallelThreads = parallelThreads;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setMaxWriteBufferNumber(int maxWriteBufferNumber) {
|
||||||
|
this.maxWriteBufferNumber = maxWriteBufferNumber;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setMinWriteBufferNumberToMerge(int minWriteBufferNumberToMerge) {
|
||||||
|
this.minWriteBufferNumberToMerge = minWriteBufferNumberToMerge;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setWriteBufferSize(long writeBufferSize) {
|
||||||
|
this.writeBufferSize = writeBufferSize;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setDelayedWriteRate(long delayedWriteRate) {
|
||||||
|
this.delayedWriteRate = delayedWriteRate;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setLevel0SlowdownWritesTrigger(int level0SlowdownWritesTrigger) {
|
||||||
|
this.level0SlowdownWritesTrigger = level0SlowdownWritesTrigger;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setLevel0StopWritesTrigger(int level0StopWritesTrigger) {
|
||||||
|
this.level0StopWritesTrigger = level0StopWritesTrigger;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setMaxBackgroundFlushes(int maxBackgroundFlushes) {
|
||||||
|
this.maxBackgroundFlushes = maxBackgroundFlushes;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setMaxBackgroundCompactions(int maxBackgroundCompactions) {
|
||||||
|
this.maxBackgroundCompactions = maxBackgroundCompactions;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setStatDumpSeconds(int statDumpSeconds) {
|
||||||
|
this.statDumpSeconds = statDumpSeconds;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setSyncMillis(long syncMillis) {
|
||||||
|
this.syncMillis = syncMillis;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setSyncWarningNanos(long syncWarningNanos) {
|
||||||
|
this.syncWarningNanos = syncWarningNanos;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public Builder setAdviseRandomOnOpen(boolean adviseRandomOnOpen) {
|
||||||
|
this.adviseRandomOnOpen = adviseRandomOnOpen;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setCreateMissingColumnFamilies(boolean createMissingColumnFamilies) {
|
||||||
|
this.createMissingColumnFamilies = createMissingColumnFamilies;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setCreateIfMissing(boolean createIfMissing) {
|
||||||
|
this.createIfMissing = createIfMissing;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setUseFsync(boolean useFsync) {
|
||||||
|
this.useFsync = useFsync;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder setPeriodicSyncEnabled(boolean periodicSyncEnabled) {
|
||||||
|
this.periodicSyncEnabled = periodicSyncEnabled;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
long getMaxTotalWalSize() {
|
||||||
|
return writeBufferSize * maxWriteBufferNumber;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,331 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.apache.nifi.rocksdb;
|
||||||
|
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.TemporaryFolder;
|
||||||
|
import org.rocksdb.ColumnFamilyHandle;
|
||||||
|
import org.rocksdb.RocksIterator;
|
||||||
|
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Random;
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
import java.util.concurrent.Future;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.TimeoutException;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertArrayEquals;
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
|
public class TestRocksDBMetronome {
|
||||||
|
|
||||||
|
@Rule
|
||||||
|
public TemporaryFolder temporaryFolder = new TemporaryFolder();
|
||||||
|
|
||||||
|
private static final byte[] KEY = "key".getBytes(StandardCharsets.UTF_8);
|
||||||
|
private static final byte[] VALUE = "value".getBytes(StandardCharsets.UTF_8);
|
||||||
|
private static final byte[] KEY_2 = "key 2".getBytes(StandardCharsets.UTF_8);
|
||||||
|
private static final byte[] VALUE_2 = "value 2".getBytes(StandardCharsets.UTF_8);
|
||||||
|
|
||||||
|
private ExecutorService executor;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void before() {
|
||||||
|
executor = Executors.newSingleThreadExecutor();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void after() {
|
||||||
|
executor.shutdownNow();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testReadWriteLong() throws Exception {
|
||||||
|
Random random = new Random();
|
||||||
|
byte[] key = new byte[8];
|
||||||
|
for (long i = 0; i < 10; i++) {
|
||||||
|
{
|
||||||
|
RocksDBMetronome.writeLong(i, key);
|
||||||
|
assertEquals(i, RocksDBMetronome.readLong(key));
|
||||||
|
}
|
||||||
|
{
|
||||||
|
long testValue = Long.MIN_VALUE + i;
|
||||||
|
RocksDBMetronome.writeLong(testValue, key);
|
||||||
|
assertEquals(testValue, RocksDBMetronome.readLong(key));
|
||||||
|
}
|
||||||
|
{
|
||||||
|
long testValue = Long.MAX_VALUE - i;
|
||||||
|
RocksDBMetronome.writeLong(testValue, key);
|
||||||
|
assertEquals(testValue, RocksDBMetronome.readLong(key));
|
||||||
|
}
|
||||||
|
{
|
||||||
|
long testValue = random.nextLong();
|
||||||
|
RocksDBMetronome.writeLong(testValue, key);
|
||||||
|
assertEquals(testValue, RocksDBMetronome.readLong(key));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPutGetDelete() throws Exception {
|
||||||
|
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.build()) {
|
||||||
|
db.initialize();
|
||||||
|
|
||||||
|
assertNull(db.get(KEY));
|
||||||
|
|
||||||
|
// test default (no sync)
|
||||||
|
db.put(KEY, VALUE);
|
||||||
|
assertArrayEquals(VALUE, db.get(KEY));
|
||||||
|
db.delete(KEY);
|
||||||
|
assertNull(db.get(KEY));
|
||||||
|
|
||||||
|
// test with "force sync"
|
||||||
|
db.put(KEY, VALUE, true);
|
||||||
|
assertArrayEquals(VALUE, db.get(KEY));
|
||||||
|
db.delete(KEY, true);
|
||||||
|
assertNull(db.get(KEY));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPutGetConfiguration() throws Exception {
|
||||||
|
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.build()) {
|
||||||
|
db.initialize();
|
||||||
|
|
||||||
|
assertNull(db.getConfiguration(KEY));
|
||||||
|
db.putConfiguration(KEY, VALUE);
|
||||||
|
assertArrayEquals(VALUE, db.getConfiguration(KEY));
|
||||||
|
db.delete(db.getColumnFamilyHandle(RocksDBMetronome.CONFIGURATION_FAMILY), KEY);
|
||||||
|
assertNull(db.getConfiguration(KEY));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = IllegalStateException.class)
|
||||||
|
public void testPutBeforeInit() throws Exception {
|
||||||
|
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.build()) {
|
||||||
|
db.put(KEY, VALUE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = IllegalStateException.class)
|
||||||
|
public void testPutClosed() throws Exception {
|
||||||
|
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.build()) {
|
||||||
|
db.initialize();
|
||||||
|
|
||||||
|
db.close();
|
||||||
|
db.put(KEY_2, VALUE_2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testColumnFamilies() throws Exception {
|
||||||
|
|
||||||
|
String secondFamilyName = "second family";
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.addColumnFamily(secondFamilyName)
|
||||||
|
.build()) {
|
||||||
|
db.initialize();
|
||||||
|
ColumnFamilyHandle secondFamily = db.getColumnFamilyHandle(secondFamilyName);
|
||||||
|
|
||||||
|
// assert nothing present
|
||||||
|
assertNull(db.get(KEY));
|
||||||
|
assertNull(db.get(KEY_2));
|
||||||
|
|
||||||
|
assertNull(db.get(secondFamily, KEY));
|
||||||
|
assertNull(db.get(secondFamily, KEY_2));
|
||||||
|
|
||||||
|
// add values
|
||||||
|
db.put(KEY, VALUE);
|
||||||
|
db.put(secondFamily, KEY_2, VALUE_2);
|
||||||
|
|
||||||
|
// assert values present in correct family
|
||||||
|
assertArrayEquals(VALUE, db.get(KEY));
|
||||||
|
assertNull(db.get(KEY_2));
|
||||||
|
|
||||||
|
assertArrayEquals(VALUE_2, db.get(secondFamily, KEY_2));
|
||||||
|
assertNull(db.get(secondFamily, KEY));
|
||||||
|
|
||||||
|
// delete from the "wrong" family
|
||||||
|
db.delete(KEY_2);
|
||||||
|
db.delete(secondFamily, KEY);
|
||||||
|
|
||||||
|
// assert values *still* present in correct family
|
||||||
|
assertArrayEquals(VALUE, db.get(KEY));
|
||||||
|
assertNull(db.get(KEY_2));
|
||||||
|
|
||||||
|
assertArrayEquals(VALUE_2, db.get(secondFamily, KEY_2));
|
||||||
|
assertNull(db.get(secondFamily, KEY));
|
||||||
|
|
||||||
|
// delete from the "right" family
|
||||||
|
db.delete(KEY);
|
||||||
|
db.delete(secondFamily, KEY_2);
|
||||||
|
|
||||||
|
// assert values removed
|
||||||
|
assertNull(db.get(KEY));
|
||||||
|
assertNull(db.get(KEY_2));
|
||||||
|
|
||||||
|
assertNull(db.get(secondFamily, KEY));
|
||||||
|
assertNull(db.get(secondFamily, KEY_2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIterator() throws Exception {
|
||||||
|
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.build()) {
|
||||||
|
db.initialize();
|
||||||
|
|
||||||
|
db.put(KEY, VALUE);
|
||||||
|
db.put(KEY_2, VALUE_2);
|
||||||
|
|
||||||
|
RocksIterator iterator = db.getIterator();
|
||||||
|
iterator.seekToFirst();
|
||||||
|
|
||||||
|
Map<String, byte[]> recovered = new HashMap<>();
|
||||||
|
|
||||||
|
while (iterator.isValid()) {
|
||||||
|
recovered.put(new String(iterator.key(), StandardCharsets.UTF_8), iterator.value());
|
||||||
|
iterator.next();
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEquals(2, recovered.size());
|
||||||
|
assertArrayEquals(VALUE, recovered.get(new String(KEY, StandardCharsets.UTF_8)));
|
||||||
|
assertArrayEquals(VALUE_2, recovered.get(new String(KEY_2, StandardCharsets.UTF_8)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCounterIncrement() throws Exception {
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.setSyncMillis(Long.MAX_VALUE) // effectively disable the auto-sync
|
||||||
|
.build()) {
|
||||||
|
db.initialize();
|
||||||
|
|
||||||
|
// get initial counter value
|
||||||
|
int counterValue = db.getSyncCounterValue();
|
||||||
|
|
||||||
|
// do the sync (which would normally happen via the db's internal executor)
|
||||||
|
db.doSync();
|
||||||
|
|
||||||
|
// assert counter value incremented
|
||||||
|
assertEquals(counterValue + 1, db.getSyncCounterValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 10_000)
|
||||||
|
public void testWaitForSync() throws Exception {
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.setSyncMillis(Long.MAX_VALUE) // effectively disable the auto-sync
|
||||||
|
.build()) {
|
||||||
|
db.initialize();
|
||||||
|
|
||||||
|
Future<Boolean> future = executor.submit(() -> {
|
||||||
|
db.waitForSync();
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
|
||||||
|
// the future should still be blocked waiting for sync to happen
|
||||||
|
assertFalse(future.isDone());
|
||||||
|
|
||||||
|
// give the future time to wake up and complete
|
||||||
|
while (!future.isDone()) {
|
||||||
|
// TESTING NOTE: this is inside a loop to address a minor *testing* race condition where our first doSync() could happen before the future runs,
|
||||||
|
// meaning waitForSync() would be left waiting on another doSync() that never comes...
|
||||||
|
|
||||||
|
// do the sync (which would normally happen via the db's internal executor)
|
||||||
|
db.doSync();
|
||||||
|
Thread.sleep(25);
|
||||||
|
}
|
||||||
|
|
||||||
|
// the future should no longer be blocked
|
||||||
|
assertTrue(future.isDone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 10_000)
|
||||||
|
public void testWaitForSyncWithValue() throws Exception {
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder()
|
||||||
|
.setStoragePath(temporaryFolder.newFolder().toPath())
|
||||||
|
.setSyncMillis(Long.MAX_VALUE) // effectively disable the auto-sync
|
||||||
|
.build()) {
|
||||||
|
db.initialize();
|
||||||
|
|
||||||
|
int syncCounterValue = db.getSyncCounterValue();
|
||||||
|
|
||||||
|
// "wait" for one before current counter value... should not block
|
||||||
|
db.waitForSync(syncCounterValue - 1);
|
||||||
|
|
||||||
|
// wait for current value... should block (because auto-sync isn't happening)
|
||||||
|
assertBlocks(db, syncCounterValue);
|
||||||
|
|
||||||
|
// do the sync (which would normally happen via the db's internal executor)
|
||||||
|
db.doSync();
|
||||||
|
|
||||||
|
// "wait" for initial value... should now not block
|
||||||
|
db.waitForSync(syncCounterValue);
|
||||||
|
|
||||||
|
// wait for current value again... should block (because auto-sync isn't happening)
|
||||||
|
assertBlocks(db, db.getSyncCounterValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertBlocks(RocksDBMetronome db, int counterValue) throws InterruptedException, java.util.concurrent.ExecutionException {
|
||||||
|
Future<Boolean> future = getWaitForSyncFuture(db, counterValue);
|
||||||
|
|
||||||
|
try {
|
||||||
|
future.get(1, TimeUnit.SECONDS);
|
||||||
|
fail();
|
||||||
|
} catch (TimeoutException expected) {
|
||||||
|
assertFalse(future.isDone());
|
||||||
|
}
|
||||||
|
future.cancel(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Future<Boolean> getWaitForSyncFuture(RocksDBMetronome db, int counterValue) {
|
||||||
|
return executor.submit(() -> {
|
||||||
|
db.waitForSync(counterValue);
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,22 @@
|
||||||
|
# Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
# contributor license agreements. See the NOTICE file distributed with
|
||||||
|
# this work for additional information regarding copyright ownership.
|
||||||
|
# The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
# (the "License"); you may not use this file except in compliance with
|
||||||
|
# the License. You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
log4j.rootLogger=INFO,console
|
||||||
|
log4j.category.org.apache.nifi=DEBUG
|
||||||
|
|
||||||
|
log4j.appender.console=org.apache.log4j.ConsoleAppender
|
||||||
|
log4j.appender.console.target=System.err
|
||||||
|
log4j.appender.console.layout=org.apache.log4j.PatternLayout
|
||||||
|
log4j.appender.console.layout.ConversionPattern=%d{yy/MM/dd HH:mm:ss} %p %c{2}: %m%n
|
|
@ -12,7 +12,9 @@
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
See the License for the specific language governing permissions and
|
See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
--><project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
-->
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
<modelVersion>4.0.0</modelVersion>
|
<modelVersion>4.0.0</modelVersion>
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>org.apache.nifi</groupId>
|
<groupId>org.apache.nifi</groupId>
|
||||||
|
@ -24,22 +26,23 @@
|
||||||
<packaging>pom</packaging>
|
<packaging>pom</packaging>
|
||||||
<modules>
|
<modules>
|
||||||
<module>nifi-data-provenance-utils</module>
|
<module>nifi-data-provenance-utils</module>
|
||||||
<module>nifi-flowfile-packager</module>
|
|
||||||
<module>nifi-expression-language</module>
|
<module>nifi-expression-language</module>
|
||||||
<module>nifi-logging-utils</module>
|
<module>nifi-flowfile-packager</module>
|
||||||
<module>nifi-properties</module>
|
|
||||||
<module>nifi-security-utils</module>
|
|
||||||
<module>nifi-socket-utils</module>
|
|
||||||
<module>nifi-utils</module>
|
|
||||||
<module>nifi-json-utils</module>
|
|
||||||
<module>nifi-web-utils</module>
|
|
||||||
<module>nifi-write-ahead-log</module>
|
|
||||||
<module>nifi-site-to-site-client</module>
|
|
||||||
<module>nifi-hl7-query-language</module>
|
<module>nifi-hl7-query-language</module>
|
||||||
<module>nifi-schema-utils</module>
|
<module>nifi-json-utils</module>
|
||||||
<module>nifi-record</module>
|
<module>nifi-logging-utils</module>
|
||||||
<module>nifi-record-path</module>
|
|
||||||
<module>nifi-metrics</module>
|
<module>nifi-metrics</module>
|
||||||
<module>nifi-parameter</module>
|
<module>nifi-parameter</module>
|
||||||
|
<module>nifi-properties</module>
|
||||||
|
<module>nifi-record</module>
|
||||||
|
<module>nifi-record-path</module>
|
||||||
|
<module>nifi-rocksdb-utils</module>
|
||||||
|
<module>nifi-schema-utils</module>
|
||||||
|
<module>nifi-security-utils</module>
|
||||||
|
<module>nifi-site-to-site-client</module>
|
||||||
|
<module>nifi-socket-utils</module>
|
||||||
|
<module>nifi-utils</module>
|
||||||
|
<module>nifi-web-utils</module>
|
||||||
|
<module>nifi-write-ahead-log</module>
|
||||||
</modules>
|
</modules>
|
||||||
</project>
|
</project>
|
||||||
|
|
|
@ -143,6 +143,11 @@
|
||||||
<artifactId>nifi-write-ahead-log</artifactId>
|
<artifactId>nifi-write-ahead-log</artifactId>
|
||||||
<version>1.10.0-SNAPSHOT</version>
|
<version>1.10.0-SNAPSHOT</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.nifi</groupId>
|
||||||
|
<artifactId>nifi-rocksdb-utils</artifactId>
|
||||||
|
<version>1.10.0-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.apache.nifi</groupId>
|
<groupId>org.apache.nifi</groupId>
|
||||||
<artifactId>nifi-flowfile-repo-serialization</artifactId>
|
<artifactId>nifi-flowfile-repo-serialization</artifactId>
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -515,7 +515,7 @@ public class WriteAheadFlowFileRepository implements FlowFileRepository, SyncLis
|
||||||
updateRepository(repoRecords, true);
|
updateRepository(repoRecords, true);
|
||||||
|
|
||||||
synchronized (this.swapLocationSuffixes) {
|
synchronized (this.swapLocationSuffixes) {
|
||||||
this.swapLocationSuffixes.add(normalizeSwapLocation(swapLocation));
|
this.swapLocationSuffixes.remove(normalizeSwapLocation(swapLocation));
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info("Repository updated to reflect that {} FlowFiles were swapped in to {}", new Object[]{swapRecords.size(), queue});
|
logger.info("Repository updated to reflect that {} FlowFiles were swapped in to {}", new Object[]{swapRecords.size(), queue});
|
||||||
|
|
|
@ -12,5 +12,6 @@
|
||||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
org.apache.nifi.controller.repository.WriteAheadFlowFileRepository
|
org.apache.nifi.controller.repository.RocksDBFlowFileRepository
|
||||||
org.apache.nifi.controller.repository.VolatileFlowFileRepository
|
org.apache.nifi.controller.repository.VolatileFlowFileRepository
|
||||||
|
org.apache.nifi.controller.repository.WriteAheadFlowFileRepository
|
||||||
|
|
|
@ -0,0 +1,807 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.apache.nifi.controller.repository;
|
||||||
|
|
||||||
|
import org.apache.nifi.connectable.Connectable;
|
||||||
|
import org.apache.nifi.connectable.Connection;
|
||||||
|
import org.apache.nifi.controller.queue.FlowFileQueue;
|
||||||
|
import org.apache.nifi.controller.queue.NopConnectionEventListener;
|
||||||
|
import org.apache.nifi.controller.queue.QueueSize;
|
||||||
|
import org.apache.nifi.controller.queue.StandardFlowFileQueue;
|
||||||
|
import org.apache.nifi.controller.repository.claim.ContentClaim;
|
||||||
|
import org.apache.nifi.controller.repository.claim.ResourceClaim;
|
||||||
|
import org.apache.nifi.controller.repository.claim.ResourceClaimManager;
|
||||||
|
import org.apache.nifi.controller.repository.claim.StandardContentClaim;
|
||||||
|
import org.apache.nifi.controller.repository.claim.StandardResourceClaimManager;
|
||||||
|
import org.apache.nifi.controller.swap.StandardSwapContents;
|
||||||
|
import org.apache.nifi.controller.swap.StandardSwapSummary;
|
||||||
|
import org.apache.nifi.rocksdb.RocksDBMetronome;
|
||||||
|
import org.apache.nifi.util.NiFiProperties;
|
||||||
|
import org.apache.nifi.util.file.FileUtils;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.TemporaryFolder;
|
||||||
|
import org.junit.rules.TestName;
|
||||||
|
import org.mockito.Mockito;
|
||||||
|
import org.mockito.stubbing.Answer;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
import java.nio.file.StandardCopyOption;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.UUID;
|
||||||
|
import java.util.concurrent.ConcurrentSkipListSet;
|
||||||
|
import java.util.function.Function;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
|
import static org.junit.Assert.assertNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.Mockito.doAnswer;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
public class TestRocksDBFlowFileRepository {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(TestRocksDBFlowFileRepository.class);
|
||||||
|
|
||||||
|
private final Map<String, String> additionalProperties = new HashMap<>();
|
||||||
|
private String nifiPropertiesPath;
|
||||||
|
|
||||||
|
@Rule
|
||||||
|
public TestName testName = new TestName();
|
||||||
|
|
||||||
|
@Rule
|
||||||
|
public TemporaryFolder temporaryFolder = new TemporaryFolder();
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void before() throws IOException {
|
||||||
|
File testRepoDir = temporaryFolder.newFolder(testName.getMethodName());
|
||||||
|
additionalProperties.put(NiFiProperties.FLOWFILE_REPOSITORY_DIRECTORY, testRepoDir.getAbsolutePath());
|
||||||
|
|
||||||
|
File properties = temporaryFolder.newFile();
|
||||||
|
Files.copy(Paths.get("src/test/resources/conf/nifi.properties"), properties.toPath(), StandardCopyOption.REPLACE_EXISTING);
|
||||||
|
nifiPropertiesPath = properties.getAbsolutePath();
|
||||||
|
|
||||||
|
logger.info("Running test: {}", testName.getMethodName());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testNormalizeSwapLocation() {
|
||||||
|
assertEquals("/", RocksDBFlowFileRepository.normalizeSwapLocation("/"));
|
||||||
|
assertEquals("", RocksDBFlowFileRepository.normalizeSwapLocation(""));
|
||||||
|
assertNull(RocksDBFlowFileRepository.normalizeSwapLocation(null));
|
||||||
|
assertEquals("test", RocksDBFlowFileRepository.normalizeSwapLocation("test.txt"));
|
||||||
|
assertEquals("test", RocksDBFlowFileRepository.normalizeSwapLocation("/test.txt"));
|
||||||
|
assertEquals("test", RocksDBFlowFileRepository.normalizeSwapLocation("/tmp/test.txt"));
|
||||||
|
assertEquals("test", RocksDBFlowFileRepository.normalizeSwapLocation("//test.txt"));
|
||||||
|
assertEquals("test", RocksDBFlowFileRepository.normalizeSwapLocation("/path/to/other/file/repository/test.txt"));
|
||||||
|
assertEquals("test", RocksDBFlowFileRepository.normalizeSwapLocation("test.txt/"));
|
||||||
|
assertEquals("test", RocksDBFlowFileRepository.normalizeSwapLocation("/path/to/test.txt/"));
|
||||||
|
assertEquals("test", WriteAheadFlowFileRepository.normalizeSwapLocation(WriteAheadFlowFileRepository.normalizeSwapLocation("/path/to/test.txt/")));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSwapLocationsRestored() throws IOException {
|
||||||
|
|
||||||
|
final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties));
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
|
||||||
|
final TestRocksDBFlowFileRepository.TestQueueProvider queueProvider = new TestRocksDBFlowFileRepository.TestQueueProvider();
|
||||||
|
repo.loadFlowFiles(queueProvider);
|
||||||
|
|
||||||
|
final Connection connection = Mockito.mock(Connection.class);
|
||||||
|
when(connection.getIdentifier()).thenReturn("1234");
|
||||||
|
|
||||||
|
final FlowFileQueue queue = Mockito.mock(FlowFileQueue.class);
|
||||||
|
when(queue.getIdentifier()).thenReturn("1234");
|
||||||
|
when(connection.getFlowFileQueue()).thenReturn(queue);
|
||||||
|
|
||||||
|
queueProvider.addConnection(connection);
|
||||||
|
|
||||||
|
StandardFlowFileRecord.Builder ffBuilder = new StandardFlowFileRecord.Builder();
|
||||||
|
ffBuilder.id(1L);
|
||||||
|
ffBuilder.size(0L);
|
||||||
|
final FlowFileRecord flowFileRecord = ffBuilder.build();
|
||||||
|
|
||||||
|
final List<RepositoryRecord> records = new ArrayList<>();
|
||||||
|
final StandardRepositoryRecord record = new StandardRepositoryRecord(queue, flowFileRecord, "swap123");
|
||||||
|
record.setDestination(queue);
|
||||||
|
records.add(record);
|
||||||
|
|
||||||
|
repo.updateRepository(records);
|
||||||
|
repo.close();
|
||||||
|
|
||||||
|
// restore
|
||||||
|
final RocksDBFlowFileRepository repo2 = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties));
|
||||||
|
repo2.initialize(new StandardResourceClaimManager());
|
||||||
|
repo2.loadFlowFiles(queueProvider);
|
||||||
|
assertTrue(repo2.isValidSwapLocationSuffix("swap123"));
|
||||||
|
assertFalse(repo2.isValidSwapLocationSuffix("other"));
|
||||||
|
repo2.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSwapLocationsUpdatedOnRepoUpdate() throws IOException {
|
||||||
|
final Path path = Paths.get("target/test-swap-repo");
|
||||||
|
if (Files.exists(path)) {
|
||||||
|
FileUtils.deleteFile(path.toFile(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties));
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
|
||||||
|
final TestRocksDBFlowFileRepository.TestQueueProvider queueProvider = new TestRocksDBFlowFileRepository.TestQueueProvider();
|
||||||
|
repo.loadFlowFiles(queueProvider);
|
||||||
|
|
||||||
|
final Connection connection = Mockito.mock(Connection.class);
|
||||||
|
when(connection.getIdentifier()).thenReturn("1234");
|
||||||
|
|
||||||
|
final FlowFileQueue queue = Mockito.mock(FlowFileQueue.class);
|
||||||
|
when(queue.getIdentifier()).thenReturn("1234");
|
||||||
|
when(connection.getFlowFileQueue()).thenReturn(queue);
|
||||||
|
|
||||||
|
queueProvider.addConnection(connection);
|
||||||
|
|
||||||
|
StandardFlowFileRecord.Builder ffBuilder = new StandardFlowFileRecord.Builder();
|
||||||
|
ffBuilder.id(1L);
|
||||||
|
ffBuilder.size(0L);
|
||||||
|
final FlowFileRecord flowFileRecord = ffBuilder.build();
|
||||||
|
|
||||||
|
final List<RepositoryRecord> records = new ArrayList<>();
|
||||||
|
final StandardRepositoryRecord record = new StandardRepositoryRecord(queue, flowFileRecord, "/tmp/swap123");
|
||||||
|
record.setDestination(queue);
|
||||||
|
records.add(record);
|
||||||
|
|
||||||
|
assertFalse(repo.isValidSwapLocationSuffix("swap123"));
|
||||||
|
repo.updateRepository(records);
|
||||||
|
assertTrue(repo.isValidSwapLocationSuffix("swap123"));
|
||||||
|
repo.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testResourceClaimsIncremented() throws IOException {
|
||||||
|
final ResourceClaimManager claimManager = new StandardResourceClaimManager();
|
||||||
|
|
||||||
|
final TestRocksDBFlowFileRepository.TestQueueProvider queueProvider = new TestRocksDBFlowFileRepository.TestQueueProvider();
|
||||||
|
final Connection connection = Mockito.mock(Connection.class);
|
||||||
|
when(connection.getIdentifier()).thenReturn("1234");
|
||||||
|
when(connection.getDestination()).thenReturn(Mockito.mock(Connectable.class));
|
||||||
|
|
||||||
|
final FlowFileSwapManager swapMgr = new TestRocksDBFlowFileRepository.MockFlowFileSwapManager();
|
||||||
|
final FlowFileQueue queue = new StandardFlowFileQueue("1234", new NopConnectionEventListener(), null, null, claimManager, null, swapMgr, null, 10000, 0L, "0 B");
|
||||||
|
|
||||||
|
when(connection.getFlowFileQueue()).thenReturn(queue);
|
||||||
|
queueProvider.addConnection(connection);
|
||||||
|
|
||||||
|
final ResourceClaim resourceClaim1 = claimManager.newResourceClaim("container", "section", "1", false, false);
|
||||||
|
final ContentClaim claim1 = new StandardContentClaim(resourceClaim1, 0L);
|
||||||
|
|
||||||
|
final ResourceClaim resourceClaim2 = claimManager.newResourceClaim("container", "section", "2", false, false);
|
||||||
|
final ContentClaim claim2 = new StandardContentClaim(resourceClaim2, 0L);
|
||||||
|
|
||||||
|
// Create a flowfile repo, update it once with a FlowFile that points to one resource claim. Then,
|
||||||
|
// indicate that a FlowFile was swapped out. We should then be able to recover these FlowFiles and the
|
||||||
|
// resource claims' counts should be updated for both the swapped out FlowFile and the non-swapped out FlowFile
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(claimManager);
|
||||||
|
repo.loadFlowFiles(queueProvider);
|
||||||
|
|
||||||
|
// Create a Repository Record that indicates that a FlowFile was created
|
||||||
|
final FlowFileRecord flowFile1 = new StandardFlowFileRecord.Builder()
|
||||||
|
.id(1L)
|
||||||
|
.addAttribute("uuid", "11111111-1111-1111-1111-111111111111")
|
||||||
|
.contentClaim(claim1)
|
||||||
|
.build();
|
||||||
|
final StandardRepositoryRecord rec1 = new StandardRepositoryRecord(queue);
|
||||||
|
rec1.setWorking(flowFile1);
|
||||||
|
rec1.setDestination(queue);
|
||||||
|
|
||||||
|
// Create a Record that we can swap out
|
||||||
|
final FlowFileRecord flowFile2 = new StandardFlowFileRecord.Builder()
|
||||||
|
.id(2L)
|
||||||
|
.addAttribute("uuid", "11111111-1111-1111-1111-111111111112")
|
||||||
|
.contentClaim(claim2)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
final StandardRepositoryRecord rec2 = new StandardRepositoryRecord(queue);
|
||||||
|
rec2.setWorking(flowFile2);
|
||||||
|
rec2.setDestination(queue);
|
||||||
|
|
||||||
|
final List<RepositoryRecord> records = new ArrayList<>();
|
||||||
|
records.add(rec1);
|
||||||
|
records.add(rec2);
|
||||||
|
repo.updateRepository(records);
|
||||||
|
|
||||||
|
final String swapLocation = swapMgr.swapOut(Collections.singletonList(flowFile2), queue, null);
|
||||||
|
repo.swapFlowFilesOut(Collections.singletonList(flowFile2), queue, swapLocation);
|
||||||
|
}
|
||||||
|
|
||||||
|
final ResourceClaimManager recoveryClaimManager = new StandardResourceClaimManager();
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(recoveryClaimManager);
|
||||||
|
final long largestId = repo.loadFlowFiles(queueProvider);
|
||||||
|
|
||||||
|
// largest ID known is 1 because this doesn't take into account the FlowFiles that have been swapped out
|
||||||
|
assertEquals(1, largestId);
|
||||||
|
}
|
||||||
|
|
||||||
|
// resource claim 1 will have a single claimant count while resource claim 2 will have no claimant counts
|
||||||
|
// because resource claim 2 is referenced only by flowfiles that are swapped out.
|
||||||
|
assertEquals(1, recoveryClaimManager.getClaimantCount(resourceClaim1));
|
||||||
|
assertEquals(0, recoveryClaimManager.getClaimantCount(resourceClaim2));
|
||||||
|
|
||||||
|
final SwapSummary summary = queue.recoverSwappedFlowFiles();
|
||||||
|
assertNotNull(summary);
|
||||||
|
assertEquals(2, summary.getMaxFlowFileId().intValue());
|
||||||
|
assertEquals(new QueueSize(1, 0L), summary.getQueueSize());
|
||||||
|
|
||||||
|
final List<ResourceClaim> swappedOutClaims = summary.getResourceClaims();
|
||||||
|
assertNotNull(swappedOutClaims);
|
||||||
|
assertEquals(1, swappedOutClaims.size());
|
||||||
|
assertEquals(claim2.getResourceClaim(), swappedOutClaims.get(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRestartWithOneRecord() throws IOException {
|
||||||
|
|
||||||
|
final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties));
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
|
||||||
|
final TestRocksDBFlowFileRepository.TestQueueProvider queueProvider = new TestRocksDBFlowFileRepository.TestQueueProvider();
|
||||||
|
repo.loadFlowFiles(queueProvider);
|
||||||
|
|
||||||
|
final List<FlowFileRecord> flowFileCollection = new ArrayList<>();
|
||||||
|
|
||||||
|
final Connection connection = Mockito.mock(Connection.class);
|
||||||
|
when(connection.getIdentifier()).thenReturn("1234");
|
||||||
|
|
||||||
|
final FlowFileQueue queue = Mockito.mock(FlowFileQueue.class);
|
||||||
|
when(queue.getIdentifier()).thenReturn("1234");
|
||||||
|
doAnswer((Answer<Object>) invocation -> {
|
||||||
|
flowFileCollection.add((FlowFileRecord) invocation.getArguments()[0]);
|
||||||
|
return null;
|
||||||
|
}).when(queue).put(any(FlowFileRecord.class));
|
||||||
|
|
||||||
|
when(connection.getFlowFileQueue()).thenReturn(queue);
|
||||||
|
|
||||||
|
queueProvider.addConnection(connection);
|
||||||
|
|
||||||
|
StandardFlowFileRecord.Builder ffBuilder = new StandardFlowFileRecord.Builder();
|
||||||
|
ffBuilder.id(1L);
|
||||||
|
ffBuilder.addAttribute("abc", "xyz");
|
||||||
|
ffBuilder.size(0L);
|
||||||
|
final FlowFileRecord flowFileRecord = ffBuilder.build();
|
||||||
|
|
||||||
|
final List<RepositoryRecord> records = new ArrayList<>();
|
||||||
|
final StandardRepositoryRecord record = new StandardRepositoryRecord(null);
|
||||||
|
record.setWorking(flowFileRecord);
|
||||||
|
record.setDestination(connection.getFlowFileQueue());
|
||||||
|
records.add(record);
|
||||||
|
|
||||||
|
repo.updateRepository(records);
|
||||||
|
|
||||||
|
// update to add new attribute
|
||||||
|
ffBuilder = new StandardFlowFileRecord.Builder().fromFlowFile(flowFileRecord).addAttribute("hello", "world");
|
||||||
|
final FlowFileRecord flowFileRecord2 = ffBuilder.build();
|
||||||
|
record.setWorking(flowFileRecord2);
|
||||||
|
repo.updateRepository(records);
|
||||||
|
|
||||||
|
// update size but no attribute
|
||||||
|
ffBuilder = new StandardFlowFileRecord.Builder().fromFlowFile(flowFileRecord2).size(40L);
|
||||||
|
final FlowFileRecord flowFileRecord3 = ffBuilder.build();
|
||||||
|
record.setWorking(flowFileRecord3);
|
||||||
|
repo.updateRepository(records);
|
||||||
|
|
||||||
|
repo.close();
|
||||||
|
|
||||||
|
// restore
|
||||||
|
final RocksDBFlowFileRepository repo2 = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties));
|
||||||
|
repo2.initialize(new StandardResourceClaimManager());
|
||||||
|
repo2.loadFlowFiles(queueProvider);
|
||||||
|
|
||||||
|
assertEquals(1, flowFileCollection.size());
|
||||||
|
final FlowFileRecord flowFile = flowFileCollection.get(0);
|
||||||
|
assertEquals(1L, flowFile.getId());
|
||||||
|
assertEquals("xyz", flowFile.getAttribute("abc"));
|
||||||
|
assertEquals(40L, flowFile.getSize());
|
||||||
|
assertEquals("world", flowFile.getAttribute("hello"));
|
||||||
|
|
||||||
|
repo2.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDoNotRemoveOrphans() throws Exception {
|
||||||
|
|
||||||
|
final TestQueue testQueue = new TestQueue();
|
||||||
|
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
|
||||||
|
repo.updateRepository(testQueue.getRepositoryRecord(new StandardFlowFileRecord.Builder()
|
||||||
|
.id(1L)
|
||||||
|
.addAttribute("abc", "xyz")
|
||||||
|
.size(0L)
|
||||||
|
.build()
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore (& confirm present)
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
assertEquals(1, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore with empty queue provider (should throw exception)
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(new TestQueueProvider());
|
||||||
|
fail();
|
||||||
|
} catch (IOException expected) {
|
||||||
|
assertTrue(expected.getMessage().contains("Found FlowFile in repository without a corresponding queue"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRemoveOrphans() throws Exception {
|
||||||
|
|
||||||
|
final TestQueue testQueue = new TestQueue();
|
||||||
|
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.REMOVE_ORPHANED_FLOWFILES.propertyName, "true");
|
||||||
|
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
|
||||||
|
repo.updateRepository(testQueue.getRepositoryRecord(new StandardFlowFileRecord.Builder()
|
||||||
|
.id(1L)
|
||||||
|
.addAttribute("abc", "xyz")
|
||||||
|
.size(0L)
|
||||||
|
.build()
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore (& confirm present)
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
assertEquals(1, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
// restore with empty queue provider (should throw exception)
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(new TestQueueProvider());
|
||||||
|
assertEquals(0, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testKnownVersion() throws Exception {
|
||||||
|
final NiFiProperties niFiProperties = NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties);
|
||||||
|
|
||||||
|
// create db with known version
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder().setStoragePath(RocksDBFlowFileRepository.getFlowFileRepoPath(niFiProperties)).build()) {
|
||||||
|
db.initialize();
|
||||||
|
db.putConfiguration(RocksDBFlowFileRepository.REPOSITORY_VERSION_KEY, RocksDBFlowFileRepository.VERSION_ONE_BYTES);
|
||||||
|
}
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = IllegalStateException.class)
|
||||||
|
public void testUnknownVersion() throws Exception {
|
||||||
|
final NiFiProperties niFiProperties = NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties);
|
||||||
|
|
||||||
|
// create db with known version
|
||||||
|
try (RocksDBMetronome db = new RocksDBMetronome.Builder().setStoragePath(RocksDBFlowFileRepository.getFlowFileRepoPath(niFiProperties)).build()) {
|
||||||
|
db.initialize();
|
||||||
|
db.putConfiguration(RocksDBFlowFileRepository.REPOSITORY_VERSION_KEY, "UNKNOWN".getBytes(StandardCharsets.UTF_8));
|
||||||
|
}
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRecoveryMode() throws Exception {
|
||||||
|
|
||||||
|
int totalFlowFiles = 50;
|
||||||
|
|
||||||
|
final TestQueue testQueue = new TestQueue();
|
||||||
|
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
|
||||||
|
// add records to the repo
|
||||||
|
for (int i = 1; i <= totalFlowFiles; i++) {
|
||||||
|
repo.updateRepository(testQueue.getRepositoryRecord(new StandardFlowFileRecord.Builder()
|
||||||
|
.id(i)
|
||||||
|
.addAttribute("abc", "xyz")
|
||||||
|
.size(0L)
|
||||||
|
.build()
|
||||||
|
));
|
||||||
|
}
|
||||||
|
assertEquals(totalFlowFiles, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore in recovery mode with varying limits
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.ENABLE_RECOVERY_MODE.propertyName, "true");
|
||||||
|
for (int recoveryLimit = 0; recoveryLimit < totalFlowFiles; recoveryLimit += 10) {
|
||||||
|
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.RECOVERY_MODE_FLOWFILE_LIMIT.propertyName, Integer.toString(recoveryLimit));
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
assertEquals(recoveryLimit, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore in recovery mode with limit equal to available files
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.RECOVERY_MODE_FLOWFILE_LIMIT.propertyName, Integer.toString(totalFlowFiles));
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
assertEquals(totalFlowFiles, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore in recovery mode with limit higher than available files
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.RECOVERY_MODE_FLOWFILE_LIMIT.propertyName, Integer.toString(Integer.MAX_VALUE));
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
assertEquals(totalFlowFiles, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore in normal mode
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.ENABLE_RECOVERY_MODE.propertyName, "false");
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.RECOVERY_MODE_FLOWFILE_LIMIT.propertyName, Integer.toString(0));
|
||||||
|
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
assertEquals(totalFlowFiles, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRecoveryModeWithContinuedLoading() throws Exception {
|
||||||
|
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.CLAIM_CLEANUP_PERIOD.propertyName, "24 hours"); // "disable" the cleanup thread, let us manually force recovery
|
||||||
|
|
||||||
|
int totalFlowFiles = 50;
|
||||||
|
int recoveryLimit = 10;
|
||||||
|
|
||||||
|
final TestQueue testQueue = new TestQueue();
|
||||||
|
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
|
||||||
|
// add records to the repo
|
||||||
|
for (int i = 1; i <= totalFlowFiles; i++) {
|
||||||
|
repo.updateRepository(testQueue.getRepositoryRecord(new StandardFlowFileRecord.Builder()
|
||||||
|
.id(i)
|
||||||
|
.addAttribute("abc", "xyz")
|
||||||
|
.size(0L)
|
||||||
|
.build()
|
||||||
|
));
|
||||||
|
}
|
||||||
|
assertEquals(totalFlowFiles, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore in recovery mode
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.ENABLE_RECOVERY_MODE.propertyName, "true");
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.RECOVERY_MODE_FLOWFILE_LIMIT.propertyName, Integer.toString(recoveryLimit));
|
||||||
|
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
assertEquals(recoveryLimit, repo.getInMemoryFlowFiles());
|
||||||
|
assertEquals(totalFlowFiles - recoveryLimit, repo.getRecordsToRestoreCount());
|
||||||
|
|
||||||
|
long flowFilesRecovered = repo.getInMemoryFlowFiles();
|
||||||
|
|
||||||
|
for (int i = 0; i < 4; i++) {
|
||||||
|
testQueue.deleteQueuedFlowFiles(repo);
|
||||||
|
assertEquals(0, repo.getInMemoryFlowFiles());
|
||||||
|
|
||||||
|
repo.doRecovery();
|
||||||
|
assertEquals(recoveryLimit, repo.getInMemoryFlowFiles());
|
||||||
|
|
||||||
|
flowFilesRecovered += repo.getInMemoryFlowFiles();
|
||||||
|
assertEquals((recoveryLimit * (i + 2)), flowFilesRecovered);
|
||||||
|
assertEquals(totalFlowFiles - flowFilesRecovered, repo.getRecordsToRestoreCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
// should have restored all files
|
||||||
|
assertEquals(0, repo.getRecordsToRestoreCount());
|
||||||
|
assertEquals(recoveryLimit, repo.getInMemoryFlowFiles());
|
||||||
|
|
||||||
|
// delete last files
|
||||||
|
testQueue.deleteQueuedFlowFiles(repo);
|
||||||
|
assertEquals(0, repo.getRecordsToRestoreCount());
|
||||||
|
assertEquals(0, repo.getInMemoryFlowFiles());
|
||||||
|
|
||||||
|
repo.doRecovery();
|
||||||
|
|
||||||
|
// should have nothing left
|
||||||
|
assertEquals(0, repo.getRecordsToRestoreCount());
|
||||||
|
assertEquals(0, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
|
||||||
|
// restore in normal mode
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.ENABLE_RECOVERY_MODE.propertyName, "false");
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.RECOVERY_MODE_FLOWFILE_LIMIT.propertyName, Integer.toString(1));
|
||||||
|
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
assertEquals(0, repo.getRecordsToRestoreCount());
|
||||||
|
assertEquals(0, repo.getInMemoryFlowFiles());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testStallStop() throws IOException {
|
||||||
|
final TestQueue testQueue = new TestQueue();
|
||||||
|
|
||||||
|
// set stall & stop properties
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.ENABLE_STALL_STOP.propertyName, "true");
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.STALL_FLOWFILE_COUNT.propertyName, "2");
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.STOP_FLOWFILE_COUNT.propertyName, "3");
|
||||||
|
|
||||||
|
// take heap usage out of the calculation
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.STALL_HEAP_USAGE_PERCENT.propertyName, "100%");
|
||||||
|
additionalProperties.put(RocksDBFlowFileRepository.RocksDbProperty.STOP_HEAP_USAGE_PERCENT.propertyName, "100%");
|
||||||
|
|
||||||
|
try (final RocksDBFlowFileRepository repo = new RocksDBFlowFileRepository(NiFiProperties.createBasicNiFiProperties(nifiPropertiesPath, additionalProperties))) {
|
||||||
|
|
||||||
|
repo.initialize(new StandardResourceClaimManager());
|
||||||
|
repo.loadFlowFiles(testQueue.provider);
|
||||||
|
|
||||||
|
StandardFlowFileRecord.Builder ffBuilder = new StandardFlowFileRecord.Builder();
|
||||||
|
ffBuilder.addAttribute("abc", "xyz");
|
||||||
|
ffBuilder.size(0L);
|
||||||
|
|
||||||
|
List<RepositoryRecord> record1 = testQueue.getRepositoryRecord(ffBuilder.id(1).build());
|
||||||
|
List<RepositoryRecord> record2 = testQueue.getRepositoryRecord(ffBuilder.id(2).build());
|
||||||
|
List<RepositoryRecord> record3 = testQueue.getRepositoryRecord(ffBuilder.id(3).build());
|
||||||
|
|
||||||
|
// CREATE one... should incur no penalty
|
||||||
|
repo.updateRepository(record1);
|
||||||
|
repo.updateStallStop();
|
||||||
|
assertFalse(repo.stallNewFlowFiles);
|
||||||
|
assertFalse(repo.stopNewFlowFiles);
|
||||||
|
|
||||||
|
// CREATE another... should stall
|
||||||
|
repo.updateRepository(record2);
|
||||||
|
repo.updateStallStop();
|
||||||
|
assertTrue(repo.stallNewFlowFiles);
|
||||||
|
assertFalse(repo.stopNewFlowFiles);
|
||||||
|
|
||||||
|
// CREATE another... should stop
|
||||||
|
repo.updateRepository(record3);
|
||||||
|
repo.updateStallStop();
|
||||||
|
assertTrue(repo.stallNewFlowFiles);
|
||||||
|
assertTrue(repo.stopNewFlowFiles);
|
||||||
|
|
||||||
|
// DELETE one... should be stalled but not stopped
|
||||||
|
((StandardRepositoryRecord) record1.get(0)).markForDelete();
|
||||||
|
repo.updateRepository(record1);
|
||||||
|
repo.updateStallStop();
|
||||||
|
assertTrue(repo.stallNewFlowFiles);
|
||||||
|
assertFalse(repo.stopNewFlowFiles);
|
||||||
|
|
||||||
|
// DELETE another... shouldn't be stalled or stopped
|
||||||
|
((StandardRepositoryRecord) record2.get(0)).markForDelete();
|
||||||
|
repo.updateRepository(record2);
|
||||||
|
repo.updateStallStop();
|
||||||
|
assertFalse(repo.stallNewFlowFiles);
|
||||||
|
assertFalse(repo.stopNewFlowFiles);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class TestQueue {
|
||||||
|
private final TestQueueProvider provider;
|
||||||
|
private final Collection<FlowFileRecord> queuedFlowFiles;
|
||||||
|
private final Connection connection;
|
||||||
|
|
||||||
|
TestQueue() {
|
||||||
|
provider = new TestQueueProvider();
|
||||||
|
queuedFlowFiles = new ConcurrentSkipListSet<>(); // potentially accessed from multiple threads
|
||||||
|
|
||||||
|
final FlowFileQueue queue = new StandardFlowFileQueue("1234", null, null, null, null, null, null, null, 0, 0, "0 B") {
|
||||||
|
@Override
|
||||||
|
public void put(final FlowFileRecord file) {
|
||||||
|
queuedFlowFiles.add(file);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
connection = Mockito.mock(Connection.class);
|
||||||
|
when(connection.getIdentifier()).thenReturn(queue.getIdentifier());
|
||||||
|
when(connection.getFlowFileQueue()).thenReturn(queue);
|
||||||
|
|
||||||
|
provider.addConnection(connection);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void deleteQueuedFlowFiles(RocksDBFlowFileRepository repo) throws IOException {
|
||||||
|
Collection<RepositoryRecord> recordsToDelete = queuedFlowFiles.stream().map((Function<FlowFileRecord, RepositoryRecord>) flowFileRecord -> {
|
||||||
|
StandardRepositoryRecord record = new StandardRepositoryRecord(null, flowFileRecord);
|
||||||
|
record.markForDelete();
|
||||||
|
return record;
|
||||||
|
}).collect(Collectors.toSet());
|
||||||
|
|
||||||
|
repo.updateRepository(recordsToDelete);
|
||||||
|
queuedFlowFiles.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<RepositoryRecord> getRepositoryRecord(final FlowFileRecord flowFileRecord) {
|
||||||
|
final StandardRepositoryRecord record = new StandardRepositoryRecord(null);
|
||||||
|
record.setWorking(flowFileRecord);
|
||||||
|
record.setDestination(connection.getFlowFileQueue());
|
||||||
|
return Collections.singletonList(record);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TestQueueProvider implements QueueProvider {
|
||||||
|
|
||||||
|
private List<Connection> connectionList = new ArrayList<>();
|
||||||
|
|
||||||
|
void addConnection(final Connection connection) {
|
||||||
|
this.connectionList.add(connection);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Collection<FlowFileQueue> getAllQueues() {
|
||||||
|
final List<FlowFileQueue> queueList = new ArrayList<>();
|
||||||
|
for (final Connection conn : connectionList) {
|
||||||
|
queueList.add(conn.getFlowFileQueue());
|
||||||
|
}
|
||||||
|
|
||||||
|
return queueList;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class MockFlowFileSwapManager implements FlowFileSwapManager {
|
||||||
|
|
||||||
|
private final Map<FlowFileQueue, Map<String, List<FlowFileRecord>>> swappedRecords = new HashMap<>();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void initialize(SwapManagerInitializationContext initializationContext) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String swapOut(List<FlowFileRecord> flowFiles, FlowFileQueue flowFileQueue, final String partitionName) {
|
||||||
|
Map<String, List<FlowFileRecord>> swapMap = swappedRecords.computeIfAbsent(flowFileQueue, k -> new HashMap<>());
|
||||||
|
|
||||||
|
final String location = UUID.randomUUID().toString();
|
||||||
|
swapMap.put(location, new ArrayList<>(flowFiles));
|
||||||
|
return location;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public SwapContents peek(String swapLocation, FlowFileQueue flowFileQueue) {
|
||||||
|
Map<String, List<FlowFileRecord>> swapMap = swappedRecords.get(flowFileQueue);
|
||||||
|
if (swapMap == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final List<FlowFileRecord> flowFiles = swapMap.get(swapLocation);
|
||||||
|
final SwapSummary summary = getSwapSummary(swapLocation);
|
||||||
|
return new StandardSwapContents(summary, flowFiles);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public SwapContents swapIn(String swapLocation, FlowFileQueue flowFileQueue) {
|
||||||
|
Map<String, List<FlowFileRecord>> swapMap = swappedRecords.get(flowFileQueue);
|
||||||
|
if (swapMap == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final List<FlowFileRecord> flowFiles = swapMap.remove(swapLocation);
|
||||||
|
final SwapSummary summary = getSwapSummary(swapLocation);
|
||||||
|
return new StandardSwapContents(summary, flowFiles);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<String> recoverSwapLocations(FlowFileQueue flowFileQueue, final String partitionName) {
|
||||||
|
Map<String, List<FlowFileRecord>> swapMap = swappedRecords.get(flowFileQueue);
|
||||||
|
if (swapMap == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new ArrayList<>(swapMap.keySet());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public SwapSummary getSwapSummary(String swapLocation) {
|
||||||
|
List<FlowFileRecord> records = null;
|
||||||
|
for (final Map<String, List<FlowFileRecord>> swapMap : swappedRecords.values()) {
|
||||||
|
records = swapMap.get(swapLocation);
|
||||||
|
if (records != null) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (records == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final List<ResourceClaim> resourceClaims = new ArrayList<>();
|
||||||
|
long size = 0L;
|
||||||
|
Long maxId = null;
|
||||||
|
for (final FlowFileRecord flowFile : records) {
|
||||||
|
size += flowFile.getSize();
|
||||||
|
|
||||||
|
if (maxId == null || flowFile.getId() > maxId) {
|
||||||
|
maxId = flowFile.getId();
|
||||||
|
}
|
||||||
|
|
||||||
|
final ContentClaim contentClaim = flowFile.getContentClaim();
|
||||||
|
if (contentClaim != null) {
|
||||||
|
final ResourceClaim resourceClaim = contentClaim.getResourceClaim();
|
||||||
|
resourceClaims.add(resourceClaim);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return new StandardSwapSummary(new QueueSize(records.size(), size), maxId, resourceClaims);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void purge() {
|
||||||
|
this.swappedRecords.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<String> getSwappedPartitionNames(FlowFileQueue queue) {
|
||||||
|
return Collections.emptySet();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String changePartitionName(String swapLocation, String newPartitionName) {
|
||||||
|
return swapLocation;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,22 @@
|
||||||
|
# Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
# contributor license agreements. See the NOTICE file distributed with
|
||||||
|
# this work for additional information regarding copyright ownership.
|
||||||
|
# The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
# (the "License"); you may not use this file except in compliance with
|
||||||
|
# the License. You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
log4j.rootLogger=INFO,console
|
||||||
|
log4j.category.org.apache.nifi=DEBUG
|
||||||
|
|
||||||
|
log4j.appender.console=org.apache.log4j.ConsoleAppender
|
||||||
|
log4j.appender.console.target=System.err
|
||||||
|
log4j.appender.console.layout=org.apache.log4j.PatternLayout
|
||||||
|
log4j.appender.console.layout.ConversionPattern=%d{yy/MM/dd HH:mm:ss} %p %c{2}: %m%n
|
Loading…
Reference in New Issue