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

org.modeshape.jcr.query.JcrQueryResult Maven / Gradle / Ivy

There is a newer version: 5.4.1.Final
Show newest version
/*
 * ModeShape (http://www.modeshape.org)
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *       http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.modeshape.jcr.query;

import java.util.Collection;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.Set;
import javax.jcr.ItemNotFoundException;
import javax.jcr.Node;
import javax.jcr.NodeIterator;
import javax.jcr.Property;
import javax.jcr.PropertyType;
import javax.jcr.RangeIterator;
import javax.jcr.RepositoryException;
import javax.jcr.Value;
import javax.jcr.query.Row;
import javax.jcr.query.RowIterator;
import org.modeshape.common.annotation.NotThreadSafe;
import org.modeshape.common.collection.Collections;
import org.modeshape.common.collection.Problem;
import org.modeshape.common.collection.Problem.Status;
import org.modeshape.jcr.JcrI18n;
import org.modeshape.jcr.cache.CachedNode;
import org.modeshape.jcr.cache.CachedNodeSupplier;
import org.modeshape.jcr.query.NodeSequence.Batch;
import org.modeshape.jcr.query.NodeSequence.Restartable;
import org.modeshape.jcr.query.QueryResults.Columns;
import org.modeshape.jcr.query.engine.process.RestartableSequence;
import org.modeshape.jcr.query.engine.process.SecureSequence;

/**
 * The results of a query. This is not thread-safe because it relies upon JcrSession, which is not thread-safe. Also, although the
 * results of a query never change, the objects returned by the iterators may vary if the session information changes.
 * 
 * @see XPathQueryResult
 * @see JcrSqlQueryResult
 */
@NotThreadSafe
public class JcrQueryResult implements org.modeshape.jcr.api.query.QueryResult {
    public static final String JCR_SCORE_COLUMN_NAME = "jcr:score";
    public static final String JCR_PATH_COLUMN_NAME = "jcr:path";
    public static final String JCR_NAME_COLUMN_NAME = "jcr:name";
    public static final String JCR_UUID_COLUMN_NAME = "jcr:uuid";
    public static final String MODE_LOCALNAME_COLUMN_NAME = "mode:localName";
    public static final String MODE_DEPTH_COLUMN_NAME = "mode:depth";
    public static final String MODE_ID_COLUMN_NAME = "mode:id";
    protected static final Set PSEUDO_COLUMNS = Collections.unmodifiableSet(JCR_SCORE_COLUMN_NAME, JCR_PATH_COLUMN_NAME,
                                                                                    JCR_NAME_COLUMN_NAME, JCR_UUID_COLUMN_NAME,
                                                                                    MODE_LOCALNAME_COLUMN_NAME,
                                                                                    MODE_DEPTH_COLUMN_NAME, MODE_ID_COLUMN_NAME);

    protected final JcrQueryContext context;
    protected final QueryResults results;
    protected final String queryStatement;
    private final NodeSequence sequence;
    private final boolean restartable;
    private List warnings;
    private boolean accessed = false;

    protected JcrQueryResult( JcrQueryContext context,
                              String query,
                              QueryResults results,
                              boolean restartable,
                              int numRowsInMemory ) {
        this.context = context;
        this.results = results;
        this.queryStatement = query;
        this.restartable = restartable;
        NodeSequence rows = results.getRows();
        if (rows.isEmpty()) {
            this.sequence = rows;
        } else if (!restartable) {
            this.sequence = new SecureSequence(rows, context);
        } else {
            String workspace = context.getWorkspaceName();
            BufferManager bufferMgr = context.getBufferManager();
            CachedNodeSupplier nodeCache = results.getCachedNodes();
            NodeSequence secureSequence = new SecureSequence(rows, context);
            this.sequence = new RestartableSequence(workspace, secureSequence, bufferMgr, nodeCache, numRowsInMemory);
        }

        assert this.context != null;
        assert this.results != null;
        assert this.queryStatement != null;
    }

    protected List getColumnNameList() {
        return results.getColumns().getColumnNames();
    }

    protected List getColumnTypeList() {
        return results.getColumns().getColumnTypes();
    }

    protected final NodeSequence sequence() {
        return sequence;
    }

    @Override
    public String[] getColumnNames() /*throws RepositoryException*/{
        List names = getColumnNameList();
        return names.toArray(new String[names.size()]); // make a defensive copy ...
    }

    @Override
    public String[] getColumnTypes() {
        List types = getColumnTypeList();
        return types.toArray(new String[types.size()]); // make a defensive copy ...
    }

    @Override
    public String[] getSelectorNames() {
        List selectorNames = results.getColumns().getSelectorNames();
        return selectorNames.toArray(new String[selectorNames.size()]); // make a defensive copy ...
    }

    @Override
    public NodeIterator getNodes() throws RepositoryException {
        if (accessed) {
            if (!restartable) {
                throw new RepositoryException(JcrI18n.multipleCallsToGetRowsOrNodesIsNotAllowed.text(queryStatement));
            }
            ((Restartable)sequence).restart();
        }
        if (getSelectorNames().length > 1) {
            throw new RepositoryException(JcrI18n.multipleSelectorsAppearInQueryUnableToCallMethod.text(queryStatement));
        }
        // Find all of the nodes in the results...
        accessed = true;
        int defaultSelectorIndex = computeDefaultSelectorIndex();
        return new QueryResultNodeIterator(context, sequence, defaultSelectorIndex);
    }

    protected int computeDefaultSelectorIndex() {
        Columns columns = results.getColumns();
        List selectorNames = columns.getSelectorNames();
        if (selectorNames.size() == 1) {
            return columns.getSelectorIndex(selectorNames.get(0));
        }
        return 0;
    }

    @Override
    public RowIterator getRows() throws RepositoryException {
        if (accessed) {
            if (!restartable) {
                throw new RepositoryException(JcrI18n.multipleCallsToGetRowsOrNodesIsNotAllowed.text(queryStatement));
            }
            ((Restartable)sequence).restart();
        }
        accessed = true;
        final Columns columns = results.getColumns();
        if (columns.getSelectorNames().size() == 1) {
            return new SingleSelectorQueryResultRowIterator(context, queryStatement, sequence, results.getColumns());
        }
        return new QueryResultRowIterator(context, queryStatement, sequence, results.getColumns());
    }

    @Override
    public String getPlan() {
        return results.getPlan();
    }

    @Override
    public Collection getWarnings() {
        if (warnings == null) {
            // Obtain the warnings ...
            if (!results.hasWarnings()) {
                warnings = java.util.Collections.emptyList();
            } else {
                List messages = new LinkedList();
                for (Problem problem : results.getProblems()) {
                    if (problem.getStatus() == Status.WARNING) {
                        String msg = problem.getMessageString();
                        if (!messages.contains(msg)) messages.add(msg);
                    }
                }
                warnings = java.util.Collections.unmodifiableList(messages);
            }
        }
        return warnings;
    }

    @Override
    public boolean isEmpty() {
        return false;
    }

    @Override
    public String toString() {
        return results.toString();
    }

    @Override
    public void close() {
        sequence.close();
    }

    /**
     * The {@link NodeIterator} implementation returned by the {@link JcrQueryResult}.
     * 
     * @see JcrQueryResult#getNodes()
     */
    @NotThreadSafe
    protected static abstract class QueryResultIterator implements RangeIterator {
        protected final JcrQueryContext context;
        private NodeSequence sequence;
        private long position = 0L;
        private Batch currentBatch;

        protected QueryResultIterator( JcrQueryContext context,
                                       NodeSequence sequence ) {
            this.context = context;
            this.sequence = sequence;
        }

        @Override
        public boolean hasNext() {
            while (findNextBatch() != null) {
                if (currentBatch.hasNext()) return true;
                currentBatch = null;
            }
            return false;
        }

        protected final Batch moveToNextRow() {
            if (findNextBatch() == null || !currentBatch.hasNext()) throw new NoSuchElementException();
            currentBatch.nextRow();
            ++position;
            return currentBatch;
        }

        protected Batch findNextBatch() {
            if (currentBatch == null) {
                currentBatch = sequence.nextBatch();
            }
            return currentBatch;
        }

        @Override
        public long getPosition() {
            return position;
        }

        @Override
        public void skip( long skipNum ) {
            for (long i = 0L; i != skipNum; ++i)
                moveToNextRow();
        }

        @Override
        public final long getSize() {
            return sequence.getRowCount();
        }
    }

    /**
     * The {@link NodeIterator} implementation returned by the {@link JcrQueryResult}.
     * 
     * @see JcrQueryResult#getNodes()
     */
    @NotThreadSafe
    protected static class QueryResultNodeIterator extends QueryResultIterator implements NodeIterator {

        private final int defaultSelectorIndex;

        protected QueryResultNodeIterator( JcrQueryContext context,
                                           NodeSequence sequence,
                                           int defaultSelectorIndex ) {
            super(context, sequence);
            this.defaultSelectorIndex = defaultSelectorIndex;
        }

        @Override
        public Node nextNode() {
            CachedNode cachedNode = moveToNextRow().getNode(defaultSelectorIndex);
            return context.getNode(cachedNode);
        }

        @Override
        public Object next() {
            return nextNode();
        }

        @Override
        public void remove() {
            throw new UnsupportedOperationException();
        }
    }

    /**
     * The {@link RowIterator} implementation returned by the {@link JcrQueryResult}.
     * 
     * @see JcrQueryResult#getRows()
     */
    @NotThreadSafe
    protected static class QueryResultRowIterator extends QueryResultIterator implements RowIterator {
        protected final List columnNames;
        private final Set selectorNames;
        protected final Columns columns;
        protected final String query;

        protected QueryResultRowIterator( JcrQueryContext context,
                                          String query,
                                          NodeSequence sequence,
                                          Columns columns ) {
            super(context, sequence);
            this.query = query;
            this.columns = columns;
            this.columnNames = this.columns.getColumnNames();
            this.selectorNames = new HashSet(columns.getSelectorNames());
        }

        public boolean hasSelector( String selectorName ) {
            return this.selectorNames.contains(selectorName);
        }

        @Override
        public Row nextRow() {
            return createRow(moveToNextRow());
        }

        @Override
        public Object next() {
            return nextRow();
        }

        protected Row createRow( Batch batch ) {
            return new MultiSelectorQueryResultRow(this, batch);
        }

        protected String getPropertyNameForColumnName( String columnName ) {
            return columns.getPropertyNameForColumnName(columnName);
        }

        protected int nodeIndexForSelector( String selector ) {
            return columns.getSelectorIndex(selector);
        }

        @Override
        public void remove() {
            throw new UnsupportedOperationException();
        }

        protected Value jcrPath( CachedNode node ) {
            assert node != null;
            // Every node has a path ...
            return context.createValue(PropertyType.PATH, context.getPath(node));
        }

        protected Value jcrUuid( CachedNode node ) {
            assert node != null;
            // only referenceable nodes have UUIDs ...
            String uuid = context.getUuid(node);
            return uuid == null ? null : context.createValue(PropertyType.STRING, uuid);
        }

        protected Value jcrName( CachedNode node ) {
            assert node != null;
            // Every node has a name; even the root has a zero-length name ...
            return context.createValue(PropertyType.NAME, context.getName(node));
        }

        protected Value jcrLocalName( CachedNode node ) {
            assert node != null;
            // Every node has a name; even the root has a zero-length name ...
            return context.createValue(PropertyType.NAME, context.getName(node).getLocalName());
        }

        protected Value jcrDepth( CachedNode node ) {
            assert node != null;
            // Every node has a depth ...
            return context.createValue(PropertyType.LONG, context.getDepth(node));
        }

        protected Value jcrId( CachedNode node ) {
            assert node != null;
            // Every node has an identifier, but we need to figure out the correct version that's exposed
            return context.createValue(PropertyType.STRING, context.getIdentifier(node));
        }

        protected Value jcrPath( String path ) {
            return context.createValue(PropertyType.PATH, path);
        }

        protected Value jcrName( String name ) {
            return context.createValue(PropertyType.NAME, name);
        }

        protected Value jcrName() {
            return context.createValue(PropertyType.NAME, "");
        }

        protected Value jcrString( String name ) {
            return context.createValue(PropertyType.STRING, name);
        }

        protected Value jcrLong( Long value ) {
            return context.createValue(PropertyType.LONG, value);
        }

        protected Value jcrDouble( Double score ) {
            return context.createValue(PropertyType.DOUBLE, score);
        }

        protected Value jcrDouble( Float score ) {
            return context.createValue(PropertyType.DOUBLE, score);
        }
    }

    /**
     * The {@link RowIterator} implementation returned by the {@link JcrQueryResult}.
     * 
     * @see JcrQueryResult#getRows()
     */
    @NotThreadSafe
    protected static class SingleSelectorQueryResultRowIterator extends QueryResultRowIterator {

        private final int selectorIndex;

        protected SingleSelectorQueryResultRowIterator( JcrQueryContext context,
                                                        String query,
                                                        NodeSequence sequence,
                                                        Columns columns ) {
            super(context, query, sequence, columns);
            this.selectorIndex = columns.getSelectorIndex(columns.getSelectorNames().get(0));
        }

        @Override
        protected Row createRow( Batch batch ) {
            return new SingleSelectorQueryResultRow(this, batch, selectorIndex);
        }
    }

    protected static abstract class AbstractRow implements javax.jcr.query.Row {
        protected final QueryResultRowIterator iterator;
        protected final Batch batchAtRow;
        private Value[] values = null;

        protected AbstractRow( QueryResultRowIterator iterator,
                               Batch batchAtRow ) {
            this.iterator = iterator;
            this.batchAtRow = batchAtRow;
            assert this.iterator != null;
            assert this.batchAtRow != null;
        }

        @Override
        public Node getNode( String selectorName ) throws RepositoryException {
            int nodeIndex = iterator.nodeIndexForSelector(selectorName);
            if (nodeIndex < 0) {
                throw new RepositoryException(JcrI18n.selectorNotUsedInQuery.text(selectorName, iterator.query));
            }
            CachedNode cachedNode = batchAtRow.getNode(nodeIndex);
            return cachedNode == null ? null : iterator.context.getNode(cachedNode);
        }

        protected Value getValue( String columnName,
                                  CachedNode cachedNode,
                                  int nodeIndex ) throws ItemNotFoundException, RepositoryException {
            if (cachedNode == null) return null;
            // Get the property name for the column. Note that if the column is aliased, the property name will be different;
            // otherwise, the property name will be the same as the column name ...
            String propertyName = iterator.getPropertyNameForColumnName(columnName);
            if (propertyName == null) return null;

            if (PSEUDO_COLUMNS.contains(propertyName)) {
                if (JCR_PATH_COLUMN_NAME.equals(propertyName)) {
                    return iterator.jcrPath(cachedNode);
                }
                if (JCR_NAME_COLUMN_NAME.equals(propertyName)) {
                    return iterator.jcrName(cachedNode);
                }
                if (MODE_LOCALNAME_COLUMN_NAME.equals(propertyName)) {
                    return iterator.jcrLocalName(cachedNode);
                }
                if (MODE_DEPTH_COLUMN_NAME.equals(propertyName)) {
                    return iterator.jcrDepth(cachedNode);
                }
                if (MODE_ID_COLUMN_NAME.equals(propertyName)) {
                    return iterator.jcrId(cachedNode);
                }
                if (JCR_SCORE_COLUMN_NAME.equals(propertyName)) {
                    float score = batchAtRow.getScore(nodeIndex);
                    return iterator.jcrDouble(score);
                }
                if (JCR_UUID_COLUMN_NAME.equals(propertyName)) {
                    return iterator.jcrUuid(cachedNode);
                }
            }
            // Get the property's value ...
            Node node = iterator.context.getNode(cachedNode);
            if (node == null || !node.hasProperty(propertyName)) return null;
            Property property = node.getProperty(propertyName);
            Value value = null;
            if (property.isMultiple()) {
                Value[] values = property.getValues();
                // The array of values might be empty ...
                if (values.length > 0) {
                    // Use only the first value of a multi-valued property ...
                    value = property.getValues()[0];
                }
                // Otherwise the value will be null
            } else {
                value = property.getValue();
            }
            return value;
        }

        @Override
        public Value[] getValues() throws RepositoryException {
            if (values == null) {
                int i = 0;
                values = new Value[iterator.columnNames.size()];
                for (String columnName : iterator.columnNames) {
                    values[i++] = getValue(columnName);
                }
            }
            return values;
        }

        @Override
        public Node getNode() throws RepositoryException {
            throw new RepositoryException(
                                          JcrI18n.multipleSelectorsAppearInQueryRequireSpecifyingSelectorName.text(iterator.query));
        }

        @Override
        public String getPath() throws RepositoryException {
            throw new RepositoryException(
                                          JcrI18n.multipleSelectorsAppearInQueryRequireSpecifyingSelectorName.text(iterator.query));
        }

        @Override
        public double getScore() throws RepositoryException {
            throw new RepositoryException(
                                          JcrI18n.multipleSelectorsAppearInQueryRequireSpecifyingSelectorName.text(iterator.query));
        }

        @Override
        public String getPath( String selectorName ) throws RepositoryException {
            Node node = getNode(selectorName);
            return node == null ? null : node.getPath();
        }

        @Override
        public double getScore( String selectorName ) throws RepositoryException {
            if (!iterator.hasSelector(selectorName)) {
                throw new RepositoryException(JcrI18n.selectorNotUsedInQuery.text(selectorName, iterator.query));
            }
            int nodeIndex = iterator.columns.getSelectorNames().indexOf(selectorName);
            return batchAtRow.getScore(nodeIndex);
        }

    }

    protected static class SingleSelectorQueryResultRow extends AbstractRow {
        protected final CachedNode cachedNode;
        protected final Node node;
        protected final int selectorIndex;

        protected SingleSelectorQueryResultRow( QueryResultRowIterator iterator,
                                                Batch batchAtRow,
                                                int selectorIndex ) {
            super(iterator, batchAtRow);
            this.selectorIndex = selectorIndex;
            this.cachedNode = batchAtRow.getNode(selectorIndex);
            this.node = iterator.context.getNode(cachedNode);
        }

        @Override
        public Node getNode( String selectorName ) throws RepositoryException {
            if (!iterator.hasSelector(selectorName)) {
                throw new RepositoryException(JcrI18n.selectorNotUsedInQuery.text(selectorName, iterator.query));
            }
            return node;
        }

        @Override
        public Value getValue( String columnName ) throws ItemNotFoundException, RepositoryException {
            return getValue(columnName, cachedNode, selectorIndex);
        }

        @Override
        public Node getNode() {
            return node;
        }

        @Override
        public String getPath() throws RepositoryException {
            return node.getPath();
        }

        @Override
        public String getPath( String selectorName ) throws RepositoryException {
            if (!iterator.hasSelector(selectorName)) {
                throw new RepositoryException(JcrI18n.selectorNotUsedInQuery.text(selectorName, iterator.query));
            }
            return node.getPath();
        }

        @Override
        public double getScore() {
            return batchAtRow.getScore(selectorIndex);
        }

        @Override
        public double getScore( String selectorName ) throws RepositoryException {
            if (!iterator.hasSelector(selectorName)) {
                throw new RepositoryException(JcrI18n.selectorNotUsedInQuery.text(selectorName, iterator.query));
            }
            return getScore();
        }
    }

    protected static class MultiSelectorQueryResultRow extends AbstractRow {

        protected MultiSelectorQueryResultRow( QueryResultRowIterator iterator,
                                               Batch batchAtRow ) {
            super(iterator, batchAtRow);
        }

        @Override
        public Value getValue( String columnName ) throws ItemNotFoundException, RepositoryException {
            String selectorName = iterator.columns.getSelectorNameForColumnName(columnName);
            int nodeIndex = iterator.columns.getSelectorIndex(selectorName);
            if (nodeIndex == -1) {
                throw new RepositoryException(JcrI18n.queryResultsDoNotIncludeColumn.text(columnName, iterator.query));
            }
            CachedNode cachedNode = batchAtRow.getNode(nodeIndex);
            return getValue(columnName, cachedNode, nodeIndex);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy