2021-10-20 01:51:34 +02:00
|
|
|
package it.cavallium.dbengine.database.disk;
|
|
|
|
|
|
|
|
import static it.cavallium.dbengine.database.LLUtils.MARKER_ROCKSDB;
|
|
|
|
|
2021-10-30 11:13:46 +02:00
|
|
|
import io.micrometer.core.instrument.MeterRegistry;
|
2022-03-16 13:47:56 +01:00
|
|
|
import io.netty5.buffer.api.Buffer;
|
|
|
|
import io.netty5.buffer.api.BufferAllocator;
|
|
|
|
import io.netty5.buffer.api.MemoryManager;
|
|
|
|
import io.netty5.buffer.api.Send;
|
2021-10-20 01:51:34 +02:00
|
|
|
import it.cavallium.dbengine.database.LLDelta;
|
|
|
|
import it.cavallium.dbengine.database.LLUtils;
|
|
|
|
import java.io.IOException;
|
2022-04-28 23:23:26 +02:00
|
|
|
import java.util.concurrent.locks.Lock;
|
2022-04-30 01:49:44 +02:00
|
|
|
import java.util.concurrent.locks.StampedLock;
|
2021-10-20 01:51:34 +02:00
|
|
|
import org.jetbrains.annotations.NotNull;
|
|
|
|
import org.jetbrains.annotations.Nullable;
|
|
|
|
import org.rocksdb.ColumnFamilyHandle;
|
|
|
|
import org.rocksdb.ReadOptions;
|
|
|
|
import org.rocksdb.RocksDBException;
|
|
|
|
import org.rocksdb.Transaction;
|
|
|
|
import org.rocksdb.TransactionDB;
|
|
|
|
import org.rocksdb.TransactionOptions;
|
|
|
|
import org.rocksdb.WriteOptions;
|
|
|
|
import reactor.core.scheduler.Schedulers;
|
|
|
|
|
|
|
|
public final class PessimisticRocksDBColumn extends AbstractRocksDBColumn<TransactionDB> {
|
|
|
|
|
|
|
|
private static final TransactionOptions DEFAULT_TX_OPTIONS = new TransactionOptions();
|
|
|
|
|
|
|
|
public PessimisticRocksDBColumn(TransactionDB db,
|
2022-04-06 02:41:32 +02:00
|
|
|
boolean nettyDirect,
|
2021-10-20 01:51:34 +02:00
|
|
|
BufferAllocator alloc,
|
2022-03-30 15:15:53 +02:00
|
|
|
String dbName,
|
2022-04-28 23:23:26 +02:00
|
|
|
ColumnFamilyHandle cfh,
|
|
|
|
MeterRegistry meterRegistry,
|
2022-04-30 01:49:44 +02:00
|
|
|
StampedLock closeLock) {
|
|
|
|
super(db, nettyDirect, alloc, dbName, cfh, meterRegistry, closeLock);
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected boolean commitOptimistically(Transaction tx) throws RocksDBException {
|
|
|
|
tx.commit();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Transaction beginTransaction(@NotNull WriteOptions writeOptions) {
|
|
|
|
return getDb().beginTransaction(writeOptions, DEFAULT_TX_OPTIONS);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2022-03-30 15:15:53 +02:00
|
|
|
public @NotNull UpdateAtomicResult updateAtomicImpl(@NotNull ReadOptions readOptions,
|
2021-10-20 01:51:34 +02:00
|
|
|
@NotNull WriteOptions writeOptions,
|
2022-04-01 01:30:56 +02:00
|
|
|
Buffer key,
|
|
|
|
BinarySerializationFunction updater,
|
2022-03-30 15:15:53 +02:00
|
|
|
UpdateAtomicResultMode returnMode) throws IOException {
|
2022-04-01 01:30:56 +02:00
|
|
|
long initNanoTime = System.nanoTime();
|
|
|
|
try {
|
|
|
|
var cfh = getCfh();
|
|
|
|
var keyArray = LLUtils.toArray(key);
|
|
|
|
if (Schedulers.isInNonBlockingThread()) {
|
|
|
|
throw new UnsupportedOperationException("Called update in a nonblocking thread");
|
|
|
|
}
|
|
|
|
try (var tx = beginTransaction(writeOptions)) {
|
|
|
|
Send<Buffer> sentPrevData;
|
|
|
|
Send<Buffer> sentCurData;
|
|
|
|
boolean changed;
|
|
|
|
if (logger.isTraceEnabled()) {
|
|
|
|
logger.trace(MARKER_ROCKSDB, "Reading {} (before update lock)", LLUtils.toStringSafe(key));
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
var prevDataArray = tx.getForUpdate(readOptions, cfh, keyArray, true);
|
|
|
|
try {
|
2021-12-29 00:31:35 +01:00
|
|
|
if (logger.isTraceEnabled()) {
|
2022-04-01 01:30:56 +02:00
|
|
|
logger.trace(MARKER_ROCKSDB,
|
|
|
|
"Reading {}: {} (before update)",
|
|
|
|
LLUtils.toStringSafe(key),
|
|
|
|
LLUtils.toStringSafe(prevDataArray)
|
|
|
|
);
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
Buffer prevData;
|
|
|
|
if (prevDataArray != null) {
|
|
|
|
readValueFoundWithoutBloomBufferSize.record(prevDataArray.length);
|
|
|
|
prevData = MemoryManager.unsafeWrap(prevDataArray);
|
|
|
|
} else {
|
|
|
|
readValueNotFoundWithoutBloomBufferSize.record(0);
|
|
|
|
prevData = null;
|
|
|
|
}
|
|
|
|
try (prevData) {
|
|
|
|
Buffer prevDataToSendToUpdater;
|
|
|
|
if (prevData != null) {
|
|
|
|
prevDataToSendToUpdater = prevData.copy().makeReadOnly();
|
2021-12-29 00:31:35 +01:00
|
|
|
} else {
|
2022-04-01 01:30:56 +02:00
|
|
|
prevDataToSendToUpdater = null;
|
2021-12-29 00:31:35 +01:00
|
|
|
}
|
2021-10-20 01:51:34 +02:00
|
|
|
|
2022-04-01 01:30:56 +02:00
|
|
|
@Nullable Buffer newData = applyUpdateAndCloseIfNecessary(updater, prevDataToSendToUpdater);
|
|
|
|
try (newData) {
|
|
|
|
var newDataArray = newData == null ? null : LLUtils.toArray(newData);
|
|
|
|
if (logger.isTraceEnabled()) {
|
|
|
|
logger.trace(MARKER_ROCKSDB,
|
|
|
|
"Updating {}. previous data: {}, updated data: {}",
|
|
|
|
LLUtils.toStringSafe(key),
|
|
|
|
LLUtils.toStringSafe(prevDataArray),
|
|
|
|
LLUtils.toStringSafe(newDataArray)
|
|
|
|
);
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
if (prevData != null && newData == null) {
|
|
|
|
if (logger.isTraceEnabled()) {
|
|
|
|
logger.trace(MARKER_ROCKSDB, "Deleting {} (after update)", LLUtils.toStringSafe(key));
|
|
|
|
}
|
|
|
|
writeValueBufferSize.record(0);
|
|
|
|
tx.delete(cfh, keyArray, true);
|
|
|
|
changed = true;
|
|
|
|
tx.commit();
|
|
|
|
} else if (newData != null && (prevData == null || !LLUtils.equals(prevData, newData))) {
|
2021-12-29 00:31:35 +01:00
|
|
|
if (logger.isTraceEnabled()) {
|
|
|
|
logger.trace(MARKER_ROCKSDB,
|
2022-04-01 01:30:56 +02:00
|
|
|
"Writing {}: {} (after update)",
|
2021-12-29 00:31:35 +01:00
|
|
|
LLUtils.toStringSafe(key),
|
2022-04-01 01:30:56 +02:00
|
|
|
LLUtils.toStringSafe(newData)
|
2021-12-29 00:31:35 +01:00
|
|
|
);
|
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
writeValueBufferSize.record(newDataArray.length);
|
|
|
|
tx.put(cfh, keyArray, newDataArray);
|
|
|
|
changed = true;
|
|
|
|
tx.commit();
|
|
|
|
} else {
|
|
|
|
changed = false;
|
|
|
|
tx.rollback();
|
2021-12-29 00:31:35 +01:00
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
sentPrevData = prevData == null ? null : prevData.send();
|
|
|
|
sentCurData = newData == null ? null : newData.send();
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
} finally {
|
|
|
|
tx.undoGetForUpdate(cfh, keyArray);
|
|
|
|
}
|
|
|
|
recordAtomicUpdateTime(changed, sentPrevData != null, sentCurData != null, initNanoTime);
|
|
|
|
return switch (returnMode) {
|
|
|
|
case NOTHING -> {
|
|
|
|
if (sentPrevData != null) {
|
|
|
|
sentPrevData.close();
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
if (sentCurData != null) {
|
|
|
|
sentCurData.close();
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
yield RESULT_NOTHING;
|
|
|
|
}
|
|
|
|
case CURRENT -> {
|
|
|
|
if (sentPrevData != null) {
|
|
|
|
sentPrevData.close();
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
yield new UpdateAtomicResultCurrent(sentCurData);
|
|
|
|
}
|
|
|
|
case PREVIOUS -> {
|
|
|
|
if (sentCurData != null) {
|
|
|
|
sentCurData.close();
|
|
|
|
}
|
|
|
|
yield new UpdateAtomicResultPrevious(sentPrevData);
|
|
|
|
}
|
|
|
|
case BINARY_CHANGED -> new UpdateAtomicResultBinaryChanged(changed);
|
|
|
|
case DELTA -> new UpdateAtomicResultDelta(LLDelta.of(sentPrevData, sentCurData).send());
|
|
|
|
};
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
2022-04-01 01:30:56 +02:00
|
|
|
} catch (Throwable ex) {
|
|
|
|
throw new IOException("Failed to update key " + LLUtils.toStringSafe(key), ex);
|
2021-10-20 01:51:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean supportsTransactions() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|