All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.modeshape.jcr.query.process.ExceptComponent Maven / Gradle / Ivy

/*
 * ModeShape (http://www.modeshape.org)
 * See the COPYRIGHT.txt file distributed with this work for information
 * regarding copyright ownership.  Some portions may be licensed
 * to Red Hat, Inc. under one or more contributor license agreements.
 * See the AUTHORS.txt file in the distribution for a full listing of 
 * individual contributors.
 *
 * ModeShape is free software. Unless otherwise indicated, all code in ModeShape
 * is licensed to you under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 2.1 of
 * the License, or (at your option) any later version.
 * 
 * ModeShape 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 this software; if not, write to the Free
 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
 */
package org.modeshape.jcr.query.process;

import java.util.Collections;
import java.util.Comparator;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import org.modeshape.jcr.query.QueryContext;
import org.modeshape.jcr.query.QueryResults.Columns;

/**
 */
public class ExceptComponent extends SetOperationComponent {

    private final Comparator comparator;

    public ExceptComponent( QueryContext context,
                            Columns columns,
                            Iterable sources,
                            boolean alreadySorted,
                            boolean all ) {
        super(context, columns, sources, alreadySorted, all);
        this.comparator = createSortComparator(context, columns);
    }

    @Override
    public List execute() {
        Iterator sources = sources().iterator();
        if (!sources.hasNext()) return emptyTuples();

        // Execute all of the source components (so we can sort the results from the smallest to the largest) ...
        // TODO: Parallelize this ???
        List> allTuples = new LinkedList>();
        while (sources.hasNext()) {
            List tuples = sources.next().execute();
            if (tuples == null) continue;
            if (tuples.isEmpty()) return emptyTuples();
            allTuples.add(tuples);
        }
        if (allTuples.isEmpty()) return emptyTuples();
        if (allTuples.size() == 1) return allTuples.get(0); // just one source
        // Sort the tuples by size, starting with the smallest ...
        Collections.sort(allTuples, new Comparator>() {
            @Override
            public int compare( List tuples1,
                                List tuples2 ) {
                return tuples1.size() - tuples2.size();
            }
        });

        // Do the sources 2 at a time ...
        Iterator> iter = allTuples.iterator();
        List tuples = iter.next(); // already sorted ...
        assert iter.hasNext();
        // Process the next source with the previous ...
        while (iter.hasNext()) {
            List next = iter.next(); // already sorted ...
            // Walk through the first and second results ...
            Iterator tupleIter = tuples.iterator();
            Iterator nextIter = next.iterator();
            // There is at least one tuple in each ...
            Object[] tuple1 = tupleIter.next();
            Object[] tuple2 = nextIter.next();
            while (true) {
                int comparison = comparator.compare(tuple1, tuple2);
                if (comparison == 0) {
                    // Both match, so remove the tuple from 'tuples' and go on ...
                    tupleIter.remove();

                    if (!tupleIter.hasNext()) {
                        // This was the last one
                        break;
                    }
                    tuple1 = tupleIter.next();
                } else if (comparison < 0) {
                    // No match, so leave tuple1 in 'tuples'

                    // tuple1 is less than tuple2, so advance tupleIter ...
                    if (!tupleIter.hasNext()) {
                        // The intersection results ('tuples') has no more tuples, so go to the next source ...
                        break;
                    }
                    tuple1 = tupleIter.next();
                } else {
                    assert comparison > 0;
                    // tuple1 is greater than tuple2, so advance nextIter ...
                    if (!nextIter.hasNext()) {
                        // The next source has no more tuples, so leave all remaining tuples, and go to the next source ...
                        break;
                    }
                    tuple2 = nextIter.next();
                }
            }
        }
        // Remove duplicates if requested to ...
        removeDuplicatesIfRequested(tuples);
        // Return all of the common tuples that were in all sources ...
        return tuples;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy