Merge pull request #5926 from eclipse/jetty-10.0.x-tck-run-12-servlet-upgrade
Implementation of HttpServletRequest.upgrade
This commit is contained in:
commit
04cb3f3736
|
@ -465,6 +465,12 @@ public class HttpGenerator
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void servletUpgrade()
|
||||||
|
{
|
||||||
|
_noContentResponse = false;
|
||||||
|
_state = State.COMMITTED;
|
||||||
|
}
|
||||||
|
|
||||||
private void prepareChunk(ByteBuffer chunk, int remaining)
|
private void prepareChunk(ByteBuffer chunk, int remaining)
|
||||||
{
|
{
|
||||||
// if we need CRLF add this to header
|
// if we need CRLF add this to header
|
||||||
|
|
|
@ -1685,8 +1685,8 @@ public class HttpParser
|
||||||
{
|
{
|
||||||
_contentChunk = buffer.asReadOnlyBuffer();
|
_contentChunk = buffer.asReadOnlyBuffer();
|
||||||
|
|
||||||
// limit content by expected size
|
// limit content by expected size if _contentLength is >= 0 (i.e.: not infinite)
|
||||||
if (remaining > content)
|
if (_contentLength > -1 && remaining > content)
|
||||||
{
|
{
|
||||||
// We can cast remaining to an int as we know that it is smaller than
|
// We can cast remaining to an int as we know that it is smaller than
|
||||||
// or equal to length which is already an int.
|
// or equal to length which is already an int.
|
||||||
|
@ -1888,6 +1888,13 @@ public class HttpParser
|
||||||
_headerComplete = false;
|
_headerComplete = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void servletUpgrade()
|
||||||
|
{
|
||||||
|
setState(State.CONTENT);
|
||||||
|
_endOfContent = EndOfContent.UNKNOWN_CONTENT;
|
||||||
|
_contentLength = -1;
|
||||||
|
}
|
||||||
|
|
||||||
protected void setState(State state)
|
protected void setState(State state)
|
||||||
{
|
{
|
||||||
if (debugEnabled)
|
if (debugEnabled)
|
||||||
|
|
|
@ -66,6 +66,7 @@ public class HttpChannelOverHttp extends HttpChannel implements HttpParser.Reque
|
||||||
// events like timeout: we get notified and either schedule onError or release the
|
// events like timeout: we get notified and either schedule onError or release the
|
||||||
// blocking semaphore.
|
// blocking semaphore.
|
||||||
private HttpInput.Content _content;
|
private HttpInput.Content _content;
|
||||||
|
private boolean _servletUpgrade;
|
||||||
|
|
||||||
public HttpChannelOverHttp(HttpConnection httpConnection, Connector connector, HttpConfiguration config, EndPoint endPoint, HttpTransport transport)
|
public HttpChannelOverHttp(HttpConnection httpConnection, Connector connector, HttpConfiguration config, EndPoint endPoint, HttpTransport transport)
|
||||||
{
|
{
|
||||||
|
@ -262,11 +263,20 @@ public class HttpChannelOverHttp extends HttpChannel implements HttpParser.Reque
|
||||||
{
|
{
|
||||||
if (LOG.isDebugEnabled())
|
if (LOG.isDebugEnabled())
|
||||||
LOG.debug("received early EOF, content = {}", _content);
|
LOG.debug("received early EOF, content = {}", _content);
|
||||||
|
if (_servletUpgrade)
|
||||||
|
{
|
||||||
|
if (_content != null)
|
||||||
|
_content.succeeded();
|
||||||
|
_content = EOF;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
EofException failure = new EofException("Early EOF");
|
EofException failure = new EofException("Early EOF");
|
||||||
if (_content != null)
|
if (_content != null)
|
||||||
_content.failed(failure);
|
_content.failed(failure);
|
||||||
_content = new HttpInput.ErrorContent(failure);
|
_content = new HttpInput.ErrorContent(failure);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected boolean eof()
|
protected boolean eof()
|
||||||
|
@ -555,6 +565,16 @@ public class HttpChannelOverHttp extends HttpChannel implements HttpParser.Reque
|
||||||
if (_content != null && !_content.isSpecial())
|
if (_content != null && !_content.isSpecial())
|
||||||
throw new AssertionError("unconsumed content: " + _content);
|
throw new AssertionError("unconsumed content: " + _content);
|
||||||
_content = null;
|
_content = null;
|
||||||
|
_servletUpgrade = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void servletUpgrade()
|
||||||
|
{
|
||||||
|
if (_content != null && (!_content.isSpecial() || !_content.isEof()))
|
||||||
|
throw new IllegalStateException("Cannot perform servlet upgrade with unconsumed content");
|
||||||
|
_content = null;
|
||||||
|
_servletUpgrade = true;
|
||||||
|
_httpConnection.getParser().servletUpgrade();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -47,6 +47,7 @@ import javax.servlet.RequestDispatcher;
|
||||||
import javax.servlet.ServletContext;
|
import javax.servlet.ServletContext;
|
||||||
import javax.servlet.ServletException;
|
import javax.servlet.ServletException;
|
||||||
import javax.servlet.ServletInputStream;
|
import javax.servlet.ServletInputStream;
|
||||||
|
import javax.servlet.ServletOutputStream;
|
||||||
import javax.servlet.ServletRequest;
|
import javax.servlet.ServletRequest;
|
||||||
import javax.servlet.ServletRequestAttributeEvent;
|
import javax.servlet.ServletRequestAttributeEvent;
|
||||||
import javax.servlet.ServletRequestAttributeListener;
|
import javax.servlet.ServletRequestAttributeListener;
|
||||||
|
@ -61,6 +62,7 @@ import javax.servlet.http.HttpSession;
|
||||||
import javax.servlet.http.HttpUpgradeHandler;
|
import javax.servlet.http.HttpUpgradeHandler;
|
||||||
import javax.servlet.http.Part;
|
import javax.servlet.http.Part;
|
||||||
import javax.servlet.http.PushBuilder;
|
import javax.servlet.http.PushBuilder;
|
||||||
|
import javax.servlet.http.WebConnection;
|
||||||
|
|
||||||
import org.eclipse.jetty.http.BadMessageException;
|
import org.eclipse.jetty.http.BadMessageException;
|
||||||
import org.eclipse.jetty.http.ComplianceViolation;
|
import org.eclipse.jetty.http.ComplianceViolation;
|
||||||
|
@ -78,6 +80,7 @@ import org.eclipse.jetty.http.HttpURI;
|
||||||
import org.eclipse.jetty.http.HttpVersion;
|
import org.eclipse.jetty.http.HttpVersion;
|
||||||
import org.eclipse.jetty.http.MetaData;
|
import org.eclipse.jetty.http.MetaData;
|
||||||
import org.eclipse.jetty.http.MimeTypes;
|
import org.eclipse.jetty.http.MimeTypes;
|
||||||
|
import org.eclipse.jetty.io.Connection;
|
||||||
import org.eclipse.jetty.io.RuntimeIOException;
|
import org.eclipse.jetty.io.RuntimeIOException;
|
||||||
import org.eclipse.jetty.server.handler.ContextHandler;
|
import org.eclipse.jetty.server.handler.ContextHandler;
|
||||||
import org.eclipse.jetty.server.handler.ContextHandler.Context;
|
import org.eclipse.jetty.server.handler.ContextHandler.Context;
|
||||||
|
@ -2140,6 +2143,11 @@ public class Request implements HttpServletRequest
|
||||||
{
|
{
|
||||||
if (_asyncNotSupportedSource != null)
|
if (_asyncNotSupportedSource != null)
|
||||||
throw new IllegalStateException("!asyncSupported: " + _asyncNotSupportedSource);
|
throw new IllegalStateException("!asyncSupported: " + _asyncNotSupportedSource);
|
||||||
|
return forceStartAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
private AsyncContextState forceStartAsync()
|
||||||
|
{
|
||||||
HttpChannelState state = getHttpChannelState();
|
HttpChannelState state = getHttpChannelState();
|
||||||
if (_async == null)
|
if (_async == null)
|
||||||
_async = new AsyncContextState(state);
|
_async = new AsyncContextState(state);
|
||||||
|
@ -2372,7 +2380,95 @@ public class Request implements HttpServletRequest
|
||||||
@Override
|
@Override
|
||||||
public <T extends HttpUpgradeHandler> T upgrade(Class<T> handlerClass) throws IOException, ServletException
|
public <T extends HttpUpgradeHandler> T upgrade(Class<T> handlerClass) throws IOException, ServletException
|
||||||
{
|
{
|
||||||
throw new ServletException("HttpServletRequest.upgrade() not supported in Jetty");
|
Response response = _channel.getResponse();
|
||||||
|
if (response.getStatus() != HttpStatus.SWITCHING_PROTOCOLS_101)
|
||||||
|
throw new IllegalStateException("Response status should be 101");
|
||||||
|
if (response.getHeader("Upgrade") == null)
|
||||||
|
throw new IllegalStateException("Missing Upgrade header");
|
||||||
|
if (!"Upgrade".equalsIgnoreCase(response.getHeader("Connection")))
|
||||||
|
throw new IllegalStateException("Invalid Connection header");
|
||||||
|
if (response.isCommitted())
|
||||||
|
throw new IllegalStateException("Cannot upgrade committed response");
|
||||||
|
if (_metaData == null || _metaData.getHttpVersion() != HttpVersion.HTTP_1_1)
|
||||||
|
throw new IllegalStateException("Only requests over HTTP/1.1 can be upgraded");
|
||||||
|
|
||||||
|
ServletOutputStream outputStream = response.getOutputStream();
|
||||||
|
ServletInputStream inputStream = getInputStream();
|
||||||
|
HttpChannelOverHttp httpChannel11 = (HttpChannelOverHttp)_channel;
|
||||||
|
HttpConnection httpConnection = (HttpConnection)_channel.getConnection();
|
||||||
|
|
||||||
|
T upgradeHandler;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
upgradeHandler = handlerClass.getDeclaredConstructor().newInstance();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
throw new ServletException("Unable to instantiate handler class", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
httpChannel11.servletUpgrade(); // tell the HTTP 1.1 channel that it is now handling an upgraded servlet
|
||||||
|
AsyncContext asyncContext = forceStartAsync(); // force the servlet in async mode
|
||||||
|
|
||||||
|
outputStream.flush(); // commit the 101 response
|
||||||
|
httpConnection.getGenerator().servletUpgrade(); // tell the generator it can send data as-is
|
||||||
|
httpConnection.addEventListener(new Connection.Listener()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public void onClosed(Connection connection)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
asyncContext.complete();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
LOG.warn("error during upgrade AsyncContext complete", e);
|
||||||
|
}
|
||||||
|
try
|
||||||
|
{
|
||||||
|
upgradeHandler.destroy();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
LOG.warn("error during upgrade HttpUpgradeHandler destroy", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onOpened(Connection connection)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
upgradeHandler.init(new WebConnection()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public void close() throws Exception
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
inputStream.close();
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
outputStream.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ServletInputStream getInputStream()
|
||||||
|
{
|
||||||
|
return inputStream;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ServletOutputStream getOutputStream()
|
||||||
|
{
|
||||||
|
return outputStream;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return upgradeHandler;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -0,0 +1,355 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2021 Mort Bay Consulting Pty Ltd and others.
|
||||||
|
//
|
||||||
|
// This program and the accompanying materials are made available under the
|
||||||
|
// terms of the Eclipse Public License v. 2.0 which is available at
|
||||||
|
// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
|
||||||
|
// which is available at https://www.apache.org/licenses/LICENSE-2.0.
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
|
||||||
|
// ========================================================================
|
||||||
|
//
|
||||||
|
|
||||||
|
package org.eclipse.jetty.servlet;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.net.Socket;
|
||||||
|
import javax.servlet.ReadListener;
|
||||||
|
import javax.servlet.ServletException;
|
||||||
|
import javax.servlet.ServletInputStream;
|
||||||
|
import javax.servlet.ServletOutputStream;
|
||||||
|
import javax.servlet.http.HttpServlet;
|
||||||
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
import javax.servlet.http.HttpUpgradeHandler;
|
||||||
|
import javax.servlet.http.WebConnection;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.server.Handler;
|
||||||
|
import org.eclipse.jetty.server.Server;
|
||||||
|
import org.eclipse.jetty.server.ServerConnector;
|
||||||
|
import org.eclipse.jetty.server.handler.DefaultHandler;
|
||||||
|
import org.eclipse.jetty.server.handler.HandlerList;
|
||||||
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import static org.eclipse.jetty.util.StringUtil.CRLF;
|
||||||
|
import static org.hamcrest.MatcherAssert.assertThat;
|
||||||
|
import static org.hamcrest.Matchers.instanceOf;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||||
|
|
||||||
|
public class ServletUpgradeTest
|
||||||
|
{
|
||||||
|
private static final Logger LOG = LoggerFactory.getLogger(ServletUpgradeTest.class);
|
||||||
|
|
||||||
|
private Server server;
|
||||||
|
private int port;
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
public void setUp() throws Exception
|
||||||
|
{
|
||||||
|
server = new Server();
|
||||||
|
|
||||||
|
ServerConnector connector = new ServerConnector(server);
|
||||||
|
server.addConnector(connector);
|
||||||
|
|
||||||
|
ServletContextHandler contextHandler = new ServletContextHandler(ServletContextHandler.NO_SESSIONS);
|
||||||
|
contextHandler.setContextPath("/");
|
||||||
|
contextHandler.addServlet(new ServletHolder(new TestServlet()), "/TestServlet");
|
||||||
|
|
||||||
|
HandlerList handlers = new HandlerList();
|
||||||
|
handlers.setHandlers(new Handler[]{contextHandler, new DefaultHandler()});
|
||||||
|
server.setHandler(handlers);
|
||||||
|
|
||||||
|
server.start();
|
||||||
|
port = connector.getLocalPort();
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
public void tearDown() throws Exception
|
||||||
|
{
|
||||||
|
server.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void upgradeTest() throws Exception
|
||||||
|
{
|
||||||
|
boolean passed1 = false;
|
||||||
|
boolean passed2 = false;
|
||||||
|
boolean passed3 = false;
|
||||||
|
String expectedResponse1 = "TCKHttpUpgradeHandler.init";
|
||||||
|
String expectedResponse2 = "onDataAvailable|Hello";
|
||||||
|
String expectedResponse3 = "onDataAvailable|World";
|
||||||
|
|
||||||
|
InputStream input = null;
|
||||||
|
OutputStream output = null;
|
||||||
|
Socket s = null;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
s = new Socket("localhost", port);
|
||||||
|
output = s.getOutputStream();
|
||||||
|
|
||||||
|
StringBuilder reqStr = new StringBuilder()
|
||||||
|
.append("POST /TestServlet HTTP/1.1").append(CRLF)
|
||||||
|
.append("User-Agent: Java/1.6.0_33").append(CRLF)
|
||||||
|
.append("Host: localhost:").append(port).append(CRLF)
|
||||||
|
.append("Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2").append(CRLF)
|
||||||
|
.append("Upgrade: YES").append(CRLF)
|
||||||
|
.append("Connection: Upgrade").append(CRLF)
|
||||||
|
.append("Content-type: application/x-www-form-urlencoded").append(CRLF)
|
||||||
|
.append(CRLF);
|
||||||
|
|
||||||
|
LOG.info("REQUEST=========" + reqStr.toString());
|
||||||
|
output.write(reqStr.toString().getBytes());
|
||||||
|
|
||||||
|
LOG.info("Writing first chunk");
|
||||||
|
writeChunk(output, "Hello");
|
||||||
|
|
||||||
|
LOG.info("Writing second chunk");
|
||||||
|
writeChunk(output, "World");
|
||||||
|
|
||||||
|
LOG.info("Consuming the response from the server");
|
||||||
|
|
||||||
|
// Consume the response from the server
|
||||||
|
input = s.getInputStream();
|
||||||
|
int len;
|
||||||
|
byte[] b = new byte[1024];
|
||||||
|
boolean receivedFirstMessage = false;
|
||||||
|
boolean receivedSecondMessage = false;
|
||||||
|
boolean receivedThirdMessage = false;
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
while ((len = input.read(b)) != -1)
|
||||||
|
{
|
||||||
|
String line = new String(b, 0, len);
|
||||||
|
sb.append(line);
|
||||||
|
LOG.info("==============Read from server:" + CRLF + sb + CRLF);
|
||||||
|
if (passed1 = compareString(expectedResponse1, sb.toString()))
|
||||||
|
{
|
||||||
|
LOG.info("==============Received first expected response!" + CRLF);
|
||||||
|
receivedFirstMessage = true;
|
||||||
|
}
|
||||||
|
if (passed2 = compareString(expectedResponse2, sb.toString()))
|
||||||
|
{
|
||||||
|
LOG.info("==============Received second expected response!" + CRLF);
|
||||||
|
receivedSecondMessage = true;
|
||||||
|
}
|
||||||
|
if (passed3 = compareString(expectedResponse3, sb.toString()))
|
||||||
|
{
|
||||||
|
LOG.info("==============Received third expected response!" + CRLF);
|
||||||
|
receivedThirdMessage = true;
|
||||||
|
}
|
||||||
|
LOG.info("receivedFirstMessage : " + receivedFirstMessage);
|
||||||
|
LOG.info("receivedSecondMessage : " + receivedSecondMessage);
|
||||||
|
LOG.info("receivedThirdMessage : " + receivedThirdMessage);
|
||||||
|
if (receivedFirstMessage && receivedSecondMessage && receivedThirdMessage)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (input != null)
|
||||||
|
{
|
||||||
|
LOG.info("Closing input...");
|
||||||
|
input.close();
|
||||||
|
LOG.info("Input closed.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
LOG.error("Failed to close input:" + ex.getMessage(), ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (output != null)
|
||||||
|
{
|
||||||
|
LOG.info("Closing output...");
|
||||||
|
output.close();
|
||||||
|
LOG.info("Output closed .");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
LOG.error("Failed to close output:" + ex.getMessage(), ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (s != null)
|
||||||
|
{
|
||||||
|
LOG.info("Closing socket..." + CRLF);
|
||||||
|
s.close();
|
||||||
|
LOG.info("Socked closed.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
LOG.error("Failed to close socket:" + ex.getMessage(), ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(passed1 && passed2 && passed3);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TestServlet extends HttpServlet
|
||||||
|
{
|
||||||
|
public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
|
||||||
|
{
|
||||||
|
if (request.getHeader("Upgrade") != null)
|
||||||
|
{
|
||||||
|
response.setStatus(101);
|
||||||
|
response.setHeader("Upgrade", "YES");
|
||||||
|
response.setHeader("Connection", "Upgrade");
|
||||||
|
TestHttpUpgradeHandler handler = request.upgrade(TestHttpUpgradeHandler.class);
|
||||||
|
assertThat(handler, instanceOf(TestHttpUpgradeHandler.class));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
response.getWriter().println("No upgrade");
|
||||||
|
response.getWriter().println("End of Test");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class TestHttpUpgradeHandler implements HttpUpgradeHandler
|
||||||
|
{
|
||||||
|
public TestHttpUpgradeHandler()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void destroy()
|
||||||
|
{
|
||||||
|
LOG.debug("===============destroy");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void init(WebConnection wc)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ServletInputStream input = wc.getInputStream();
|
||||||
|
ServletOutputStream output = wc.getOutputStream();
|
||||||
|
TestReadListener readListener = new TestReadListener("/", input, output);
|
||||||
|
input.setReadListener(readListener);
|
||||||
|
output.println("===============TCKHttpUpgradeHandler.init");
|
||||||
|
output.flush();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
throw new RuntimeException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TestReadListener implements ReadListener
|
||||||
|
{
|
||||||
|
private final ServletInputStream input;
|
||||||
|
private final ServletOutputStream output;
|
||||||
|
private final String delimiter;
|
||||||
|
|
||||||
|
TestReadListener(String del, ServletInputStream in, ServletOutputStream out)
|
||||||
|
{
|
||||||
|
input = in;
|
||||||
|
output = out;
|
||||||
|
delimiter = del;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void onAllDataRead()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
output.println("=onAllDataRead");
|
||||||
|
output.close();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
throw new IllegalStateException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void onDataAvailable()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
output.println("=onDataAvailable");
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
int len;
|
||||||
|
byte[] b = new byte[1024];
|
||||||
|
while (input.isReady() && (len = input.read(b)) != -1)
|
||||||
|
{
|
||||||
|
String data = new String(b, 0, len);
|
||||||
|
sb.append(data);
|
||||||
|
}
|
||||||
|
output.println(delimiter + sb.toString());
|
||||||
|
output.flush();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
throw new IllegalStateException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void onError(final Throwable t)
|
||||||
|
{
|
||||||
|
LOG.error("TestReadListener error", t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean compareString(String expected, String actual)
|
||||||
|
{
|
||||||
|
String[] listExpected = expected.split("[|]");
|
||||||
|
boolean found = true;
|
||||||
|
for (int i = 0, n = listExpected.length, startIdx = 0, bodyLength = actual.length(); i < n; i++)
|
||||||
|
{
|
||||||
|
String search = listExpected[i];
|
||||||
|
if (startIdx >= bodyLength)
|
||||||
|
{
|
||||||
|
startIdx = bodyLength;
|
||||||
|
}
|
||||||
|
|
||||||
|
int searchIdx = actual.toLowerCase().indexOf(search.toLowerCase(), startIdx);
|
||||||
|
|
||||||
|
LOG.debug("[ServletTestUtil] Scanning response for " + "search string: '" + search + "' starting at index " + "location: " + startIdx);
|
||||||
|
if (searchIdx < 0)
|
||||||
|
{
|
||||||
|
found = false;
|
||||||
|
String s = "[ServletTestUtil] Unable to find the following " +
|
||||||
|
"search string in the server's " +
|
||||||
|
"response: '" + search + "' at index: " +
|
||||||
|
startIdx +
|
||||||
|
"\n[ServletTestUtil] Server's response:\n" +
|
||||||
|
"-------------------------------------------\n" +
|
||||||
|
actual +
|
||||||
|
"\n-------------------------------------------\n";
|
||||||
|
LOG.debug(s);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG.debug("[ServletTestUtil] Found search string: '" + search + "' at index '" + searchIdx + "' in the server's " + "response");
|
||||||
|
// the new searchIdx is the old index plus the lenght of the
|
||||||
|
// search string.
|
||||||
|
startIdx = searchIdx + search.length();
|
||||||
|
}
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void writeChunk(OutputStream out, String data) throws IOException
|
||||||
|
{
|
||||||
|
if (data != null)
|
||||||
|
{
|
||||||
|
out.write(data.getBytes());
|
||||||
|
}
|
||||||
|
out.flush();
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue