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

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

There is a newer version: 5.4.1.Final
Show newest version
/*
 * 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.ArrayList;
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;
import org.modeshape.jcr.query.QueryResults.TupleReformatter;

/**
 */
public abstract class SetOperationComponent extends ProcessingComponent {

    private final Iterable sources;
    protected final List sourceReformatters;
    protected final Comparator removeDuplicatesComparator;
    private final Columns columns;

    protected SetOperationComponent( QueryContext context,
                                     Columns columns,
                                     Iterable sources,
                                     boolean alreadySorted,
                                     boolean all ) {
        super(context, columns);
        assert unionCompatible(columns, sources);
        this.sources = wrapWithLocationOrdering(sources, alreadySorted);
        // Use for sorting the columns that may have been wrapped with location ordering ...
        List reformatters = new ArrayList();
        Columns reformattedColumns = null;
        for (ProcessingComponent component : sources) {
            TupleReformatter reformatter = component.getColumns().getTupleReformatter();
            reformatters.add(reformatter);
            if (reformattedColumns == null && reformatter != null) {
                reformattedColumns = reformatter.getColumns();
            }
        }
        Columns newColumns = this.sources.iterator().next().getColumns();
        if (Collections.frequency(reformatters, reformatters.get(0)) == reformatters.size()) {
            // They are all the same, so don't reformat anything ...
            this.sourceReformatters = Collections.nCopies(reformatters.size(), null);
        } else {
            // We have to reformat the tuples, and use the corresponding columns with proper indexes
            this.sourceReformatters = reformatters;
            if (reformattedColumns != null) newColumns = reformattedColumns;
        }
        this.columns = newColumns;
        this.removeDuplicatesComparator = all ? null : createSortComparator(context, this.columns);
    }

    protected static boolean unionCompatible( Columns columns,
                                              Iterable sources ) {
        for (ProcessingComponent source : sources) {
            if (!columns.isUnionCompatible(source.getColumns())) return false;
        }
        return true;
    }

    @Override
    public Columns getColumns() {
        return columns;
    }

    /**
     * @return sources
     */
    protected Iterable sources() {
        return sources;
    }

    /**
     * The sources' results must be sorted before the intersection can be computed. Ensure that the sources' results are indeed
     * sorted, and if not wrap them in a sorting component.
     * 
     * @param sources the sources being intersected; may not be null
     * @param alreadySorted true if the sources' results are already sorted, or false if they must be sorted by this component
     * @return the sources (or their wrappers); never null
     */
    protected static Iterable wrapWithLocationOrdering( Iterable sources,
                                                                             boolean alreadySorted ) {
        assert sources != null;
        if (alreadySorted) return sources;
        List wrapped = new LinkedList();
        for (ProcessingComponent source : sources) {
            wrapped.add(new SortLocationsComponent(source));
        }
        return wrapped;
    }

    protected void removeDuplicatesIfRequested( List tuples ) {
        if (removeDuplicatesComparator != null) {
            Iterator iter = tuples.iterator();
            Object[] previous = null;
            while (iter.hasNext()) {
                Object[] current = iter.next();
                if (previous != null && removeDuplicatesComparator.compare(previous, current) == 0) {
                    iter.remove();
                } else {
                    previous = current;
                }
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy