Introduce Jackson 3 support for spring-websocket
This commit introduces a JacksonJsonSockJsMessageCodec Jackson 3 variant of Jackson2SockJsMessageCodec. See gh-33798
This commit is contained in:
parent
a0ed3f052e
commit
ac3c1b8762
|
@ -17,6 +17,7 @@ dependencies {
|
||||||
}
|
}
|
||||||
optional("org.eclipse.jetty.websocket:jetty-websocket-jetty-api")
|
optional("org.eclipse.jetty.websocket:jetty-websocket-jetty-api")
|
||||||
optional("org.eclipse.jetty:jetty-client")
|
optional("org.eclipse.jetty:jetty-client")
|
||||||
|
optional("tools.jackson.core:jackson-databind")
|
||||||
testImplementation(testFixtures(project(":spring-core")))
|
testImplementation(testFixtures(project(":spring-core")))
|
||||||
testImplementation(testFixtures(project(":spring-web")))
|
testImplementation(testFixtures(project(":spring-web")))
|
||||||
testImplementation("io.projectreactor.netty:reactor-netty-http")
|
testImplementation("io.projectreactor.netty:reactor-netty-http")
|
||||||
|
|
|
@ -23,8 +23,6 @@ import org.springframework.beans.factory.config.CustomScopeConfigurer;
|
||||||
import org.springframework.context.ApplicationContext;
|
import org.springframework.context.ApplicationContext;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.core.task.TaskExecutor;
|
import org.springframework.core.task.TaskExecutor;
|
||||||
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
|
|
||||||
import org.springframework.messaging.converter.MappingJackson2MessageConverter;
|
|
||||||
import org.springframework.messaging.simp.SimpMessagingTemplate;
|
import org.springframework.messaging.simp.SimpMessagingTemplate;
|
||||||
import org.springframework.messaging.simp.SimpSessionScope;
|
import org.springframework.messaging.simp.SimpSessionScope;
|
||||||
import org.springframework.messaging.simp.annotation.support.SimpAnnotationMethodMessageHandler;
|
import org.springframework.messaging.simp.annotation.support.SimpAnnotationMethodMessageHandler;
|
||||||
|
@ -162,17 +160,4 @@ public abstract class WebSocketMessageBrokerConfigurationSupport extends Abstrac
|
||||||
return stats;
|
return stats;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
protected MappingJackson2MessageConverter createJacksonConverter() {
|
|
||||||
MappingJackson2MessageConverter messageConverter = super.createJacksonConverter();
|
|
||||||
// Use Jackson builder in order to have well-known modules registered automatically.
|
|
||||||
Jackson2ObjectMapperBuilder builder = Jackson2ObjectMapperBuilder.json();
|
|
||||||
ApplicationContext applicationContext = getApplicationContext();
|
|
||||||
if (applicationContext != null) {
|
|
||||||
builder.applicationContext(applicationContext);
|
|
||||||
}
|
|
||||||
messageConverter.setObjectMapper(builder.build());
|
|
||||||
return messageConverter;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2024 the original author or authors.
|
* Copyright 2002-2025 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -47,6 +47,7 @@ import org.springframework.web.socket.sockjs.frame.SockJsMessageCodec;
|
||||||
*
|
*
|
||||||
* @author Rossen Stoyanchev
|
* @author Rossen Stoyanchev
|
||||||
* @author Juergen Hoeller
|
* @author Juergen Hoeller
|
||||||
|
* @author Sebastien Deleuze
|
||||||
* @since 4.1
|
* @since 4.1
|
||||||
*/
|
*/
|
||||||
public abstract class AbstractClientSockJsSession implements WebSocketSession {
|
public abstract class AbstractClientSockJsSession implements WebSocketSession {
|
||||||
|
@ -268,7 +269,7 @@ public abstract class AbstractClientSockJsSession implements WebSocketSession {
|
||||||
try {
|
try {
|
||||||
messages = getMessageCodec().decode(frameData);
|
messages = getMessageCodec().decode(frameData);
|
||||||
}
|
}
|
||||||
catch (IOException ex) {
|
catch (RuntimeException | IOException ex) {
|
||||||
if (logger.isErrorEnabled()) {
|
if (logger.isErrorEnabled()) {
|
||||||
logger.error("Failed to decode data for SockJS \"message\" frame: " + frame + " in " + this, ex);
|
logger.error("Failed to decode data for SockJS \"message\" frame: " + frame + " in " + this, ex);
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,6 +40,7 @@ import org.springframework.web.socket.WebSocketHttpHeaders;
|
||||||
import org.springframework.web.socket.WebSocketSession;
|
import org.springframework.web.socket.WebSocketSession;
|
||||||
import org.springframework.web.socket.client.WebSocketClient;
|
import org.springframework.web.socket.client.WebSocketClient;
|
||||||
import org.springframework.web.socket.sockjs.frame.Jackson2SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.Jackson2SockJsMessageCodec;
|
||||||
|
import org.springframework.web.socket.sockjs.frame.JacksonJsonSockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.frame.SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.SockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.transport.TransportType;
|
import org.springframework.web.socket.sockjs.transport.TransportType;
|
||||||
import org.springframework.web.util.UriComponentsBuilder;
|
import org.springframework.web.util.UriComponentsBuilder;
|
||||||
|
@ -62,6 +63,9 @@ import org.springframework.web.util.UriComponentsBuilder;
|
||||||
*/
|
*/
|
||||||
public class SockJsClient implements WebSocketClient, Lifecycle {
|
public class SockJsClient implements WebSocketClient, Lifecycle {
|
||||||
|
|
||||||
|
private static final boolean jacksonPresent = ClassUtils.isPresent(
|
||||||
|
"tools.jackson.databind.ObjectMapper", SockJsClient.class.getClassLoader());
|
||||||
|
|
||||||
private static final boolean jackson2Present = ClassUtils.isPresent(
|
private static final boolean jackson2Present = ClassUtils.isPresent(
|
||||||
"com.fasterxml.jackson.databind.ObjectMapper", SockJsClient.class.getClassLoader());
|
"com.fasterxml.jackson.databind.ObjectMapper", SockJsClient.class.getClassLoader());
|
||||||
|
|
||||||
|
@ -97,7 +101,10 @@ public class SockJsClient implements WebSocketClient, Lifecycle {
|
||||||
Assert.notEmpty(transports, "No transports provided");
|
Assert.notEmpty(transports, "No transports provided");
|
||||||
this.transports = new ArrayList<>(transports);
|
this.transports = new ArrayList<>(transports);
|
||||||
this.infoReceiver = initInfoReceiver(transports);
|
this.infoReceiver = initInfoReceiver(transports);
|
||||||
if (jackson2Present) {
|
if (jacksonPresent) {
|
||||||
|
this.messageCodec = new JacksonJsonSockJsMessageCodec();
|
||||||
|
}
|
||||||
|
else if (jackson2Present) {
|
||||||
this.messageCodec = new Jackson2SockJsMessageCodec();
|
this.messageCodec = new Jackson2SockJsMessageCodec();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,78 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2025 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.web.socket.sockjs.frame;
|
||||||
|
|
||||||
|
import java.io.InputStream;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.core.io.JsonStringEncoder;
|
||||||
|
import org.jspecify.annotations.Nullable;
|
||||||
|
import tools.jackson.databind.ObjectMapper;
|
||||||
|
import tools.jackson.databind.cfg.MapperBuilder;
|
||||||
|
import tools.jackson.databind.json.JsonMapper;
|
||||||
|
|
||||||
|
import org.springframework.util.Assert;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A Jackson 3.x codec for encoding and decoding SockJS messages.
|
||||||
|
*
|
||||||
|
* <p>The default constructor loads {@link tools.jackson.databind.JacksonModule}s
|
||||||
|
* found by {@link MapperBuilder#findModules(ClassLoader)}.
|
||||||
|
*
|
||||||
|
* @author Sebastien Deleuze
|
||||||
|
* @since 7.0
|
||||||
|
*/
|
||||||
|
public class JacksonJsonSockJsMessageCodec extends AbstractSockJsMessageCodec {
|
||||||
|
|
||||||
|
private final ObjectMapper objectMapper;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a new instance with a {@link JsonMapper} customized with the
|
||||||
|
* {@link tools.jackson.databind.JacksonModule}s found by
|
||||||
|
* {@link MapperBuilder#findModules(ClassLoader)}.
|
||||||
|
*/
|
||||||
|
public JacksonJsonSockJsMessageCodec() {
|
||||||
|
this.objectMapper = JsonMapper.builder().findAndAddModules(JacksonJsonSockJsMessageCodec.class.getClassLoader()).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a new instance with the provided {@link ObjectMapper}.
|
||||||
|
* @see JsonMapper#builder()
|
||||||
|
* @see MapperBuilder#findAndAddModules(ClassLoader)
|
||||||
|
*/
|
||||||
|
public JacksonJsonSockJsMessageCodec(ObjectMapper objectMapper) {
|
||||||
|
Assert.notNull(objectMapper, "ObjectMapper must not be null");
|
||||||
|
this.objectMapper = objectMapper;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String @Nullable [] decode(String content) {
|
||||||
|
return this.objectMapper.readValue(content, String[].class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String @Nullable [] decodeInputStream(InputStream content) {
|
||||||
|
return this.objectMapper.readValue(content, String[].class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected char[] applyJsonQuoting(String content) {
|
||||||
|
return JsonStringEncoder.getInstance().quoteAsString(content);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -51,6 +51,7 @@ import org.springframework.web.socket.server.HandshakeInterceptor;
|
||||||
import org.springframework.web.socket.server.support.HandshakeInterceptorChain;
|
import org.springframework.web.socket.server.support.HandshakeInterceptorChain;
|
||||||
import org.springframework.web.socket.sockjs.SockJsException;
|
import org.springframework.web.socket.sockjs.SockJsException;
|
||||||
import org.springframework.web.socket.sockjs.frame.Jackson2SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.Jackson2SockJsMessageCodec;
|
||||||
|
import org.springframework.web.socket.sockjs.frame.JacksonJsonSockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.frame.SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.SockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.support.AbstractSockJsService;
|
import org.springframework.web.socket.sockjs.support.AbstractSockJsService;
|
||||||
|
|
||||||
|
@ -70,6 +71,9 @@ import org.springframework.web.socket.sockjs.support.AbstractSockJsService;
|
||||||
*/
|
*/
|
||||||
public class TransportHandlingSockJsService extends AbstractSockJsService implements SockJsServiceConfig, Lifecycle {
|
public class TransportHandlingSockJsService extends AbstractSockJsService implements SockJsServiceConfig, Lifecycle {
|
||||||
|
|
||||||
|
private static final boolean jacksonPresent = ClassUtils.isPresent(
|
||||||
|
"tools.jackson.databind.ObjectMapper", TransportHandlingSockJsService.class.getClassLoader());
|
||||||
|
|
||||||
private static final boolean jackson2Present = ClassUtils.isPresent(
|
private static final boolean jackson2Present = ClassUtils.isPresent(
|
||||||
"com.fasterxml.jackson.databind.ObjectMapper", TransportHandlingSockJsService.class.getClassLoader());
|
"com.fasterxml.jackson.databind.ObjectMapper", TransportHandlingSockJsService.class.getClassLoader());
|
||||||
|
|
||||||
|
@ -118,7 +122,10 @@ public class TransportHandlingSockJsService extends AbstractSockJsService implem
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (jackson2Present) {
|
if (jacksonPresent) {
|
||||||
|
this.messageCodec = new JacksonJsonSockJsMessageCodec();
|
||||||
|
}
|
||||||
|
else if (jackson2Present) {
|
||||||
this.messageCodec = new Jackson2SockJsMessageCodec();
|
this.messageCodec = new Jackson2SockJsMessageCodec();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,7 +29,7 @@ import org.springframework.web.socket.TextMessage;
|
||||||
import org.springframework.web.socket.WebSocketExtension;
|
import org.springframework.web.socket.WebSocketExtension;
|
||||||
import org.springframework.web.socket.WebSocketHandler;
|
import org.springframework.web.socket.WebSocketHandler;
|
||||||
import org.springframework.web.socket.WebSocketSession;
|
import org.springframework.web.socket.WebSocketSession;
|
||||||
import org.springframework.web.socket.sockjs.frame.Jackson2SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.JacksonJsonSockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.frame.SockJsFrame;
|
import org.springframework.web.socket.sockjs.frame.SockJsFrame;
|
||||||
import org.springframework.web.socket.sockjs.transport.TransportType;
|
import org.springframework.web.socket.sockjs.transport.TransportType;
|
||||||
|
|
||||||
|
@ -48,7 +48,7 @@ import static org.mockito.Mockito.verifyNoMoreInteractions;
|
||||||
*/
|
*/
|
||||||
class ClientSockJsSessionTests {
|
class ClientSockJsSessionTests {
|
||||||
|
|
||||||
private static final Jackson2SockJsMessageCodec CODEC = new Jackson2SockJsMessageCodec();
|
private static final JacksonJsonSockJsMessageCodec CODEC = new JacksonJsonSockJsMessageCodec();
|
||||||
|
|
||||||
private WebSocketHandler handler = mock();
|
private WebSocketHandler handler = mock();
|
||||||
|
|
||||||
|
|
|
@ -30,7 +30,7 @@ import org.mockito.ArgumentCaptor;
|
||||||
import org.springframework.http.HttpHeaders;
|
import org.springframework.http.HttpHeaders;
|
||||||
import org.springframework.scheduling.TaskScheduler;
|
import org.springframework.scheduling.TaskScheduler;
|
||||||
import org.springframework.web.socket.WebSocketSession;
|
import org.springframework.web.socket.WebSocketSession;
|
||||||
import org.springframework.web.socket.sockjs.frame.Jackson2SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.JacksonJsonSockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.transport.TransportType;
|
import org.springframework.web.socket.sockjs.transport.TransportType;
|
||||||
|
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
@ -47,7 +47,7 @@ import static org.mockito.Mockito.verifyNoMoreInteractions;
|
||||||
*/
|
*/
|
||||||
class DefaultTransportRequestTests {
|
class DefaultTransportRequestTests {
|
||||||
|
|
||||||
private final Jackson2SockJsMessageCodec CODEC = new Jackson2SockJsMessageCodec();
|
private final JacksonJsonSockJsMessageCodec CODEC = new JacksonJsonSockJsMessageCodec();
|
||||||
|
|
||||||
private CompletableFuture<WebSocketSession> connectFuture = new CompletableFuture<>();
|
private CompletableFuture<WebSocketSession> connectFuture = new CompletableFuture<>();
|
||||||
|
|
||||||
|
|
|
@ -50,7 +50,7 @@ import org.springframework.web.socket.CloseStatus;
|
||||||
import org.springframework.web.socket.TextMessage;
|
import org.springframework.web.socket.TextMessage;
|
||||||
import org.springframework.web.socket.WebSocketHandler;
|
import org.springframework.web.socket.WebSocketHandler;
|
||||||
import org.springframework.web.socket.WebSocketSession;
|
import org.springframework.web.socket.WebSocketSession;
|
||||||
import org.springframework.web.socket.sockjs.frame.Jackson2SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.JacksonJsonSockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.frame.SockJsFrame;
|
import org.springframework.web.socket.sockjs.frame.SockJsFrame;
|
||||||
import org.springframework.web.socket.sockjs.transport.TransportType;
|
import org.springframework.web.socket.sockjs.transport.TransportType;
|
||||||
|
|
||||||
|
@ -69,7 +69,7 @@ import static org.mockito.Mockito.verifyNoMoreInteractions;
|
||||||
*/
|
*/
|
||||||
class RestTemplateXhrTransportTests {
|
class RestTemplateXhrTransportTests {
|
||||||
|
|
||||||
private static final Jackson2SockJsMessageCodec CODEC = new Jackson2SockJsMessageCodec();
|
private static final JacksonJsonSockJsMessageCodec CODEC = new JacksonJsonSockJsMessageCodec();
|
||||||
|
|
||||||
private final WebSocketHandler webSocketHandler = mock();
|
private final WebSocketHandler webSocketHandler = mock();
|
||||||
|
|
||||||
|
@ -114,7 +114,7 @@ class RestTemplateXhrTransportTests {
|
||||||
Message<byte[]> message = MessageBuilder.createMessage("body".getBytes(UTF_8), headers);
|
Message<byte[]> message = MessageBuilder.createMessage("body".getBytes(UTF_8), headers);
|
||||||
byte[] bytes = new StompEncoder().encode(message);
|
byte[] bytes = new StompEncoder().encode(message);
|
||||||
TextMessage textMessage = new TextMessage(bytes);
|
TextMessage textMessage = new TextMessage(bytes);
|
||||||
SockJsFrame frame = SockJsFrame.messageFrame(new Jackson2SockJsMessageCodec(), textMessage.getPayload());
|
SockJsFrame frame = SockJsFrame.messageFrame(new JacksonJsonSockJsMessageCodec(), textMessage.getPayload());
|
||||||
|
|
||||||
String body = """
|
String body = """
|
||||||
o
|
o
|
||||||
|
|
|
@ -49,7 +49,7 @@ class SockJsFrameTests {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void messageArrayFrame() {
|
void messageArrayFrame() {
|
||||||
SockJsFrame frame = SockJsFrame.messageFrame(new Jackson2SockJsMessageCodec(), "m1", "m2");
|
SockJsFrame frame = SockJsFrame.messageFrame(new JacksonJsonSockJsMessageCodec(), "m1", "m2");
|
||||||
|
|
||||||
assertThat(frame.getContent()).isEqualTo("a[\"m1\",\"m2\"]");
|
assertThat(frame.getContent()).isEqualTo("a[\"m1\",\"m2\"]");
|
||||||
assertThat(frame.getType()).isEqualTo(SockJsFrameType.MESSAGE);
|
assertThat(frame.getType()).isEqualTo(SockJsFrameType.MESSAGE);
|
||||||
|
|
|
@ -18,7 +18,7 @@ package org.springframework.web.socket.sockjs.transport.session;
|
||||||
|
|
||||||
import org.springframework.scheduling.TaskScheduler;
|
import org.springframework.scheduling.TaskScheduler;
|
||||||
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
|
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
|
||||||
import org.springframework.web.socket.sockjs.frame.Jackson2SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.JacksonJsonSockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.frame.SockJsMessageCodec;
|
import org.springframework.web.socket.sockjs.frame.SockJsMessageCodec;
|
||||||
import org.springframework.web.socket.sockjs.transport.SockJsServiceConfig;
|
import org.springframework.web.socket.sockjs.transport.SockJsServiceConfig;
|
||||||
|
|
||||||
|
@ -33,7 +33,7 @@ public class StubSockJsServiceConfig implements SockJsServiceConfig {
|
||||||
|
|
||||||
private TaskScheduler taskScheduler = new ThreadPoolTaskScheduler();
|
private TaskScheduler taskScheduler = new ThreadPoolTaskScheduler();
|
||||||
|
|
||||||
private SockJsMessageCodec messageCodec = new Jackson2SockJsMessageCodec();
|
private SockJsMessageCodec messageCodec = new JacksonJsonSockJsMessageCodec();
|
||||||
|
|
||||||
private int httpMessageCacheSize = 100;
|
private int httpMessageCacheSize = 100;
|
||||||
|
|
||||||
|
|
|
@ -16,13 +16,13 @@
|
||||||
|
|
||||||
package org.springframework.web.socket.sockjs.transport.session;
|
package org.springframework.web.socket.sockjs.transport.session;
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.junit.jupiter.api.BeforeEach;
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
|
import tools.jackson.core.JacksonException;
|
||||||
|
|
||||||
import org.springframework.web.socket.CloseStatus;
|
import org.springframework.web.socket.CloseStatus;
|
||||||
import org.springframework.web.socket.TextMessage;
|
import org.springframework.web.socket.TextMessage;
|
||||||
|
@ -118,7 +118,7 @@ class WebSocketServerSockJsSessionTests extends AbstractSockJsSessionTests<TestW
|
||||||
this.session.handleMessage(message, this.webSocketSession);
|
this.session.handleMessage(message, this.webSocketSession);
|
||||||
|
|
||||||
this.session.isClosed();
|
this.session.isClosed();
|
||||||
verify(this.webSocketHandler).handleTransportError(same(this.session), any(IOException.class));
|
verify(this.webSocketHandler).handleTransportError(same(this.session), any(JacksonException.class));
|
||||||
verifyNoMoreInteractions(this.webSocketHandler);
|
verifyNoMoreInteractions(this.webSocketHandler);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue