Skip to content

Commit 300eda0

Browse files
authored
Merge branch 'master' into validation-extensions
2 parents de8f59e + 15995c2 commit 300eda0

15 files changed

+769
-166
lines changed

graphql-dgs-client/build.gradle.kts

+4-2
Original file line numberDiff line numberDiff line change
@@ -25,14 +25,16 @@ dependencies {
2525
implementation("org.springframework:spring-web")
2626
implementation("com.fasterxml.jackson.module:jackson-module-kotlin")
2727
implementation("com.fasterxml.jackson.datatype:jackson-datatype-jsr310")
28+
implementation("com.fasterxml.jackson.module:jackson-module-parameter-names")
29+
implementation("com.fasterxml.jackson.datatype:jackson-datatype-jdk8")
2830
implementation("com.graphql-java:graphql-java")
2931

3032
testImplementation("org.springframework.boot:spring-boot-starter-web")
3133
testImplementation("org.springframework:spring-test")
3234
testImplementation("org.springframework.boot:spring-boot-starter-webflux")
3335
testImplementation("com.graphql-java:graphql-java-extended-scalars")
3436
testImplementation("io.projectreactor:reactor-test")
35-
testImplementation("com.squareup.okhttp3:mockwebserver")
36-
testImplementation(project(":graphql-dgs-webflux-starter"))
3737
testImplementation(project(":graphql-dgs-subscriptions-sse-autoconfigure"))
38+
testImplementation(project(":graphql-dgs-spring-boot-oss-autoconfigure"))
39+
testImplementation(project(":graphql-dgs-spring-webmvc-autoconfigure"))
3840
}

graphql-dgs-client/dependencies.lock

+28-121
Large diffs are not rendered by default.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
/*
2+
* Copyright 2021 Netflix, Inc.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package com.netflix.graphql.dgs.client
18+
19+
/**
20+
* Blocking implementation of a GraphQL client.
21+
* The user is responsible for doing the actual HTTP request, making this pluggable with any HTTP client.
22+
* For a more convenient option, use [WebClientGraphQLClient] instead.
23+
*/
24+
class CustomGraphQLClient(private val url: String, private val requestExecutor: RequestExecutor) : GraphQLClient {
25+
override fun executeQuery(query: String): GraphQLResponse {
26+
return executeQuery(query, emptyMap(), null)
27+
}
28+
29+
override fun executeQuery(query: String, variables: Map<String, Any>): GraphQLResponse {
30+
return executeQuery(query, variables, null)
31+
}
32+
33+
override fun executeQuery(query: String, variables: Map<String, Any>, operationName: String?): GraphQLResponse {
34+
val serializedRequest = GraphQLClients.objectMapper.writeValueAsString(
35+
Request(
36+
query,
37+
variables,
38+
operationName
39+
)
40+
)
41+
42+
val response = requestExecutor.execute(url, GraphQLClients.defaultHeaders, serializedRequest)
43+
return GraphQLClients.handleResponse(response, serializedRequest, url)
44+
}
45+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
/*
2+
* Copyright 2021 Netflix, Inc.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package com.netflix.graphql.dgs.client
18+
19+
import reactor.core.publisher.Mono
20+
21+
/**
22+
* Non-blocking implementation of a GraphQL client, based on the [Mono] type.
23+
* The user is responsible for doing the actual HTTP request, making this pluggable with any HTTP client.
24+
* For a more convenient option, use [WebClientGraphQLClient] instead.
25+
*/
26+
class CustomMonoGraphQLClient(private val url: String, private val monoRequestExecutor: MonoRequestExecutor) : MonoGraphQLClient {
27+
override fun reactiveExecuteQuery(query: String): Mono<GraphQLResponse> {
28+
return reactiveExecuteQuery(query, emptyMap(), null)
29+
}
30+
31+
override fun reactiveExecuteQuery(query: String, variables: Map<String, Any>): Mono<GraphQLResponse> {
32+
return reactiveExecuteQuery(query, variables, null)
33+
}
34+
35+
override fun reactiveExecuteQuery(
36+
query: String,
37+
variables: Map<String, Any>,
38+
operationName: String?
39+
): Mono<GraphQLResponse> {
40+
val serializedRequest = GraphQLClients.objectMapper.writeValueAsString(
41+
Request(
42+
query,
43+
variables,
44+
operationName
45+
)
46+
)
47+
return monoRequestExecutor.execute(url, GraphQLClients.defaultHeaders, serializedRequest).map { response ->
48+
GraphQLClients.handleResponse(response, serializedRequest, url)
49+
}
50+
}
51+
}

graphql-dgs-client/src/main/kotlin/com/netflix/graphql/dgs/client/DefaultGraphQLClient.kt

+37-36
Original file line numberDiff line numberDiff line change
@@ -16,15 +16,13 @@
1616

1717
package com.netflix.graphql.dgs.client
1818

19-
import com.fasterxml.jackson.databind.ObjectMapper
20-
import com.fasterxml.jackson.module.kotlin.KotlinModule
21-
import com.fasterxml.jackson.module.kotlin.registerKotlinModule
2219
import reactor.core.publisher.Mono
2320

2421
/**
25-
* Default GraphQLClient implementation. Use this class to execute GraphQL queries against a standalone DGS or the gateway.
22+
* Default [GraphQLClient] implementation. Use this class to execute GraphQL queries against a standalone DGS or the gateway.
2623
* The value of this client is in it's JSON parsing of responses.
2724
* The client is not tied to any particular HTTP client library. The actual HTTP request code is provided by the user.
25+
* Note that if you want to use WebClient, there is the [WebClientGraphQLClient] available, which is simpler to use.
2826
*
2927
* Example:
3028
*
@@ -40,23 +38,9 @@ import reactor.core.publisher.Mono
4038
* return new HttpResponse(exchange.getStatusCodeValue(), exchange.getBody());
4139
* });
4240
*/
41+
@Deprecated("This has been replaced by [CustomGraphQLClient], [CustomReactiveGraphQLClient] and [WebClientGraphQLClient]")
4342
class DefaultGraphQLClient(private val url: String) : GraphQLClient, MonoGraphQLClient {
4443

45-
companion object {
46-
private val objectMapper: ObjectMapper = try {
47-
Class.forName("com.fasterxml.jackson.module.kotlin.KotlinModule\$Builder")
48-
ObjectMapper().registerModule(KotlinModule.Builder().nullIsSameAsDefault(true).build())
49-
} catch (ex: ClassNotFoundException) {
50-
ObjectMapper().registerKotlinModule()
51-
}
52-
private val defaultHeaders = mapOf(
53-
"Accept" to listOf("application/json"),
54-
"Content-type" to listOf("application/json")
55-
)
56-
57-
private data class Request(val query: String, val variables: Map<String, Any>, val operationName: String?)
58-
}
59-
6044
/**
6145
* Executes a query and returns a GraphQLResponse.
6246
* The actual HTTP request is done by an implementation of RequestExecutor, which is user provided.
@@ -75,10 +59,21 @@ class DefaultGraphQLClient(private val url: String) : GraphQLClient, MonoGraphQL
7559
operationName: String?,
7660
requestExecutor: RequestExecutor
7761
): GraphQLResponse {
78-
val serializedRequest = objectMapper.writeValueAsString(Request(query, variables, operationName))
62+
val serializedRequest = GraphQLClients.objectMapper.writeValueAsString(Request(query, variables, operationName))
63+
val response = requestExecutor.execute(url, GraphQLClients.defaultHeaders, serializedRequest)
64+
return GraphQLClients.handleResponse(response, serializedRequest, url)
65+
}
66+
67+
override fun executeQuery(query: String): GraphQLResponse {
68+
throw UnsupportedOperationException("Please move to [BlockingGraphQLClient] to use this method")
69+
}
70+
71+
override fun executeQuery(query: String, variables: Map<String, Any>): GraphQLResponse {
72+
throw UnsupportedOperationException("Please move to [BlockingGraphQLClient] to use this method")
73+
}
7974

80-
val response = requestExecutor.execute(url, defaultHeaders, serializedRequest)
81-
return handleResponse(response, serializedRequest)
75+
override fun executeQuery(query: String, variables: Map<String, Any>, operationName: String?): GraphQLResponse {
76+
throw UnsupportedOperationException("Please move to [BlockingGraphQLClient] to use this method")
8277
}
8378

8479
/**
@@ -97,9 +92,26 @@ class DefaultGraphQLClient(private val url: String) : GraphQLClient, MonoGraphQL
9792
variables: Map<String, Any>,
9893
requestExecutor: RequestExecutor
9994
): GraphQLResponse {
95+
@Suppress("DEPRECATION", "BlockingMethodInNonBlockingContext")
10096
return executeQuery(query, variables, null, requestExecutor)
10197
}
10298

99+
override fun reactiveExecuteQuery(query: String): Mono<GraphQLResponse> {
100+
throw UnsupportedOperationException("Please move to [CustomGraphQLClient] to use this method")
101+
}
102+
103+
override fun reactiveExecuteQuery(query: String, variables: Map<String, Any>): Mono<GraphQLResponse> {
104+
throw UnsupportedOperationException("Please move to [CustomGraphQLClient] to use this method")
105+
}
106+
107+
override fun reactiveExecuteQuery(
108+
query: String,
109+
variables: Map<String, Any>,
110+
operationName: String?
111+
): Mono<GraphQLResponse> {
112+
throw UnsupportedOperationException("Please move to [CustomGraphQLClient] to use this method")
113+
}
114+
103115
/**
104116
* Executes a query and returns a reactive Mono<GraphQLResponse>.
105117
* The actual HTTP request is done by an implementation of RequestExecutor, which is user provided.
@@ -137,20 +149,9 @@ class DefaultGraphQLClient(private val url: String) : GraphQLClient, MonoGraphQL
137149
operationName: String?,
138150
requestExecutor: MonoRequestExecutor
139151
): Mono<GraphQLResponse> {
140-
val serializedRequest = objectMapper.writeValueAsString(Request(query, variables, operationName))
141-
142-
return requestExecutor.execute(url, defaultHeaders, serializedRequest).map { response ->
143-
handleResponse(response, serializedRequest)
144-
}
145-
}
146-
147-
private fun handleResponse(response: HttpResponse, requestBody: String): GraphQLResponse {
148-
val (statusCode, body) = response
149-
val headers = response.headers
150-
if (statusCode !in 200..299) {
151-
throw GraphQLClientException(statusCode, url, body ?: "", requestBody)
152+
val serializedRequest = GraphQLClients.objectMapper.writeValueAsString(Request(query, variables, operationName))
153+
return requestExecutor.execute(url, GraphQLClients.defaultHeaders, serializedRequest).map { response ->
154+
GraphQLClients.handleResponse(response, serializedRequest, url)
152155
}
153-
154-
return GraphQLResponse(body ?: "", headers)
155156
}
156157
}

0 commit comments

Comments
 (0)