org.neo4j.driver.async.ResultCursor Maven / Gradle / Ivy
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [http://neo4j.com]
*
* This file is part of Neo4j.
*
* 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.neo4j.driver.async;
import java.util.List;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.Executor;
import java.util.function.Consumer;
import java.util.function.Function;
import org.neo4j.driver.Record;
import org.neo4j.driver.Records;
import org.neo4j.driver.Result;
import org.neo4j.driver.exceptions.NoSuchRecordException;
import org.neo4j.driver.summary.ResultSummary;
/**
* The result of asynchronous execution of a Cypher query, conceptually an asynchronous stream of
* {@link Record records}.
*
* Result can be eagerly fetched in a list using {@link #listAsync()} or navigated lazily using
* {@link #forEachAsync(Consumer)} or {@link #nextAsync()}.
*
* Results are valid until the next query is run or until the end of the current transaction,
* whichever comes first. To keep a result around while further queries are run, or to use a result outside the scope
* of the current transaction, see {@link #listAsync()}.
*
Important note on semantics
*
* In order to handle very large results, and to minimize memory overhead and maximize
* performance, results are retrieved lazily. Please see {@link AsyncQueryRunner} for
* important details on the effects of this.
*
* The short version is that, if you want a hard guarantee that the underlying query
* has completed, you need to either call {@link AsyncTransaction#commitAsync()} on the {@link AsyncTransaction transaction}
* or {@link AsyncSession#closeAsync()} on the {@link AsyncSession session} that created this result, or you need to use
* the result.
*
* Note: Every returned {@link CompletionStage} can be completed by an IO thread which should never block.
* Otherwise IO operations on this and potentially other network connections might deadlock. Please do not chain
* blocking operations like {@link CompletableFuture#get()} on the returned stages. Consider using asynchronous calls
* throughout the chain or offloading blocking operation to a different {@link Executor}. This can be done using
* methods with "Async" suffix like {@link CompletionStage#thenApplyAsync(java.util.function.Function)} or
* {@link CompletionStage#thenApplyAsync(java.util.function.Function, Executor)}.
*
* @since 1.5
*/
public interface ResultCursor {
/**
* Retrieve the keys of the records this result cursor contains.
*
* @return list of all keys.
*/
List keys();
/**
* Asynchronously retrieve the result summary.
*
* If the records in the result is not fully consumed, then calling this method will exhausts the result.
*
* If you want to access unconsumed records after summary, you shall use {@link Result#list()} to buffer all records into memory before summary.
*
* @return a {@link CompletionStage} completed with a summary for the whole query result. Stage can also be
* completed exceptionally if query execution fails.
*/
CompletionStage consumeAsync();
/**
* Asynchronously navigate to and retrieve the next {@link Record} in this result. Returned stage can contain
* {@code null} if end of records stream has been reached.
*
* @return a {@link CompletionStage} completed with a record or {@code null}. Stage can also be
* completed exceptionally if query execution fails.
*/
CompletionStage nextAsync();
/**
* Asynchronously investigate the next upcoming {@link Record} without moving forward in the result. Returned
* stage can contain {@code null} if end of records stream has been reached.
*
* @return a {@link CompletionStage} completed with a record or {@code null}. Stage can also be
* completed exceptionally if query execution fails.
*/
CompletionStage peekAsync();
/**
* Asynchronously return the first record in the result, failing if there is not exactly
* one record left in the stream.
*
* @return a {@link CompletionStage} completed with the first and only record in the stream. Stage will be
* completed exceptionally with {@link NoSuchRecordException} if there is not exactly one record left in the
* stream. It can also be completed exceptionally if query execution fails.
*/
CompletionStage singleAsync();
/**
* Asynchronously apply the given {@link Consumer action} to every record in the result, yielding a summary of it.
*
* @param action the function to be applied to every record in the result. Provided function should not block.
* @return a {@link CompletionStage} completed with a summary for the whole query result. Stage can also be
* completed exceptionally if query execution or provided function fails.
*/
CompletionStage forEachAsync(Consumer action);
/**
* Asynchronously retrieve and store the entire result stream.
* This can be used if you want to iterate over the stream multiple times or to store the
* whole result for later use.
*
* Note that this method can only be used if you know that the query that
* yielded this result returns a finite stream. Some queries can yield
* infinite results, in which case calling this method will lead to running
* out of memory.
*
* Calling this method exhausts the result.
*
* @return a {@link CompletionStage} completed with a list of all remaining immutable records. Stage can also be
* completed exceptionally if query execution fails.
*/
CompletionStage> listAsync();
/**
* Asynchronously retrieve and store a projection of the entire result.
* This can be used if you want to iterate over the stream multiple times or to store the
* whole result for later use.
*
* Note that this method can only be used if you know that the query that
* yielded this result returns a finite stream. Some queries can yield
* infinite results, in which case calling this method will lead to running
* out of memory.
*
* Calling this method exhausts the result.
*
* @param mapFunction a function to map from Record to T. See {@link Records} for some predefined functions.
* @param the type of result list elements
* @return a {@link CompletionStage} completed with a list of all remaining immutable records. Stage can also be
* completed exceptionally if query execution or provided function fails.
*/
CompletionStage> listAsync(Function mapFunction);
}