/*
* Copyright (c) OSGi Alliance 2015. All Rights Reserved.
*
* 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.osgi.util.promise;
import java.lang.reflect.InvocationTargetException;
import org.osgi.util.function.Callback;
import org.osgi.util.function.Function;
import org.osgi.util.function.Predicate;
/**
* A Promise of a value.
* <p/>
* A Promise represents a future value. It handles the interactions to for asynchronous processing. A Deferred object
* can be used to create a Promise and later resolve the Promise. A Promise is used by the caller of an asynchronous
* function to get the result or handle the error. The caller can either get a callback when the Promise is resolved
* with a value or an error, or the Promise can be used in chaining. In chaining, callbacks are provided that receive
* the resolved Promise, and a new Promise is generated that resolves based upon the result of a callback.
* <p/>
* Both callbacks and chaining can be repeated any number of times, even after the Promise has been resolved.
* <p/>
* Example callback usage:
* <pre>
* final Promise<String> foo= foo();
* foo.onResolve(new Runnable() {
* public void run() {
* System.out.println(foo.getValue());
* }
* });
* </pre>
* <p/>
* Example chaining usage;
* <pre>
* Success<String,String> doubler = new Success<String,String>() {
* public Promise<String> call(Promise<String> p) throws Exception {
* return Promises.resolved(p.getValue()+p.getValue());
* }
* };
* final Promise<String> foo = foo().then(doubler).then(doubler);
* foo.onResolve(new Runnable() {
* public void run() {
* System.out.println(foo.getValue());
* }
* });
* </pre>
*
* @param <T> The value type associated with this Promise.
*/
@org.osgi.annotation.versioning.ProviderType
public interface Promise<T> {
/**
* Returns whether this Promise has been resolved.
* <p/>
* This Promise may be successfully resolved or resolved with a failure.
*
* @return true if this Promise was resolved either successfully or with a failure; false if this Promise is
* unresolved.
*/
boolean isDone();
/**
* Returns the value of this Promise.
* <p/>
* If this Promise is not resolved, this method must block and wait for this Promise to be resolved before
* completing.
* <p/>
* If this Promise was successfully resolved, this method returns with the value of this Promise. If this Promise
* was resolved with a failure, this method must throw an InvocationTargetException with the failure exception as
* the cause.
*
* @return The value of this resolved Promise.
* @throws InvocationTargetException If this Promise was resolved with a failure. The cause of the
* InvocationTargetException is the failure exception.
* @throws InterruptedException If the current thread was interrupted while waiting.
*/
T getValue() throws InvocationTargetException, InterruptedException;
/**
* Returns the failure of this Promise.
* <p/>
* If this Promise is not resolved, this method must block and wait for this Promise to be resolved before
* completing.
* <p/>
* If this Promise was resolved with a failure, this method returns with the failure of this Promise. If this
* Promise was successfully resolved, this method must return null.
*
* @return The failure of this resolved Promise or null if this Promise was successfully resolved.
* @throws InterruptedException If the current thread was interrupted while waiting.
*/
Throwable getFailure() throws InterruptedException;
/**
* Register a callback to be called when this Promise is resolved.
* <p/>
* The specified callback is called when this Promise is resolved either successfully or with a failure.
* <p/>
* This method may be called at any time including before and after this Promise has been resolved.
* <p/>
* Resolving this Promise happens-before any registered callback is called. That is, in a registered callback,
* isDone() must return true and getValue() and getFailure() must not block.
* <p/>
* A callback may be called on a different thread than the thread which registered the callback. So the callback
* must be thread safe but can rely upon that the registration of the callback happens-before the registered
* callback is called.
*
* @param callback A callback to be called when this Promise is resolved. Must not be null.
* @return This Promise.
*/
Promise<T> onResolve(Runnable callback);
/**
* Chain a new Promise to this Promise with Success and Failure callbacks.
* <p/>
* The specified Success callback is called when this Promise is successfully resolved and the specified Failure
* callback is called when this Promise is resolved with a failure.
* <p/>
* This method returns a new Promise which is chained to this Promise. The returned Promise must be resolved when
* this Promise is resolved after the specified Success or Failure callback is executed. The result of the executed
* callback must be used to resolve the returned Promise. Multiple calls to this method can be used to create a
* chain of promises which are resolved in sequence.
* <p/>
* If this Promise is successfully resolved, the Success callback is executed and the result Promise, if any, or
* thrown exception is used to resolve the returned Promise from this method. If this Promise is resolved with a
* failure, the Failure callback is executed and the returned Promise from this method is failed.
* <p/>
* This method may be called at any time including before and after this Promise has been resolved.
* <p/>
* Resolving this Promise happens-before any registered callback is called. That is, in a registered callback,
* isDone() must return true and getValue() and getFailure() must not block.
* <p/>
* A callback may be called on a different thread than the thread which registered the callback. So the callback
* must be thread safe but can rely upon that the registration of the callback happens-before the registered
* callback is called.
*
* @param <R> The value type associated with the returned Promise.
* @param success A Success callback to be called when this Promise is successfully resolved. May be null if no
* Success callback is required. In thi
* @param failure A Failure callback to be called when this Promise is resolved with a failure. May be null if no
* Failure callback is required.
* @return A new Promise which is chained to this Promise. The returned Promise must be resolved when this Promise
* is resolved after the specified Success or Failure callback, if any, is executed
*/
<R> Promise<R> then(Success<? super T, ? extends R> success, Failure failure);
/**
* Chain a new Promise to this Promise with a Success callback.
* <p/>
* This method performs the same function as calling then(Success, Failure) with the specified Success callback and
* null for the Failure callback
*
* @param success A Success callback to be called when this Promise is successfully resolved. May be null if no
* Success callback is required. In this case, the returned Promise must be resolved with the value
* null when this Promise is successfully resolved.
* @param <R> The value type associated with the returned Promise.
* @return A new Promise which is chained to this Promise. The returned Promise must be resolved when this Promise
* is resolved after the specified Success, if any, is executed.
* @see #then(Success, Failure)
*/
<R> Promise<R> then(Success<? super T, ? extends R> success);
/**
* Chain a new Promise to this Promise with a callback.
* <p>
* The specified {@link Callback} is called when this Promise is resolved
* either successfully or with a failure.
* <p>
* This method returns a new Promise which is chained to this Promise. The
* returned Promise must be resolved when this Promise is resolved after the
* specified callback is executed. If the callback throws an exception, the
* returned Promise is failed with that exception. Otherwise the returned
* Promise is resolved with this Promise.
* <p>
* This method may be called at any time including before and after this
* Promise has been resolved.
* <p>
* Resolving this Promise <i>happens-before</i> any registered callback is
* called. That is, in a registered callback, {@link #isDone()} must return
* {@code true} and {@link #getValue()} and {@link #getFailure()} must not
* block.
* <p>
* A callback may be called on a different thread than the thread which
* registered the callback. So the callback must be thread safe but can rely
* upon that the registration of the callback <i>happens-before</i> the
* registered callback is called.
*
* @param callback A callback to be called when this Promise is resolved.
* Must not be {@code null}.
* @return A new Promise which is chained to this Promise. The returned
* Promise must be resolved when this Promise is resolved after the
* specified callback is executed.
* @since 1.1
*/
Promise<T> then(Callback callback);
/**
* Filter the value of this Promise.
* <p/>
* If this Promise is successfully resolved, the returned Promise will either be resolved with the value of this
* Promise if the specified Predicate accepts that value or failed with a NoSuchElementException if the specified
* Predicate does not accept that value. If the specified Predicate throws an exception, the returned Promise will
* be failed with the exception.
* <p/>
* If this Promise is resolved with a failure, the returned Promise will be failed with that failure.
* <p/>
* This method may be called at any time including before and after this Promise has been resolved.
*
* @param predicate The Predicate to evaluate the value of this Promise. Must not be null.
* @return A Promise that filters the value of this Promise.
*/
Promise<T> filter(Predicate<? super T> predicate);
/**
* Map the value of this Promise.
* <p/>
* If this Promise is successfully resolved, the returned Promise will be resolved with the value of specified
* Function as applied to the value of this Promise. If the specified Function throws an exception, the returned
* Promise will be failed with the exception.
* <p/>
* If this Promise is resolved with a failure, the returned Promise will be failed with that failure.
* <p/>
* This method may be called at any time including before and after this Promise has been resolved.
*
* @param mapper The Function that will map the value of this Promise to the value that will be used to resolve the
* returned Promise. Must not be null.
* @param <R> The value type associated with the returned Promise.
* @return A Promise that returns the value of this Promise as mapped by the specified Function.
*/
<R> Promise<R> map(Function<? super T, ? extends R> mapper);
/**
* FlatMap the value of this Promise.
* <p/>
* If this Promise is successfully resolved, the returned Promise will be resolved with the Promise from the
* specified Function as applied to the value of this Promise. If the specified Function throws an exception, the
* returned Promise will be failed with the exception.
* <p/>
* If this Promise is resolved with a failure, the returned Promise will be failed with that failure.
* <p/>
* This method may be called at any time including before and after this Promise has been resolved.
*
* @param mapper The Function that will flatMap the value of this Promise to a Promise that will be used to resolve
* the returned Promise. Must not be null.
* @param <R> The value type associated with the returned Promise.
* @return A Promise that returns the value of this Promise as mapped by the specified Function.
*/
<R> Promise<R> flatMap(Function<? super T, Promise<? extends R>> mapper);
/**
* Recover from a failure of this Promise with a recovery value.
* <p/>
* If this Promise is successfully resolved, the returned Promise will be resolved with the value of this Promise.
* <p/>
* If this Promise is resolved with a failure, the specified Function is applied to this Promise to produce a
* recovery value.
* <p/>
* If the recovery value is not null, the returned Promise will be resolved with the recovery value.
* <p/>
* If the recovery value is null, the returned Promise will be failed with the failure of this Promise.
* <p/>
* If the specified Function throws an exception, the returned Promise will be failed with that exception.
* <p/>
* To recover from a failure of this Promise with a recovery value of null, the recoverWith(Function) method must be
* used. The specified Function for recoverWith(Function) can return Promises.resolved(null) to supply the desired
* null value.
* <p/>
* This method may be called at any time including before and after this Promise has been resolved.
*
* @param recovery If this Promise resolves with a failure, the specified Function is called to produce a recovery
* value to be used to resolve the returned Promise. Must not be null.
* @return A Promise that resolves with the value of this Promise or recovers from the failure of this Promise.
*/
Promise<T> recover(Function<Promise<?>, ? extends T> recovery);
/**
* Recover from a failure of this Promise with a recovery Promise.
* <p/>
* If this Promise is successfully resolved, the returned Promise will be resolved with the value of this Promise.
* <p/>
* If this Promise is resolved with a failure, the specified Function is applied to this Promise to produce a
* recovery Promise.
* <p/>
* If the recovery Promise is not null, the returned Promise will be resolved with the recovery Promise.
* <p/>
* If the recovery Promise is null, the returned Promise will be failed with the failure of this Promise.
* <p/>
* If the specified Function throws an exception, the returned Promise will be failed with that exception.
* <p/>
* This method may be called at any time including before and after this Promise has been resolved.
*
* @param recovery If this Promise resolves with a failure, the specified Function is called to produce a recovery
* Promise to be used to resolve the returned Promise. Must not be null.
* @return A Promise that resolves with the value of this Promise or recovers from the failure of this Promise.
*/
Promise<T> recoverWith(Function<Promise<?>, Promise<? extends T>> recovery);
/**
* Fall back to the value of the specified Promise if this Promise fails.
* <p/>
* If this Promise is successfully resolved, the returned Promise will be resolved with the value of this Promise.
* <p/>
* If this Promise is resolved with a failure, the successful result of the specified Promise is used to resolve the
* returned Promise. If the specified Promise is resolved with a failure, the returned Promise will be failed with
* the failure of this Promise rather than the failure of the specified Promise.
* <p/>
* This method may be called at any time including before and after this Promise has been resolved.
*
* @param fallback The Promise whose value will be used to resolve the returned Promise if this Promise resolves
* with a failure. Must not be null.
* @return A Promise that returns the value of this Promise or falls back to the value of the specified Promise.
*/
Promise<T> fallbackTo(Promise<? extends T> fallback);
/**
* Time out the resolution of this Promise.
* <p>
* If this Promise is successfully resolved before the timeout, the returned
* Promise is resolved with the value of this Promise. If this Promise is
* resolved with a failure before the timeout, the returned Promise is
* resolved with the failure of this Promise. If the timeout is reached
* before this Promise is resolved, the returned Promise is failed with a
* {@link TimeoutException}.
*
* @param milliseconds The time to wait in milliseconds. Zero and negative
* time is treated as an immediate timeout.
* @return A Promise that is resolved when either this Promise is resolved
* or the specified timeout is reached.
* @since 1.1
*/
Promise<T> timeout(long milliseconds);
/**
* Delay after the resolution of this Promise.
* <p>
* Once this Promise is resolved, resolve the returned Promise with this
* Promise after the specified delay.
*
* @param milliseconds The time to delay in milliseconds. Zero and negative
* time is treated as no delay.
* @return A Promise that is resolved with this Promise after this Promise
* is resolved and the specified delay has elapsed.
* @since 1.1
*/
Promise<T> delay(long milliseconds);
}