-
-
Notifications
You must be signed in to change notification settings - Fork 251
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
ghostdogpr
merged 12 commits into
ghostdogpr:master
from
frekw:feat/add-relay-connections
Jan 14, 2022
Merged
Changes from 3 commits
Commits
Show all changes
12 commits
Select commit
Hold shift + click to select a range
caa418d
wip cursors
frekw ac47a0e
update docs
frekw ae4735b
specify implicit type
frekw 8cccf53
Update core/src/main/scala/caliban/relay/Connection.scala
frekw 928a908
Update vuepress/docs/docs/relay-connections.md
frekw 2d9ecbd
PR review
frekw cc1c563
Update core/src/main/scala/caliban/relay/Base64Cursor.scala
frekw 3e35a39
shorten
frekw 2c1a2cf
clearer error message
frekw 9303993
make errors consistent
frekw d997789
validate cursor format
frekw 21b9d79
pr comments:
frekw File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,40 @@ | ||
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 { | ||
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.scalarSchema( | ||
ghostdogpr marked this conversation as resolved.
Show resolved
Hide resolved
|
||
"String", | ||
None, | ||
None, | ||
cursor => Value.StringValue(Cursor[Base64Cursor].encode(cursor)) | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 f translates a [[caliban.relay.PageInfo]] object and a list of edges to a `Connection` | ||
* @param g translates an entity and an offseet to an `Edge` | ||
frekw marked this conversation as resolved.
Show resolved
Hide resolved
|
||
* @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, _], R <: Connection[E]]( | ||
f: (PageInfo, List[E]) => R | ||
ghostdogpr marked this conversation as resolved.
Show resolved
Hide resolved
|
||
)(g: (A, Int) => E)( | ||
items: List[A], | ||
args: Pagination[Base64Cursor] | ||
): R = { | ||
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(g.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) | ||
) | ||
f(pageInfo, edges) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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("both first and last cannot be set") | ||
ghostdogpr marked this conversation as resolved.
Show resolved
Hide resolved
|
||
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) |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
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 😄
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍