This function makes a GraphQL request using GRAPHQL
and
VARIABLES
as inputs. GRAPHQL
is a GraphQL string. VARIABLES
is a JSON-like alist. The other arguments behave as for
ghub-request
(which see).
The response is returned as a JSON-like alist. Even if the response
contains errors
, this function does not raise an error.
Cursor-handling is likewise left to the caller.
ghub-graphql
is a thin convenience wrapper around ghub-request
,
similar to ghub-post
and friends. While the latter only hard-code
the value of the METHOD
argument, the former also hard-codes RESOURCE
and constructs PAYLOAD
from GRAPHQL
and VARIABLES
. It also drops
UNPAGINATE
, NOERROR
, READER
(internal functions expect alist-ified
JSON) and FORGE
(only Github currently supports GraphQL).
ghub-graphql
does not account for the fact that pagination works
differently in GraphQL than it does in REST, so users of this function
have to deal with that themselves. Likewise error handling works
differently and has to be done by the caller too.
An early attempt at implementing automatic unpaginating for GraphQL
can be found in the faithful-graphql
branch, provided I haven’t
deleted that by now. On that branch I try to do things as intended by
the designers of GraphQL, using variables and fragments, and drowning
in a sea of boilerplate.
The problem with that approach is that it only works for applications that fetch specific information on demand and actually want things to be paginated. I am convinced that GraphQL is very nice for web apps.
However the Forge package for which I have implemented all of this has
very different needs. It wants to fetch "all the data" and "cache"
it locally, so that it is available even when there is no internet
connection. GraphQL was designed around the idea that you should be
able to "ask for what you need and get exactly that". But when that
boils down to "look, if I persist, then you are going to hand me over
all the data anyway, so just caught it up already", then things start
to fall apart. If Github’s GraphQL allowed pagination to be turned
off completely, then teaching ghub-graphql
about error handling would
be enough.
But it doesn’t and when doing things as intended, then that leads to huge amounts of repetitive boilerplate, which is so boring to write that doing it without introducing bugs left and right is near impossible; so I decided to give up on GraphQL variables, fragments and conditions, and instead implement something more powerful, though also more opinionated.
This function is an opinionated alternative to ghub-graphql
.
It relies on dark magic to get the job done.
It makes an initial request using QUERY
. It then looks for
paginated edges in the returned data and makes more requests to
resolve them. In order to do so it automatically transforms the
initial QUERY
into another query suitable for that particular edge.
The data retrieved by subsequent requests is then injected into the
data of the original request before that is returned or passed to
the callback. If subsequently retrieved data features new paginated
edges, then those are followed recursively.
The end result is essentially the same as using ghub-graphql
, if
only it were possible to say "do not paginate anything". The
implementation is much more complicated because it is not possible
to do that.
QUERY
is a GraphQL query expressed as an s-expression. The bundled
gsexp
library is used to turn that into a GraphQL query string.
Only a subset of the GraphQL features are supported; fragments for
example are not, and magical stuff happens to variables. This is
not documented yet, I am afraid. Look at existing callers.
VARIABLES
is a JSON-like alist as for ghub-graphql
.
UNTIL
is an alist ((EDGE-until . VALUE)...)
. When unpaginating EDGE
try not to fetch beyond the element whose first field has the value
VALUE
and remove that element as well as all "lesser" elements from
the retrieved data if necessary. Look at forge--pull-repository
for
an example. This is only useful if you "cache" the response locally
and want to avoid fetching data again that you already have.
Other arguments behave as for ghub-graphql
and ghub-request
, more or
less. If CALLBACK
is nil, pretty-print the response.
Using ghub--graphql-vacuum
, the following resource specific functions
are implemented. These functions are not part of the public API yet
and are very much subject to change.
This function asynchronously fetches forge data about the specified
repository. Once all data has been collected, CALLBACK
is called
with the data as the only argument.
This function asynchronously fetches forge data about the specified
issue. Once all data has been collected, CALLBACK
is called
with the data as the only argument.
This function asynchronously fetches forge data about the specified
pull-request. Once all data has been collected, CALLBACK
is called
with the data as the only argument.
Note that in order to avoid duplication all of these functions base
their initial query on the query stored in ghub-fetch-repository
. The
latter two pass that query through ghub--graphql-prepare-query
, which
then uses ghub--graphql-narrow-query
to remove parts the caller is not
interested in. These two functions are also used internally, when
unpaginating, but as demonstrated here they can be useful even before
making an initial request.