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

org.modeshape.jcr.query.process.QueryResults 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.Collections;
import java.util.Iterator;
import java.util.List;
import org.modeshape.common.annotation.Immutable;
import org.modeshape.common.collection.ImmutableProblems;
import org.modeshape.common.collection.Problems;
import org.modeshape.common.collection.SimpleProblems;
import org.modeshape.common.util.StringUtil;
import org.modeshape.jcr.GraphI18n;
import org.modeshape.jcr.query.model.Column;
import org.modeshape.jcr.query.model.TypeSystem;
import org.modeshape.jcr.query.model.TypeSystem.TypeFactory;

/**
 * The resulting output of a query.
 */
@Immutable
public class QueryResults implements org.modeshape.jcr.query.QueryResults {
    private static final Problems NO_PROBLEMS = new ImmutableProblems(new SimpleProblems());

    private static final long serialVersionUID = 1L;

    private final Problems problems;
    private final Columns columns;
    private final List tuples;
    private final int[] tupleIndexesForColumns;
    private final Statistics statistics;
    private final String plan;

    /**
     * Create a results object for the supplied context, command, and result columns and with the supplied tuples.
     * 
     * @param columns the definition of the query result columns
     * @param statistics the statistics for this query; may not be null
     * @param tuples the tuples
     * @param problems the problems; may be null if there are no problems
     * @param plan the text representation of the query plan, if the hints asked for it
     */
    public QueryResults( Columns columns,
                         Statistics statistics,
                         List tuples,
                         Problems problems,
                         String plan ) {
        assert columns != null;
        assert statistics != null;
        this.problems = problems != null ? problems : NO_PROBLEMS;
        this.columns = columns;
        this.tuples = tuples;
        this.statistics = statistics;
        this.plan = plan;
        // Precompute the indexes for each tuple, given the desired columns ...
        int numLocations = this.columns.getLocationCount();
        int numScores = this.columns.hasFullTextSearchScores() ? numLocations : 0;
        this.tupleIndexesForColumns = new int[this.columns.getColumnCount() + numLocations + numScores];
        int i = 0;
        for (Column column : this.columns) {
            this.tupleIndexesForColumns[i++] = this.columns.getColumnIndexForProperty(column.selectorName().getString(),
                                                                                      column.getPropertyName());
        }
        for (String selectorName : this.columns.getSelectorNames()) {
            this.tupleIndexesForColumns[i++] = this.columns.getLocationIndex(selectorName);
        }
        if (this.columns.hasFullTextSearchScores()) {
            for (String selectorName : this.columns.getSelectorNames()) {
                this.tupleIndexesForColumns[i++] = this.columns.getFullTextSearchScoreIndexFor(selectorName);
            }
        }
    }

    /**
     * Create a results object for the supplied context, command, and result columns and with the supplied tuples.
     * 
     * @param columns the definition of the query result columns
     * @param statistics the statistics for this query; may not be null
     * @param tuples the tuples
     */
    public QueryResults( Columns columns,
                         Statistics statistics,
                         List tuples ) {
        this(columns, statistics, tuples, NO_PROBLEMS, null);
    }

    /**
     * Create an empty {@link QueryResults} object for the supplied context, command, and result columns.
     * 
     * @param columns the definition of the query result columns
     * @param statistics the statistics for this query; may not be null
     * @param problems the problems; may be null if there are no problems
     */
    public QueryResults( Columns columns,
                         Statistics statistics,
                         Problems problems ) {
        this(columns, statistics, Collections.emptyList(), problems, null);
    }

    /**
     * Create an empty {@link QueryResults} object for the supplied context, command, and result columns.
     * 
     * @param columns the definition of the query result columns
     * @param statistics the statistics for this query; may not be null
     */
    public QueryResults( Columns columns,
                         Statistics statistics ) {
        this(columns, statistics, Collections.emptyList(), null, null);
    }

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

    @Override
    public Cursor getCursor() {
        return new TupleCursor(columns, tuples.iterator());
    }

    @Override
    public List getTuples() {
        return tuples;
    }

    @Override
    public int getRowCount() {
        return tuples.size();
    }

    @Override
    public String getPlan() {
        return plan;
    }

    @Override
    public Problems getProblems() {
        return problems;
    }

    @Override
    public boolean hasErrors() {
        return getProblems().hasErrors();
    }

    @Override
    public boolean hasWarnings() {
        return getProblems().hasWarnings();
    }

    @Override
    public Statistics getStatistics() {
        return statistics;
    }

    @Override
    public String toString() {
        return toString(null, Integer.MAX_VALUE);
    }

    /**
     * Get a string representation of this result object, with a maximum number of tuples to include.
     * 
     * @param typeSystem the type system that can be used to convert the values to a string; may be null if
     *        {@link Object#toString()} should be used
     * @param maxTuples the maximum number of tuples to print, or {@link Integer#MAX_VALUE} if all the tuples are to be printed
     * @return the string representation; never null
     */
    public String toString( TypeSystem typeSystem,
                            int maxTuples ) {
        StringBuilder sb = new StringBuilder();
        toString(typeSystem, sb, maxTuples);
        return sb.toString();
    }

    /**
     * Get a string representation of this result object.
     * 
     * @param typeSystem the type system that can be used to convert the values to a string; may be null if
     *        {@link Object#toString()} should be used
     * @param sb the string builder to which the results should be written; may not be null
     */
    public void toString( TypeSystem typeSystem,
                          StringBuilder sb ) {
        toString(typeSystem, sb, Integer.MAX_VALUE);
    }

    /**
     * Get a string representation of this result object, with a maximum number of tuples to include.
     * 
     * @param typeSystem the type system that can be used to convert the values to a string; may be null if
     *        {@link Object#toString()} should be used
     * @param sb the string builder to which the results should be written; may not be null
     * @param maxTuples the maximum number of tuples to print, or {@link Integer#MAX_VALUE} if all the tuples are to be printed
     */
    public void toString( TypeSystem typeSystem,
                          StringBuilder sb,
                          int maxTuples ) {
        int[] columnWidths = determineColumnWidths(typeSystem, Integer.MAX_VALUE, true);
        printDelimiterLine(sb, columnWidths, true);
        printHeader(sb, columnWidths);
        printDelimiterLine(sb, columnWidths, true);
        printLines(typeSystem, sb, columnWidths, maxTuples);
        printDelimiterLine(sb, columnWidths, false);
    }

    /**
     * Determine the width of each column.
     * 
     * @param typeSystem the type system that can be used to convert the values to a string; may be null if
     *        {@link Object#toString()} should be used
     * @param maxWidth the maximum width; must be positive
     * @param useData true if the data should be used to compute the length, or false if just the column names should be used
     * @return the array of widths for each column, excluding any decorating characters; never null
     */
    protected int[] determineColumnWidths( TypeSystem typeSystem,
                                           int maxWidth,
                                           boolean useData ) {
        assert maxWidth > 0;
        int tupleLength = columns.getTupleSize();
        int[] columnWidths = new int[tupleLength + 1]; // +1 for the row number column
        for (int i = 0; i != columnWidths.length; ++i) {
            columnWidths[i] = 0;
        }
        // Determine the width of the first column that shows the row number ...
        String rowNumber = Integer.toString(getTuples().size());
        columnWidths[0] = rowNumber.length();

        // Compute the column names ...
        List tupleValueNames = columns.getTupleValueNames();
        for (int i = 0, j = 1, max = tupleValueNames.size(); i != max; ++i, ++j) {
            String name = tupleValueNames.get(i);
            columnWidths[j] = Math.max(Math.min(maxWidth, name.length()), columnWidths[j]);
        }
        // Look at the data ...
        if (useData) {
            for (Object[] tuple : getTuples()) {
                for (int i = 0, j = 1; i != tupleLength; ++i, ++j) {
                    String valueStr = stringOf(typeSystem, tuple[this.tupleIndexesForColumns[i]]);
                    if (valueStr == null) continue;
                    columnWidths[j] = Math.max(Math.min(maxWidth, valueStr.length()), columnWidths[j]);
                }
            }
        }
        return columnWidths;
    }

    protected String stringOf( TypeSystem typeSystem,
                               Object value ) {
        if (value == null) return null;
        if (value instanceof Object[]) {
            // Multi-valued ...
            StringBuilder sb = new StringBuilder();
            Object[] array = (Object[])value;
            int len = array.length;
            for (int i = 0; i != len; ++i) {
                if (i != 0) sb.append(", ");
                sb.append(stringOf(typeSystem, array[i]));
            }
            return sb.toString();
        }
        if (typeSystem == null) return value.toString();
        TypeFactory typeFactory = typeSystem.getTypeFactory(value);
        return typeFactory.asReadableString(value);
    }

    protected void printHeader( StringBuilder sb,
                                int[] columnWidths ) {
        // Print the row number column ...
        sb.append("| ").append(StringUtil.justifyLeft("#", columnWidths[0], ' ')).append(' ');
        // Print the name line ...
        sb.append('|');
        int i = 1;
        for (String name : columns.getTupleValueNames()) {
            sb.append(' ');
            sb.append(StringUtil.justifyLeft(name, columnWidths[i], ' '));
            sb.append(" |");
            ++i;
        }
        sb.append('\n');
    }

    protected void printLines( TypeSystem typeSystem,
                               StringBuilder sb,
                               int[] columnWidths,
                               int maxRowsToPrint ) {
        int rowNumber = 1;
        int tupleLength = columns.getTupleSize();
        // Should they all be printed ?
        if (maxRowsToPrint > tuples.size()) {
            // Print all tuples ...
            for (Object[] tuple : getTuples()) {
                printTuple(typeSystem, sb, columnWidths, rowNumber, tupleLength, tuple);
                ++rowNumber;
            }
        } else {
            // Print max number of rows ...
            for (Object[] tuple : getTuples()) {
                printTuple(typeSystem, sb, columnWidths, rowNumber, tupleLength, tuple);
                if (rowNumber >= maxRowsToPrint) break;
                ++rowNumber;
            }
        }

    }

    private final void printTuple( TypeSystem typeSystem,
                                   StringBuilder sb,
                                   int[] columnWidths,
                                   int rowNumber,
                                   int tupleLength,
                                   Object[] tuple ) {
        // Print the row number column ...
        sb.append("| ").append(StringUtil.justifyLeft(Integer.toString(rowNumber), columnWidths[0], ' ')).append(' ');
        // Print the remaining columns ...
        for (int i = 0, j = 1; i != tupleLength; ++i, ++j) {
            String valueStr = stringOf(typeSystem, tuple[tupleIndexesForColumns[i]]);
            valueStr = StringUtil.justifyLeft(valueStr, columnWidths[j], ' ');
            sb.append('|').append(' ').append(valueStr).append(' ');
        }
        sb.append('|');
        sb.append('\n');
    }

    protected void printDelimiterLine( StringBuilder sb,
                                       int[] columnWidths,
                                       boolean includeLineFeed ) {
        sb.append('+');
        for (int i = 0, max = columnWidths.length; i != max; ++i) {
            for (int j = 0, width = columnWidths[i] + 2; j != width; ++j) { // +1 for space before, +1 for space after
                sb.append('-');
            }
            sb.append('+');
        }
        if (includeLineFeed) sb.append('\n');
    }

    /**
     * An interface used to walk through the results.
     */
    public final class TupleCursor implements Cursor {
        private final Columns columns;
        private final Iterator iterator;
        private Object[] currentTuple;
        private int tupleIndex;

        protected TupleCursor( Columns columns,
                               Iterator iterator ) {
            this.iterator = iterator;
            this.columns = columns;
            this.tupleIndex = -1;
        }

        @Override
        public boolean hasNext() {
            return iterator.hasNext();
        }

        @Override
        public void next() {
            currentTuple = iterator.next();
            ++tupleIndex;
        }

        @Override
        public Location getLocation( int columnNumber ) {
            return (Location)currentTuple[columns.getLocationIndexForColumn(columnNumber)];
        }

        @Override
        public Location getLocation( String selectorName ) {
            return (Location)currentTuple[columns.getLocationIndex(selectorName)];
        }

        @Override
        public int getRowIndex() {
            return tupleIndex;
        }

        @Override
        public Object getValue( int columnNumber ) {
            if (columnNumber >= columns.getColumnCount()) {
                throw new IndexOutOfBoundsException();
            }
            if (currentTuple == null) {
                throw new IllegalStateException(GraphI18n.nextMethodMustBeCalledBeforeGettingValue.text());
            }
            return currentTuple[columnNumber];
        }

        @Override
        public Object getValue( String columnName ) {
            if (currentTuple == null) {
                throw new IllegalStateException(GraphI18n.nextMethodMustBeCalledBeforeGettingValue.text());
            }
            return currentTuple[columns.getColumnIndexForName(columnName)];
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy