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

io.permazen.kv.simple.SimpleKVTransaction Maven / Gradle / Ivy


/*
 * Copyright (C) 2015 Archie L. Cobbs. All rights reserved.
 */

package io.permazen.kv.simple;

import com.google.common.base.Preconditions;
import com.google.common.util.concurrent.ListenableFuture;

import io.permazen.kv.AbstractKVStore;
import io.permazen.kv.CloseableKVStore;
import io.permazen.kv.KVPair;
import io.permazen.kv.KVPairIterator;
import io.permazen.kv.KVTransaction;
import io.permazen.kv.KeyRange;
import io.permazen.kv.mvcc.AtomicKVStore;
import io.permazen.kv.mvcc.LockOwner;
import io.permazen.kv.mvcc.MutableView;
import io.permazen.kv.util.CloseableForwardingKVStore;
import io.permazen.kv.util.NavigableMapKVStore;
import io.permazen.util.ByteUtil;
import io.permazen.util.CloseableIterator;

import java.util.Arrays;
import java.util.SortedSet;
import java.util.TreeSet;

import org.slf4j.LoggerFactory;

/**
 * {@link KVTransaction} implementation for {@link SimpleKVDatabase}.
 *
 * 

* Locking note: all fields in this class are protected by the Java monitor of the associated {@link SimpleKVDatabase}, * not the Java monitor of this instance. */ public class SimpleKVTransaction extends AbstractKVStore implements KVTransaction { final SimpleKVDatabase kvdb; final TreeSet mutations = new TreeSet<>(KeyRange.SORT_BY_MIN); final LockOwner lockOwner = new LockOwner(); boolean stale; long waitTimeout; private volatile boolean readOnly; /** * Constructor. * * @param kvdb associated database * @param waitTimeout wait timeout for this transaction * @throws IllegalArgumentException if {@code kvdb} is null * @throws IllegalArgumentException if {@code waitTimeout} is negative */ protected SimpleKVTransaction(SimpleKVDatabase kvdb, long waitTimeout) { Preconditions.checkArgument(kvdb != null, "null kvdb"); this.kvdb = kvdb; this.setTimeout(waitTimeout); } @Override public SimpleKVDatabase getKVDatabase() { return this.kvdb; } @Override public void setTimeout(long timeout) { Preconditions.checkArgument(timeout >= 0, "timeout < 0"); this.waitTimeout = timeout; } @Override public ListenableFuture watchKey(byte[] key) { return this.kvdb.watchKey(key); } @Override public byte[] get(byte[] key) { return this.kvdb.get(this, key); } @Override public KVPair getAtLeast(byte[] min, byte[] max) { return this.kvdb.getAtLeast(this, min, max); } @Override public KVPair getAtMost(byte[] max, byte[] min) { return this.kvdb.getAtMost(this, max, min); } @Override public CloseableIterator getRange(byte[] minKey, byte[] maxKey, boolean reverse) { if (minKey == null) minKey = ByteUtil.EMPTY; return new KVPairIterator(this, new KeyRange(minKey, maxKey), null, reverse); } @Override public void put(byte[] key, byte[] value) { this.kvdb.put(this, key, value); } @Override public void remove(byte[] key) { this.kvdb.remove(this, key); } @Override public void removeRange(byte[] minKey, byte[] maxKey) { this.kvdb.removeRange(this, minKey, maxKey); } @Override public boolean isReadOnly() { return this.readOnly; } @Override public void setReadOnly(boolean readOnly) { this.readOnly = readOnly; } @Override public void commit() { this.kvdb.commit(this, this.readOnly); } @Override public void rollback() { this.kvdb.rollback(this); } @Override public CloseableKVStore mutableSnapshot() { // Build copy CloseableKVStore kvstore; if (this.kvdb.kv instanceof NavigableMapKVStore) { final NavigableMapKVStore kv; synchronized (this.kvdb) { kv = ((NavigableMapKVStore)this.kvdb.kv).clone(); } kvstore = new CloseableForwardingKVStore(kv.clone()); } else if (this.kvdb.kv instanceof AtomicKVStore) { final AtomicKVStore kv = (AtomicKVStore)this.kvdb.kv; final CloseableKVStore snapshot = kv.snapshot(); final MutableView view = new MutableView(snapshot); view.disableReadTracking(); kvstore = new CloseableForwardingKVStore(view, snapshot); } else { throw new UnsupportedOperationException("underlying KVStore " + this.kvdb.kv.getClass().getSimpleName() + " is not an AtomicKVStore"); } // Apply mutations synchronized (this.kvdb) { for (Mutation mutation : this.mutations) mutation.apply(kvstore); } // Done return kvstore; } // Find the mutation that overlaps with the given key, if any. // This method assumes we are already synchronized on the associated database. Mutation findMutation(byte[] key) { // Sanity check during unit testing assert Thread.holdsLock(this.kvdb); assert !this.hasOverlaps() && !this.hasEmpties(); // Get all mutations starting at or prior to `key' and look for overlap final SortedSet left = this.mutations.headSet(Mutation.key(ByteUtil.getNextKey(key))); if (!left.isEmpty()) { final Mutation last = left.last(); if (last.contains(key)) return last; } return null; } /** * Ensure transaction is eventually rolled back if leaked due to an application bug. */ @Override protected void finalize() throws Throwable { try { if (!this.stale) LoggerFactory.getLogger(this.getClass()).warn(this + " leaked without commit() or rollback()"); this.rollback(); } finally { super.finalize(); } } private boolean hasEmpties() { for (Mutation mutation : this.mutations) { final byte[] minKey = mutation.getMin(); final byte[] maxKey = mutation.getMax(); if (minKey != null && maxKey != null && Arrays.equals(minKey, maxKey)) return true; } return false; } private boolean hasOverlaps() { Mutation previous = null; for (Mutation mutation : this.mutations) { if (previous != null && mutation.overlaps(previous)) return true; previous = mutation; } return false; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy