org.modeshape.jcr.index.local.LocalDuplicateIndex Maven / Gradle / Ivy
/*
* ModeShape (http://www.modeshape.org)
*
* Licensed 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.modeshape.jcr.index.local;
import java.util.Comparator;
import java.util.NavigableMap;
import java.util.concurrent.atomic.AtomicLong;
import javax.jcr.query.qom.StaticOperand;
import org.mapdb.DB;
import org.mapdb.Fun;
import org.mapdb.Serializer;
import org.modeshape.common.logging.Logger;
import org.modeshape.jcr.cache.NodeKey;
import org.modeshape.jcr.index.local.IndexValues.Converter;
import org.modeshape.jcr.index.local.MapDB.UniqueKey;
import org.modeshape.jcr.spi.index.Index;
/**
* An {@link Index} that associates a single value with multiple {@link NodeKey}s. All values are stored in natural sort order,
* allowing finding all the node keys for a range of values.
*
* @param the type of values
* @author Randall Hauch ([email protected])
*/
final class LocalDuplicateIndex extends LocalMapIndex, T> {
/**
* Create a new index that allows duplicate values across all keys.
*
* @param name the name of the index; may not be null or empty
* @param workspaceName the name of the workspace; may not be null
* @param db the database in which the index information is to be stored; may not be null
* @param converter the converter from {@link StaticOperand} to values being indexed; may not be null
* @param valueSerializer the serializer for the type of value being indexed
* @param comparator the comparator for the values; may not be null
* @return the new index; never null
*/
static LocalDuplicateIndex create( String name,
String workspaceName,
DB db,
Converter converter,
Serializer valueSerializer,
Comparator comparator ) {
return new LocalDuplicateIndex<>(name, workspaceName, db, converter, valueSerializer, comparator);
}
private static final String NEXT_COUNTER = "next-counter";
private final Logger logger = Logger.getLogger(getClass());
private final AtomicLong counter;
protected LocalDuplicateIndex( String name,
String workspaceName,
DB db,
Converter converter,
Serializer valueSerializer,
Comparator comparator ) {
super(name, workspaceName, db, IndexValues.uniqueKeyConverter(converter), MapDB.uniqueKeyBTreeSerializer(valueSerializer,
comparator),
MapDB.uniqueKeySerializer(valueSerializer, comparator));
Long nextCounter = (Long)options.get(NEXT_COUNTER);
this.counter = new AtomicLong(nextCounter != null ? nextCounter.longValue() : 0L);
}
@Override
public void shutdown( boolean destroyed ) {
// Store the value of the next counter in the options map ...
options.put(NEXT_COUNTER, counter.get());
super.shutdown(destroyed);
}
@Override
public void add( String nodeKey,
T value ) {
logger.trace("Adding node '{0}' to '{1}' index with value '{2}'", nodeKey, name, value);
keysByValue.put(new UniqueKey(value, counter.getAndIncrement()), nodeKey);
}
@Override
public void remove( String nodeKey,
T value ) {
// Find all of the actual unique values for the given value ...
UniqueKey fromKey = new UniqueKey(value, 0);
UniqueKey toKey = new UniqueKey(value, Long.MAX_VALUE);
NavigableMap, String> matching = keysByValue.subMap(fromKey, true, toKey, true);
if (logger.isTraceEnabled()) {
for (UniqueKey actualValue : matching.keySet()) {
logger.trace("Removing node '{0}' from '{1}' index with value '{2}'", nodeKey, name, actualValue.actualKey);
keysByValue.remove(actualValue);
}
} else {
for (UniqueKey actualValue : matching.keySet()) {
keysByValue.remove(actualValue);
}
}
}
@Override
public void remove( String nodeKey ) {
// Find all of the T values (entry keys) for the given node key (entry values) ...
for (UniqueKey key : Fun.filter(valuesByKey, nodeKey)) {
logger.trace("Removing node '{0}' from '{1}' index with value '{2}'", nodeKey, name, key.actualKey);
keysByValue.remove(key);
}
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy