com.hazelcast.query.impl.IndexImpl Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2015, Hazelcast, Inc. All Rights Reserved.
*
* 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 com.hazelcast.query.impl;
import com.hazelcast.nio.serialization.Data;
import com.hazelcast.query.QueryException;
import com.hazelcast.query.impl.TypeConverters.TypeConverter;
import java.util.HashSet;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
/**
* Implementation for {@link com.hazelcast.query.impl.Index}
*/
public class IndexImpl implements Index {
/**
* Creates instance from NullObject is a inner class.
*/
public static final NullObject NULL = new NullObject();
// indexKey -- indexValue
private final ConcurrentMap recordValues = new ConcurrentHashMap(1000);
private final IndexStore indexStore;
private final String attribute;
private final boolean ordered;
private volatile TypeConverter converter;
public IndexImpl(String attribute, boolean ordered) {
this.attribute = attribute;
this.ordered = ordered;
indexStore = (ordered) ? new SortedIndexStore() : new UnsortedIndexStore();
}
@Override
public void removeEntryIndex(Data indexKey) {
Comparable oldValue = recordValues.remove(indexKey);
if (oldValue != null) {
indexStore.removeIndex(oldValue, indexKey);
}
}
@Override
public void clear() {
recordValues.clear();
indexStore.clear();
// Clear converter
converter = null;
}
ConcurrentMap getRecordMap(Comparable indexValue) {
return indexStore.getRecordMap(indexValue);
}
@Override
public void saveEntryIndex(QueryableEntry e) throws QueryException {
/*
* At first, check if converter is not initialized, initialize it before saving an entry index
* Because, if entity index is saved before,
* that thread can be blocked before executing converter setting code block,
* another thread can query over indexes without knowing the converter and
* this causes to class cast exceptions.
*/
if (converter == null) {
// Initialize attribute type by using entry index
AttributeType attributeType = e.getAttributeType(attribute);
converter = attributeType == null ? new IdentityConverter() : attributeType.getConverter();
}
Data key = e.getIndexKey();
Comparable oldValue = recordValues.remove(key);
Comparable newValue = e.getAttribute(attribute);
if (newValue == null) {
newValue = NULL;
} else if (newValue.getClass().isEnum()) {
newValue = TypeConverters.ENUM_CONVERTER.convert(newValue);
}
recordValues.put(key, newValue);
if (oldValue == null) {
// new
indexStore.newIndex(newValue, e);
} else {
// update
indexStore.updateIndex(oldValue, newValue, e);
}
}
@Override
public Set getRecords(Comparable[] values) {
if (values.length == 1) {
if (converter != null) {
return indexStore.getRecords(convert(values[0]));
} else {
return new SingleResultSet(null);
}
} else {
MultiResultSet results = new MultiResultSet();
if (converter != null) {
Set convertedValues = new HashSet(values.length);
for (Comparable value : values) {
convertedValues.add(convert(value));
}
indexStore.getRecords(results, convertedValues);
}
return results;
}
}
@Override
public Set getRecords(Comparable value) {
if (converter != null) {
return indexStore.getRecords(convert(value));
} else {
return new SingleResultSet(null);
}
}
@Override
public Set getSubRecordsBetween(Comparable from, Comparable to) {
MultiResultSet results = new MultiResultSet();
if (converter != null) {
indexStore.getSubRecordsBetween(results, convert(from), convert(to));
}
return results;
}
@Override
public Set getSubRecords(ComparisonType comparisonType, Comparable searchedValue) {
MultiResultSet results = new MultiResultSet();
if (converter != null) {
indexStore.getSubRecords(results, comparisonType, convert(searchedValue));
}
return results;
}
private Comparable convert(Comparable value) {
return converter.convert(value);
}
public ConcurrentMap getRecordValues() {
return recordValues;
}
@Override
public String getAttributeName() {
return attribute;
}
@Override
public boolean isOrdered() {
return ordered;
}
/**
* Provides comparable null object.
*/
public static final class NullObject implements Comparable {
@Override
public int compareTo(Object o) {
if (o == this || o instanceof NullObject) {
return 0;
}
return -1;
}
@Override
public int hashCode() {
return 0;
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
return true;
}
}
}