org.modeshape.jcr.index.local.IndexValues 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.Collections;
import java.util.Map;
import javax.jcr.query.qom.BindVariableValue;
import javax.jcr.query.qom.StaticOperand;
import org.modeshape.jcr.api.index.IndexDefinition.IndexKind;
import org.modeshape.jcr.index.local.MapDB.UniqueKey;
import org.modeshape.jcr.query.model.Literal;
import org.modeshape.jcr.query.model.LiteralValue;
import org.modeshape.jcr.value.ValueFactory;
/**
* Utilities for working with individual values.
*
* @author Randall Hauch ([email protected])
*/
public class IndexValues {
/**
* Converts between {@link StaticOperand} and an index's key type.
*
* Note that {@link IndexKind#VALUE} indexes may have multiple node keys for a given value, which means that any given value
* might have a range of keys. Some MapDB-based indexes store such entries using an index key that consists of the actual
* property value plus a counter. In this way, there are multiple index keys for a given {@link StaticOperand}, and the
* {@link #toLowerValue(StaticOperand)} will return the lowest-possible index key given the {@link StaticOperand}, while the
* {@link #toUpperValue(StaticOperand)} will return the highest-possible index key for the given {@link StaticOperand}
*
*
* {@link IndexKind#UNIQUE_VALUE} indexes will use the actual property value as the index key, and thus the
* {@link #toLowerValue(StaticOperand)} and {@link #toUpperValue(StaticOperand)} methods will both return the same index key
* for the same {@link StaticOperand}.
*
* @param the index's key type
* @author Randall Hauch ([email protected])
*/
public static interface Converter {
/**
* Create the lowest possible index key for the supplied {@link StaticOperand}.
*
* @param operand the static operand; may be null
* @return the lowest/smallest possible index key; or null if the operand is null
*/
public T toLowerValue( StaticOperand operand );
/**
* Create the highest possible index key for the supplied {@link StaticOperand}.
*
* @param operand the static operand; may be null
* @return the highest/largest possible index key; or null if the operand is null
*/
T toUpperValue( StaticOperand operand );
}
private static final Map NO_VARIABLES = Collections.emptyMap();
public static Converter> uniqueKeyConverter( Converter converter ) {
return new UniqueKeyConverter(converter);
}
public static Converter converter( ValueFactory factory,
Map variables ) {
return new StandardConverter(factory, variables);
}
public static Converter converter( ValueFactory factory ) {
return new StandardConverter(factory, NO_VARIABLES);
}
static final class UniqueKeyConverter implements Converter> {
private final Converter valueConverter;
protected UniqueKeyConverter( Converter valueConverter ) {
this.valueConverter = valueConverter;
}
@Override
public UniqueKey toLowerValue( StaticOperand operand ) {
T value = valueConverter.toLowerValue(operand);
return value != null ? new UniqueKey(value, 0L) : null;
}
@Override
public UniqueKey toUpperValue( StaticOperand operand ) {
T value = valueConverter.toLowerValue(operand);
return value != null ? new UniqueKey(value, Long.MAX_VALUE) : null;
}
}
protected static class StandardConverter implements Converter {
private final ValueFactory factory;
private final Map variables;
protected StandardConverter( ValueFactory factory,
Map variables ) {
this.factory = factory;
this.variables = variables;
}
@Override
public T toLowerValue( StaticOperand operand ) {
if (operand instanceof Literal) {
Literal literal = (Literal)operand;
return factory.create(literal.value());
}
if (operand instanceof BindVariableValue) {
BindVariableValue bind = (BindVariableValue)operand;
return factory.create(variables.get(bind.getBindVariableName()));
}
throw new LocalIndexException("Unexpected static operand: " + operand);
}
@Override
public T toUpperValue( StaticOperand operand ) {
if (operand instanceof LiteralValue) {
LiteralValue literal = (LiteralValue)operand;
return factory.create(literal.value());
}
if (operand instanceof BindVariableValue) {
BindVariableValue bind = (BindVariableValue)operand;
return factory.create(variables.get(bind.getBindVariableName()));
}
throw new LocalIndexException("Unexpected static operand: " + operand);
}
}
private IndexValues() {
}
}