package org.infinispan.lucene; import java.io.IOException; import java.io.ObjectInput; import java.io.ObjectOutput; import java.util.Collections; import java.util.Set; import org.infinispan.commons.io.UnsignedNumeric; import org.infinispan.commons.marshall.AbstractExternalizer; /** * Used as a key to distinguish file chunk in cache. * * @since 4.0 * @author Lukasz Moren * @author Sanne Grinovero */ public final class ChunkCacheKey implements IndexScopedKey { private final int chunkId; private final String indexName; private final String fileName; private final int bufferSize; private final int hashCode; private final int affinitySegmentId; public ChunkCacheKey(final String indexName, final String fileName, final int chunkId, final int bufferSize, final int affinitySegmentId) { if (fileName == null) throw new IllegalArgumentException("filename must not be null"); this.indexName = indexName; this.fileName = fileName; this.chunkId = chunkId; this.bufferSize = bufferSize; this.affinitySegmentId = affinitySegmentId; this.hashCode = generatedHashCode(); } /** * Get the chunkId. * * @return the chunkId. */ public int getChunkId() { return chunkId; } /** * Get the bufferSize. * * @return the bufferSize. */ public int getBufferSize() { return bufferSize; } /** * Get the indexName. * * @return the indexName. */ @Override public String getIndexName() { return indexName; } @Override public int getAffinitySegmentId() { return affinitySegmentId; } @Override public Object accept(KeyVisitor visitor) throws Exception { return visitor.visit(this); } /** * Get the fileName. * * @return the fileName. */ public String getFileName() { return fileName; } @Override public int hashCode() { return hashCode; } private int generatedHashCode() { //bufferSize and indexName excluded from computation: not very relevant final int prime = 31; int result = prime + fileName.hashCode(); return prime * result + chunkId; //adding chunkId as last is better } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (ChunkCacheKey.class != obj.getClass()) return false; ChunkCacheKey other = (ChunkCacheKey) obj; if (chunkId != other.chunkId) return false; if (!fileName.equals(other.fileName)) return false; return indexName.equals(other.indexName); } /** * Changing the encoding could break backwards compatibility * * @see LuceneKey2StringMapper#getKeyMapping(String) */ @Override public String toString() { return "C|" + fileName + "|" + chunkId + "|" + bufferSize + "|" + indexName + "|" + affinitySegmentId; } public static final class Externalizer extends AbstractExternalizer<ChunkCacheKey> { @Override public void writeObject(final ObjectOutput output, final ChunkCacheKey key) throws IOException { output.writeUTF(key.indexName); output.writeUTF(key.fileName); UnsignedNumeric.writeUnsignedInt(output, key.chunkId); UnsignedNumeric.writeUnsignedInt(output, key.bufferSize); UnsignedNumeric.writeUnsignedInt(output, key.affinitySegmentId); } @Override public ChunkCacheKey readObject(final ObjectInput input) throws IOException { final String indexName = input.readUTF(); final String fileName = input.readUTF(); final int chunkId = UnsignedNumeric.readUnsignedInt(input); final int bufferSize = UnsignedNumeric.readUnsignedInt(input); final int affinitySegmentId = UnsignedNumeric.readUnsignedInt(input); return new ChunkCacheKey(indexName, fileName, chunkId, bufferSize, affinitySegmentId); } @Override public Integer getId() { return ExternalizerIds.CHUNK_CACHE_KEY; } @Override public Set<Class<? extends ChunkCacheKey>> getTypeClasses() { return Collections.singleton(ChunkCacheKey.class); } } }