/*
 * Copyright 2002-2017 the original author or authors.
 *
 * 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.springframework.util.concurrent;

import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

import org.springframework.lang.Nullable;
import org.springframework.util.Assert;

Abstract class that adapts a Future parameterized over S into a Future parameterized over T. All methods are delegated to the adaptee, where get() and get(long, TimeUnit) call adapt(Object) on the adaptee's result.
Author:Arjen Poutsma
Type parameters:
  • <T> – the type of this Future
  • <S> – the type of the adaptee's Future
Since:4.0
/** * Abstract class that adapts a {@link Future} parameterized over S into a {@code Future} * parameterized over T. All methods are delegated to the adaptee, where {@link #get()} * and {@link #get(long, TimeUnit)} call {@link #adapt(Object)} on the adaptee's result. * * @author Arjen Poutsma * @since 4.0 * @param <T> the type of this {@code Future} * @param <S> the type of the adaptee's {@code Future} */
public abstract class FutureAdapter<T, S> implements Future<T> { private final Future<S> adaptee; @Nullable private Object result; private State state = State.NEW; private final Object mutex = new Object();
Constructs a new FutureAdapter with the given adaptee.
Params:
  • adaptee – the future to delegate to
/** * Constructs a new {@code FutureAdapter} with the given adaptee. * @param adaptee the future to delegate to */
protected FutureAdapter(Future<S> adaptee) { Assert.notNull(adaptee, "Delegate must not be null"); this.adaptee = adaptee; }
Returns the adaptee.
/** * Returns the adaptee. */
protected Future<S> getAdaptee() { return this.adaptee; } @Override public boolean cancel(boolean mayInterruptIfRunning) { return this.adaptee.cancel(mayInterruptIfRunning); } @Override public boolean isCancelled() { return this.adaptee.isCancelled(); } @Override public boolean isDone() { return this.adaptee.isDone(); } @Override @Nullable public T get() throws InterruptedException, ExecutionException { return adaptInternal(this.adaptee.get()); } @Override @Nullable public T get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { return adaptInternal(this.adaptee.get(timeout, unit)); } @SuppressWarnings("unchecked") @Nullable final T adaptInternal(S adapteeResult) throws ExecutionException { synchronized (this.mutex) { switch (this.state) { case SUCCESS: return (T) this.result; case FAILURE: Assert.state(this.result instanceof ExecutionException, "Failure without exception"); throw (ExecutionException) this.result; case NEW: try { T adapted = adapt(adapteeResult); this.result = adapted; this.state = State.SUCCESS; return adapted; } catch (ExecutionException ex) { this.result = ex; this.state = State.FAILURE; throw ex; } catch (Throwable ex) { ExecutionException execEx = new ExecutionException(ex); this.result = execEx; this.state = State.FAILURE; throw execEx; } default: throw new IllegalStateException(); } } }
Adapts the given adaptee's result into T.
Returns:the adapted result
/** * Adapts the given adaptee's result into T. * @return the adapted result */
@Nullable protected abstract T adapt(S adapteeResult) throws ExecutionException; private enum State {NEW, SUCCESS, FAILURE} }