ratpack.func.Pair Maven / Gradle / Ivy
/*
* Copyright 2014 the original author or authors.
*
* Licensed under the Apache 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://www.apache.org/licenses/LICENSE-2.0
*
* 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 ratpack.func;
/**
* A generic pair implementation that can be used to cumulatively aggregate a data structure during a promise pipeline.
*
* This can sometimes be useful when collecting facts about something as part of a data stream without using mutable data structures.
*
{@code
* import ratpack.func.Pair;
* import ratpack.exec.Promise;
* import ratpack.test.embed.EmbeddedApp;
*
* import static org.junit.Assert.assertEquals;
*
* public class Example {
*
* public static void main(String[] args) throws Exception {
* EmbeddedApp.fromHandler(ctx -> {
* int id = 1;
* int age = 21;
* String name = "John";
*
* Promise.value(id)
* .map(idValue -> Pair.of(idValue, age))
* .flatMap(pair -> Promise.value(name).map(pair::nestRight))
* .then(pair -> {
* int receivedId = pair.left;
* int receivedAge = pair.right.right;
* String receivedName = pair.right.left;
* ctx.render(receivedName + " [" + receivedId + "] - age: " + receivedAge);
* });
* }).test(httpClient -> {
* assertEquals("John [1] - age: 21", httpClient.getText());
* });
* }
* }
* }
*
*
* @param the left data type
* @param the right data type
*/
public final class Pair {
/**
* The left item of the pair.
*/
public final L left;
/**
* The right item of the pair.
*/
public final R right;
private Pair(L left, R right) {
this.left = left;
this.right = right;
}
/**
* Creates a new pair.
*
* @param left the left item
* @param right the right item
* @param the type of the left item
* @param the type of the right item
* @return a new pair
*/
public static Pair of(L left, R right) {
return new Pair<>(left, right);
}
/**
* The left item of the pair.
*
* @return the left item of the pair
*/
public L getLeft() {
return left;
}
/**
* The right item of the pair.
*
* @return the right item of the pair
*/
public R getRight() {
return right;
}
/**
* The left item of the pair.
*
* @return the left item of the pair
*/
public L left() {
return left;
}
/**
* The right item of the pair.
*
* @return the right item of the pair
*/
public R right() {
return right;
}
/**
* Replaces the left item with the given item.
*
* @param newLeft the replacement left side of the pair
* @param the new left type
* @return a new pair, with the given item replacing the left of this
*/
public Pair left(T newLeft) {
return of(newLeft, right);
}
/**
* Replaces the right item with the given item.
*
* @param newRight the replacement right side of the pair
* @param the new right type
* @return a new pair, with the given item replacing the right of this
*/
public Pair right(T newRight) {
return of(left, newRight);
}
/**
* Creates a new pair.
*
* @param left the left item
* @param right the right item
* @param the type of the left item
* @param the type of the right item
* @return a new pair
*/
public static Pair pair(L left, R right) {
return of(left, right);
}
/**
* Creates a new pair, with {@code this} as the right item and the given value as the left.
*
* @param t the left value for the returned pair
* @param the type of the left value for the returned pair
* @return a new pair, with {@code this} as the right item and the given value as the left
*/
public Pair> pushLeft(T t) {
return of(t, this);
}
/**
* Creates a new pair, with {@code this} as the left item and the given value as the right.
*
* @param t the right value for the returned pair
* @param the type of the right value for the returned pair
* @return a new pair, with {@code this} as the left item and the given value as the right
*/
public Pair, T> pushRight(T t) {
return of(this, t);
}
/**
* Creates a new pair, with {@code pair(t, this.left)} as the left item and the the right value of {@code this} as the right.
*
* @param t the item to nest with the left item of {@code this} pair
* @param the type of item to nest with the left item of {@code this} pair
* @return a new pair, with {@code pair(t, this.left)} as the left item and the the right value of {@code this} as the right
*/
public Pair, R> nestLeft(T t) {
return of(of(t, left), right);
}
/**
* Creates a new pair, with {@code pair(t, this.right)} as the right item and the the left value of {@code this} as the left.
*
* @param t the item to nest with the right item of {@code this} pair
* @param the type of item to nest with the right item of {@code this} pair
* @return a new pair, with {@code pair(t, this.right)} as the right item and the the left value of {@code this} as the left
*/
public Pair> nestRight(T t) {
return of(left, of(t, right));
}
/**
* Creates a new pair, with the left item being the result of applying the given function to the left item of {@code this}.
*
* The right value is unchanged.
*
* @param function a transformer for the left value
* @param the type of the new left value
* @return a new pair, with the left item being the result of applying the given function to the left item of {@code this}
* @throws Exception any thrown by {@code function}
*/
public Pair mapLeft(Function super L, ? extends T> function) throws Exception {
T t = function.apply(left);
return of(t, right);
}
/**
* Creates a new pair, with the right item being the result of applying the given function to the right item of {@code this}.
*
* The left value is unchanged.
*
* @param function a transformer for the right value
* @param the type of the new right value
* @return a new pair, with the right item being the result of applying the given function to the right item of {@code this}
* @throws Exception any thrown by {@code function}
*/
public Pair mapRight(Function super R, ? extends T> function) throws Exception {
T t = function.apply(right);
return of(left, t);
}
/**
* Applies the given function to {@code this}, returning the result.
*
* @param function a function to apply to {@code this}
* @param the function result type
* @return the result of applying {@code function} to {@code this}
* @throws Exception any thrown by {@code function}
*/
public T map(Function super Pair, ? extends T> function) throws Exception {
return function.apply(this);
}
/**
* Convenience function for returning the left item of a pair.
*
* @param the type of the left item
* @param the pair type
* @return a function that when applied to a pair returns the left item
*/
public static > Function unpackLeft() {
return pair -> pair.left;
}
/**
* Convenience function for returning the right item of a pair.
*
* @param the type of the right item
* @param the pair type
* @return a function that when applied to a pair returns the right item
*/
public static > Function unpackRight() {
return pair -> pair.right;
}
/**
* A pair is equal if its left and right items are equal to the left and right items of {@code this} respectively.
*
* @param o the object to compare to
* @return the equality
*/
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
Pair, ?> pair = (Pair, ?>) o;
return !(left != null ? !left.equals(pair.left) : pair.left != null) && !(right != null ? !right.equals(pair.right) : pair.right != null);
}
/**
* Hash code.
*
* @return hash code.
*/
@Override
public int hashCode() {
int result = left != null ? left.hashCode() : 0;
result = 31 * result + (right != null ? right.hashCode() : 0);
return result;
}
/**
* Returns "Pair[«left.toString()»,«right.toString()»]
*
* @return "Pair[«left.toString()»,«right.toString()»]
*/
@Override
public String toString() {
return "Pair[" + left + "," + right + ']';
}
}