227 lines
7.6 KiB
Java
227 lines
7.6 KiB
Java
/*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
* this work for additional information regarding copyright ownership.
|
|
* The ASF licenses this file to You 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
|
|
*
|
|
* http://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.apache.coyote.http2;
|
|
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.io.PrintWriter;
|
|
import java.nio.ByteBuffer;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
|
|
import javax.servlet.AsyncContext;
|
|
import javax.servlet.ServletException;
|
|
import javax.servlet.http.HttpServlet;
|
|
import javax.servlet.http.HttpServletRequest;
|
|
import javax.servlet.http.HttpServletResponse;
|
|
|
|
import org.junit.Assert;
|
|
import org.junit.Test;
|
|
|
|
import org.apache.catalina.Context;
|
|
import org.apache.catalina.Wrapper;
|
|
import org.apache.catalina.startup.Tomcat;
|
|
import org.apache.tomcat.util.compat.JrePlatform;
|
|
import org.apache.tomcat.util.http.FastHttpDateFormat;
|
|
|
|
public class TestStreamProcessor extends Http2TestBase {
|
|
|
|
@Test
|
|
public void testAsyncComplete() throws Exception {
|
|
enableHttp2();
|
|
|
|
Tomcat tomcat = getTomcatInstance();
|
|
|
|
// Map the async servlet to /simple so we can re-use the HTTP/2 handling
|
|
// logic from the super class.
|
|
Context ctxt = tomcat.addContext("", null);
|
|
Tomcat.addServlet(ctxt, "simple", new SimpleServlet());
|
|
ctxt.addServletMappingDecoded("/simple", "simple");
|
|
Wrapper w = Tomcat.addServlet(ctxt, "async", new AsyncComplete());
|
|
w.setAsyncSupported(true);
|
|
ctxt.addServletMappingDecoded("/async", "async");
|
|
|
|
tomcat.start();
|
|
|
|
openClientConnection();
|
|
doHttpUpgrade();
|
|
sendClientPreface();
|
|
validateHttp2InitialResponse();
|
|
|
|
byte[] frameHeader = new byte[9];
|
|
ByteBuffer headersPayload = ByteBuffer.allocate(128);
|
|
buildGetRequest(frameHeader, headersPayload, null, 3, "/async");
|
|
writeFrame(frameHeader, headersPayload);
|
|
|
|
readSimpleGetResponse();
|
|
// Flush before startAsync means body is written in two packets so an
|
|
// additional frame needs to be read
|
|
parser.readFrame(true);
|
|
|
|
Assert.assertEquals(
|
|
"3-HeadersStart\n" +
|
|
"3-Header-[:status]-[200]\n" +
|
|
"3-Header-[content-type]-[text/plain;charset=UTF-8]\n" +
|
|
"3-Header-[date]-[Wed, 11 Nov 2015 19:18:42 GMT]\n" +
|
|
"3-HeadersEnd\n" +
|
|
"3-Body-17\n" +
|
|
"3-Body-8\n" +
|
|
"3-EndOfStream\n", output.getTrace());
|
|
}
|
|
|
|
|
|
@Test
|
|
public void testAsyncDispatch() throws Exception {
|
|
enableHttp2();
|
|
|
|
Tomcat tomcat = getTomcatInstance();
|
|
|
|
// Map the async servlet to /simple so we can re-use the HTTP/2 handling
|
|
// logic from the super class.
|
|
Context ctxt = tomcat.addContext("", null);
|
|
Tomcat.addServlet(ctxt, "simple", new SimpleServlet());
|
|
ctxt.addServletMappingDecoded("/simple", "simple");
|
|
Wrapper w = Tomcat.addServlet(ctxt, "async", new AsyncDispatch());
|
|
w.setAsyncSupported(true);
|
|
ctxt.addServletMappingDecoded("/async", "async");
|
|
|
|
tomcat.start();
|
|
|
|
openClientConnection();
|
|
doHttpUpgrade();
|
|
sendClientPreface();
|
|
validateHttp2InitialResponse();
|
|
|
|
byte[] frameHeader = new byte[9];
|
|
ByteBuffer headersPayload = ByteBuffer.allocate(128);
|
|
buildGetRequest(frameHeader, headersPayload, null, 3, "/async");
|
|
writeFrame(frameHeader, headersPayload);
|
|
|
|
readSimpleGetResponse();
|
|
Assert.assertEquals(getSimpleResponseTrace(3), output.getTrace());
|
|
}
|
|
|
|
|
|
@Test
|
|
public void testPrepareHeaders() throws Exception {
|
|
enableHttp2();
|
|
|
|
Tomcat tomcat = getTomcatInstance();
|
|
|
|
File appDir = new File("test/webapp");
|
|
Context ctxt = tomcat.addWebapp(null, "", appDir.getAbsolutePath());
|
|
|
|
Tomcat.addServlet(ctxt, "simple", new SimpleServlet());
|
|
ctxt.addServletMappingDecoded("/simple", "simple");
|
|
|
|
tomcat.start();
|
|
|
|
openClientConnection();
|
|
doHttpUpgrade();
|
|
sendClientPreface();
|
|
validateHttp2InitialResponse();
|
|
|
|
byte[] frameHeader = new byte[9];
|
|
ByteBuffer headersPayload = ByteBuffer.allocate(128);
|
|
|
|
List<Header> headers = new ArrayList<>(3);
|
|
headers.add(new Header(":method", "GET"));
|
|
headers.add(new Header(":scheme", "http"));
|
|
headers.add(new Header(":path", "/index.html"));
|
|
headers.add(new Header(":authority", "localhost:" + getPort()));
|
|
headers.add(new Header("if-modified-since", FastHttpDateFormat.getCurrentDate()));
|
|
|
|
buildGetRequest(frameHeader, headersPayload, null, headers, 3);
|
|
|
|
writeFrame(frameHeader, headersPayload);
|
|
|
|
parser.readFrame(true);
|
|
|
|
StringBuilder expected = new StringBuilder();
|
|
expected.append("3-HeadersStart\n");
|
|
expected.append("3-Header-[:status]-[304]\n");
|
|
// Different line-endings -> different files size -> different weak eTag
|
|
if (JrePlatform.IS_WINDOWS) {
|
|
expected.append("3-Header-[etag]-[W/\"957-1447269522000\"]\n");
|
|
} else {
|
|
expected.append("3-Header-[etag]-[W/\"934-1447269522000\"]\n");
|
|
}
|
|
expected.append("3-Header-[date]-[Wed, 11 Nov 2015 19:18:42 GMT]\n");
|
|
expected.append("3-HeadersEnd\n");
|
|
|
|
Assert.assertEquals(expected.toString(), output.getTrace());
|
|
}
|
|
|
|
|
|
private static final class AsyncComplete extends HttpServlet {
|
|
|
|
private static final long serialVersionUID = 1L;
|
|
|
|
@Override
|
|
protected void doGet(HttpServletRequest request, HttpServletResponse response)
|
|
throws ServletException, IOException {
|
|
|
|
response.setContentType("text/plain");
|
|
response.setCharacterEncoding("UTF-8");
|
|
|
|
PrintWriter pw = response.getWriter();
|
|
pw.print("Enter-");
|
|
|
|
final AsyncContext asyncContext = request.startAsync(request, response);
|
|
pw.print("StartAsync-");
|
|
pw.flush();
|
|
|
|
asyncContext.start(new Runnable() {
|
|
|
|
@Override
|
|
public void run() {
|
|
try {
|
|
asyncContext.getResponse().getWriter().print("Complete");
|
|
asyncContext.complete();
|
|
} catch (Exception e) {
|
|
e.printStackTrace();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
|
|
private static final class AsyncDispatch extends HttpServlet {
|
|
|
|
private static final long serialVersionUID = 1L;
|
|
|
|
@Override
|
|
protected void doGet(HttpServletRequest request, HttpServletResponse response)
|
|
throws ServletException, IOException {
|
|
|
|
final AsyncContext asyncContext = request.startAsync(request, response);
|
|
asyncContext.start(new Runnable() {
|
|
|
|
@Override
|
|
public void run() {
|
|
try {
|
|
asyncContext.dispatch("/simple");
|
|
} catch (Exception e) {
|
|
e.printStackTrace();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
}
|