org.alfresco.repo.search.impl.lucene.query.DeltaReader.java Source code

Java tutorial

Introduction

Here is the source code for org.alfresco.repo.search.impl.lucene.query.DeltaReader.java

Source

/*
 * #%L
 * Alfresco Legacy Lucene
 * %%
 * Copyright (C) 2005 - 2016 Alfresco Software Limited
 * %%
 * This file is part of the Alfresco software. 
 * If the software was purchased under a paid Alfresco license, the terms of 
 * the paid license agreement will prevail.  Otherwise, the software is 
 * provided under the following open source license terms:
 * 
 * Alfresco is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 * 
 * Alfresco is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 * 
 * You should have received a copy of the GNU Lesser General Public License
 * along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
 * #L%
 */
package org.alfresco.repo.search.impl.lucene.query;

import java.io.IOException;
import java.util.Arrays;

import org.apache.lucene.index.IndexReader;
import org.apache.lucene.index.MultiReader;
import org.apache.lucene.index.Term;
import org.apache.lucene.index.TermDocs;
import org.apache.lucene.index.TermEnum;
import org.apache.lucene.index.TermPositions;

public class DeltaReader extends MultiReader {
    int[][] deletions;

    Boolean hasExclusions = null;

    private IndexReader[] subReaders;

    private int maxDoc = 0;

    private int[] starts;

    public DeltaReader(IndexReader[] readers, int[][] deletions) throws IOException {
        super(readers);
        this.deletions = deletions;
        initialize(readers);
    }

    private void initialize(IndexReader[] subReaders) throws IOException {
        this.subReaders = subReaders;
        starts = new int[subReaders.length + 1]; // build starts array
        for (int i = 0; i < subReaders.length; i++) {
            starts[i] = maxDoc;
            maxDoc += subReaders[i].maxDoc(); // compute maxDocs
        }
        starts[subReaders.length] = maxDoc;
    }

    protected void doCommit() throws IOException {
        // TODO Auto-generated method stub
        throw new UnsupportedOperationException();
    }

    protected void doDelete(int arg0) throws IOException {
        // TODO Auto-generated method stub
        throw new UnsupportedOperationException();
    }

    protected void doUndeleteAll() throws IOException {
        // TODO Auto-generated method stub
        throw new UnsupportedOperationException();
    }

    public boolean hasDeletions() {
        return super.hasDeletions() || hasSearchExclusions();
    }

    private boolean hasSearchExclusions() {
        if (hasExclusions == null) {
            for (int i = 0; i < deletions.length; i++) {
                if (deletions[i].length > 0) {
                    hasExclusions = new Boolean(true);
                    break;
                }
            }
            hasExclusions = new Boolean(false);
        }
        return hasExclusions.booleanValue();
    }

    public boolean isDeleted(int docNumber) {
        int i = readerIndex(docNumber);
        return super.isDeleted(docNumber) || (Arrays.binarySearch(deletions[i], docNumber - starts[i]) != -1);
    }

    private int readerIndex(int n) { // find reader for doc n:
        int lo = 0; // search starts array
        int hi = subReaders.length - 1; // for first element less

        while (hi >= lo) {
            int mid = (lo + hi) >> 1;
            int midValue = starts[mid];
            if (n < midValue)
                hi = mid - 1;
            else if (n > midValue)
                lo = mid + 1;
            else { // found a match
                while (mid + 1 < subReaders.length && starts[mid + 1] == midValue) {
                    mid++; // scan to last match
                }
                return mid;
            }
        }
        return hi;
    }

    public TermDocs termDocs() throws IOException {
        return new DeletingTermDocs(super.termDocs());
    }

    public TermPositions termPositions() throws IOException {
        // TODO Auto-generated method stub
        throw new UnsupportedOperationException();
    }

    private class DeletingTermDocs implements TermDocs {
        TermDocs delegate;

        DeletingTermDocs(TermDocs delegate) {
            super();
            this.delegate = delegate;
        }

        public void seek(Term term) throws IOException {
            delegate.seek(term);
        }

        public void seek(TermEnum termEnum) throws IOException {
            delegate.seek(termEnum);
        }

        public int doc() {
            return delegate.doc();
        }

        public int freq() {
            return delegate.freq();
        }

        public boolean next() throws IOException {
            while (delegate.next()) {
                if (!isDeleted(doc())) {
                    return true;
                }
            }
            return false;
        }

        public int read(int[] docs, int[] freqs) throws IOException {
            int end;
            int deletedCount;
            do {
                end = delegate.read(docs, freqs);
                if (end == 0) {
                    return end;
                }
                deletedCount = 0;
                for (int i = 0; i < end; i++) {
                    if (!isDeleted(docs[i])) {
                        deletedCount++;
                    }
                }
            } while (end == deletedCount);
            // fix up for deleted
            int position = 0;
            for (int i = 0; i < end; i++) {
                if (!isDeleted(i)) {
                    docs[position] = docs[i];
                    freqs[position] = freqs[i];
                    position++;
                }
            }
            return position;
        }

        public boolean skipTo(int docNumber) throws IOException {
            delegate.skipTo(docNumber);
            if (!isDeleted(doc())) {
                return true;
            } else {
                return next();
            }
        }

        public void close() throws IOException {
            delegate.close();
        }

    }
}