forked from quarkusio/quarkus
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
WebSockets Next: create a new event loop context for each client
- we want to avoid a situation where if multiple clients/connections are created in a row, the same event loop is used and so writing/receiving messages is de-facto serialized
- Loading branch information
Showing
4 changed files
with
343 additions
and
93 deletions.
There are no files selected for viewing
89 changes: 89 additions & 0 deletions
89
...yment/src/test/java/io/quarkus/websockets/next/test/client/BasicConnectorContextTest.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,89 @@ | ||
package io.quarkus.websockets.next.test.client; | ||
|
||
import static org.junit.jupiter.api.Assertions.assertEquals; | ||
import static org.junit.jupiter.api.Assertions.assertTrue; | ||
|
||
import java.net.URI; | ||
import java.util.Set; | ||
import java.util.concurrent.ConcurrentHashMap; | ||
import java.util.concurrent.CountDownLatch; | ||
import java.util.concurrent.TimeUnit; | ||
|
||
import org.junit.jupiter.api.Test; | ||
import org.junit.jupiter.api.extension.RegisterExtension; | ||
|
||
import io.quarkus.test.QuarkusUnitTest; | ||
import io.quarkus.test.common.http.TestHTTPResource; | ||
import io.quarkus.websockets.next.BasicWebSocketConnector; | ||
import io.quarkus.websockets.next.OnClose; | ||
import io.quarkus.websockets.next.OnOpen; | ||
import io.quarkus.websockets.next.WebSocket; | ||
import io.quarkus.websockets.next.WebSocketClientConnection; | ||
|
||
public class BasicConnectorContextTest { | ||
|
||
@RegisterExtension | ||
public static final QuarkusUnitTest test = new QuarkusUnitTest() | ||
.withApplicationRoot(root -> { | ||
root.addClasses(ServerEndpoint.class); | ||
}); | ||
|
||
@TestHTTPResource("/end") | ||
URI uri; | ||
|
||
static final CountDownLatch MESSAGE_LATCH = new CountDownLatch(2); | ||
|
||
static final Set<String> THREADS = ConcurrentHashMap.newKeySet(); | ||
|
||
static final CountDownLatch CLOSED_LATCH = new CountDownLatch(2); | ||
|
||
@Test | ||
void testClient() throws InterruptedException { | ||
BasicWebSocketConnector connector = BasicWebSocketConnector.create(); | ||
connector | ||
.executionModel(BasicWebSocketConnector.ExecutionModel.NON_BLOCKING) | ||
.onTextMessage((c, m) -> { | ||
String thread = Thread.currentThread().getName(); | ||
THREADS.add(thread); | ||
MESSAGE_LATCH.countDown(); | ||
}) | ||
.onClose((c, cr) -> { | ||
CLOSED_LATCH.countDown(); | ||
}) | ||
.baseUri(uri); | ||
WebSocketClientConnection conn1 = connector.connectAndAwait(); | ||
WebSocketClientConnection conn2 = connector.connectAndAwait(); | ||
assertTrue(MESSAGE_LATCH.await(10, TimeUnit.SECONDS)); | ||
if (Runtime.getRuntime().availableProcessors() > 1) { | ||
// Each client should be executed on a dedicated event loop thread | ||
assertEquals(2, THREADS.size()); | ||
} else { | ||
// Single core - the event pool is shared | ||
// Due to some CI weirdness it might happen that the system incorrectly reports single core | ||
// Therefore, the assert checks if the number of threads used is >= 1 | ||
assertTrue(THREADS.size() >= 1); | ||
} | ||
conn1.closeAndAwait(); | ||
conn2.closeAndAwait(); | ||
assertTrue(ServerEndpoint.CLOSED_LATCH.await(5, TimeUnit.SECONDS)); | ||
assertTrue(CLOSED_LATCH.await(5, TimeUnit.SECONDS)); | ||
} | ||
|
||
@WebSocket(path = "/end") | ||
public static class ServerEndpoint { | ||
|
||
static final CountDownLatch CLOSED_LATCH = new CountDownLatch(1); | ||
|
||
@OnOpen | ||
String open() { | ||
return "Hello!"; | ||
} | ||
|
||
@OnClose | ||
void close() { | ||
CLOSED_LATCH.countDown(); | ||
} | ||
|
||
} | ||
|
||
} |
104 changes: 104 additions & 0 deletions
104
...xt/deployment/src/test/java/io/quarkus/websockets/next/test/client/ClientContextTest.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,104 @@ | ||
package io.quarkus.websockets.next.test.client; | ||
|
||
import static org.junit.jupiter.api.Assertions.assertEquals; | ||
import static org.junit.jupiter.api.Assertions.assertTrue; | ||
|
||
import java.net.URI; | ||
import java.util.Set; | ||
import java.util.concurrent.ConcurrentHashMap; | ||
import java.util.concurrent.CountDownLatch; | ||
import java.util.concurrent.TimeUnit; | ||
|
||
import jakarta.inject.Inject; | ||
|
||
import org.junit.jupiter.api.Test; | ||
import org.junit.jupiter.api.extension.RegisterExtension; | ||
|
||
import io.quarkus.test.QuarkusUnitTest; | ||
import io.quarkus.test.common.http.TestHTTPResource; | ||
import io.quarkus.websockets.next.OnClose; | ||
import io.quarkus.websockets.next.OnOpen; | ||
import io.quarkus.websockets.next.OnTextMessage; | ||
import io.quarkus.websockets.next.WebSocket; | ||
import io.quarkus.websockets.next.WebSocketClient; | ||
import io.quarkus.websockets.next.WebSocketClientConnection; | ||
import io.quarkus.websockets.next.WebSocketConnector; | ||
import io.smallrye.mutiny.Uni; | ||
|
||
public class ClientContextTest { | ||
|
||
@RegisterExtension | ||
public static final QuarkusUnitTest test = new QuarkusUnitTest() | ||
.withApplicationRoot(root -> { | ||
root.addClasses(ServerEndpoint.class, ClientEndpoint.class); | ||
}); | ||
|
||
@Inject | ||
WebSocketConnector<ClientEndpoint> connector; | ||
|
||
@TestHTTPResource("/") | ||
URI uri; | ||
|
||
@Test | ||
void testClient() throws InterruptedException { | ||
connector.baseUri(uri); | ||
WebSocketClientConnection conn1 = connector.connectAndAwait(); | ||
WebSocketClientConnection conn2 = connector.connectAndAwait(); | ||
assertTrue(ClientEndpoint.MESSAGE_LATCH.await(10, TimeUnit.SECONDS)); | ||
if (Runtime.getRuntime().availableProcessors() > 1) { | ||
// Each client should be executed on a dedicated event loop thread | ||
assertEquals(2, ClientEndpoint.THREADS.size()); | ||
} else { | ||
// Single core - the event pool is shared | ||
// Due to some CI weirdness it might happen that the system incorrectly reports single core | ||
// Therefore, the assert checks if the number of threads used is >= 1 | ||
assertTrue(ClientEndpoint.THREADS.size() >= 1); | ||
} | ||
conn1.closeAndAwait(); | ||
conn2.closeAndAwait(); | ||
assertTrue(ClientEndpoint.CLOSED_LATCH.await(5, TimeUnit.SECONDS)); | ||
assertTrue(ServerEndpoint.CLOSED_LATCH.await(5, TimeUnit.SECONDS)); | ||
} | ||
|
||
@WebSocket(path = "/end") | ||
public static class ServerEndpoint { | ||
|
||
static final CountDownLatch CLOSED_LATCH = new CountDownLatch(1); | ||
|
||
@OnOpen | ||
String open() { | ||
return "Hello!"; | ||
} | ||
|
||
@OnClose | ||
void close() { | ||
CLOSED_LATCH.countDown(); | ||
} | ||
|
||
} | ||
|
||
@WebSocketClient(path = "/end") | ||
public static class ClientEndpoint { | ||
|
||
static final CountDownLatch MESSAGE_LATCH = new CountDownLatch(2); | ||
|
||
static final Set<String> THREADS = ConcurrentHashMap.newKeySet(); | ||
|
||
static final CountDownLatch CLOSED_LATCH = new CountDownLatch(2); | ||
|
||
@OnTextMessage | ||
Uni<Void> onMessage(String message) { | ||
String thread = Thread.currentThread().getName(); | ||
THREADS.add(thread); | ||
MESSAGE_LATCH.countDown(); | ||
return Uni.createFrom().voidItem(); | ||
} | ||
|
||
@OnClose | ||
void close() { | ||
CLOSED_LATCH.countDown(); | ||
} | ||
|
||
} | ||
|
||
} |
Oops, something went wrong.