SmallRye GraphQL Client
This guide demonstrates how your Quarkus application can use the GraphQL client library. The client is implemented by the SmallRye GraphQL project. This guide is specifically geared towards the client side, so if you need an introduction to GraphQL in general, first refer to the SmallRye GraphQL guide, which provides an introduction to the GraphQL query language, general concepts and server-side development.
The guide will walk you through developing and running a simple application that uses both supported types of GraphQL clients to retrieve data from a remote resource, that being a database related to Star Wars. It’s available at this webpage if you want to experiment with it manually. The web UI allows you to write and execute GraphQL queries against it.
准备
要完成本指南,您需要:
-
Roughly 15 minutes
-
An IDE
-
JDK 11+ installed with
JAVA_HOME
configured appropriately -
Apache Maven 3.9.6
-
Optionally the Quarkus CLI if you want to use it
-
Optionally Mandrel or GraalVM installed and configured appropriately if you want to build a native executable (or Docker if you use a native container build)
GraphQL client types introduction
Two types of GraphQL clients are supported.
The typesafe client works very much like the MicroProfile REST Client adjusted for calling GraphQL endpoints. A client instance is basically a proxy that you can call like a regular Java object, but under the hood, the call will be translated to a GraphQL operation. It works with domain classes directly. Any input and output objects for the operation will be translated to/from their representations in the GraphQL query language.
The dynamic client, on the other hand, works rather like an equivalent of
the Jakarta REST client from the jakarta.ws.rs.client
package. It does not
require the domain classes to work, it works with abstract representations
of GraphQL documents instead. Documents are built using a domain-specific
language (DSL). The exchanged objects are treated as an abstract
JsonObject
, but, when necessary, it is possible to convert them to
concrete model objects (if suitable model classes are available).
The typesafe client can be viewed as a rather high-level and more declarative approach designed for ease of use, whereas the dynamic client is lower-level, more imperative, somewhat more verbose to use, but allows finer grained control over operations and responses.
完整源码
We recommend that you follow the instructions in the next sections and create the application step by step. However, you can go right to the completed example.
Clone the Git repository: git clone https://github.com/quarkusio/quarkus-quickstarts.git
, or download
an archive.
The solution is located in the microprofile-graphql-client-quickstart
directory.
Creating the Maven Project
First, we need a new project. Create a new project with the following command:
For Windows users:
-
If using cmd, (don’t use backward slash
\
and put everything on the same line) -
If using Powershell, wrap
-D
parameters in double quotes e.g."-DprojectArtifactId=microprofile-graphql-client-quickstart"
This command generates a project, importing the smallrye-graphql-client
extension, and also the resteasy-reactive
extension, because we will use
that too - a REST endpoint will be the entrypoint to allow you to manually
trigger the GraphQL client to do its work.
If you already have your Quarkus project configured, you can add the
smallrye-graphql-client
extension to your project by running the following
command in your project base directory:
quarkus extension add graphql-client
./mvnw quarkus:add-extension -Dextensions='graphql-client'
./gradlew addExtension --extensions='graphql-client'
This will add the following to your build file:
<dependency>
<groupId>io.quarkus</groupId>
<artifactId>quarkus-smallrye-graphql-client</artifactId>
</dependency>
implementation("io.quarkus:quarkus-smallrye-graphql-client")
The application
The application we will build makes use of both types of GraphQL clients. In both cases, they will connect to the Star Wars service at SWAPI and query it for a list of Star Wars films, and, for each film, the names of the planets which appear in that film.
The corresponding GraphQL query looks like this:
{
allFilms {
films {
title
planetConnection {
planets {
name
}
}
}
}
}
You may go to the webpage to execute this query manually.
Using the Typesafe client
To use the typesafe client, we need the corresponding model classes that are compatible with the schema. There are two ways to obtain them. First is to use the client generator offered by SmallRye GraphQL, which generates classes from the schema document and a list of queries. This generator is considered highly experimental for now, and is not covered in this example. If interested, refer to the Client Generator and its documentation.
In this example, we will create a slimmed down version of the model classes
manually, with only the fields that we need, and ignore all the stuff that
we don’t need. We will need the classes for Film
and Planet
. But, the
service is also using specific wrappers named FilmConnection
and
PlanetConnection
, which, for our purpose, will serve just to contain the
actual list of Film
and Planet
instances, respectively.
Let’s create all the model classes and put them into the
org.acme.microprofile.graphql.client.model
package:
public class FilmConnection {
private List<Film> films;
public List<Film> getFilms() {
return films;
}
public void setFilms(List<Film> films) {
this.films = films;
}
}
public class Film {
private String title;
private PlanetConnection planetConnection;
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public PlanetConnection getPlanetConnection() {
return planetConnection;
}
public void setPlanetConnection(PlanetConnection planetConnection) {
this.planetConnection = planetConnection;
}
}
public class PlanetConnection {
private List<Planet> planets;
public List<Planet> getPlanets() {
return planets;
}
public void setPlanets(List<Planet> planets) {
this.planets = planets;
}
}
public class Planet {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
Now that we have the model classes, we can create the interface that represents the actual set of operations we want to call on the remote GraphQL service.
@GraphQLClientApi(configKey = "star-wars-typesafe") public interface StarWarsClientApi { FilmConnection allFilms(); }
For simplicity, we’re only calling the query named allFilms
. We named our
corresponding method allFilms
too. If we named the method differently, we
would need to annotate it with @Query(value="allFilms")
to specify the
name of the query that should be executed when this method is called.
The client also needs some configuration, namely at least the URL of the
remote service. We can either specify that within the @GraphQLClientApi
annotation (by setting the endpoint
parameter), or move this over to the
configuration file, application.properties
:
quarkus.smallrye-graphql-client.star-wars-typesafe.url=https://swapi-graphql.netlify.app/.netlify/functions/index
During tests only, the URL is an optional property, and if it’s not
specified, Quarkus will assume that the target of the client is the
application that is being tested (typically,
http://localhost:8081/graphql ). This is useful if your application
contains a GraphQL server-side API as well as a GraphQL client that is used
for testing the API.
|
star-wars-typesafe
is the name of the configured client instance, and
corresponds to the configKey
in the @GraphQLClientApi
annotation. If you
don’t want to specify a custom name, you can leave out the configKey
, and
then refer to it by using the fully qualified name of the interface.
Now that we have the client instance properly configured, we need a way to have it perform something when we start the application. For that, we will use a REST endpoint that, when called by a user, obtains the client instance and lets it execute the query.
@Path("/")
public class StarWarsResource {
@Inject
StarWarsClientApi typesafeClient;
@GET
@Path("/typesafe")
@Produces(MediaType.APPLICATION_JSON)
@Blocking
public List<Film> getAllFilmsUsingTypesafeClient() {
return typesafeClient.allFilms().getFilms();
}
}
With this REST endpoint included in your application, you can simply send a
GET request to /typesafe
, and the application will use an injected
typesafe client instance to call the remote service, obtain the films and
planets, and return the JSON representation of the resulting list.
Using the Dynamic client
For the dynamic client, the model classes are optional, because we can work with abstract representations of the GraphQL types and documents. The client API interface is not needed at all.
We still need to configure the URL for the client, so let’s put this into
application.properties
:
quarkus.smallrye-graphql-client.star-wars-dynamic.url=https://swapi-graphql.netlify.app/.netlify/functions/index
We decided to name the client star-wars-dynamic
. We will use this name
when injecting a dynamic client to properly qualify the injection point.
If you need to add an authorization header, or any other custom HTTP header (in our case it’s not required), this can be done by:
quarkus.smallrye-graphql-client.star-wars-dynamic.header.HEADER-KEY=HEADER-VALUE"
Add this to the StarWarsResource
created earlier:
import static io.smallrye.graphql.client.core.Document.document;
import static io.smallrye.graphql.client.core.Field.field;
import static io.smallrye.graphql.client.core.Operation.operation;
// ....
@Inject
@GraphQLClient("star-wars-dynamic") (1)
DynamicGraphQLClient dynamicClient;
@GET
@Path("/dynamic")
@Produces(MediaType.APPLICATION_JSON)
@Blocking
public List<Film> getAllFilmsUsingDynamicClient() throws Exception {
Document query = document( (2)
operation(
field("allFilms",
field("films",
field("title"),
field("planetConnection",
field("planets",
field("name")
)
)
)
)
)
);
Response response = dynamicClient.executeSync(query); (3)
return response.getObject(FilmConnection.class, "allFilms").getFilms(); (4)
}
1 | Qualifies the injection point so that we know which named client needs to be injected here. |
2 | Here we build a document representing the GraphQL query, using the provided DSL language. We use static imports to make the code easier to read. The DSL is designed in a way that it looks quite similar to writing a GraphQL query as a string. |
3 | Execute the query and block while waiting for the response. There is also an
asynchronous variant that returns a Uni<Response> . |
4 | Here we did the optional step of converting the response to instances of our
model classes, because we have the classes available. If you don’t have the
classes available or don’t want to use them, simply calling
response.getData() would get you a JsonObject representing all the
returned data. |
运行应用
Launch the application in dev mode using:
quarkus dev
./mvnw quarkus:dev
./gradlew --console=plain quarkusDev
To execute the queries, you need to send GET requests to our REST endpoint:
curl -s http://localhost:8080/dynamic # to use the dynamic client
curl -s http://localhost:8080/typesafe # to use the typesafe client
Whether you use dynamic or typesafe, the result should be the same. If the
JSON document is hard to read, you might want to run it through a tool that
formats it for better readability by humans, for example by piping the
output through jq
.