/* * Copyright 2016 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 net.jodah.failsafe; import java.util.concurrent.Callable; import java.util.concurrent.Semaphore; import net.jodah.failsafe.function.AsyncCallable; import net.jodah.failsafe.function.AsyncRunnable; import net.jodah.failsafe.function.CheckedBiConsumer; import net.jodah.failsafe.function.CheckedBiFunction; import net.jodah.failsafe.function.CheckedConsumer; import net.jodah.failsafe.function.CheckedFunction; import net.jodah.failsafe.function.CheckedRunnable; import net.jodah.failsafe.function.ContextualCallable; import net.jodah.failsafe.function.ContextualRunnable; import net.jodah.failsafe.internal.util.Assert; /** * Utilities and adapters for creating functions. * * @author Jonathan Halterman */ final class Functions { static abstract class AsyncCallableWrapper<T> implements Callable<T> { protected AsyncExecution execution; void inject(AsyncExecution execution) { this.execution = execution; } } static abstract class ContextualCallableWrapper<T> implements Callable<T> { protected ExecutionContext context; void inject(ExecutionContext context) { this.context = context; } } static <T> AsyncCallableWrapper<T> asyncOf(final AsyncCallable<T> callable) { Assert.notNull(callable, "callable"); return new AsyncCallableWrapper<T>() { @Override public synchronized T call() throws Exception { try { execution.before(); T result = callable.call(execution); return result; } catch (Throwable e) { execution.completeOrRetry(null, e); return null; } } }; } static <T> AsyncCallableWrapper<T> asyncOf(final AsyncRunnable runnable) { Assert.notNull(runnable, "runnable"); return new AsyncCallableWrapper<T>() { @Override public synchronized T call() throws Exception { try { execution.before(); runnable.run(execution); } catch (Throwable e) { execution.completeOrRetry(null, e); } return null; } }; } static <T> AsyncCallableWrapper<T> asyncOf(final Callable<T> callable) { Assert.notNull(callable, "callable"); return new AsyncCallableWrapper<T>() { @Override public T call() throws Exception { try { execution.before(); T result = callable.call(); execution.completeOrRetry(result, null); return result; } catch (Throwable e) { execution.completeOrRetry(null, e); return null; } } }; } static <T> AsyncCallableWrapper<T> asyncOf(final CheckedRunnable runnable) { Assert.notNull(runnable, "runnable"); return new AsyncCallableWrapper<T>() { @Override public T call() throws Exception { try { execution.before(); runnable.run(); execution.completeOrRetry(null, null); } catch (Throwable e) { execution.completeOrRetry(null, e); } return null; } }; } static <T> AsyncCallableWrapper<T> asyncOf(final ContextualCallable<T> callable) { Assert.notNull(callable, "callable"); return new AsyncCallableWrapper<T>() { @Override public T call() throws Exception { try { execution.before(); T result = callable.call(execution); execution.completeOrRetry(result, null); return result; } catch (Throwable e) { execution.completeOrRetry(null, e); return null; } } }; } static <T> AsyncCallableWrapper<T> asyncOf(final ContextualRunnable runnable) { Assert.notNull(runnable, "runnable"); return new AsyncCallableWrapper<T>() { @Override public T call() throws Exception { try { execution.before(); runnable.run(execution); execution.completeOrRetry(null, null); } catch (Throwable e) { execution.completeOrRetry(null, e); } return null; } }; } static <T> AsyncCallableWrapper<T> asyncOfFuture( final AsyncCallable<java.util.concurrent.CompletableFuture<T>> callable) { Assert.notNull(callable, "callable"); return new AsyncCallableWrapper<T>() { Semaphore asyncFutureLock = new Semaphore(1); @Override public T call() throws Exception { try { execution.before(); asyncFutureLock.acquire(); callable.call(execution).whenComplete(new java.util.function.BiConsumer<T, Throwable>() { @Override public void accept(T innerResult, Throwable failure) { try { if (failure != null) execution.completeOrRetry(innerResult, failure instanceof java.util.concurrent.CompletionException ? failure.getCause() : failure); } finally { asyncFutureLock.release(); } } }); } catch (Throwable e) { try { execution.completeOrRetry(null, e); } finally { asyncFutureLock.release(); } } return null; } }; } static <T> AsyncCallableWrapper<T> asyncOfFuture(final Callable<java.util.concurrent.CompletableFuture<T>> callable) { Assert.notNull(callable, "callable"); return new AsyncCallableWrapper<T>() { @Override public T call() throws Exception { try { execution.before(); callable.call().whenComplete(new java.util.function.BiConsumer<T, Throwable>() { @Override public void accept(T innerResult, Throwable failure) { // Unwrap CompletionException cause if (failure != null && failure instanceof java.util.concurrent.CompletionException) failure = failure.getCause(); execution.completeOrRetry(innerResult, failure); } }); } catch (Throwable e) { execution.completeOrRetry(null, e); } return null; } }; } static <T> AsyncCallableWrapper<T> asyncOfFuture( final ContextualCallable<java.util.concurrent.CompletableFuture<T>> callable) { Assert.notNull(callable, "callable"); return new AsyncCallableWrapper<T>() { @Override public T call() throws Exception { try { execution.before(); callable.call(execution).whenComplete(new java.util.function.BiConsumer<T, Throwable>() { @Override public void accept(T innerResult, Throwable failure) { // Unwrap CompletionException cause if (failure != null && failure instanceof java.util.concurrent.CompletionException) failure = failure.getCause(); execution.completeOrRetry(innerResult, failure); } }); } catch (Throwable e) { execution.completeOrRetry(null, e); } return null; } }; } static <T> Callable<T> callableOf(final CheckedRunnable runnable) { Assert.notNull(runnable, "runnable"); return new Callable<T>() { @Override public T call() throws Exception { runnable.run(); return null; } }; } static <T> Callable<T> callableOf(final ContextualCallable<T> callable) { Assert.notNull(callable, "callable"); return new ContextualCallableWrapper<T>() { @Override public T call() throws Exception { T result = callable.call(context); return result; } }; } static <T> Callable<T> callableOf(final ContextualRunnable runnable) { Assert.notNull(runnable, "runnable"); return new ContextualCallableWrapper<T>() { @Override public T call() throws Exception { runnable.run(context); return null; } }; } static <T, U, R> CheckedBiFunction<T, U, R> fnOf(final Callable<R> callable) { return new CheckedBiFunction<T, U, R>() { @Override public R apply(T t, U u) throws Exception { return callable.call(); } }; } static <T, U, R> CheckedBiFunction<T, U, R> fnOf(final CheckedBiConsumer<T, U> consumer) { return new CheckedBiFunction<T, U, R>() { @Override public R apply(T t, U u) throws Exception { consumer.accept(t, u); return null; } }; } static <T, U, R> CheckedBiFunction<T, U, R> fnOf(final CheckedConsumer<U> consumer) { return new CheckedBiFunction<T, U, R>() { @Override public R apply(T t, U u) throws Exception { consumer.accept(u); return null; } }; } static <T, U, R> CheckedBiFunction<T, U, R> fnOf(final CheckedFunction<U, R> function) { return new CheckedBiFunction<T, U, R>() { @Override public R apply(T t, U u) throws Exception { return function.apply(u); } }; } static <T, U, R> CheckedBiFunction<T, U, R> fnOf(final CheckedRunnable runnable) { return new CheckedBiFunction<T, U, R>() { @Override public R apply(T t, U u) throws Exception { runnable.run(); return null; } }; } static <T, U, R> CheckedBiFunction<T, U, R> fnOf(final R result) { return new CheckedBiFunction<T, U, R>() { @Override public R apply(T t, U u) throws Exception { return result; } }; } }