package org.infinispan.query.indexmanager;
import java.io.Serializable;
import org.hibernate.search.backend.AddLuceneWork;
import org.hibernate.search.backend.DeleteLuceneWork;
import org.hibernate.search.backend.FlushLuceneWork;
import org.hibernate.search.backend.IndexWorkVisitor;
import org.hibernate.search.backend.LuceneWork;
import org.hibernate.search.backend.OptimizeLuceneWork;
import org.hibernate.search.backend.PurgeAllLuceneWork;
import org.hibernate.search.backend.UpdateLuceneWork;
import org.hibernate.search.backend.spi.DeleteByQueryLuceneWork;
import org.infinispan.query.backend.KeyTransformationHandler;
/**
* The serialized form of LuceneWork needs to be adjusted after deserialization to apply
* our custom keyTransformers. LuceneWork instances are immutable, so we have to replace them
* with new instances iff an id transformation is needed.
*
* @author Sanne Grinovero <sanne@hibernate.org> (C) 2012 Red Hat Inc.
*/
public class LuceneWorkTransformationVisitor implements IndexWorkVisitor<KeyTransformationHandler, LuceneWork> {
static final LuceneWorkTransformationVisitor INSTANCE = new LuceneWorkTransformationVisitor();
private static final LuceneWorkIdTransformer<AddLuceneWork> addDuplicator = new AddWorkDuplicator();
private static final LuceneWorkIdTransformer<DeleteLuceneWork> deleteDuplicator = new DeleteWorkDuplicator();
private static final LuceneWorkIdTransformer<UpdateLuceneWork> updateDuplicator = new UpdateWorkDuplicator();
private static final LuceneWorkIdTransformer<LuceneWork> returnSameDuplicator = new NotReallyDuplicator();
private LuceneWorkTransformationVisitor() {
//no instances needed
}
@Override
public LuceneWork visitAddWork(AddLuceneWork addLuceneWork, KeyTransformationHandler keyTransformationHandler) {
return addDuplicator.cloneOverridingIdString(addLuceneWork, keyTransformationHandler);
}
@Override
public LuceneWork visitDeleteWork(DeleteLuceneWork deleteLuceneWork, KeyTransformationHandler keyTransformationHandler) {
return deleteDuplicator.cloneOverridingIdString(deleteLuceneWork, keyTransformationHandler);
}
@Override
public LuceneWork visitUpdateWork(UpdateLuceneWork updateLuceneWork, KeyTransformationHandler keyTransformationHandler) {
return updateDuplicator.cloneOverridingIdString(updateLuceneWork, keyTransformationHandler);
}
@Override
public LuceneWork visitOptimizeWork(OptimizeLuceneWork optimizeLuceneWork, KeyTransformationHandler keyTransformationHandler) {
return returnSameDuplicator.cloneOverridingIdString(optimizeLuceneWork, keyTransformationHandler);
}
@Override
public LuceneWork visitFlushWork(FlushLuceneWork flushLuceneWork, KeyTransformationHandler keyTransformationHandler) {
return returnSameDuplicator.cloneOverridingIdString(flushLuceneWork, keyTransformationHandler);
}
@Override
public LuceneWork visitPurgeAllWork(PurgeAllLuceneWork purgeAllLuceneWork, KeyTransformationHandler keyTransformationHandler) {
return returnSameDuplicator.cloneOverridingIdString(purgeAllLuceneWork, keyTransformationHandler);
}
@Override
public LuceneWork visitDeleteByQueryWork(DeleteByQueryLuceneWork work, KeyTransformationHandler p) {
throw new UnsupportedOperationException("delete-by-query is not supported");
}
private static class AddWorkDuplicator implements LuceneWorkIdTransformer<AddLuceneWork> {
@Override
public AddLuceneWork cloneOverridingIdString(final AddLuceneWork lw, final KeyTransformationHandler keyTransformationHandler) {
final Serializable id = lw.getId();
if (id == null) {
//this is serialized work received from a remote node: take the getIdAsString instead
final String idInString = lw.getIdInString();
return new AddLuceneWork(idInString, idInString, lw.getEntityClass(), lw.getDocument(), lw.getFieldToAnalyzerMap());
}
else {
return lw;
}
}
}
private static class UpdateWorkDuplicator implements LuceneWorkIdTransformer<UpdateLuceneWork> {
@Override
public UpdateLuceneWork cloneOverridingIdString(final UpdateLuceneWork lw, final KeyTransformationHandler keyTransformationHandler) {
final Serializable id = lw.getId();
if (id == null) {
//this is serialized work received from a remote node: take the getIdAsString instead
final String idInString = lw.getIdInString();
return new UpdateLuceneWork(idInString, idInString, lw.getEntityClass(), lw.getDocument(), lw.getFieldToAnalyzerMap());
}
else {
return lw;
}
}
}
private static class DeleteWorkDuplicator implements LuceneWorkIdTransformer<DeleteLuceneWork> {
@Override
public DeleteLuceneWork cloneOverridingIdString(final DeleteLuceneWork lw, final KeyTransformationHandler keyTransformationHandler) {
final Serializable id = lw.getId();
if (id == null) {
//this is serialized work received from a remote node: take the getIdAsString instead
final String idInString = lw.getIdInString();
return new DeleteLuceneWork(idInString, idInString, lw.getEntityClass());
}
else {
return lw;
}
}
}
/**
* The remaining cases don't need any key transformation, so we return the same instance.
* Not particularly tricky since these are immutable anyway.
*/
private static class NotReallyDuplicator implements LuceneWorkIdTransformer<LuceneWork> {
@Override
public LuceneWork cloneOverridingIdString(final LuceneWork lw, final KeyTransformationHandler keyTransformationHandler) {
return lw;
}
}
}