use of com.webpieces.http2.api.streaming.StreamRef in project webpieces by deanhiller.
the class TestC4FrameSizeAndHeaders method testSection4_2FrameTooLarge.
/**
* An endpoint MUST send an error code of FRAME_SIZE_ERROR if a frame
* exceeds the size defined in SETTINGS_MAX_FRAME_SIZE, exceeds any
* limit defined for the frame type, or is too small to contain
* mandatory frame data. A frame size error in a frame that could alter
* the state of the entire connection MUST be treated as a connection
* error (Section 5.4.1); this includes any frame carrying a header
* block (Section 4.3) (that is, HEADERS, PUSH_PROMISE, and
* CONTINUATION), SETTINGS, and any frame with a stream identifier of 0.
*/
@Test
public void testSection4_2FrameTooLarge() {
MockStreamWriter mockStreamWriter = new MockStreamWriter();
MockResponseListener listener1 = new MockResponseListener();
listener1.setIncomingRespDefault(XFuture.<StreamWriter>completedFuture(mockStreamWriter));
Http2Request request = sendRequestToServer(listener1);
sendResponseFromServer(listener1, request);
DataFrame dataFrame = new DataFrame(request.getStreamId(), false);
byte[] buf = new byte[localSettings.getMaxFrameSize() + 4];
dataFrame.setData(DATA_GEN.wrapByteArray(buf));
// endOfStream=false
mockChannel.write(dataFrame);
// remote receives goAway
GoAwayFrame goAway = (GoAwayFrame) mockChannel.getFrameAndClear();
Assert.assertEquals(Http2ErrorCode.FRAME_SIZE_ERROR, goAway.getKnownErrorCode());
DataWrapper debugData = goAway.getDebugData();
String msg = debugData.createStringFromUtf8(0, debugData.getReadableSize());
Assert.assertEquals("ConnectionException: stream1:(EXCEEDED_MAX_FRAME_SIZE) Frame size=16389 was greater than max=16385", msg);
Assert.assertTrue(mockChannel.isClosed());
ShutdownStream failResp = (ShutdownStream) listener1.getSingleRstStream();
Assert.assertEquals(CancelReasonCode.EXCEEDED_MAX_FRAME_SIZE, failResp.getCause().getReasonCode());
// send new request on closed connection
Http2Request request1 = Requests.createRequest();
StreamRef streamRef = httpSocket.openStream().process(request1, listener1);
XFuture<StreamWriter> future = streamRef.getWriter();
ConnectionClosedException intercept = (ConnectionClosedException) TestAssert.intercept(future);
Assert.assertTrue(intercept.getMessage().contains("Connection closed or closing"));
Assert.assertEquals(0, mockChannel.getFramesAndClear().size());
}
use of com.webpieces.http2.api.streaming.StreamRef in project webpieces by deanhiller.
the class EScopedRouter method invokeRouter.
private RouterStreamRef invokeRouter(AbstractRouter router, RequestContext ctx, ProxyStreamHandle handler, boolean isCorsRequest) {
RouterStreamRef streamRef = invokeWithProtection(router, ctx, handler, isCorsRequest);
XFuture<StreamWriter> writer = streamRef.getWriter().handle((r, t) -> {
if (t == null)
return XFuture.completedFuture(r);
XFuture<StreamWriter> fut = new XFuture<>();
Throwable exc = convert(router.getMatchInfo(), t);
fut.completeExceptionally(exc);
return fut;
}).thenCompose(Function.identity());
return new RouterStreamRef("eScoped2", writer, streamRef);
}
use of com.webpieces.http2.api.streaming.StreamRef in project webpieces by deanhiller.
the class TestHttp1Backpressure method testBasicBackpressureChunked.
@Test
public void testBasicBackpressureChunked() throws InterruptedException, ExecutionException, TimeoutException {
MockResponseListener listener = new MockResponseListener();
RequestStreamHandle handle = socket.openStream();
mockChannel.addWriteResponse(XFuture.completedFuture(null));
Http2Request request = Requests.createRequest();
StreamRef streamRef = handle.process(request, listener);
XFuture<StreamWriter> writer = streamRef.getWriter();
Assert.assertTrue(writer.isDone());
Assert.assertEquals(request, mockChannel.getLastWriteParam());
Http2Response response1 = Requests.createChunkedResponse(1);
HttpChunk response2 = Requests.createHttpChunk(250);
HttpLastChunk lastChunk = new HttpLastChunk();
List<ByteBuffer> buffers = create4BuffersWith3Messags(response1, response2, lastChunk);
DataListener dataListener = mockChannel.getConnectedListener();
XFuture<Void> fut1 = dataListener.incomingData(mockChannel, buffers.get(0));
// resolved since it never made it to the client.
Assert.assertTrue(fut1.isDone());
// this keeps bytes in-memory that is less than bufPool.getSuggestedSize. ie. < 5k
XFuture<StreamWriter> requestFuture = new XFuture<StreamWriter>();
listener.addProcessResponse(requestFuture);
XFuture<Void> fut2 = dataListener.incomingData(mockChannel, buffers.get(1));
// not resolved yet since client only has part of the data
Assert.assertFalse(fut2.isDone());
XFuture<Void> streamWriterFuture = new XFuture<Void>();
MockStreamWriter mockWriter = new MockStreamWriter();
mockWriter.addProcessResponse(streamWriterFuture);
// This releases the response BUT 'some' data was with response so can't ack yet
requestFuture.complete(mockWriter);
// not resolved yet since requestFuture AND streamWriterFutuer BOTH need to be resolved
Assert.assertFalse(fut2.isDone());
// This is BECAUSE 68 bytes resulted in TWO messages and both futures can ack the 68 bytes.
streamWriterFuture.complete(null);
// NOW it's resolved
fut2.get(2, TimeUnit.SECONDS);
// feed the rest of first chunk in and feed part of last chunk
XFuture<Void> firstChunkAck = new XFuture<Void>();
mockWriter.addProcessResponse(firstChunkAck);
XFuture<Void> fut3 = dataListener.incomingData(mockChannel, buffers.get(2));
Assert.assertFalse(fut3.isDone());
// ack the http chunk packet
firstChunkAck.complete(null);
fut3.get(2, TimeUnit.SECONDS);
XFuture<Void> lastChunkAck = new XFuture<Void>();
mockWriter.addProcessResponse(lastChunkAck);
XFuture<Void> fut4 = dataListener.incomingData(mockChannel, buffers.get(3));
Assert.assertFalse(fut4.isDone());
lastChunkAck.complete(null);
fut4.get(2, TimeUnit.SECONDS);
}
use of com.webpieces.http2.api.streaming.StreamRef in project webpieces by deanhiller.
the class Layer2Http11Handler method processInitialPieceOfRequest.
private XFuture<Void> processInitialPieceOfRequest(FrontendSocketImpl socket, HttpRequest http1Req, Http2Request headers) {
int id = counter.getAndAdd(2);
PermitQueue permitQueue = socket.getPermitQueue();
return permitQueue.runRequest(() -> {
Http11StreamImpl currentStream = new Http11StreamImpl(id, socket, httpParser, permitQueue, http1Req, headers);
HttpStream streamHandle = httpListener.openStream(socket);
currentStream.setStreamHandle(streamHandle);
socket.setCurrentStream(currentStream);
if (!headers.isEndOfStream()) {
// in this case, we are NOT at the end of the request so we must let the next piece of
// data run right after the request
// TODO(dhiller): Replace this section with futureUtil.trySuccessFinally
StreamRef streamRef = streamHandle.incomingRequest(headers, currentStream);
currentStream.setStreamRef(streamRef);
return streamRef.getWriter().thenApply(w -> {
// must release the permit so the next data piece(which may be cached) can come in
permitQueue.releasePermit();
return null;
});
} else {
// in this case, since this is the END of the request, we cannot release the permit in the
// permit queue as we do not want to let the next request to start until the full response is
// sent back to the client
currentStream.setSentFullRequest(true);
StreamRef streamRef = streamHandle.incomingRequest(headers, currentStream);
currentStream.setStreamRef(streamRef);
return streamRef.getWriter().thenApply(w -> null);
}
});
}
use of com.webpieces.http2.api.streaming.StreamRef in project webpieces by deanhiller.
the class FrontendSocketImpl method cancelAllStreams.
private void cancelAllStreams(StreamListener httpListener, FarEndClosedConnection f) {
Http11StreamImpl stream = getCurrentStream();
if (stream == null)
return;
ShutdownStream shutdown = new ShutdownStream(stream.getStreamId(), f);
StreamRef streamRef = stream.getStreamRef();
streamRef.cancel(shutdown);
}
Aggregations