Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Relay cursors/connections #1196

Merged
merged 12 commits into from
Jan 14, 2022
36 changes: 36 additions & 0 deletions core/src/main/scala/caliban/relay/Base64Cursor.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
package caliban.relay

import caliban.schema.Schema
import caliban.Value
import scala.util.Try

/**
* A cursor implementation that models an index/offset as an
* opaque base64 cursor.
*/
case class Base64Cursor(value: Int)
object Base64Cursor {
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you mind adding a line break, just to match the style of the rest of the code 😄

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍

import java.util.Base64
lazy val encoder = Base64.getEncoder()
lazy val decoder = Base64.getDecoder()

private val prefix = "cursor:"

implicit val cursor: Cursor[Base64Cursor] = new Cursor[Base64Cursor] {
type T = Int
def encode(a: Base64Cursor): String =
encoder.encodeToString(s"$prefix${a.value}".getBytes("UTF-8"))

def decode(raw: String): Either[String, Base64Cursor] =
Try({
val bytes = decoder.decode(raw)
val s = new String(bytes, "UTF-8")
Base64Cursor(s.replaceFirst(prefix, "").toInt)
ghostdogpr marked this conversation as resolved.
Show resolved Hide resolved
}).toEither.left.map(_.getMessage())

def value(cursor: Base64Cursor): Int = cursor.value
}

implicit val schema: Schema[Any, Base64Cursor] =
Schema.stringSchema.contramap(Cursor[Base64Cursor].encode)
}
80 changes: 80 additions & 0 deletions core/src/main/scala/caliban/relay/Connection.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,80 @@
package caliban.relay

/**
* The Relay PageInfo type which models pagination info
* for a connection.
*/
case class PageInfo(
hasNextPage: Boolean,
hasPreviousPage: Boolean,
startCursor: Option[String],
endCursor: Option[String]
)

/**
* An abstract class representing a Relay connection edge
* for some type `T`.
*/
abstract class Edge[+C: Cursor, +T] {
def cursor: C
def node: T

def encodeCursor = Cursor[C].encode(cursor)
}

/**
* An abstract class representing a Relay connection for
* some edge `T`.
*/
abstract class Connection[T <: Edge[_, _]] {
val pageInfo: PageInfo
val edges: List[T]
}

object Connection {

/**
* A function that returns a sliced result set based
* on some mapping functions, a full input list of entities
* and pagination information.
*
* @param makeConnection translates a [[caliban.relay.PageInfo]] object and a list of edges to a `Connection`
* @param makeEdge translates an entity and an offset to an `Edge`
* @param items the list of items to paginate
* @param args a set of [[caliban.relay.Pagination]] arguments
* @return a paginated connection
*/
def fromList[A, E <: Edge[Base64Cursor, _], C <: Connection[E]](
makeConnection: (PageInfo, List[E]) => C
)(makeEdge: (A, Int) => E)(
items: List[A],
args: Pagination[Base64Cursor]
): C = {
val itemsWithIndex = items.zipWithIndex
ghostdogpr marked this conversation as resolved.
Show resolved Hide resolved

val sliced = (args.cursor match {
case PaginationCursor.NoCursor => itemsWithIndex
case PaginationCursor.After(cursor) => itemsWithIndex.drop(cursor.value + 1)
case PaginationCursor.Before(cursor) => itemsWithIndex.dropRight(cursor.value + 1)
})

val dropped = args.count match {
case PaginationCount.First(count) => sliced.take(count)
case PaginationCount.Last(count) => sliced.takeRight(count)
}

val edges = dropped.map(makeEdge.tupled)

val pageInfo = PageInfo(
hasNextPage = edges.headOption
.map(e => e.cursor.value + args.count.count < items.size)
.getOrElse(false),
hasPreviousPage = edges.lastOption
.map(e => e.cursor.value > args.count.count)
.getOrElse(false),
startCursor = edges.headOption.map(start => start.encodeCursor),
endCursor = edges.lastOption.map(end => end.encodeCursor)
)
makeConnection(pageInfo, edges)
}
}
15 changes: 15 additions & 0 deletions core/src/main/scala/caliban/relay/Cursor.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
package caliban.relay

/**
* A trait representing an abstract Relay Connection cursor.
*/
trait Cursor[A] {
type T
def encode(a: A): String
def decode(s: String): Either[String, A]
def value(cursor: A): T
}

object Cursor {
def apply[A](implicit c: Cursor[A]): Cursor[A] = c
}
91 changes: 91 additions & 0 deletions core/src/main/scala/caliban/relay/PaginationArgs.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
package caliban.relay

import caliban.CalibanError
import zio._

object Pagination {
import PaginationCount._
import PaginationCursor._

def apply[C: Cursor](
args: PaginationArgs[C]
): ZIO[Any, CalibanError, Pagination[C]] =
apply(args.first, args.last, args.before, args.after)

def apply[C: Cursor](
first: Option[Int],
last: Option[Int],
before: Option[String],
after: Option[String]
): ZIO[Any, CalibanError, Pagination[C]] =
ZIO
.mapParN(
validateFirstLast(first, last),
validateCursors(before, after)
)((count, cursor) => new Pagination[C](count, cursor))
.parallelErrors
.mapError((errors: ::[String]) => CalibanError.ValidationError(msg = errors.mkString(", "), explanatoryText = ""))

private def validateCursors[C: Cursor](
before: Option[String],
after: Option[String]
) =
(before, after) match {
case (Some(_), Some(_)) =>
ZIO.fail("both before and after may not be set")
case (Some(x), _) =>
ZIO.fromEither(Cursor[C].decode(x)).map(Before(_))
case (_, Some(x)) =>
ZIO.fromEither(Cursor[C].decode(x)).map(After(_))
case (None, None) => ZIO.succeed(NoCursor)
}

private def validateFirstLast(first: Option[Int], last: Option[Int]) =
(first, last) match {
case (None, None) =>
ZIO.fail("first and last cannot both be empty")
case (Some(_), Some(_)) =>
ZIO.fail("first and last cannot both be set")
case (Some(a), _) =>
validatePositive("first", a).map(First(_))
case (_, Some(b)) =>
validatePositive("last", b).map(Last(_))
}

private def validatePositive(which: String, i: Int) =
if (i < 0) ZIO.fail(s"$which cannot be negative")
else ZIO.succeed(i)
ghostdogpr marked this conversation as resolved.
Show resolved Hide resolved
}

sealed trait PaginationCount extends Product with Serializable {
def count: Int
}
object PaginationCount {
case class First(count: Int) extends PaginationCount
case class Last(count: Int) extends PaginationCount
}

sealed trait PaginationCursor[+C]
object PaginationCursor {
case class After[C](cursor: C) extends PaginationCursor[C]
case class Before[C](cursor: C) extends PaginationCursor[C]
case object NoCursor extends PaginationCursor[Nothing]
}

case class Pagination[+C](
count: PaginationCount,
cursor: PaginationCursor[C]
)

abstract class PaginationArgs[C: Cursor] { self =>
val first: Option[Int]
val last: Option[Int]
val before: Option[String]
val after: Option[String]

def toPagination: ZIO[Any, CalibanError, Pagination[C]] = Pagination(
self
)
}

case class PaginationError(reason: String)
Loading