2024-01-03 02:14:13 +01:00
|
|
|
package fn
|
|
|
|
|
|
|
|
// Either is a type that can be either left or right.
|
|
|
|
type Either[L any, R any] struct {
|
2024-04-22 22:52:19 +02:00
|
|
|
isRight bool
|
|
|
|
left L
|
|
|
|
right R
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewLeft returns an Either with a left value.
|
|
|
|
func NewLeft[L any, R any](l L) Either[L, R] {
|
2024-04-22 22:52:19 +02:00
|
|
|
return Either[L, R]{left: l}
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewRight returns an Either with a right value.
|
|
|
|
func NewRight[L any, R any](r R) Either[L, R] {
|
2024-04-22 22:52:19 +02:00
|
|
|
return Either[L, R]{isRight: true, right: r}
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
|
2024-04-17 21:00:42 +02:00
|
|
|
// ElimEither is the universal Either eliminator. It can be used to safely
|
|
|
|
// handle all possible values inside the Either by supplying two continuations,
|
|
|
|
// one for each side of the Either.
|
|
|
|
func ElimEither[L, R, O any](f func(L) O, g func(R) O, e Either[L, R]) O {
|
2024-04-22 22:52:19 +02:00
|
|
|
if !e.isRight {
|
|
|
|
return f(e.left)
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
|
2024-04-22 22:52:19 +02:00
|
|
|
return g(e.right)
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
|
2024-01-03 02:14:13 +01:00
|
|
|
// WhenLeft executes the given function if the Either is left.
|
|
|
|
func (e Either[L, R]) WhenLeft(f func(L)) {
|
2024-04-22 22:52:19 +02:00
|
|
|
if !e.isRight {
|
|
|
|
f(e.left)
|
|
|
|
}
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// WhenRight executes the given function if the Either is right.
|
|
|
|
func (e Either[L, R]) WhenRight(f func(R)) {
|
2024-04-22 22:52:19 +02:00
|
|
|
if e.isRight {
|
|
|
|
f(e.right)
|
|
|
|
}
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// IsLeft returns true if the Either is left.
|
|
|
|
func (e Either[L, R]) IsLeft() bool {
|
2024-04-22 22:52:19 +02:00
|
|
|
return !e.isRight
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// IsRight returns true if the Either is right.
|
|
|
|
func (e Either[L, R]) IsRight() bool {
|
2024-04-22 22:52:19 +02:00
|
|
|
return e.isRight
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
|
2024-04-17 21:00:42 +02:00
|
|
|
// LeftToOption converts a Left value to an Option, returning None if the inner
|
|
|
|
// Either value is a Right value.
|
|
|
|
func (e Either[L, R]) LeftToOption() Option[L] {
|
2024-04-22 22:52:19 +02:00
|
|
|
if e.isRight {
|
|
|
|
return None[L]()
|
|
|
|
}
|
|
|
|
|
|
|
|
return Some(e.left)
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// RightToOption converts a Right value to an Option, returning None if the
|
|
|
|
// inner Either value is a Left value.
|
|
|
|
func (e Either[L, R]) RightToOption() Option[R] {
|
2024-04-22 22:52:19 +02:00
|
|
|
if !e.isRight {
|
|
|
|
return None[R]()
|
|
|
|
}
|
|
|
|
|
|
|
|
return Some(e.right)
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// UnwrapLeftOr will extract the Left value from the Either if it is present
|
|
|
|
// returning the supplied default if it is not.
|
|
|
|
func (e Either[L, R]) UnwrapLeftOr(l L) L {
|
2024-04-22 22:52:19 +02:00
|
|
|
if e.isRight {
|
|
|
|
return l
|
|
|
|
}
|
|
|
|
|
|
|
|
return e.left
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// UnwrapRightOr will extract the Right value from the Either if it is present
|
|
|
|
// returning the supplied default if it is not.
|
|
|
|
func (e Either[L, R]) UnwrapRightOr(r R) R {
|
2024-04-22 22:52:19 +02:00
|
|
|
if !e.isRight {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
return e.right
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Swap reverses the type argument order. This can be useful as an adapter
|
|
|
|
// between APIs.
|
|
|
|
func (e Either[L, R]) Swap() Either[R, L] {
|
|
|
|
return Either[R, L]{
|
2024-04-22 22:52:19 +02:00
|
|
|
isRight: !e.isRight,
|
|
|
|
left: e.right,
|
|
|
|
right: e.left,
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-03 02:14:13 +01:00
|
|
|
// MapLeft maps the left value of the Either to a new value.
|
2024-04-17 21:00:42 +02:00
|
|
|
func MapLeft[L, R, O any](f func(L) O) func(Either[L, R]) Either[O, R] {
|
|
|
|
return func(e Either[L, R]) Either[O, R] {
|
2024-04-22 22:52:19 +02:00
|
|
|
if !e.isRight {
|
2024-04-17 21:00:42 +02:00
|
|
|
return Either[O, R]{
|
2024-04-22 22:52:19 +02:00
|
|
|
isRight: false,
|
|
|
|
left: f(e.left),
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
|
2024-04-17 21:00:42 +02:00
|
|
|
return Either[O, R]{
|
2024-04-22 22:52:19 +02:00
|
|
|
isRight: true,
|
|
|
|
right: e.right,
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MapRight maps the right value of the Either to a new value.
|
|
|
|
func MapRight[L, R, O any](f func(R) O) func(Either[L, R]) Either[L, O] {
|
|
|
|
return func(e Either[L, R]) Either[L, O] {
|
2024-04-22 22:52:19 +02:00
|
|
|
if e.isRight {
|
2024-04-17 21:00:42 +02:00
|
|
|
return Either[L, O]{
|
2024-04-22 22:52:19 +02:00
|
|
|
isRight: true,
|
|
|
|
right: f(e.right),
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Either[L, O]{
|
2024-04-22 22:52:19 +02:00
|
|
|
isRight: false,
|
|
|
|
left: e.left,
|
2024-04-17 21:00:42 +02:00
|
|
|
}
|
2024-01-03 02:14:13 +01:00
|
|
|
}
|
|
|
|
}
|