/** * Licensed to The Apereo Foundation under one or more contributor license * agreements. See the NOTICE file distributed with this work for additional * information regarding copyright ownership. * * * The Apereo Foundation licenses this file to you under the Educational * Community 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://opensource.org/licenses/ecl2.txt * * 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 org.opencastproject.util.data; import static org.opencastproject.util.data.Either.left; import static org.opencastproject.util.data.Either.right; import static org.opencastproject.util.data.functions.Misc.chuck; import org.opencastproject.util.data.functions.Functions; import com.entwinemedia.fn.Fn; /** * Function of arity 1. * * A general note on function implementations: Each function has an associated X version with the only difference that a * checked exception may be thrown on application. This behaviour could be nicely combined into the main function class * but with the cost of having no abstract method. Some IDEs such as IntelliJ support nice code folding for SAM (single * abstract method) classes. * * @see X */ public abstract class Function<A, B> { /** Apply function to <code>a</code>. */ public abstract B apply(A a); /** Currying. */ public Function0<B> curry(final A a) { return Functions.curry(this, a); } /** Currying. */ public Function<A, Function0<B>> curry() { return Functions.curry(this); } /** Function composition. <code>g.o(f).apply(x) == g(f(x))</code> */ public <C> Function<C, B> o(final Function<? super C, ? extends A> f) { return Functions.o(this, f); } /** Function composition. <code>g.o(f).apply() == g(f())</code> */ public Function0<B> o(final Function0<? extends A> f) { return Functions.o(this, f); } /** Left to right composition: <code>f then g = g(f(x))</code> */ public <C> Function<A, C> then(final Function<? super B, ? extends C> f) { return Functions.then(this, f); } /** @see Functions#rethrow(Function, Function) */ public Function<A, B> rethrow(final Function<? super Exception, ? extends Exception> transformer) { return Functions.rethrow(this, transformer); } /** @see Functions#handle(Function, Function) */ public Function<A, B> handle(final Function<? super Exception, B> handler) { return Functions.handle(this, handler); } /** @see Functions#either(Function, Function) */ public <X> Function<A, Either<X, B>> either(final Function<Exception, X> handler) { return Functions.either(this, handler); } /** Turn this function into an effect by discarding its result. */ public Effect<A> toEffect() { return Functions.toEffect(this); } /** Convert this function into a google guava function. */ public com.google.common.base.Function<A, B> toGuava() { return Functions.toGuava(this); } public Fn<A, B> toFn() { return new Fn<A, B>() { @Override public B apply(A a) { return Function.this.apply(a); } }; } /** Version of {@link Function} that allows for throwing a checked exception. */ public abstract static class X<A, B> extends Function<A, B> { @Override public final B apply(A a) { try { return xapply(a); } catch (Exception e) { return chuck(e); } } /** * Apply function to <code>a</code>. Any thrown exception gets "chucked" so that you may catch them as is. See * {@link Functions#chuck(Throwable)} for details. */ protected abstract B xapply(A a) throws Exception; public final Function<A, Either<Exception, B>> either() { return new Function<A, Either<Exception, B>>() { @Override public Either<Exception, B> apply(A a) { try { return right(xapply(a)); } catch (Exception e) { return left(e); } } }; } } /** Version of {@link Function} that allows for throwing a checked exception. */ public abstract static class Xe<A, B, Err> extends Function<A, Either<Err, B>> { private final Function<Exception, Err> toErr; protected Xe(Function<Exception, Err> toErr) { this.toErr = toErr; } @Override public final Either<Err, B> apply(A a) { try { return right(xapply(a)); } catch (Exception e) { return left(toErr.apply(e)); } } protected abstract B xapply(A a) throws Exception; } }