/** * 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.functions; import static org.opencastproject.util.data.Monadics.mlist; import static org.opencastproject.util.data.Option.none; import static org.opencastproject.util.data.Option.some; import org.opencastproject.util.data.Effect; import org.opencastproject.util.data.Function; import org.opencastproject.util.data.Function0; import org.opencastproject.util.data.Function2; import org.opencastproject.util.data.Option; import java.util.ArrayList; import java.util.List; /** {@link Option} related functions. */ public final class Options { private Options() { } /** m (m a) -> m a */ public static <A> Option<A> join(Option<Option<A>> a) { return a.bind(Functions.<Option<A>> identity()); } public static <A> Function<Option<A>, List<A>> asList() { return new Function<Option<A>, List<A>>() { @Override public List<A> apply(Option<A> a) { return a.list(); } }; } public static <A, B> Function<A, Option<B>> never() { return new Function<A, Option<B>>() { @Override public Option<B> apply(A a) { return none(); } }; } public static <A> Function0<Option<A>> never2() { return new Function0<Option<A>>() { @Override public Option<A> apply() { return none(); } }; } /** Function that turns <code>true</code> into <code>some(true)</code> and false into <code>none</code>. */ public static final Function<Boolean, Option<Boolean>> toOption = new Function<Boolean, Option<Boolean>>() { @Override public Option<Boolean> apply(Boolean a) { return a ? some(true) : Option.<Boolean> none(); } }; /** Returns some(message) if predicate is false, none otherwise. */ public static Option<String> toOption(boolean predicate, String message) { return predicate ? Option.<String> none() : some(message); } /** Sequence a list of options. [Option a] -> Option [a] */ public static <A> Option<List<A>> sequenceOpt(List<Option<A>> as) { final List<A> seq = mlist(as).foldl(new ArrayList<A>(), new Function2<List<A>, Option<A>, List<A>>() { @Override public List<A> apply(List<A> sum, Option<A> o) { for (A a : o) { sum.add(a); return sum; } return sum; } }); return some(seq); } /** Map <code>g</code> over the result of <code>f</code>. */ public static <A, B, C> Function<A, Option<C>> map(final Function<? super A, ? extends Option<? extends B>> f, final Function<? super B, ? extends C> g) { return new Function<A, Option<C>>() { @Override public Option<C> apply(A a) { return f.apply(a).map(g); } }; } /** Apply effect <code>e</code> to the result of <code>f</code> which is then returned. */ public static <A, B> Function<A, Option<B>> foreach(final Function<? super A, ? extends Option<? extends B>> f, final Effect<? super B> e) { return new Function<A, Option<B>>() { @Override public Option<B> apply(A a) { return (Option<B>) f.apply(a).foreach(e); } }; } /** {@link org.opencastproject.util.data.Option#isNone()} as a function. */ public static <A> Function<Option<A>, Boolean> isNone() { return new Function<Option<A>, Boolean>() { @Override public Boolean apply(Option<A> a) { return a.isNone(); } }; } /** {@link org.opencastproject.util.data.Option#isSome()} as a function. */ public static <A> Function<Option<A>, Boolean> isSome() { return new Function<Option<A>, Boolean>() { @Override public Boolean apply(Option<A> a) { return a.isSome(); } }; } public static <A> Function<Option<A>, A> getOrElse(final A none) { return new Function<Option<A>, A>() { @Override public A apply(Option<A> as) { return as.getOrElse(none); } }; } public static <A> Function<Option<A>, A> getF() { return new Function<Option<A>, A>() { @Override public A apply(Option<A> as) { return as.get(); } }; } }