All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ratpack.func.Function Maven / Gradle / Ivy

There is a newer version: 2.0.0-rc-1
Show newest version
/*
 * Copyright 2013 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 ratpack.func;

import com.google.common.collect.ImmutableList;
import ratpack.func.internal.ConditionalFunction;
import ratpack.util.Exceptions;

import java.util.Objects;

/**
 * A single argument function.
 * 

* This type serves the same purpose as the JDK's {@link java.util.function.Function}, but allows throwing checked exceptions. * It contains methods for bridging to and from the JDK type. * * @param the type of the input * @param the type of the output */ @FunctionalInterface public interface Function { /** * The function implementation. * * @param i the input to the function * @return the output of the function * @throws Exception any */ O apply(I i) throws Exception; /** * Joins {@code this} function with the given function. * *

{@code
   * import ratpack.func.Function;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String[] args) throws Exception {
   *     Function function = in -> in + "-bar";
   *     assertEquals("FOO-BAR", function.andThen(String::toUpperCase).apply("foo"));
   *   }
   * }
   * }
*

* Analogous to {@link java.util.function.Function#andThen(java.util.function.Function)}. * * @param after the function to apply to the result of {@code this} function * @param the type of the final output * @return the result of applying the given function to {@code this} function * @throws Exception any thrown by {@code this} or {@code after} */ default Function andThen(Function after) throws Exception { Objects.requireNonNull(after); return (I i) -> { O apply = apply(i); return after.apply(apply); }; } /** * Joins the given function with {@code this} function. * *

{@code
   * import ratpack.func.Function;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     Function function = String::toUpperCase;
   *     assertEquals("FOO-BAR", function.compose(in -> in + "-BAR").apply("foo"));
   *   }
   * }
   * }
*

* Analogous to {@link java.util.function.Function#compose(java.util.function.Function)}. * * @param before the function to apply {@code this} function to the result of * @param the type of the new input * @return the result of applying {@code this} function to the result of the given function * @throws Exception any thrown by {@code this} or {@code before} */ default Function compose(Function before) throws Exception { Objects.requireNonNull(before); return (T t) -> apply(before.apply(t)); } /** * Converts {@code this} function into the equivalent JDK type. *

* Any exceptions thrown by {@code this} function will be unchecked via {@link ratpack.util.Exceptions#uncheck(Throwable)} and rethrown. * * @return this function as a JDK style function. */ default java.util.function.Function toFunction() { return (t) -> { try { return apply(t); } catch (Exception e) { throw Exceptions.uncheck(e); } }; } /** * Converts {@code this} function into the equivalent Guava type. *

* Any exceptions thrown by {@code this} function will be unchecked via {@link ratpack.util.Exceptions#uncheck(Throwable)} and rethrown. * * @return this function as a Guava style function. */ default com.google.common.base.Function toGuavaFunction() { return (t) -> { try { return apply(t); } catch (Exception e) { throw Exceptions.uncheck(e); } }; } /** * Creates a function of this type from a JDK style function. * * @param function a JDK style function * @param the input type * @param the output type * @return a Ratpack style function wrapping the given JDK function */ static Function from(java.util.function.Function function) { Objects.requireNonNull(function); return function::apply; } /** * Creates a function of this type from a Guava style function. * * @param function a Guava style function * @param the input type * @param the output type * @return a Ratpack style function wrapping the given Guava function */ static Function fromGuava(com.google.common.base.Function function) { Objects.requireNonNull(function); return function::apply; } /** * Returns an identity function (return value always same as input). * * @param the type of the input and output objects to the function * @return a function that always returns its input argument */ static Function identity() { return t -> t; } /** * Returns a function that always returns the given argument. * * @param t the value to always return * @param the type of returned value * @return a function that returns the given value */ static Function constant(T t) { return i -> t; } /** * Creates a function that delegates to the given function if the given predicate applies, else delegates to {@link #identity()}. *

* This is equivalent to {@link #when(Predicate, Function, Function) when(predicate, function, identity())}. * * @param predicate the condition for the argument * @param function the function to apply if the predicate applies * @param the type of argument and return value * @return a function that delegates to the given function if the predicate applies, else returns the argument * @see #when(Predicate, Function, Function) * @see #conditional(Function, Action) * @since 1.5 */ static Function when(Predicate predicate, Function function) { return when(predicate, function, Function.identity()); } /** * Creates a function that delegates to the first function if the given predicate applies, else the second function. * * @param predicate the condition for the argument * @param onTrue the function to apply if the predicate applies * @param onFalse the function to apply if the predicate DOES NOT apply * @param the type of argument * @param the type of return value * @return a function that delegates to the first function if the predicate applies, else the second argument * @see #when(Predicate, Function) * @see #conditional(Function, Action) * @since 1.5 */ static Function when(Predicate predicate, Function onTrue, Function onFalse) { return Exceptions.uncheck(() -> conditional(onFalse, s -> s.when(predicate, onTrue))); } /** * A spec for adding conditions to a conditional function. * * @param the input type * @param the output type * @see #conditional(Function, Action) * @since 1.5 */ interface ConditionalSpec { /** * Adds a conditional function. * * @param predicate the condition predicate * @param function the function to apply if the predicate applies * @return {@code this} */ ConditionalSpec when(Predicate predicate, Function function); } /** * Creates a function that delegates based on the specified conditions. *

* If no conditions match, an {@link IllegalArgumentException} will be thrown. * Use {@link #conditional(Function, Action)} alternatively to specify a different “else” strategy. * * @param conditions the conditions * @param the input type * @param the output type * @return a conditional function * @see #conditional(Function, Action) * @throws Exception any thrown by {@code conditions} * @since 1.5 */ static Function conditional(Action> conditions) throws Exception { return conditional(i -> { throw new IllegalArgumentException("Unhandled argument: " + i); }, conditions); } /** * Creates a function that delegates based on the specified conditions. *

* If no condition applies, the {@code onElse} function will be delegated to. * * @param onElse the function to delegate to if no condition matches * @param conditions the conditions * @param the input type * @param the output type * @return a conditional function * @see #conditional(Action) * @throws Exception any thrown by {@code conditions} * @since 1.5 */ static Function conditional(Function onElse, Action> conditions) throws Exception { ImmutableList.Builder> builder = ImmutableList.builder(); conditions.execute(new ConditionalSpec() { @Override public ConditionalSpec when(Predicate predicate, Function function) { builder.add(new ConditionalFunction.Branch<>(predicate, function)); return this; } }); return new ConditionalFunction<>(builder.build(), onElse); } }