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

net.calledtoconstruct.Left Maven / Gradle / Ivy

Go to download

A small collection of potentially useful classes for streamlining process flow.

The newest version!
package net.calledtoconstruct;

import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier;
import java.util.stream.Collectors;

public class Left implements Either {

    private final TLeft value;

    /**
     * An implementation of {@link Either} in which the {@link Left} value of type {@code TLeft} is retained.
     * 
     * @param value The value of type {@code TLeft} to store.
     */
    public Left(TLeft value) {
        this.value = value;
    }

    @Override
    public  Either onLeftApply(Function function) {
        final TOther other = function.apply(value);
        return new Left<>(other);
    }

    @Override
    public  Either onRightApply(Function function) {
        return new Left<>(value);
    }

    @Override
    public Either onLeftAccept(Consumer consumer) {
        consumer.accept(value);
        return this;
    }

    @Override
    public Either onRightAccept(Consumer consumer) {
        return this;
    }

    @Override
    public  Either onLeftSupply(Supplier supplier) {
        final var other = supplier.get();
        return new Left<>(other);
    }

    @Override
    public  Either onRightSupply(Supplier supplier) {
        return new Left<>(value);
    }

    /**
     * Get the contained value of type {@code TLeft}.
     * 
     * @return the value of type {@code TLeft}
     */
    public TLeft getValue() {
        return value;
    }
    
    /**
     * A function which accepts zero to many instances of {@link Either}, collects all the values from those of type {@link Left}, and invokes the supplied
     * {@code consumer}, passing the {@link java.util.List} of {@code TLeft} values as the parameter.
     * 
     * @param  The type for the {@link Left} value.
     * @param  The type for the {@link Right} value.
     * @param consumer A method which receives a {@link java.util.List} of {@code TLeft} values collected from any of the supplied {@code input} which are of type {@link Left}.
     * @param input Zero or more instances of {@link Either} to be evaluated.
     * @return {@code true} when {@code input} contains at least one instance of {@link Left}, otherwise, {@code false}.
     */
    @SafeVarargs
    public static  boolean acceptAll(Consumer> consumer, Either... input) {
        final var values = Arrays.stream(input)
            .map(either -> (either instanceof Left left) ? left.getValue() : null)
            .filter(Objects::nonNull)
            .collect(Collectors.toList());

        if (values.size() == input.length) {
            consumer.accept(values);
            return true;
        }

        return false;
    }

    /**
     * A function which accepts zero to many instances of {@link Either}, searches for any instance of type {@link Left}, then extracts its value and returns
     * it as an {@link java.util.Optional}.  When no instance of type {@link Left} is provided, an empty {@link java.util.Optional} is returned.
     * 
     * @param  The type for the {@link Left} value.
     * @param  The type for the {@link Right} value.
     * @param input Zero or more instances of {@link Either} to be evaluated.
     * @return And {@link java.util.Optional} of type {@code TLeft} or empty.
     */
    @SafeVarargs
    public static  Optional any(Either... input) {
        for (var either : input) {
            if (either instanceof Left left) {
                return Optional.of(left.getValue());
            }
        }
        return Optional.empty();
    }

    @Override
    public  Either mergeFailToLeft(
        Either other,
        BiFunction functionMergeLeft,
        BiFunction functionMergeRight,
        Function transformThis,
        Function transformOther
    ) {
        if (other instanceof Left left) {
            return new Left<>(functionMergeLeft.apply(value, left.value));
        } else if (other instanceof Right) {
            return new Left<>(transformThis.apply(value));
        } else {
            throw new UnexpectedNeitherException();
        }
    }

    @Override
    public  Either mergeFailToRight(
        Either other,
        BiFunction functionLeft,
        BiFunction functionRight,
        Function transformThis,
        Function transformOther
    ) {
        if (other instanceof Left left) {
            return new Left<>(functionLeft.apply(value, left.value));
        } else if (other instanceof Right right) {
            return new Right<>(transformOther.apply(right.getValue()));
        } else {
            throw new UnexpectedNeitherException();
        }
    }

    @Override
    public Either flip() {
        return new Right<>(value);
    }

    @Override
    public  Either onLeftFlatMap(
        Function> function,
        Function otherwise
    ) {
        return function.apply(value)
            .map(out -> new Left(out))
            .orElse(new Left<>(otherwise.apply(value)));
    }

    @Override
    public  Either onRightFlatMap(
        Function> function,
        Function otherwise
    ) {
        return new Left<>(value);
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy