From b8c177b46a6a48fb028efc4dd82f12213641d235 Mon Sep 17 00:00:00 2001 From: Martin Bonnin Date: Tue, 26 Nov 2024 10:05:04 +0100 Subject: [PATCH] Added websockets support for ktor (#44) * Added websockets support for ktor * update docs * add changelog entry --- CHANGELOG.md | 1 + Writerside/topics/ktor.md | 29 ++++++------ .../api/apollo-execution-ktor.api | 2 + apollo-execution-ktor/build.gradle.kts | 1 + .../com/apollographql/execution/ktor/main.kt | 46 +++++++++++++++++++ .../api/apollo-execution-runtime.api | 4 +- .../api/apollo-execution-runtime.klib.api | 2 +- .../execution/internal/OperationContext.kt | 2 +- .../websocket/SubscriptionWebSocketHandler.kt | 10 ++-- gradle/libs.versions.toml | 1 + 10 files changed, 75 insertions(+), 23 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 142a9426..e4b31f27 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,7 @@ * Add `Ftv1Instrumentation` and `ApolloReportingInstrumentation` for respectively federated and monograph operation and fields insights. * Publish to GCS +* Add `apolloSubscriptionModule` for subscription support with Ktor. # Version 0.1.0 _2024-10-23_ diff --git a/Writerside/topics/ktor.md b/Writerside/topics/ktor.md index 3e09167b..6eddaca7 100644 --- a/Writerside/topics/ktor.md +++ b/Writerside/topics/ktor.md @@ -3,6 +3,8 @@ > See [sample-ktor](https://github.com/apollographql/apollo-kotlin-execution/tree/main/sample-ktor) for a project using the Ktor integration. +## Add apollo-execution-ktor to your project + To use the Ktor integration, add `apollo-execution-ktor` to your dependencies and a Ktor engine: ```kotlin @@ -15,28 +17,23 @@ dependencies { } ``` -`apollo-execution-ktor` provides an `apolloModule(ExecutableSchema)` function that adds a `/graphql` route to your application: - -```kotlin -embeddedServer(Netty, port = 8080) { - // /graphql route - apolloModule(ServiceExecutableSchemaBuilder().build()) -}.start(wait = true) -``` - -> `ServiceExecutableSchemaBuilder` is generated by the KSP processor. See ["Getting started"](getting-started.md) for more details. +`apollo-execution-ktor` provides 3 modules: -You can also opt in the Apollo Sandbox route by using `apolloSandboxModule()` +- `apolloModule(ExecutableSchema)` adds the main `/graphql` route for queries/mutations. +- `apolloSubscriptionModule(ExecutableSchema)` adds the `/subscription` route for subscriptions. +- `apolloSandboxModule(ExecutableSchema)` adds the `/sandbox/index.html` for the online IDE ```kotlin embeddedServer(Netty, port = 8080) { + val executableSchema = ServiceExecutableSchemaBuilder().build() + // /graphql route + apolloModule(executableSchema) + // /subscription route + apolloSubscriptionModule(executableSchema) // /sandbox/index.html route apolloSandboxModule() -} +}.start(wait = true) ``` -`apolloSandboxModule()` adds a `sandbox/index.html` route to your application. - -Open [`http://localhost:8080/sandbox/index.html`](http://localhost:8080/sandbox/index.html) and try out your API in the [Apollo sandbox](https://www.apollographql.com/docs/graphos/explorer/sandbox/) +> `ServiceExecutableSchemaBuilder` is generated by the KSP processor. See ["Getting started"](getting-started.md) for more details. -[![Apollo Sandbox](sandbox.png)](http://localhost:8080/sandbox/index.html) \ No newline at end of file diff --git a/apollo-execution-ktor/api/apollo-execution-ktor.api b/apollo-execution-ktor/api/apollo-execution-ktor.api index 7b9264ca..edb1471b 100644 --- a/apollo-execution-ktor/api/apollo-execution-ktor.api +++ b/apollo-execution-ktor/api/apollo-execution-ktor.api @@ -3,6 +3,8 @@ public final class com/apollographql/execution/ktor/MainKt { public static synthetic fun apolloModule$default (Lio/ktor/server/application/Application;Lcom/apollographql/execution/ExecutableSchema;Ljava/lang/String;Lkotlin/jvm/functions/Function1;ILjava/lang/Object;)V public static final fun apolloSandboxModule (Lio/ktor/server/application/Application;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)V public static synthetic fun apolloSandboxModule$default (Lio/ktor/server/application/Application;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;ILjava/lang/Object;)V + public static final fun apolloSubscriptionModule (Lio/ktor/server/application/Application;Lcom/apollographql/execution/ExecutableSchema;Ljava/lang/String;Lkotlin/jvm/functions/Function1;)V + public static synthetic fun apolloSubscriptionModule$default (Lio/ktor/server/application/Application;Lcom/apollographql/execution/ExecutableSchema;Ljava/lang/String;Lkotlin/jvm/functions/Function1;ILjava/lang/Object;)V public static final fun parseAsGraphQLRequest (Lio/ktor/server/request/ApplicationRequest;Lkotlin/coroutines/Continuation;)Ljava/lang/Object; public static final fun respondGraphQL (Lio/ktor/server/application/ApplicationCall;Lcom/apollographql/execution/ExecutableSchema;Lcom/apollographql/apollo/api/ExecutionContext;Lkotlin/jvm/functions/Function2;Lkotlin/coroutines/Continuation;)Ljava/lang/Object; public static synthetic fun respondGraphQL$default (Lio/ktor/server/application/ApplicationCall;Lcom/apollographql/execution/ExecutableSchema;Lcom/apollographql/apollo/api/ExecutionContext;Lkotlin/jvm/functions/Function2;Lkotlin/coroutines/Continuation;ILjava/lang/Object;)Ljava/lang/Object; diff --git a/apollo-execution-ktor/build.gradle.kts b/apollo-execution-ktor/build.gradle.kts index 77be2c98..719cd435 100644 --- a/apollo-execution-ktor/build.gradle.kts +++ b/apollo-execution-ktor/build.gradle.kts @@ -16,6 +16,7 @@ kotlin { implementation(libs.atomicfu) api(libs.coroutines) api(libs.ktor.server.core) + implementation(libs.ktor.server.websockets) } } getByName("jvmTest") { diff --git a/apollo-execution-ktor/src/commonMain/kotlin/com/apollographql/execution/ktor/main.kt b/apollo-execution-ktor/src/commonMain/kotlin/com/apollographql/execution/ktor/main.kt index 9da03a88..35951db8 100644 --- a/apollo-execution-ktor/src/commonMain/kotlin/com/apollographql/execution/ktor/main.kt +++ b/apollo-execution-ktor/src/commonMain/kotlin/com/apollographql/execution/ktor/main.kt @@ -2,6 +2,11 @@ package com.apollographql.execution.ktor import com.apollographql.apollo.api.ExecutionContext import com.apollographql.execution.* +import com.apollographql.execution.websocket.ConnectionInitAck +import com.apollographql.execution.websocket.SubscriptionWebSocketHandler +import com.apollographql.execution.websocket.WebSocketBinaryMessage +import com.apollographql.execution.websocket.WebSocketHandler +import com.apollographql.execution.websocket.WebSocketTextMessage import io.ktor.http.* import io.ktor.http.content.* import io.ktor.server.application.* @@ -9,7 +14,12 @@ import io.ktor.server.request.* import io.ktor.server.response.* import io.ktor.server.routing.* import io.ktor.server.util.* +import io.ktor.server.websocket.WebSockets +import io.ktor.server.websocket.webSocket import io.ktor.utils.io.* +import io.ktor.websocket.Frame +import io.ktor.websocket.readText +import kotlinx.coroutines.coroutineScope import okio.Buffer suspend fun ApplicationCall.respondGraphQL( @@ -86,6 +96,42 @@ fun Application.apolloModule( } } +fun Application.apolloSubscriptionModule( + executableSchema: ExecutableSchema, + path: String = "/subscription", + executionContext: (ApplicationRequest) -> ExecutionContext = { ExecutionContext.Empty } +) { + install(WebSockets) + + routing { + webSocket(path) { + coroutineScope { + val handler = SubscriptionWebSocketHandler( + executableSchema = executableSchema, + scope = this, + executionContext = executionContext(call.request), + sendMessage = { + when (it) { + is WebSocketBinaryMessage -> send(Frame.Binary(true, it.data)) + is WebSocketTextMessage -> send(Frame.Text(it.data)) + } + }, + connectionInitHandler = { + ConnectionInitAck + } + ) + + for (frame in incoming) { + if (frame !is Frame.Text) { + continue + } + handler.handleMessage(WebSocketTextMessage(frame.readText())) + } + } + } + } +} + fun Application.apolloSandboxModule( title: String = "API sandbox", sandboxPath: String = "/sandbox", diff --git a/apollo-execution-runtime/api/apollo-execution-runtime.api b/apollo-execution-runtime/api/apollo-execution-runtime.api index 09411fce..04714458 100644 --- a/apollo-execution-runtime/api/apollo-execution-runtime.api +++ b/apollo-execution-runtime/api/apollo-execution-runtime.api @@ -279,8 +279,8 @@ public abstract interface class com/apollographql/execution/websocket/Connection } public final class com/apollographql/execution/websocket/SubscriptionWebSocketHandler : com/apollographql/execution/websocket/WebSocketHandler { - public fun (Lcom/apollographql/execution/ExecutableSchema;Lkotlinx/coroutines/CoroutineScope;Lcom/apollographql/apollo/api/ExecutionContext;Lkotlin/jvm/functions/Function1;Lkotlin/jvm/functions/Function2;)V - public synthetic fun (Lcom/apollographql/execution/ExecutableSchema;Lkotlinx/coroutines/CoroutineScope;Lcom/apollographql/apollo/api/ExecutionContext;Lkotlin/jvm/functions/Function1;Lkotlin/jvm/functions/Function2;ILkotlin/jvm/internal/DefaultConstructorMarker;)V + public fun (Lcom/apollographql/execution/ExecutableSchema;Lkotlinx/coroutines/CoroutineScope;Lcom/apollographql/apollo/api/ExecutionContext;Lkotlin/jvm/functions/Function2;Lkotlin/jvm/functions/Function2;)V + public synthetic fun (Lcom/apollographql/execution/ExecutableSchema;Lkotlinx/coroutines/CoroutineScope;Lcom/apollographql/apollo/api/ExecutionContext;Lkotlin/jvm/functions/Function2;Lkotlin/jvm/functions/Function2;ILkotlin/jvm/internal/DefaultConstructorMarker;)V public final fun close ()V public fun handleMessage (Lcom/apollographql/execution/websocket/WebSocketMessage;)V } diff --git a/apollo-execution-runtime/api/apollo-execution-runtime.klib.api b/apollo-execution-runtime/api/apollo-execution-runtime.klib.api index 165f2e60..c0675af2 100644 --- a/apollo-execution-runtime/api/apollo-execution-runtime.klib.api +++ b/apollo-execution-runtime/api/apollo-execution-runtime.klib.api @@ -44,7 +44,7 @@ final class com.apollographql.execution.websocket/ConnectionInitError : com.apol final fun (): com.apollographql.apollo.api/Optional // com.apollographql.execution.websocket/ConnectionInitError.payload.|(){}[0] } final class com.apollographql.execution.websocket/SubscriptionWebSocketHandler : com.apollographql.execution.websocket/WebSocketHandler { // com.apollographql.execution.websocket/SubscriptionWebSocketHandler|null[0] - constructor (com.apollographql.execution/ExecutableSchema, kotlinx.coroutines/CoroutineScope, com.apollographql.apollo.api/ExecutionContext, kotlin/Function1, kotlin.coroutines/SuspendFunction1 = ...) // com.apollographql.execution.websocket/SubscriptionWebSocketHandler.|(com.apollographql.execution.ExecutableSchema;kotlinx.coroutines.CoroutineScope;com.apollographql.apollo.api.ExecutionContext;kotlin.Function1;kotlin.coroutines.SuspendFunction1){}[0] + constructor (com.apollographql.execution/ExecutableSchema, kotlinx.coroutines/CoroutineScope, com.apollographql.apollo.api/ExecutionContext, kotlin.coroutines/SuspendFunction1, kotlin.coroutines/SuspendFunction1 = ...) // com.apollographql.execution.websocket/SubscriptionWebSocketHandler.|(com.apollographql.execution.ExecutableSchema;kotlinx.coroutines.CoroutineScope;com.apollographql.apollo.api.ExecutionContext;kotlin.coroutines.SuspendFunction1;kotlin.coroutines.SuspendFunction1){}[0] final fun close() // com.apollographql.execution.websocket/SubscriptionWebSocketHandler.close|close(){}[0] final fun handleMessage(com.apollographql.execution.websocket/WebSocketMessage) // com.apollographql.execution.websocket/SubscriptionWebSocketHandler.handleMessage|handleMessage(com.apollographql.execution.websocket.WebSocketMessage){}[0] } diff --git a/apollo-execution-runtime/src/commonMain/kotlin/com/apollographql/execution/internal/OperationContext.kt b/apollo-execution-runtime/src/commonMain/kotlin/com/apollographql/execution/internal/OperationContext.kt index a3df409b..445b0dc9 100644 --- a/apollo-execution-runtime/src/commonMain/kotlin/com/apollographql/execution/internal/OperationContext.kt +++ b/apollo-execution-runtime/src/commonMain/kotlin/com/apollographql/execution/internal/OperationContext.kt @@ -76,7 +76,7 @@ internal class OperationContext( "query" -> queryRoot?.resolveRoot() "mutation" -> mutationRoot?.resolveRoot() "subscription" -> { - return graphqlErrorResponse("Use executeSubscription() to execute subscriptions") + return graphqlErrorResponse("Use subscribe() to execute subscriptions") } else -> { diff --git a/apollo-execution-runtime/src/commonMain/kotlin/com/apollographql/execution/websocket/SubscriptionWebSocketHandler.kt b/apollo-execution-runtime/src/commonMain/kotlin/com/apollographql/execution/websocket/SubscriptionWebSocketHandler.kt index 67d177ac..a16f99ee 100644 --- a/apollo-execution-runtime/src/commonMain/kotlin/com/apollographql/execution/websocket/SubscriptionWebSocketHandler.kt +++ b/apollo-execution-runtime/src/commonMain/kotlin/com/apollographql/execution/websocket/SubscriptionWebSocketHandler.kt @@ -32,7 +32,7 @@ class SubscriptionWebSocketHandler( private val executableSchema: ExecutableSchema, private val scope: CoroutineScope, private val executionContext: ExecutionContext, - private val sendMessage: (WebSocketMessage) -> Unit, + private val sendMessage: suspend (WebSocketMessage) -> Unit, private val connectionInitHandler: ConnectionInitHandler = { ConnectionInitAck }, ) : WebSocketHandler { private val lock = reentrantLock() @@ -67,7 +67,9 @@ class SubscriptionWebSocketHandler( activeSubscriptions.containsKey(clientMessage.id) } if (isActive) { - sendMessage(SubscriptionWebsocketError(id = clientMessage.id, error = Error.Builder("Subscription ${clientMessage.id} is already active").build()).toWsMessage()) + scope.launch { + sendMessage(SubscriptionWebsocketError(id = clientMessage.id, error = Error.Builder("Subscription ${clientMessage.id} is already active").build()).toWsMessage()) + } return } @@ -107,7 +109,9 @@ class SubscriptionWebSocketHandler( } is SubscriptionWebsocketClientMessageParseError -> { - sendMessage(SubscriptionWebsocketError(null, Error.Builder("Cannot handle message (${clientMessage.message})").build()).toWsMessage()) + scope.launch { + sendMessage(SubscriptionWebsocketError(null, Error.Builder("Cannot handle message (${clientMessage.message})").build()).toWsMessage()) + } } } } diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index a907ff2a..3c7e3886 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -23,6 +23,7 @@ ktor-server-netty = { group = "io.ktor", name = "ktor-server-netty", version.ref ktor-server-core = { group = "io.ktor", name = "ktor-server-core", version.ref ="ktor" } ktor-server-cors = { group = "io.ktor", name = "ktor-server-cors", version.ref ="ktor" } ktor-client-cio = { module = "io.ktor:ktor-client-cio", version.ref = "ktor" } +ktor-server-websockets = { group = "io.ktor", name = "ktor-server-websockets", version.ref ="ktor" } slf4j-simple = "org.slf4j:slf4j-simple:2.0.13" slf4j-nop = "org.slf4j:slf4j-nop:2.0.13" ksp-api = { module = "com.google.devtools.ksp:symbol-processing-api", version.ref = "ksp" }