blob: 10a2d5e89330162deb54f6c7697e5d7a3d653a82 [file] [log] [blame]
/****************************************************************************
* Copyright (c) 2014 Composent, Inc. and others.
*
* This program and the accompanying materials are made
* available under the terms of the Eclipse Public License 2.0
* which is available at https://www.eclipse.org/legal/epl-2.0/
*
* Contributors:
* Composent, Inc. - initial API and implementation
*
* SPDX-License-Identifier: EPL-2.0
*****************************************************************************/
package org.eclipse.ecf.remoteservice.asyncproxy;
import java.lang.reflect.Method;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.Future;
import org.eclipse.equinox.concurrent.future.IFuture;
import org.osgi.util.promise.Deferred;
import org.osgi.util.promise.Promise;
public abstract class AbstractAsyncProxyRemoteService {
@SuppressWarnings("rawtypes")
protected abstract IFuture callAsync(AbstractAsyncProxyRemoteCall call);
@SuppressWarnings("rawtypes")
protected abstract Future callFutureAsync(AbstractAsyncProxyRemoteCall call);
protected abstract void callCompletableAsync(AbstractAsyncProxyRemoteCall call, IAsyncProxyCompletable completable);
protected boolean isReturnAsync(Object proxy, Method method, Object[] args) {
@SuppressWarnings("rawtypes")
Class returnType = method.getReturnType();
return (CompletableFuture.class.isAssignableFrom(returnType) || CompletionStage.class.isAssignableFrom(returnType) ||
Future.class.isAssignableFrom(returnType) || IFuture.class.isAssignableFrom(returnType) || Promise.class.isAssignableFrom(returnType));
}
@SuppressWarnings("unchecked")
protected Object callFuture(AbstractAsyncProxyRemoteCall call, @SuppressWarnings("rawtypes") Class returnType) {
if (Promise.class.isAssignableFrom(returnType)) {
@SuppressWarnings("rawtypes")
Deferred d = new Deferred();
callCompletableAsync(call, (r,hadException,exception) -> {
if (hadException) d.fail(exception);
else d.resolve(r);
});
return d.getPromise();
}
// If the result value is a CompletableFuture then
// we callCompletableAsync
if (CompletableFuture.class.isAssignableFrom(returnType) || CompletionStage.class.isAssignableFrom(returnType)) {
@SuppressWarnings("rawtypes")
CompletableFuture result = new CompletableFuture();
callCompletableAsync(call, (r,hadException,exception) -> {
if (hadException) result.completeExceptionally(exception);
else result.complete(r);
});
// And return the CompletableFuture
return result;
}
// Else if it's an IFuture then return
// IFuture result of callAsync
if (IFuture.class.isAssignableFrom(returnType))
return callAsync(call);
// Else it must be a Future return value
return callFutureAsync(call);
}
}