com.tencent.angel.ipc.CallFuture Maven / Gradle / Ivy
/*
* Tencent is pleased to support the open source community by making Angel available.
*
* Copyright (C) 2017-2018 THL A29 Limited, a Tencent company. 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
*
* https://opensource.org/licenses/Apache-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 com.tencent.angel.ipc;
import java.util.concurrent.*;
/**
* A Future implementation for RPCs.
*/
public class CallFuture implements Future, Callback {
private final CountDownLatch latch = new CountDownLatch(1);
private final Callback chainedCallback;
private T result = null;
private Throwable error = null;
/**
* Creates a CallFuture.
*/
public CallFuture() {
this(null);
}
/**
* Creates a CallFuture with a chained Callback which will be invoked when this CallFuture's
* Callback methods are invoked.
*
* @param chainedCallback the chained Callback to set.
*/
public CallFuture(Callback chainedCallback) {
this.chainedCallback = chainedCallback;
}
/**
* Sets the RPC response, and unblocks all threads waiting on {@link #get()} or
* {@link #get(long, java.util.concurrent.TimeUnit)}.
*
* @param result the RPC result to set.
*/
@Override public void handleResult(T result) {
this.result = result;
latch.countDown();
if (chainedCallback != null) {
chainedCallback.handleResult(result);
}
}
/**
* Sets an error thrown during RPC execution, and unblocks all threads waiting on {@link #get()}
* or {@link #get(long, java.util.concurrent.TimeUnit)}.
*
* @param error the RPC error to set.
*/
@Override public void handleError(Throwable error) {
this.error = error;
latch.countDown();
if (chainedCallback != null) {
chainedCallback.handleError(error);
}
}
/**
* Gets the value of the RPC result without blocking. Using {@link #get()} or
* {@link #get(long, java.util.concurrent.TimeUnit)} is usually preferred because these methods
* block until the result is available or an error occurs.
*
* @return the value of the response, or null if no result was returned or the RPC has not yet
* completed.
*/
public T getResult() {
return result;
}
/**
* Gets the error that was thrown during RPC execution. Does not block. Either {@link #get()} or
* {@link #get(long, java.util.concurrent.TimeUnit)} should be called first because these methods
* block until the RPC has completed.
*
* @return the RPC error that was thrown, or null if no error has occurred or if the RPC has not
* yet completed.
*/
public Throwable getError() {
return error;
}
@Override public boolean cancel(boolean mayInterruptIfRunning) {
return false;
}
@Override public boolean isCancelled() {
return false;
}
@Override public T get() throws InterruptedException, ExecutionException {
latch.await();
if (error != null) {
throw new ExecutionException(error);
}
return result;
}
@Override public T get(long timeout, TimeUnit unit)
throws InterruptedException, ExecutionException, TimeoutException {
if (latch.await(timeout, unit)) {
if (error != null) {
throw new ExecutionException(error);
}
return result;
} else {
throw new TimeoutException();
}
}
/**
* Waits for the CallFuture to complete without returning the result.
*
* @throws InterruptedException if interrupted.
*/
public void await() throws InterruptedException {
latch.await();
}
/**
* Waits for the CallFuture to complete without returning the result.
*
* @param timeout the maximum time to wait.
* @param unit the time unit of the timeout argument.
* @throws InterruptedException if interrupted.
* @throws java.util.concurrent.TimeoutException if the wait timed out.
*/
public void await(long timeout, TimeUnit unit) throws InterruptedException, TimeoutException {
if (!latch.await(timeout, unit)) {
throw new TimeoutException();
}
}
@Override public boolean isDone() {
return latch.getCount() <= 0;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy