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

functionalj.function.Func1 Maven / Gradle / Ivy

// ============================================================================
// Copyright(c) 2017-2019 Nawapunth Manusitthipol (NawaMan - http://nawaman.net)
// ----------------------------------------------------------------------------
// MIT License
// 
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
// 
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
// 
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
// ============================================================================
package functionalj.function;

import java.io.PrintStream;
import java.io.PrintWriter;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
import java.util.stream.Stream;

import functionalj.functions.ThrowFuncs;
import functionalj.list.FuncList;
import functionalj.map.FuncMap;
import functionalj.promise.DeferAction;
import functionalj.promise.HasPromise;
import functionalj.promise.Promise;
import functionalj.result.Result;
import functionalj.stream.StreamPlus;
import functionalj.task.Task;
import lombok.val;
import nullablej.nullable.Nullable;

/**
 * Function of one parameter.
 * 
 * @param    the input data type.
 * @param   the output data type.
 * 
 * @author NawaMan -- [email protected]
 */
@FunctionalInterface
public interface Func1 extends Function {
    
    /**
     * Constructs a Func1 from function or lambda.
     * 
     * @param  function  the function or lambda.
     * @param     the input data type.
     * @param    the output data type.
     * @return           the result Func1.
     **/
    public static  
            Func1 of(Func1 function) {
        return function;
    }
    public static  
            Func1 func1(Func1 function) {
        return function;
    }
    
    public static  Func1 from(Function func) {
        return func::apply;
    }
    public static  Func1 from(
            Func2 func,
            Func1 input1,
            Func1 input2) {
        return source -> {
            val i1 = input1.apply(source);
            val i2 = input2.apply(source);
            return func.apply(i1, i2);
        };
    }
    public static  Func1 from(
            Func3 func,
            Func1 input1,
            Func1 input2,
            Func1 input3) {
        return source -> {
            val i1 = input1.apply(source);
            val i2 = input2.apply(source);
            val i3 = input3.apply(source);
            return func.apply(i1, i2, i3);
        };
    }
    public static  Func1 from(
            Func4 func,
            Func1 input1,
            Func1 input2,
            Func1 input3,
            Func1 input4) {
        return source -> {
            val i1 = input1.apply(source);
            val i2 = input2.apply(source);
            val i3 = input3.apply(source);
            val i4 = input4.apply(source);
            return func.apply(i1, i2, i3, i4);
        };
    }
    public static  Func1 from(
            Func5 func,
            Func1 input1,
            Func1 input2,
            Func1 input3,
            Func1 input4,
            Func1 input5) {
        return source -> {
            val i1 = input1.apply(source);
            val i2 = input2.apply(source);
            val i3 = input3.apply(source);
            val i4 = input4.apply(source);
            val i5 = input5.apply(source);
            return func.apply(i1, i2, i3, i4, i5);
        };
    }
    public static  Func1 from(
            Func6 func,
            Func1 input1,
            Func1 input2,
            Func1 input3,
            Func1 input4,
            Func1 input5,
            Func1 input6) {
        return source -> {
            val i1 = input1.apply(source);
            val i2 = input2.apply(source);
            val i3 = input3.apply(source);
            val i4 = input4.apply(source);
            val i5 = input5.apply(source);
            val i6 = input6.apply(source);
            return func.apply(i1, i2, i3, i4, i5, i6);
        };
    }
    
    public OUTPUT applyUnsafe(INPUT input) throws Exception;
    
    
    /**
     * Applies this function to the given input value.
     *
     * @param input  the input function.
     * @return the function result.
     */
    public default OUTPUT apply(INPUT input) {
        try {
            return applyUnsafe(input);
        } catch (RuntimeException e) {
            throw e;
        } catch (Exception e) {
            throw ThrowFuncs.exceptionTransformer.value().apply(e);
        }
    }
    
    public default OUTPUT applyToNull() {
        return apply((INPUT)null);
    }
    public default Result applyTo(Result input) {
        return input.map(this);
    }
    public default Optional applyTo(Optional input) {
        return input.map(this);
    }
    public default Nullable applyTo(Nullable input) {
        return input.map(this);
    }
    public default Promise applyTo(HasPromise input) {
        return input.getPromise().map(this);
    }
    public default Task applyTo(Task input) {
        return input.map(this);
    }
    public default StreamPlus applyTo(Stream input) {
        return StreamPlus.from(input).map(this);
    }
    public default FuncList applyTo(List input) {
        return FuncList.from(input).map(this);
    }
    public default  FuncMap applyTo(Map input) {
        return FuncMap.from(input).map(this);
    }
    public default FuncList applyTo(FuncList input) {
        return FuncList.from(input).map(this);
    }
    public default  FuncMap applyTo(FuncMap input) {
        return FuncMap.from(input).map(this);
    }
    public default Func0 applyTo(Supplier input) {
        return ()->apply(input.get());
    }
    public default  Func1 applyTo(Function input) {
        return t -> apply(input.apply(t));
    }
    
    public default Result applySafely(INPUT input) {
        try {
            val output = applyUnsafe(input);
            return Result.valueOf(output);
        } catch (Exception exception) {
            return Result.ofException(exception);
        }
    }
    
    public default Func1 memoize() {
        return Func.cacheFor(this);
    }
    
    /**
     * Compose this function to the given function.
     * NOTE: Too bad the name 'compose' is already been taken :-(
     * 
     * @param    the target result value.
     * @param  after     the function to be run after this function.
     * @return           the composed function.
     */
    public default  Func1 then(Function after) {
        return input -> {
            OUTPUT output = this.applyUnsafe(input);
            TARGET target = Func.applyUnsafe(after, output);
            return target;
        };
    }
    public default  Func1 map(Function after) {
        return input -> {
            OUTPUT output = this.applyUnsafe(input);
            TARGET target = (output != null)
                          ? Func.applyUnsafe(after, output)
                          : null;
            return target;
        };
    }
    
    public default Func1 ifException(Consumer exceptionHandler) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                return outputValue;
            } catch (Exception e) {
                exceptionHandler.accept(e);
                return null;
            }
        };
    }
    public default Func1 ifExceptionThenPrint() {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                return outputValue;
            } catch (Exception e) {
                e.printStackTrace();
                return null;
            }
        };
    }
    public default Func1 ifExceptionThenPrint(PrintStream printStream) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                return outputValue;
            } catch (Exception e) {
                e.printStackTrace(printStream);
                return null;
            }
        };
    }
    public default Func1 ifExceptionThenPrint(PrintWriter printWriter) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                return outputValue;
            } catch (Exception e) {
                e.printStackTrace(printWriter);
                return null;
            }
        };
    }
    
    public default Func1 whenAbsentUse(OUTPUT defaultValue) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : defaultValue;
                return returnValue;
            } catch (Exception e) {
                return defaultValue;
            }
        };
    }
    public default Func1 whenAbsentGet(Supplier defaultSupplier) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : defaultSupplier.get();
                return returnValue;
            } catch (Exception e) {
                return defaultSupplier.get();
            }
        };
    }
    public default Func1 whenAbsentApply(Func1 exceptionMapper) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(null);
                return returnValue;
            } catch (Exception e) {
                return exceptionMapper.apply(e);
            }
        };
    }
    public default Func1 whenAbsentApply(Func2 exceptionMapper) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(input, null);
                return returnValue;
            } catch (Exception e) {
                return exceptionMapper.apply(input, e);
            }
        };
    }
    
    public default Func1 whenAbsentUse(Consumer exceptionHandler, OUTPUT defaultValue) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : defaultValue;
                return returnValue;
            } catch (Exception e) {
                return defaultValue;
            }
        };
    }
    public default Func1 whenAbsentGet(Consumer exceptionHandler, Supplier defaultSupplier) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : defaultSupplier.get();
                return returnValue;
            } catch (Exception e) {
                return defaultSupplier.get();
            }
        };
    }
    public default Func1 whenAbsentApply(Consumer exceptionHandler, Func1 exceptionMapper) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(null);
                return returnValue;
            } catch (Exception e) {
                return exceptionMapper.apply(e);
            }
        };
    }
    public default Func1 whenAbsentApply(Consumer exceptionHandler, Func2 exceptionMapper) {
        return (input)->{
            try {
                val outputValue = this.applyUnsafe(input);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(input, null);
                return returnValue;
            } catch (Exception e) {
                return exceptionMapper.apply(input, e);
            }
        };
    }
    
    public default OUTPUT orElse(INPUT input, OUTPUT defaultValue) {
        return applySafely(input).orElse(defaultValue);
    }
    
    public default OUTPUT orGet(INPUT input, Supplier defaultSupplier) {
        return applySafely(input).orGet(defaultSupplier);
    }
    
    public default Func1> safely() {
        return Func.of(this::applySafely);
    }
    
    public default Func1> optionally() {
        return (input) -> {
            try {
                return Optional.ofNullable(this.applyUnsafe(input));
            } catch (Exception e) {
                return Optional.empty();
            }
        };
    }
    
    public default Func1> async() {
        return input -> {
            val supplier = (Func0)()->{
                return this.applyUnsafe(input);
            };
            return DeferAction.from(supplier)
                    .start().getPromise();
        };
    }
    public default Func1, Promise> defer() {
        return input -> input.getPromise().map(this);
    }
    
    public default FuncUnit1 ignoreResult() {
        return FuncUnit1.of((input1)->applyUnsafe(input1));
    }
    
    public default Predicate toPredicate() {
        return toPredicate(Boolean.TRUE::equals);
    }
    public default Predicate toPredicate(Func1 converter) {
        val converted = this.then(converter);
        return Func.toPredicate(converted);
    }
    
    /**
     * Create a bind function (a supplier) of the this function.
     * 
     * @param   input  the input value.
     * @return         the Supplier.
     */
    public default Func0 bind(INPUT input) {
        return () -> {
            return this.applyUnsafe(input);
        };
    }
    
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy