com.tectonica.collections.InMemKeyValueStore Maven / Gradle / Ivy
package com.tectonica.collections;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import com.tectonica.util.SerializeUtil;
public class InMemKeyValueStore extends KeyValueStore
{
private final ConcurrentHashMap entries;
private final ConcurrentHashMap locks;
private final List> indexes;
/**
* creates an in-memory data store, suitable mostly for development.
*
* @param keyMapper
* this optional parameter is suitable in situations where the key of an entry can be inferred from its value directly
* (as opposed to when the key and value are stored separately). when provided, several convenience methods become applicable
*/
public InMemKeyValueStore(KeyMapper keyMapper)
{
super(keyMapper);
this.entries = new ConcurrentHashMap<>();
this.locks = new ConcurrentHashMap<>();
this.indexes = new ArrayList<>();
}
protected class InMemEntry implements Modifier, KeyValue
{
private final K _key; // never null
private V _value; // never null
public InMemEntry(K key, V value)
{
if (key == null || value == null)
throw new NullPointerException();
_key = key;
_value = value;
}
@Override
public K getKey()
{
return _key;
}
@Override
public V getValue()
{
return _value;
}
@Override
public V getModifiableValue()
{
return SerializeUtil.copyOf(_value); // TODO: replace with a more efficient implementation
// return KryoUtil.copyOf(_value);
}
@Override
public void dbPut(V value)
{
V oldEntry = _value;
_value = value;
reindex(_key, oldEntry, value);
}
}
/***********************************************************************************
*
* GETTERS
*
***********************************************************************************/
@Override
protected V dbGet(K key)
{
KeyValue kv = entries.get(key);
if (kv == null)
return null;
return kv.getValue();
}
@SuppressWarnings({ "unchecked", "rawtypes" })
@Override
public Iterator> iterator()
{
return (Iterator) entries.values().iterator();
}
@Override
public Iterator keyIterator()
{
return entries.keySet().iterator();
}
@Override
protected Iterator> dbOrderedIterator(Collection keys)
{
List> list = new ArrayList<>();
for (K key : keys)
{
KeyValue kv = entries.get(key);
if (kv != null)
list.add(kv);
}
return list.iterator();
}
@Override
public Set keySet()
{
return entries.keySet();
}
/***********************************************************************************
*
* SETTERS (UTILS)
*
***********************************************************************************/
@Override
protected Modifier getModifier(K key, ModificationType purpose)
{
return entries.get(key);
}
@Override
public Lock getModificationLock(K key)
{
Lock lock;
Lock existing = locks.putIfAbsent(key, lock = new ReentrantLock());
if (existing != null)
lock = existing;
return lock;
}
/***********************************************************************************
*
* SETTERS
*
***********************************************************************************/
@Override
protected void dbInsert(K key, V value)
{
Modifier existing = entries.putIfAbsent(key, new InMemEntry(key, value));
if (existing == null)
reindex(key, null, value);
else
throw new RuntimeException("attempted to insert entry with existing key " + key);
}
/***********************************************************************************
*
* DELETERS
*
***********************************************************************************/
@Override
protected boolean dbDelete(K key)
{
InMemEntry removed = null;
if (indexes.size() == 0)
removed = entries.remove(key); // without indexes to update, this is a primitive operation
else
{
KeyValue kv = entries.get(key);
if (kv != null)
{
V oldValue = kv.getValue();
removed = entries.remove(key);
reindex(key, oldValue, null);
}
}
return (removed != null);
}
@Override
protected int dbDeleteAll()
{
int removed = entries.size();
entries.clear();
locks.clear();
clearIndices();
return removed;
}
/***********************************************************************************
*
* INDEXES
*
***********************************************************************************/
@Override
public Index createIndex(String indexName, IndexMapper mapper)
{
if (entries.size() > 0)
throw new RuntimeException("adding indexes on non-empty data set is not supported yet");
InMemIndexImpl index = new InMemIndexImpl<>(mapper, indexName);
indexes.add(index);
return index;
}
/**
* straightforward in-memory implementation of an index
*
* @author Zach Melamed
*/
public class InMemIndexImpl extends Index
{
private ConcurrentMultimap
© 2015 - 2025 Weber Informatics LLC | Privacy Policy