Fix #437 Avoid NPE on receiving empty message though MessageHandler.Partial
Signed-off-by: Pavel Baranchikov <p.a.baranchikov@gmail.com>
This commit is contained in:
parent
df9de50f96
commit
beccc5ee80
|
@ -51,14 +51,16 @@ public class BinaryPartialMessage implements MessageAppender
|
||||||
// Supported Partial<> Type #1: ByteBuffer
|
// Supported Partial<> Type #1: ByteBuffer
|
||||||
if (msgWrapper.isMessageType(ByteBuffer.class))
|
if (msgWrapper.isMessageType(ByteBuffer.class))
|
||||||
{
|
{
|
||||||
partialHandler.onMessage(payload.slice(),isLast);
|
partialHandler.onMessage(payload==null?ByteBuffer.allocate(0):
|
||||||
|
payload.slice(),isLast);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Supported Partial<> Type #2: byte[]
|
// Supported Partial<> Type #2: byte[]
|
||||||
if (msgWrapper.isMessageType(byte[].class))
|
if (msgWrapper.isMessageType(byte[].class))
|
||||||
{
|
{
|
||||||
partialHandler.onMessage(BufferUtil.toArray(payload),isLast);
|
partialHandler.onMessage(payload==null?new byte[0]:
|
||||||
|
BufferUtil.toArray(payload),isLast);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,308 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2016 Mort Bay Consulting Pty. Ltd.
|
||||||
|
// ------------------------------------------------------------------------
|
||||||
|
// All rights reserved. This program and the accompanying materials
|
||||||
|
// are made available under the terms of the Eclipse Public License v1.0
|
||||||
|
// and Apache License v2.0 which accompanies this distribution.
|
||||||
|
//
|
||||||
|
// The Eclipse Public License is available at
|
||||||
|
// http://www.eclipse.org/legal/epl-v10.html
|
||||||
|
//
|
||||||
|
// The Apache License v2.0 is available at
|
||||||
|
// http://www.opensource.org/licenses/apache2.0.php
|
||||||
|
//
|
||||||
|
// You may elect to redistribute this code under either of these licenses.
|
||||||
|
// ========================================================================
|
||||||
|
//
|
||||||
|
|
||||||
|
package org.eclipse.jetty.websocket.jsr356;
|
||||||
|
|
||||||
|
import static org.hamcrest.Matchers.instanceOf;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import javax.websocket.ContainerProvider;
|
||||||
|
import javax.websocket.Endpoint;
|
||||||
|
import javax.websocket.EndpointConfig;
|
||||||
|
import javax.websocket.MessageHandler;
|
||||||
|
import javax.websocket.Session;
|
||||||
|
import javax.websocket.WebSocketContainer;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.server.Server;
|
||||||
|
import org.eclipse.jetty.server.ServerConnector;
|
||||||
|
import org.eclipse.jetty.server.handler.ContextHandler;
|
||||||
|
import org.eclipse.jetty.util.log.Log;
|
||||||
|
import org.eclipse.jetty.util.log.Logger;
|
||||||
|
import org.junit.AfterClass;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class tests receiving of messages by different types of {@link MessageHandler}
|
||||||
|
*/
|
||||||
|
public class MessageReceivingTest {
|
||||||
|
private static final Logger LOG = Log.getLogger(EndpointEchoTest.class);
|
||||||
|
private static Server server;
|
||||||
|
private static EchoHandler handler;
|
||||||
|
private static URI serverUri;
|
||||||
|
private WebSocketContainer container;
|
||||||
|
private final String VERY_LONG_STRING;
|
||||||
|
|
||||||
|
public MessageReceivingTest() {
|
||||||
|
final StringBuilder sb = new StringBuilder();
|
||||||
|
for (int i = 0; i < 1024 * 1024; i++) {
|
||||||
|
sb.append(i & 1);
|
||||||
|
}
|
||||||
|
VERY_LONG_STRING = sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void startServer() throws Exception {
|
||||||
|
server = new Server();
|
||||||
|
ServerConnector connector = new ServerConnector(server);
|
||||||
|
server.addConnector(connector);
|
||||||
|
|
||||||
|
handler = new EchoHandler();
|
||||||
|
|
||||||
|
ContextHandler context = new ContextHandler();
|
||||||
|
context.setContextPath("/");
|
||||||
|
context.setHandler(handler);
|
||||||
|
server.setHandler(context);
|
||||||
|
|
||||||
|
// Start Server
|
||||||
|
server.start();
|
||||||
|
|
||||||
|
String host = connector.getHost();
|
||||||
|
if (host == null) {
|
||||||
|
host = "localhost";
|
||||||
|
}
|
||||||
|
int port = connector.getLocalPort();
|
||||||
|
serverUri = new URI(String.format("ws://%s:%d/", host, port));
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterClass
|
||||||
|
public static void stopServer() {
|
||||||
|
try {
|
||||||
|
server.stop();
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace(System.err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void configureTest() {
|
||||||
|
container = ContainerProvider.getWebSocketContainer();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method tests receiving of text messages at once.
|
||||||
|
*
|
||||||
|
* @throws Exception on exception occur
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testWholeTextMessage() throws Exception {
|
||||||
|
final TestEndpoint echoer = new TestEndpoint(new WholeStringCaptureHandler());
|
||||||
|
Assert.assertThat(echoer, instanceOf(javax.websocket.Endpoint.class));
|
||||||
|
// Issue connect using instance of class that extends Endpoint
|
||||||
|
final Session session = container.connectToServer(echoer, serverUri);
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Client Connected: {}", session);
|
||||||
|
session.getBasicRemote().sendText("");
|
||||||
|
session.getBasicRemote().sendText("Echo");
|
||||||
|
session.getBasicRemote().sendText(VERY_LONG_STRING);
|
||||||
|
session.getBasicRemote().sendText("Echo");
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Client Message Sent");
|
||||||
|
echoer.handler.getMessageQueue().awaitMessages(2, 1000, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method tests receiving of text messages by parts.
|
||||||
|
*
|
||||||
|
* @throws Exception on exception occur
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testPartialTextMessage() throws Exception {
|
||||||
|
final TestEndpoint echoer = new TestEndpoint(new PartialStringCaptureHandler());
|
||||||
|
Assert.assertThat(echoer, instanceOf(javax.websocket.Endpoint.class));
|
||||||
|
// Issue connect using instance of class that extends Endpoint
|
||||||
|
final Session session = container.connectToServer(echoer, serverUri);
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Client Connected: {}", session);
|
||||||
|
session.getBasicRemote().sendText("");
|
||||||
|
session.getBasicRemote().sendText("Echo");
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Client Message Sent");
|
||||||
|
echoer.handler.getMessageQueue().awaitMessages(2, 1000, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method tests receiving of binary messages at once.
|
||||||
|
*
|
||||||
|
* @throws Exception on exception occur
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testWholeBinaryMessage() throws Exception {
|
||||||
|
final TestEndpoint echoer = new TestEndpoint(new WholeByteBufferCaptureHandler());
|
||||||
|
Assert.assertThat(echoer, instanceOf(javax.websocket.Endpoint.class));
|
||||||
|
// Issue connect using instance of class that extends Endpoint
|
||||||
|
final Session session = container.connectToServer(echoer, serverUri);
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Client Connected: {}", session);
|
||||||
|
sendBinary(session, "");
|
||||||
|
sendBinary(session, "Echo");
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Client Message Sent");
|
||||||
|
echoer.handler.getMessageQueue().awaitMessages(2, 1000, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method tests receiving of binary messages by parts.
|
||||||
|
*
|
||||||
|
* @throws Exception on exception occur
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testPartialBinaryMessage() throws Exception {
|
||||||
|
final TestEndpoint echoer = new TestEndpoint(new PartialByteBufferCaptureHandler());
|
||||||
|
Assert.assertThat(echoer, instanceOf(javax.websocket.Endpoint.class));
|
||||||
|
// Issue connect using instance of class that extends Endpoint
|
||||||
|
final Session session = container.connectToServer(echoer, serverUri);
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Client Connected: {}", session);
|
||||||
|
sendBinary(session, "");
|
||||||
|
sendBinary(session, "Echo");
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Client Message Sent");
|
||||||
|
echoer.handler.getMessageQueue().awaitMessages(2, 1000, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void sendBinary(Session session, String message) throws IOException {
|
||||||
|
final ByteBuffer bb = ByteBuffer.wrap(message.getBytes());
|
||||||
|
session.getBasicRemote().sendBinary(bb);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TestEndpoint extends Endpoint {
|
||||||
|
public final AbstractHandler handler;
|
||||||
|
|
||||||
|
public TestEndpoint(AbstractHandler handler) {
|
||||||
|
this.handler = handler;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onOpen(Session session, EndpointConfig config) {
|
||||||
|
session.addMessageHandler(handler);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Abstract message handler implementation, used for tests.
|
||||||
|
*/
|
||||||
|
private static abstract class AbstractHandler implements MessageHandler {
|
||||||
|
/**
|
||||||
|
* Message queue to put the result messages.
|
||||||
|
*/
|
||||||
|
private final MessageQueue messageQueue = new MessageQueue();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns message queue to test received messages count.
|
||||||
|
*
|
||||||
|
* @return message queue object
|
||||||
|
*/
|
||||||
|
public MessageQueue getMessageQueue() {
|
||||||
|
return messageQueue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Partial message handler for receiving binary messages.
|
||||||
|
*/
|
||||||
|
public static class PartialByteBufferCaptureHandler extends AbstractHandler implements
|
||||||
|
MessageHandler.Partial<ByteBuffer> {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parts of the current message. This list is appended with every non-last part and is
|
||||||
|
* cleared after last part of a message has been received.
|
||||||
|
*/
|
||||||
|
private final List<ByteBuffer> currentMessage = new ArrayList<>();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onMessage(ByteBuffer messagePart, boolean last) {
|
||||||
|
final ByteBuffer bufferCopy = ByteBuffer.allocate(messagePart.capacity());
|
||||||
|
bufferCopy.put(messagePart);
|
||||||
|
currentMessage.add(bufferCopy);
|
||||||
|
if (last) {
|
||||||
|
int totalSize = 0;
|
||||||
|
for (ByteBuffer bb : currentMessage) {
|
||||||
|
totalSize += bb.capacity();
|
||||||
|
}
|
||||||
|
final ByteBuffer result = ByteBuffer.allocate(totalSize);
|
||||||
|
for (ByteBuffer bb : currentMessage) {
|
||||||
|
result.put(bb);
|
||||||
|
}
|
||||||
|
final String stringResult = new String(result.array());
|
||||||
|
getMessageQueue().add(stringResult);
|
||||||
|
currentMessage.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Whole message handler for receiving binary messages.
|
||||||
|
*/
|
||||||
|
public class WholeByteBufferCaptureHandler extends AbstractHandler implements
|
||||||
|
MessageHandler.Whole<ByteBuffer> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onMessage(ByteBuffer message) {
|
||||||
|
final String stringResult = new String(message.array());
|
||||||
|
getMessageQueue().add(stringResult);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Partial message handler for receiving text messages.
|
||||||
|
*/
|
||||||
|
public static class PartialStringCaptureHandler extends AbstractHandler implements
|
||||||
|
MessageHandler.Partial<String> {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parts of the current message. This list is appended with every non-last part and is
|
||||||
|
* cleared after last part of a message has been received.
|
||||||
|
*/
|
||||||
|
private StringBuilder sb = new StringBuilder();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onMessage(String messagePart, boolean last) {
|
||||||
|
sb.append(messagePart);
|
||||||
|
if (last) {
|
||||||
|
getMessageQueue().add(sb.toString());
|
||||||
|
sb = new StringBuilder();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Whole message handler for receiving text messages.
|
||||||
|
*/
|
||||||
|
public class WholeStringCaptureHandler extends AbstractHandler implements
|
||||||
|
MessageHandler.Whole<String> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onMessage(String message) {
|
||||||
|
getMessageQueue().add(message);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue