
endpoints4s.Tupler.scala Maven / Gradle / Ivy
The newest version!
package endpoints4s
/**
* Defines a strategy for tupling `A` and `B` values, according to types `A` and `B`.
*
* The actual implementation avoids nested tuples and eliminates `Unit`, so that instead of ending with, e.g.,
* the following type:
*
* {{{
* ((Unit, Int), (((Unit, Unit), String)))
* }}}
*
* We just get:
*
* {{{
* (Int, String)
* }}}
*
* The following rules are implemented (by increasing priority):
* - A, B -> (A, B)
* - A, (B, C) -> (A, B, C)
* - (A, B), C -> (A, B, C)
* - (A, B), (C, D) -> (A, B, C, D)
* - A, (B, C, D, E) -> (A, B, C, D, E)
* - (A, B), (C, D, E) -> (A, B, C, D, E)
* - (A, B, C), D -> (A, B, C, D)
* - (A, B, C, D), E -> (A, B, C, D, E)
* - (A, B, C, D, E), F -> (A, B, C, D, E, F)
* - A, Unit -> A
* - Unit, A -> A
*/
//#definition
trait Tupler[A, B] {
type Out
//#definition
def apply(a: A, b: B): Out
def unapply(out: Out): (A, B)
//#definition
}
//#definition
object Tupler extends Tupler5
trait Tupler1 {
type Aux[A, B, Out0] = Tupler[A, B] { type Out = Out0 }
implicit def ab[A, B]: Aux[A, B, (A, B)] =
new Tupler[A, B] {
type Out = (A, B)
def apply(a: A, b: B): (A, B) = (a, b)
def unapply(out: (A, B)): (A, B) = out
}
}
trait Tupler2 extends Tupler1 {
implicit def tupler1And2[A, B, C]: Aux[A, (B, C), (A, B, C)] =
new Tupler[A, (B, C)] {
type Out = (A, B, C)
def apply(a: A, bc: (B, C)): (A, B, C) = (a, bc._1, bc._2)
def unapply(out: (A, B, C)): (A, (B, C)) = {
val (a, b, c) = out
(a, (b, c))
}
}
implicit def tupler2And1[A, B, C]: Aux[(A, B), C, (A, B, C)] =
new Tupler[(A, B), C] {
type Out = (A, B, C)
def apply(ab: (A, B), c: C): (A, B, C) = (ab._1, ab._2, c)
def unapply(out: (A, B, C)): ((A, B), C) = {
val (a, b, c) = out
((a, b), c)
}
}
}
trait Tupler3 extends Tupler2 {
implicit def tupler2And2[A, B, C, D]: Aux[(A, B), (C, D), (A, B, C, D)] =
new Tupler[(A, B), (C, D)] {
type Out = (A, B, C, D)
def apply(ab: (A, B), cd: (C, D)): (A, B, C, D) =
(ab._1, ab._2, cd._1, cd._2)
def unapply(out: (A, B, C, D)): ((A, B), (C, D)) = {
val (a, b, c, d) = out
((a, b), (c, d))
}
}
implicit def tupler1And4[A, B, C, D, E]: Tupler[A, (B, C, D, E)] { type Out = (A, B, C, D, E) } =
new Tupler[A, (B, C, D, E)] {
type Out = (A, B, C, D, E)
def apply(a: A, bcde: (B, C, D, E)): (A, B, C, D, E) =
(a, bcde._1, bcde._2, bcde._3, bcde._4)
def unapply(out: (A, B, C, D, E)): (A, (B, C, D, E)) = {
val (a, b, c, d, e) = out
(a, (b, c, d, e))
}
}
implicit def tupler2And3[A, B, C, D, E]
: Tupler[(A, B), (C, D, E)] { type Out = (A, B, C, D, E) } =
new Tupler[(A, B), (C, D, E)] {
type Out = (A, B, C, D, E)
def apply(ab: (A, B), cde: (C, D, E)): (A, B, C, D, E) =
(ab._1, ab._2, cde._1, cde._2, cde._3)
def unapply(out: (A, B, C, D, E)): ((A, B), (C, D, E)) = {
val (a, b, c, d, e) = out
((a, b), (c, d, e))
}
}
}
trait Tupler4 extends Tupler3 with TuplerAppend {
implicit def leftUnit[A]: Aux[Unit, A, A] =
new Tupler[Unit, A] {
type Out = A
def apply(a: Unit, b: A): A = b
def unapply(out: Out): (Unit, A) = ((), out)
}
}
trait Tupler5 extends Tupler4 {
implicit def rightUnit[A]: Aux[A, Unit, A] =
new Tupler[A, Unit] {
type Out = A
def apply(a: A, b: Unit): A = a
def unapply(out: Out): (A, Unit) = (out, ())
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy