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

functionalj.function.Func3 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.Optional;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier;

import functionalj.functions.ThrowFuncs;
import functionalj.promise.DeferAction;
import functionalj.promise.HasPromise;
import functionalj.promise.Promise;
import functionalj.result.Result;
import functionalj.task.Task;
import functionalj.tuple.Tuple;
import functionalj.tuple.Tuple3;
import lombok.val;
import nullablej.nullable.Nullable;

/**
 * Function of three parameters.
 * 
 * @param   the first input data type.
 * @param   the second input data type.
 * @param   the third input data type.
 * @param   the output data type.
 * 
 * @author NawaMan -- [email protected]
 */
@FunctionalInterface
public interface Func3 {

    public static  Func3 of(Func3 func) {
        return func;
    }
    public static  Func3 func3(Func3 func) {
        return func;
    }
    
    public OUTPUT applyUnsafe(INPUT1 input1, INPUT2 input2, INPUT3 input3) throws Exception;
    
    /**
     * Applies this function to the given input values.
     *
     * @param  input1  the first input.
     * @param  input2  the second input.
     * @param  input3  the third input.
     * @return         the function result.
     */
    public default OUTPUT apply(INPUT1 input1, INPUT2 input2, INPUT3 input3) {
        try {
            return applyUnsafe(input1, input2, input3);
        } catch (RuntimeException e) {
            throw e;
        } catch (Exception e) {
            throw ThrowFuncs.exceptionTransformer.value().apply(e);
        }
    }
    
    /**
     * Applies this function to the given input values.
     *
     * @param  input the tuple input.
     * @return       the function result.
     */
    public default OUTPUT applyTo(Tuple3 input) {
        return apply(input._1(), input._2(), input._3());
    }
    public default Func2 applyTo(INPUT1 input1) {
        return (input2, input3) -> apply(input1, input2, input3);
    }
    public default Result applyTo(Result input1, Result input2, Result input3) {
        return Result.ofResults(input1, input2, input3, this);
    }
    public default Optional applyTo(Optional input1, Optional input2, Optional input3) {
        return input1.flatMap(i1 -> {
            return input2.flatMap(i2 -> {
                return input3.map(i3 -> {
                    return Func3.this.apply(i1, i2, i3);
                });
            });
        });
    }
    public default Nullable applyTo(Nullable input1, Nullable input2, Nullable input3) {
        return input1.flatMap(i1 -> {
            return input2.flatMap(i2 -> {
                return input3.map(i3 -> {
                    return Func3.this.apply(i1, i2, i3);
                });
            });
        });
    }
    public default Promise applyTo(HasPromise input1, HasPromise input2, HasPromise input3) {
        return Promise.from(input1, input2, input3, this);
    }
    public default Task applyTo(Task input1, Task input2, Task input3) {
        return Task.from(input1, input2, input3, this);
    }
    public default Func0 applyTo(Supplier input1, Supplier input2, Supplier input3) {
        return ()->apply(input1.get(), input2.get(), input3.get());
    }
    public default  Func1 applyTo(Func1 input1, Func1 input2, Func1 input3) {
        return source -> {
            val i1 = input1.apply(source);
            val i2 = input2.apply(source);
            val i3 = input3.apply(source);
            return apply(i1, i2, i3);
        };
    }
    
    public default Result applySafely(INPUT1 input1, INPUT2 input2, INPUT3 input3) {
        try {
            val output = applyUnsafe(input1, input2, input3);
            return Result.valueOf(output);
        } catch (Exception exception) {
            return Result.ofException(exception);
        }
    }
    
    /**
     * 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  Func3 then(Function after) {
        return (input1, input2, input3) -> {
            OUTPUT output = this.applyUnsafe(input1, input2, input3);
            TARGET target = Func.applyUnsafe(after, output);
            return target;
        };
    }
    public default  Func3 map(Function after) {
        return (input1, input2, input3) -> {
            OUTPUT output = this.applyUnsafe(input1, input2, input3);
            TARGET target = (output != null)
                          ? Func.applyUnsafe(after, output)
                          : null;
            return target;
        };
    }
    
    public default Func3 ifException(Consumer exceptionHandler) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                return outputValue;
            } catch (Exception e) {
                exceptionHandler.accept(e);
                return null;
            }
        };
    }
    public default Func3 ifExceptionThenPrint() {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                return outputValue;
            } catch (Exception e) {
                e.printStackTrace();
                return null;
            }
        };
    }
    public default Func3 ifExceptionThenPrint(PrintStream printStream) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                return outputValue;
            } catch (Exception e) {
                e.printStackTrace(printStream);
                return null;
            }
        };
    }
    public default Func3 ifExceptionThenPrint(PrintWriter printWriter) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                return outputValue;
            } catch (Exception e) {
                e.printStackTrace(printWriter);
                return null;
            }
        };
    }
    
    public default Func3 whenAbsentUse(OUTPUT defaultValue) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : defaultValue;
                return returnValue;
            } catch (Exception e) {
                return defaultValue;
            }
        };
    }
    public default Func3 whenAbsentGet(Supplier defaultSupplier) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : defaultSupplier.get();
                return returnValue;
            } catch (Exception e) {
                return defaultSupplier.get();
            }
        };
    }
    public default Func3 whenAbsentApply(Func1 exceptionMapper) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(null);
                return returnValue;
            } catch (Exception e) {
                return exceptionMapper.apply(e);
            }
        };
    }
    public default Func3 whenAbsentApply(Func4 exceptionMapper) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(input1, input2, input3, null);
                return returnValue;
            } catch (Exception e) {
                return exceptionMapper.apply(input1, input2, input3, e);
            }
        };
    }
    public default Func3 whenAbsentApply(Func2, Exception, OUTPUT> exceptionMapper) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(Tuple.of(input1, input2, input3), null);
                return returnValue;
            } catch (Exception e) {
                return exceptionMapper.apply(Tuple.of(input1, input2, input3), e);
            }
        };
    }
    
    public default Func3 whenAbsentUse(Consumer exceptionHandler, OUTPUT defaultValue) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : defaultValue;
                return returnValue;
            } catch (Exception e) {
                exceptionHandler.accept(e);
                return defaultValue;
            }
        };
    }
    public default Func3 whenAbsentGet(Consumer exceptionHandler, Supplier defaultSupplier) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : defaultSupplier.get();
                return returnValue;
            } catch (Exception e) {
                exceptionHandler.accept(e);
                return defaultSupplier.get();
            }
        };
    }
    public default Func3 whenAbsentApply(Consumer exceptionHandler, Func1 exceptionMapper) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(null);
                return returnValue;
            } catch (Exception e) {
                exceptionHandler.accept(e);
                return exceptionMapper.apply(e);
            }
        };
    }
    public default Func3 whenAbsentApply(Consumer exceptionHandler, Func4 exceptionMapper) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(input1, input2, input3, null);
                return returnValue;
            } catch (Exception e) {
                exceptionHandler.accept(e);
                return exceptionMapper.apply(input1, input2, input3, e);
            }
        };
    }
    public default Func3 whenAbsentApply(Consumer exceptionHandler, Func2, Exception, OUTPUT> exceptionMapper) {
        return (input1, input2, input3)->{
            try {
                val outputValue = this.applyUnsafe(input1, input2, input3);
                val returnValue 
                        = (outputValue != null)
                        ? outputValue
                        : exceptionMapper.apply(Tuple.of(input1, input2, input3), null);
                return returnValue;
            } catch (Exception e) {
                exceptionHandler.accept(e);
                return exceptionMapper.apply(Tuple.of(input1, input2, input3), e);
            }
        };
    }
    
    public default OUTPUT orElse(INPUT1 input1, INPUT2 input2, INPUT3 input3, OUTPUT defaultValue) {
        return applySafely(input1, input2, input3).orElse(defaultValue);
    }
    
    public default OUTPUT orGet(INPUT1 input1, INPUT2 input2, INPUT3 input3, Supplier defaultSupplier) {
        return applySafely(input1, input2, input3).orGet(defaultSupplier);
    }
    
    public default Func3> safely() {
        return Func.of(this::applySafely);
    }
    
    public default Func3> optionally() {
        return (input1, input2, input3) -> {
            try {
                return Optional.ofNullable(this.applyUnsafe(input1, input2, input3));
            } catch (Exception e) {
                return Optional.empty();
            }
        };
    }
    
    public default Func3> async() {
        return (input1, input2, input3) -> {
            val supplier = (Func0)()->{
                return this.applyUnsafe(input1, input2, input3);
            };
            return DeferAction.from(supplier)
                    .start().getPromise();
        };
    }
    public default Func3, HasPromise, HasPromise, Promise> defer() {
        return (promise1, promise2, promise3) -> {
            return Promise.from(promise1, promise2, promise3, this);
        };
    }
    
    public default FuncUnit3 ignoreResult() {
        return FuncUnit3.of((input1, input2, input3)->applyUnsafe(input1, input2, input3));
    }
    
    public default Func1, OUTPUT> wholly() {
        return t -> this.applyUnsafe(t._1(), t._2(), t._3());
    }
    
    /**
     * Flip the parameter order.
     * 
     * @return  the Func3 with parameter in a flipped order.
     */
    public default Func3 flip() {
        return (i3, i2, i1) -> this.applyUnsafe(i1, i2, i3);
    }
    
    public default Func2> elevate() {
        return (i2, i3) -> (i1) -> this.applyUnsafe(i1, i2, i3);
    }
    
    public default Func1 elevateWith(INPUT2 i2, INPUT3 i3) {
        return (i1) -> this.applyUnsafe(i1, i2, i3);
    }
    
    public default Func1> split() {
        return split1();
    }
    public default Func1> split1() {
        return (i1) -> (i2, i3) -> this.applyUnsafe(i1, i2, i3);
    }
    public default Func2> split2() {
        return (i1, i2) -> (i3) -> this.applyUnsafe(i1, i2, i3);
    }
    
    //== Partially apply functions ==
    
    @SuppressWarnings("javadoc")
    public default Func0 bind(INPUT1 i1, INPUT2 i2, INPUT3 i3) {
        return () -> this.applyUnsafe(i1, i2, i3);
    }
    @SuppressWarnings("javadoc")
    public default Func2 bind1(INPUT1 i1) {
        return (i2,i3) -> this.applyUnsafe(i1, i2, i3);
    }
    @SuppressWarnings("javadoc")
    public default Func2 bind2(INPUT2 i2) {
        return (i1,i3) -> this.applyUnsafe(i1, i2, i3);
    }
    @SuppressWarnings("javadoc")
    public default Func2 bind3(INPUT3 i3) {
        return (i1,i2) -> this.applyUnsafe(i1, i2, i3);
    }
    
    @SuppressWarnings("javadoc")
    public default Func1 bind(Absent a1, INPUT2 i2, INPUT3 i3) {
        return i1 -> this.applyUnsafe(i1, i2, i3);
    }
    @SuppressWarnings("javadoc")
    public default Func1 bind(INPUT1 i1, Absent a2, INPUT3 i3) {
        return i2 -> this.applyUnsafe(i1, i2, i3);
    }
    @SuppressWarnings("javadoc")
    public default Func1 bind(INPUT1 i1, INPUT2 i2, Absent a3) {
        return i3 -> this.applyUnsafe(i1, i2, i3);
    }
    
    @SuppressWarnings("javadoc")
    public default Func2 bind(Absent a1, Absent a2, INPUT3 i3) {
        return (i1, i2) -> this.applyUnsafe(i1, i2, i3);
    }
    @SuppressWarnings("javadoc")
    public default Func2 bind(Absent a1, INPUT2 i2, Absent a3) {
        return (i1, i3) -> this.applyUnsafe(i1, i2, i3);
    }
    @SuppressWarnings("javadoc")
    public default Func2 bind(INPUT1 i1, Absent a2, Absent a3) {
        return (i2, i3) -> this.applyUnsafe(i1, i2, i3);
    }
    
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy