Auto-configure RSocketGraphQlClient
This commit contributes a `RSocketGraphQlClient.Builder` component to the context, pre-configured with the `RSocketStrategies`, a customized `RSocketConnector` and the expected data MIME type. See gh-30453
This commit is contained in:
parent
9c3cce58ea
commit
2dc2e5ab11
|
|
@ -0,0 +1,56 @@
|
|||
/*
|
||||
* Copyright 2012-2022 the original author or authors.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* https://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package org.springframework.boot.autoconfigure.graphql.rsocket;
|
||||
|
||||
import graphql.GraphQL;
|
||||
import io.rsocket.RSocket;
|
||||
import io.rsocket.transport.netty.client.TcpClientTransport;
|
||||
|
||||
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
||||
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
|
||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
||||
import org.springframework.boot.autoconfigure.rsocket.RSocketRequesterAutoConfiguration;
|
||||
import org.springframework.context.annotation.Bean;
|
||||
import org.springframework.context.annotation.Scope;
|
||||
import org.springframework.graphql.client.RSocketGraphQlClient;
|
||||
import org.springframework.messaging.rsocket.RSocketRequester;
|
||||
import org.springframework.util.MimeTypeUtils;
|
||||
|
||||
/**
|
||||
* {@link EnableAutoConfiguration Auto-configuration} for {@link RSocketGraphQlClient}.
|
||||
* This auto-configuration creates {@link RSocketGraphQlClient.Builder} prototype beans,
|
||||
* as the builders are stateful and should not be reused to build client instances with
|
||||
* different configurations.
|
||||
*
|
||||
* @author Brian Clozel
|
||||
* @since 2.7.0
|
||||
*/
|
||||
@AutoConfiguration(after = RSocketRequesterAutoConfiguration.class)
|
||||
@ConditionalOnClass({ GraphQL.class, RSocketGraphQlClient.class, RSocketRequester.class, RSocket.class,
|
||||
TcpClientTransport.class })
|
||||
public class RSocketGraphQlClientAutoConfiguration {
|
||||
|
||||
@Bean
|
||||
@Scope("prototype")
|
||||
@ConditionalOnMissingBean
|
||||
public RSocketGraphQlClient.Builder<?> rsocketGraphQlClientBuilder(
|
||||
RSocketRequester.Builder rsocketRequesterBuilder) {
|
||||
return RSocketGraphQlClient.builder(rsocketRequesterBuilder.dataMimeType(MimeTypeUtils.APPLICATION_GRAPHQL));
|
||||
}
|
||||
|
||||
}
|
||||
|
|
@ -0,0 +1,76 @@
|
|||
/*
|
||||
* Copyright 2012-2022 the original author or authors.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* https://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package org.springframework.boot.autoconfigure.graphql.rsocket;
|
||||
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
||||
import org.springframework.boot.autoconfigure.rsocket.RSocketRequesterAutoConfiguration;
|
||||
import org.springframework.boot.autoconfigure.rsocket.RSocketStrategiesAutoConfiguration;
|
||||
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
||||
import org.springframework.context.annotation.Bean;
|
||||
import org.springframework.context.annotation.Configuration;
|
||||
import org.springframework.graphql.client.RSocketGraphQlClient;
|
||||
|
||||
import static org.assertj.core.api.Assertions.assertThat;
|
||||
|
||||
/**
|
||||
* Tests for {@link RSocketGraphQlClientAutoConfiguration}.
|
||||
*
|
||||
* @author Brian Clozel
|
||||
*/
|
||||
class RSocketGraphQlClientAutoConfigurationTests {
|
||||
|
||||
private static final RSocketGraphQlClient.Builder<?> builderInstance = RSocketGraphQlClient.builder();
|
||||
|
||||
private final ApplicationContextRunner contextRunner = new ApplicationContextRunner()
|
||||
.withConfiguration(AutoConfigurations.of(RSocketStrategiesAutoConfiguration.class,
|
||||
RSocketRequesterAutoConfiguration.class, RSocketGraphQlClientAutoConfiguration.class));
|
||||
|
||||
@Test
|
||||
void shouldCreateBuilder() {
|
||||
this.contextRunner.run((context) -> assertThat(context).hasSingleBean(RSocketGraphQlClient.Builder.class));
|
||||
}
|
||||
|
||||
@Test
|
||||
void shouldGetPrototypeScopedBean() {
|
||||
this.contextRunner.run((context) -> {
|
||||
RSocketGraphQlClient.Builder<?> first = context.getBean(RSocketGraphQlClient.Builder.class);
|
||||
RSocketGraphQlClient.Builder<?> second = context.getBean(RSocketGraphQlClient.Builder.class);
|
||||
assertThat(first).isNotEqualTo(second);
|
||||
});
|
||||
}
|
||||
|
||||
@Test
|
||||
void shouldNotCreateBuilderIfAlreadyPresent() {
|
||||
this.contextRunner.withUserConfiguration(CustomRSocketGraphQlClientBuilder.class).run((context) -> {
|
||||
RSocketGraphQlClient.Builder<?> builder = context.getBean(RSocketGraphQlClient.Builder.class);
|
||||
assertThat(builder).isEqualTo(builderInstance);
|
||||
});
|
||||
}
|
||||
|
||||
@Configuration(proxyBeanMethods = false)
|
||||
static class CustomRSocketGraphQlClientBuilder {
|
||||
|
||||
@Bean
|
||||
RSocketGraphQlClient.Builder<?> myRSocketGraphQlClientBuilder() {
|
||||
return builderInstance;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
|
@ -117,13 +117,14 @@ Spring Boot supports many configuration properties under the `spring.graphql.cor
|
|||
|
||||
RSocket is also supported as a transport, on top of WebSocket or TCP.
|
||||
Once the <<messaging#messaging.rsocket.server-auto-configuration,RSocket server is configured>>, we can configure our GraphQL handler on a particular route using configprop:spring.graphql.rsocket.mapping[].
|
||||
For example, configuring that mapping as `"graphql"` means we can use the `RSocketGraphQlClient` as follows.
|
||||
For example, configuring that mapping as `"graphql"` means we can use that as a route when sending requests with the `RSocketGraphQlClient`.
|
||||
|
||||
For RSocket over TCP:
|
||||
include::code:RSocketGraphQlClientExample[tag=tcp]
|
||||
Spring Boot auto-configures a `RSocketGraphQlClient.Builder<?>` bean that you can inject in your components:
|
||||
|
||||
For RSocket over WebSocket:
|
||||
include::code:RSocketGraphQlClientExample[tag=websocket]
|
||||
include::code:RSocketGraphQlClientExample[tag=builder]
|
||||
|
||||
And then send a request:
|
||||
include::code:RSocketGraphQlClientExample[tag=request]
|
||||
|
||||
|
||||
[[web.graphql.exception-handling]]
|
||||
|
|
|
|||
|
|
@ -16,32 +16,29 @@
|
|||
|
||||
package org.springframework.boot.docs.web.graphql.transports.rsocket;
|
||||
|
||||
import java.net.URI;
|
||||
import java.time.Duration;
|
||||
|
||||
import reactor.core.publisher.Mono;
|
||||
|
||||
import org.springframework.graphql.client.RSocketGraphQlClient;
|
||||
import org.springframework.stereotype.Component;
|
||||
|
||||
// tag::builder[]
|
||||
@Component
|
||||
public class RSocketGraphQlClientExample {
|
||||
|
||||
public void rsocketOverTcp() {
|
||||
// tag::tcp[]
|
||||
RSocketGraphQlClient client = RSocketGraphQlClient.builder().tcp("example.spring.io", 8181).route("graphql")
|
||||
.build();
|
||||
Mono<Book> book = client.document("{ bookById(id: \"book-1\"){ id name pageCount author } }")
|
||||
.retrieve("bookById").toEntity(Book.class);
|
||||
// end::tcp[]
|
||||
book.block(Duration.ofSeconds(5));
|
||||
}
|
||||
private final RSocketGraphQlClient graphQlClient;
|
||||
|
||||
public void rsocketOverWebSocket() {
|
||||
// tag::websocket[]
|
||||
RSocketGraphQlClient client = RSocketGraphQlClient.builder()
|
||||
.webSocket(URI.create("wss://example.spring.io/rsocket")).route("graphql").build();
|
||||
Mono<Book> book = client.document("{ bookById(id: \"book-1\"){ id name pageCount author } }")
|
||||
public RSocketGraphQlClientExample(RSocketGraphQlClient.Builder<?> builder) {
|
||||
this.graphQlClient = builder.tcp("example.spring.io", 8181).route("graphql").build();
|
||||
}
|
||||
// end::builder[]
|
||||
|
||||
public void rsocketOverTcp() {
|
||||
// tag::request[]
|
||||
Mono<Book> book = this.graphQlClient.document("{ bookById(id: \"book-1\"){ id name pageCount author } }")
|
||||
.retrieve("bookById").toEntity(Book.class);
|
||||
// end::websocket[]
|
||||
// end::request[]
|
||||
book.block(Duration.ofSeconds(5));
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -17,19 +17,21 @@
|
|||
package org.springframework.boot.docs.web.graphql.transports.rsocket
|
||||
|
||||
import org.springframework.graphql.client.RSocketGraphQlClient
|
||||
import java.net.URI
|
||||
import org.springframework.stereotype.Component
|
||||
import java.time.Duration
|
||||
|
||||
// tag::builder[]
|
||||
@Component
|
||||
class RSocketGraphQlClientExample(private val builder: RSocketGraphQlClient.Builder<*>) {
|
||||
// end::builder[]
|
||||
|
||||
class RSocketGraphQlClientExample {
|
||||
val graphQlClient = builder.tcp("example.spring.io", 8181)
|
||||
.route("graphql")
|
||||
.build()
|
||||
|
||||
fun rsocketOverTcp() {
|
||||
// tag::tcp[]
|
||||
val client = RSocketGraphQlClient.builder()
|
||||
.tcp("example.spring.io", 8181)
|
||||
.route("graphql")
|
||||
.build()
|
||||
val book = client.document(
|
||||
// tag::request[]
|
||||
val book = graphQlClient.document(
|
||||
"""
|
||||
{
|
||||
bookById(id: "book-1"){
|
||||
|
|
@ -39,31 +41,10 @@ class RSocketGraphQlClientExample {
|
|||
author
|
||||
}
|
||||
}
|
||||
""")
|
||||
.retrieve("bookById").toEntity(Book::class.java)
|
||||
// end::tcp[]
|
||||
book.block(Duration.ofSeconds(5))
|
||||
}
|
||||
|
||||
fun rsocketOverWebSocket() {
|
||||
// tag::websocket[]
|
||||
val client = RSocketGraphQlClient.builder()
|
||||
.webSocket(URI.create("wss://example.spring.io/rsocket"))
|
||||
.route("graphql")
|
||||
.build()
|
||||
val book = client.document(
|
||||
"""
|
||||
{
|
||||
bookById(id: "book-1"){
|
||||
id
|
||||
name
|
||||
pageCount
|
||||
author
|
||||
}
|
||||
}
|
||||
""")
|
||||
)
|
||||
.retrieve("bookById").toEntity(Book::class.java)
|
||||
// end::websocket[]
|
||||
// end::request[]
|
||||
book.block(Duration.ofSeconds(5))
|
||||
}
|
||||
|
||||
|
|
|
|||
Loading…
Reference in New Issue