| /**************************************************************************** |
| * 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); |
| } |
| |
| } |