com.firefly.codec.http2.stream.HTTP2Session Maven / Gradle / Ivy
package com.firefly.codec.http2.stream;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.ClosedChannelException;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Queue;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import com.firefly.codec.http2.decode.Parser;
import com.firefly.codec.http2.encode.Generator;
import com.firefly.codec.http2.frame.DataFrame;
import com.firefly.codec.http2.frame.DisconnectFrame;
import com.firefly.codec.http2.frame.ErrorCode;
import com.firefly.codec.http2.frame.Frame;
import com.firefly.codec.http2.frame.FrameType;
import com.firefly.codec.http2.frame.GoAwayFrame;
import com.firefly.codec.http2.frame.HeadersFrame;
import com.firefly.codec.http2.frame.PingFrame;
import com.firefly.codec.http2.frame.PriorityFrame;
import com.firefly.codec.http2.frame.PushPromiseFrame;
import com.firefly.codec.http2.frame.ResetFrame;
import com.firefly.codec.http2.frame.SettingsFrame;
import com.firefly.codec.http2.frame.WindowUpdateFrame;
import com.firefly.utils.concurrent.Atomics;
import com.firefly.utils.concurrent.Callback;
import com.firefly.utils.concurrent.CountingCallback;
import com.firefly.utils.concurrent.Promise;
import com.firefly.utils.concurrent.Scheduler;
import com.firefly.utils.lang.Pair;
import com.firefly.utils.log.Log;
import com.firefly.utils.log.LogFactory;
import com.firefly.utils.time.Millisecond100Clock;
public abstract class HTTP2Session implements SessionSPI, Parser.Listener {
private static Log log = LogFactory.getInstance().getLog("firefly-system");
private final ConcurrentMap streams = new ConcurrentHashMap<>();
private final AtomicInteger streamIds = new AtomicInteger();
private final AtomicInteger lastStreamId = new AtomicInteger();
private final AtomicInteger localStreamCount = new AtomicInteger();
private final AtomicInteger remoteStreamCount = new AtomicInteger();
private final AtomicInteger sendWindow = new AtomicInteger();
private final AtomicInteger recvWindow = new AtomicInteger();
private final AtomicReference closed = new AtomicReference<>(CloseState.NOT_CLOSED);
private final Scheduler scheduler;
private final com.firefly.net.Session endPoint;
private final Generator generator;
private final Session.Listener listener;
private final FlowControlStrategy flowControl;
private final HTTP2Flusher flusher;
private int maxLocalStreams;
private int maxRemoteStreams;
private long streamIdleTimeout;
private int initialSessionRecvWindow;
private boolean pushEnabled;
private long idleTime;
public HTTP2Session(Scheduler scheduler, com.firefly.net.Session endPoint, Generator generator,
Session.Listener listener, FlowControlStrategy flowControl, int initialStreamId, int streamIdleTimeout) {
this.scheduler = scheduler;
this.endPoint = endPoint;
this.generator = generator;
this.listener = listener;
this.flowControl = flowControl;
this.flusher = new HTTP2Flusher(this);
this.maxLocalStreams = -1;
this.maxRemoteStreams = -1;
this.streamIds.set(initialStreamId);
this.streamIdleTimeout = streamIdleTimeout;
this.sendWindow.set(FlowControlStrategy.DEFAULT_WINDOW_SIZE);
this.recvWindow.set(FlowControlStrategy.DEFAULT_WINDOW_SIZE);
this.pushEnabled = true; // SPEC: by default, push is enabled.
this.idleTime = Millisecond100Clock.currentTimeMillis();
}
public FlowControlStrategy getFlowControlStrategy() {
return flowControl;
}
public int getMaxLocalStreams() {
return maxLocalStreams;
}
public void setMaxLocalStreams(int maxLocalStreams) {
this.maxLocalStreams = maxLocalStreams;
}
public int getMaxRemoteStreams() {
return maxRemoteStreams;
}
public void setMaxRemoteStreams(int maxRemoteStreams) {
this.maxRemoteStreams = maxRemoteStreams;
}
public long getStreamIdleTimeout() {
return streamIdleTimeout;
}
public void setStreamIdleTimeout(long streamIdleTimeout) {
this.streamIdleTimeout = streamIdleTimeout;
}
public int getInitialSessionRecvWindow() {
return initialSessionRecvWindow;
}
public void setInitialSessionRecvWindow(int initialSessionRecvWindow) {
this.initialSessionRecvWindow = initialSessionRecvWindow;
}
public com.firefly.net.Session getEndPoint() {
return endPoint;
}
public Generator getGenerator() {
return generator;
}
@Override
public void onData(final DataFrame frame) {
if (log.isDebugEnabled())
log.debug("Received {}", frame);
int streamId = frame.getStreamId();
final StreamSPI stream = getStream(streamId);
// SPEC: the session window must be updated even if the stream is null.
// The flow control length includes the padding bytes.
final int flowControlLength = frame.remaining() + frame.padding();
flowControl.onDataReceived(this, stream, flowControlLength);
if (stream != null) {
if (getRecvWindow() < 0) {
close(ErrorCode.FLOW_CONTROL_ERROR.code, "session_window_exceeded", Callback.NOOP);
} else {
stream.process(frame, new Callback() {
@Override
public void succeeded() {
complete();
}
@Override
public void failed(Throwable x) {
// Consume also in case of failures, to free the
// session flow control window for other streams.
complete();
}
private void complete() {
notIdle();
stream.notIdle();
flowControl.onDataConsumed(HTTP2Session.this, stream, flowControlLength);
}
@Override
public boolean isNonBlocking() {
return false;
}
});
}
} else {
if (log.isDebugEnabled())
log.debug("Ignoring {}, stream #{} not found", frame, streamId);
// We must enlarge the session flow control window,
// otherwise other requests will be stalled.
flowControl.onDataConsumed(this, null, flowControlLength);
}
}
@Override
public abstract void onHeaders(HeadersFrame frame);
@Override
public void onPriority(PriorityFrame frame) {
if (log.isDebugEnabled())
log.debug("Received {}", frame);
}
@Override
public void onReset(ResetFrame frame) {
if (log.isDebugEnabled())
log.debug("Received {}", frame);
StreamSPI stream = getStream(frame.getStreamId());
if (stream != null)
stream.process(frame, Callback.NOOP);
else
notifyReset(this, frame);
}
@Override
public void onSettings(SettingsFrame frame) {
// SPEC: SETTINGS frame MUST be replied.
onSettings(frame, true);
}
public void onSettings(SettingsFrame frame, boolean reply) {
if (log.isDebugEnabled())
log.debug("Received {}", frame);
if (frame.isReply())
return;
// Iterate over all settings
for (Map.Entry entry : frame.getSettings().entrySet()) {
int key = entry.getKey();
int value = entry.getValue();
switch (key) {
case SettingsFrame.HEADER_TABLE_SIZE: {
if (log.isDebugEnabled())
log.debug("Update HPACK header table size to {}", value);
generator.setHeaderTableSize(value);
break;
}
case SettingsFrame.ENABLE_PUSH: {
// SPEC: check the value is sane.
if (value != 0 && value != 1) {
onConnectionFailure(ErrorCode.PROTOCOL_ERROR.code, "invalid_settings_enable_push");
return;
}
pushEnabled = value == 1;
break;
}
case SettingsFrame.MAX_CONCURRENT_STREAMS: {
maxLocalStreams = value;
if (log.isDebugEnabled())
log.debug("Update max local concurrent streams to {}", maxLocalStreams);
break;
}
case SettingsFrame.INITIAL_WINDOW_SIZE: {
if (log.isDebugEnabled())
log.debug("Update initial window size to {}", value);
flowControl.updateInitialStreamWindow(this, value, false);
break;
}
case SettingsFrame.MAX_FRAME_SIZE: {
if (log.isDebugEnabled())
log.debug("Update max frame size to {}", value);
// SPEC: check the max frame size is sane.
if (value < Frame.DEFAULT_MAX_LENGTH || value > Frame.MAX_MAX_LENGTH) {
onConnectionFailure(ErrorCode.PROTOCOL_ERROR.code, "invalid_settings_max_frame_size");
return;
}
generator.setMaxFrameSize(value);
break;
}
case SettingsFrame.MAX_HEADER_LIST_SIZE: {
if (log.isDebugEnabled())
log.debug("Update max header list size to {}", value);
generator.setMaxHeaderListSize(value);
break;
}
default: {
if (log.isDebugEnabled())
log.debug("Unknown setting {}:{}", key, value);
break;
}
}
}
notifySettings(this, frame);
if (reply) {
SettingsFrame replyFrame = new SettingsFrame(Collections. emptyMap(), true);
settings(replyFrame, Callback.NOOP);
}
}
@Override
public void onPing(PingFrame frame) {
if (log.isDebugEnabled())
log.debug("Received {}", frame);
if (frame.isReply()) {
notifyPing(this, frame);
} else {
PingFrame reply = new PingFrame(frame.getPayload(), true);
control(null, Callback.NOOP, reply);
}
}
/**
* This method is called when receiving a GO_AWAY from the other peer. We
* check the close state to act appropriately:
*
* * NOT_CLOSED: we move to REMOTELY_CLOSED and queue a disconnect, so that
* the content of the queue is written, and then the connection closed. We
* notify the application after being terminated. See
* HTTP2Session.ControlEntry#succeeded()
*
* * In all other cases, we do nothing since other methods are already
* performing their actions.
*
* @param frame
* the GO_AWAY frame that has been received.
* @see #close(int, String, Callback)
* @see #onShutdown()
* @see #onIdleTimeout()
*/
@Override
public void onGoAway(final GoAwayFrame frame) {
if (log.isDebugEnabled())
log.debug("Received {}", frame);
while (true) {
CloseState current = closed.get();
switch (current) {
case NOT_CLOSED: {
if (closed.compareAndSet(current, CloseState.REMOTELY_CLOSED)) {
// We received a GO_AWAY, so try to write
// what's in the queue and then disconnect.
control(null, new Callback() {
@Override
public void succeeded() {
notifyClose(HTTP2Session.this, frame);
}
@Override
public void failed(Throwable x) {
notifyClose(HTTP2Session.this, frame);
}
@Override
public boolean isNonBlocking() {
return false;
}
}, new DisconnectFrame());
return;
}
break;
}
default: {
if (log.isDebugEnabled())
log.debug("Ignored {}, already closed", frame);
return;
}
}
}
}
@Override
public void onWindowUpdate(WindowUpdateFrame frame) {
if (log.isDebugEnabled())
log.debug("Received {}", frame);
int streamId = frame.getStreamId();
if (streamId > 0) {
StreamSPI stream = getStream(streamId);
if (stream != null) {
stream.process(frame, Callback.NOOP);
onWindowUpdate(stream, frame);
}
} else {
onWindowUpdate(null, frame);
}
}
@Override
public void onConnectionFailure(int error, String reason) {
close(error, reason, Callback.NOOP);
notifyFailure(this, new IOException(String.format("%d/%s", error, reason)));
}
@Override
public void newStream(HeadersFrame frame, Promise promise, Stream.Listener listener) {
// Synchronization is necessary to atomically create
// the stream id and enqueue the frame to be sent.
boolean queued;
synchronized (this) {
int streamId = frame.getStreamId();
if (streamId <= 0) {
streamId = streamIds.getAndAdd(2);
PriorityFrame priority = frame.getPriority();
priority = priority == null ? null
: new PriorityFrame(streamId, priority.getParentStreamId(), priority.getWeight(),
priority.isExclusive());
frame = new HeadersFrame(streamId, frame.getMetaData(), priority, frame.isEndStream());
}
final StreamSPI stream = createLocalStream(streamId, promise);
if (stream == null)
return;
stream.setListener(listener);
ControlEntry entry = new ControlEntry(frame, stream, new PromiseCallback<>(promise, stream));
queued = flusher.append(entry);
}
// Iterate outside the synchronized block.
if (queued)
flusher.iterate();
}
@Override
public int priority(PriorityFrame frame, Callback callback) {
int streamId = frame.getStreamId();
StreamSPI stream = streams.get(streamId);
if (stream == null) {
streamId = streamIds.getAndAdd(2);
frame = new PriorityFrame(streamId, frame.getParentStreamId(), frame.getWeight(), frame.isExclusive());
}
control(stream, callback, frame);
return streamId;
}
@Override
public void push(StreamSPI stream, Promise promise, PushPromiseFrame frame, Stream.Listener listener) {
// Synchronization is necessary to atomically create
// the stream id and enqueue the frame to be sent.
boolean queued;
synchronized (this) {
int streamId = streamIds.getAndAdd(2);
frame = new PushPromiseFrame(frame.getStreamId(), streamId, frame.getMetaData());
final StreamSPI pushStream = createLocalStream(streamId, promise);
if (pushStream == null)
return;
pushStream.setListener(listener);
ControlEntry entry = new ControlEntry(frame, pushStream, new PromiseCallback<>(promise, pushStream));
queued = flusher.append(entry);
}
// Iterate outside the synchronized block.
if (queued)
flusher.iterate();
}
@Override
public void settings(SettingsFrame frame, Callback callback) {
control(null, callback, frame);
}
@Override
public void ping(PingFrame frame, Callback callback) {
if (frame.isReply())
callback.failed(new IllegalArgumentException());
else
control(null, callback, frame);
}
protected void reset(ResetFrame frame, Callback callback) {
control(getStream(frame.getStreamId()), callback, frame);
}
/**
* Invoked internally and by applications to send a GO_AWAY frame to the
* other peer. We check the close state to act appropriately:
*
* * NOT_CLOSED: we move to LOCALLY_CLOSED and queue a GO_AWAY. When the
* GO_AWAY has been written, it will only cause the output to be shut down
* (not the connection closed), so that the application can still read
* frames arriving from the other peer. Ideally the other peer will notice
* the GO_AWAY and close the connection. When that happen, we close the
* connection from {@link #onShutdown()}. Otherwise, the idle timeout
* mechanism will close the connection, see {@link #onIdleTimeout()}.
*
* * In all other cases, we do nothing since other methods are already
* performing their actions.
*
* @param error
* the error code
* @param reason
* the reason
* @param callback
* the callback to invoke when the operation is complete
* @see #onGoAway(GoAwayFrame)
* @see #onShutdown()
* @see #onIdleTimeout()
*/
@Override
public boolean close(int error, String reason, Callback callback) {
while (true) {
CloseState current = closed.get();
switch (current) {
case NOT_CLOSED: {
if (closed.compareAndSet(current, CloseState.LOCALLY_CLOSED)) {
byte[] payload = null;
if (reason != null) {
// Trim the reason to avoid attack vectors.
reason = reason.substring(0, Math.min(reason.length(), 32));
payload = reason.getBytes(StandardCharsets.UTF_8);
}
GoAwayFrame frame = new GoAwayFrame(lastStreamId.get(), error, payload);
control(null, callback, frame);
return true;
}
break;
}
default: {
if (log.isDebugEnabled())
log.debug("Ignoring close {}/{}, already closed", error, reason);
callback.succeeded();
return false;
}
}
}
}
@Override
public boolean isClosed() {
return closed.get() != CloseState.NOT_CLOSED;
}
private void control(StreamSPI stream, Callback callback, Frame frame) {
frames(stream, callback, frame, Frame.EMPTY_ARRAY);
}
@Override
public void frames(StreamSPI stream, Callback callback, Frame frame, Frame... frames) {
// We want to generate as late as possible to allow re-prioritization;
// generation will happen while processing the entries.
// The callback needs to be notified only when the last frame completes.
int length = frames.length;
if (length == 0) {
frame(new ControlEntry(frame, stream, callback), true);
} else {
callback = new CountingCallback(callback, 1 + length);
frame(new ControlEntry(frame, stream, callback), false);
for (int i = 1; i <= length; ++i)
frame(new ControlEntry(frames[i - 1], stream, callback), i == length);
}
}
@Override
public void data(StreamSPI stream, Callback callback, DataFrame frame) {
// We want to generate as late as possible to allow re-prioritization.
frame(new DataEntry(frame, stream, callback), true);
}
private void frame(HTTP2Flusher.Entry entry, boolean flush) {
if (log.isDebugEnabled())
log.debug("{} {}", flush ? "Sending" : "Queueing", entry.frame);
// Ping frames are prepended to process them as soon as possible.
boolean queued = entry.frame.getType() == FrameType.PING ? flusher.prepend(entry) : flusher.append(entry);
if (queued && flush) {
if (entry.stream != null)
entry.stream.notIdle();
flusher.iterate();
}
}
protected StreamSPI createLocalStream(int streamId, Promise promise) {
while (true) {
int localCount = localStreamCount.get();
int maxCount = maxLocalStreams;
if (maxCount >= 0 && localCount >= maxCount) {
promise.failed(new IllegalStateException("Max local stream count " + maxCount + " exceeded"));
return null;
}
if (localStreamCount.compareAndSet(localCount, localCount + 1))
break;
}
StreamSPI stream = newStream(streamId, true);
if (streams.putIfAbsent(streamId, stream) == null) {
stream.setIdleTimeout(getStreamIdleTimeout());
flowControl.onStreamCreated(stream);
if (log.isDebugEnabled())
log.debug("Created local {}", stream);
return stream;
} else {
promise.failed(new IllegalStateException("Duplicate stream " + streamId));
return null;
}
}
protected StreamSPI createRemoteStream(int streamId) {
// SPEC: exceeding max concurrent streams is treated as stream error.
while (true) {
int remoteCount = remoteStreamCount.get();
int maxCount = getMaxRemoteStreams();
if (maxCount >= 0 && remoteCount >= maxCount) {
reset(new ResetFrame(streamId, ErrorCode.REFUSED_STREAM_ERROR.code), Callback.NOOP);
return null;
}
if (remoteStreamCount.compareAndSet(remoteCount, remoteCount + 1))
break;
}
StreamSPI stream = newStream(streamId, false);
// SPEC: duplicate stream is treated as connection error.
if (streams.putIfAbsent(streamId, stream) == null) {
updateLastStreamId(streamId);
stream.setIdleTimeout(getStreamIdleTimeout());
flowControl.onStreamCreated(stream);
if (log.isDebugEnabled())
log.debug("Created remote {}", stream);
return stream;
} else {
close(ErrorCode.PROTOCOL_ERROR.code, "duplicate_stream", Callback.NOOP);
return null;
}
}
protected StreamSPI newStream(int streamId, boolean local) {
return new HTTP2Stream(scheduler, this, streamId, local);
}
@Override
public void removeStream(StreamSPI stream) {
StreamSPI removed = streams.remove(stream.getId());
if (removed != null) {
assert removed == stream;
boolean local = stream.isLocal();
if (local)
localStreamCount.decrementAndGet();
else
remoteStreamCount.decrementAndGet();
flowControl.onStreamDestroyed(stream);
if (log.isDebugEnabled())
log.debug("Removed {} {}", local ? "local" : "remote", stream);
}
}
@Override
public Collection getStreams() {
List result = new ArrayList<>();
result.addAll(streams.values());
return result;
}
public int getStreamCount() {
return streams.size();
}
@Override
public StreamSPI getStream(int streamId) {
return streams.get(streamId);
}
public int getSendWindow() {
return sendWindow.get();
}
public int getRecvWindow() {
return recvWindow.get();
}
@Override
public int updateSendWindow(int delta) {
return sendWindow.getAndAdd(delta);
}
@Override
public int updateRecvWindow(int delta) {
return recvWindow.getAndAdd(delta);
}
@Override
public void onWindowUpdate(StreamSPI stream, WindowUpdateFrame frame) {
// WindowUpdateFrames arrive concurrently with writes.
// Increasing (or reducing) the window size concurrently
// with writes requires coordination with the flusher, that
// decides how many frames to write depending on the available
// window sizes. If the window sizes vary concurrently, the
// flusher may take non-optimal or wrong decisions.
// Here, we "queue" window updates to the flusher, so it will
// be the only component responsible for window updates, for
// both increments and reductions.
flusher.window(stream, frame);
}
@Override
public boolean isPushEnabled() {
return pushEnabled;
}
/**
* A typical close by a remote peer involves a GO_AWAY frame followed by TCP
* FIN. This method is invoked when the TCP FIN is received, or when an
* exception is thrown while reading, and we check the close state to act
* appropriately:
*
* * NOT_CLOSED: means that the remote peer did not send a GO_AWAY (abrupt
* close) or there was an exception while reading, and therefore we
* terminate.
*
* * LOCALLY_CLOSED: we have sent the GO_AWAY to the remote peer, which
* received it and closed the connection; we queue a disconnect to close the
* connection on the local side. The GO_AWAY just shutdown the output, so we
* need this step to make sure the connection is closed. See
* {@link #close(int, String, Callback)}.
*
* * REMOTELY_CLOSED: we received the GO_AWAY, and the TCP FIN afterwards,
* so we do nothing since the handling of the GO_AWAY will take care of
* closing the connection. See {@link #onGoAway(GoAwayFrame)}.
*
* @see #onGoAway(GoAwayFrame)
* @see #close(int, String, Callback)
* @see #onIdleTimeout()
*/
@Override
public void onShutdown() {
if (log.isDebugEnabled())
log.debug("Shutting down {}", this);
switch (closed.get()) {
case NOT_CLOSED: {
// The other peer did not send a GO_AWAY, no need to be gentle.
if (log.isDebugEnabled())
log.debug("Abrupt close for {}", this);
abort(new ClosedChannelException());
break;
}
case LOCALLY_CLOSED: {
// We have closed locally, and only shutdown
// the output; now queue a disconnect.
control(null, Callback.NOOP, new DisconnectFrame());
break;
}
case REMOTELY_CLOSED: {
// Nothing to do, the GO_AWAY frame we
// received will close the connection.
break;
}
default: {
break;
}
}
}
/**
* This method is invoked when the idle timeout triggers. We check the close
* state to act appropriately:
*
* * NOT_CLOSED: it's a real idle timeout, we just initiate a close, see
* {@link #close(int, String, Callback)}.
*
* * LOCALLY_CLOSED: we have sent a GO_AWAY and only shutdown the output,
* but the other peer did not close the connection so we never received the
* TCP FIN, and therefore we terminate.
*
* * REMOTELY_CLOSED: the other peer sent us a GO_AWAY, we should have
* queued a disconnect, but for some reason it was not processed (for
* example, queue was stuck because of TCP congestion), therefore we
* terminate. See {@link #onGoAway(GoAwayFrame)}.
*
* @return true if the session should be closed, false otherwise
* @see #onGoAway(GoAwayFrame)
* @see #close(int, String, Callback)
* @see #onShutdown()
*/
@Override
public boolean onIdleTimeout() {
switch (closed.get()) {
case NOT_CLOSED: {
long elapsed = Millisecond100Clock.currentTimeMillis() - idleTime;
if (elapsed < endPoint.getIdleTimeout())
return false;
return notifyIdleTimeout(this);
}
case LOCALLY_CLOSED:
case REMOTELY_CLOSED: {
abort(new TimeoutException("Idle timeout " + endPoint.getIdleTimeout() + " ms"));
return false;
}
default: {
return false;
}
}
}
private void notIdle() {
idleTime = Millisecond100Clock.currentTimeMillis();
}
@Override
public void onFrame(Frame frame) {
onConnectionFailure(ErrorCode.PROTOCOL_ERROR.code, "upgrade");
}
public void disconnect() {
if (log.isDebugEnabled())
log.debug("Disconnecting {}", this);
endPoint.close();
}
private void terminate() {
while (true) {
CloseState current = closed.get();
switch (current) {
case NOT_CLOSED:
case LOCALLY_CLOSED:
case REMOTELY_CLOSED: {
if (closed.compareAndSet(current, CloseState.CLOSED)) {
flusher.terminate();
for (StreamSPI stream : streams.values())
stream.close();
streams.clear();
disconnect();
return;
}
break;
}
default: {
return;
}
}
}
}
protected void abort(Throwable failure) {
terminate();
notifyFailure(this, failure);
}
public boolean isDisconnected() {
return !endPoint.isOpen();
}
private void updateLastStreamId(int streamId) {
Atomics.updateMax(lastStreamId, streamId);
}
protected Stream.Listener notifyNewStream(Stream stream, HeadersFrame frame) {
try {
return listener.onNewStream(stream, frame);
} catch (Throwable x) {
log.info("Failure while notifying listener " + listener, x);
return null;
}
}
protected void notifySettings(Session session, SettingsFrame frame) {
try {
listener.onSettings(session, frame);
} catch (Throwable x) {
log.info("Failure while notifying listener " + listener, x);
}
}
protected void notifyPing(Session session, PingFrame frame) {
try {
listener.onPing(session, frame);
} catch (Throwable x) {
log.info("Failure while notifying listener " + listener, x);
}
}
protected void notifyReset(Session session, ResetFrame frame) {
try {
listener.onReset(session, frame);
} catch (Throwable x) {
log.info("Failure while notifying listener " + listener, x);
}
}
protected void notifyClose(Session session, GoAwayFrame frame) {
try {
listener.onClose(session, frame);
} catch (Throwable x) {
log.info("Failure while notifying listener " + listener, x);
}
}
protected boolean notifyIdleTimeout(Session session) {
try {
return listener.onIdleTimeout(session);
} catch (Throwable x) {
log.info("Failure while notifying listener " + listener, x);
return true;
}
}
protected void notifyFailure(Session session, Throwable failure) {
try {
listener.onFailure(session, failure);
} catch (Throwable x) {
log.info("Failure while notifying listener " + listener, x);
}
}
@Override
public String toString() {
return String.format("%s@%x{l:%s <-> r:%s,queueSize=%d,sendWindow=%s,recvWindow=%s,streams=%d,%s}",
getClass().getSimpleName(), hashCode(), getEndPoint().getLocalAddress(),
getEndPoint().getRemoteAddress(), flusher.getQueueSize(), sendWindow, recvWindow, streams.size(),
closed);
}
private class ControlEntry extends HTTP2Flusher.Entry {
private ControlEntry(Frame frame, StreamSPI stream, Callback callback) {
super(frame, stream, callback);
}
protected boolean generate(Queue buffers) {
buffers.addAll(generator.control(frame));
if (log.isDebugEnabled())
log.debug("Generated {}", frame);
prepare();
return true;
}
/**
*
* Performs actions just before writing the frame to the network.
*
*
* Some frame, when sent over the network, causes the receiver to react
* and send back frames that may be processed by the original sender
* *before* {@link #succeeded()} is called.
*
* If the action to perform updates some state, this update may not be
* seen by the received frames and cause errors.
*
*
* For example, suppose the action updates the stream window to a larger
* value; the sender sends the frame; the receiver is now entitled to
* send back larger data; when the data is received by the original
* sender, the action may have not been performed yet, causing the
* larger data to be rejected, when it should have been accepted.
*
*/
private void prepare() {
switch (frame.getType()) {
case SETTINGS: {
SettingsFrame settingsFrame = (SettingsFrame) frame;
Integer initialWindow = settingsFrame.getSettings().get(SettingsFrame.INITIAL_WINDOW_SIZE);
if (initialWindow != null)
flowControl.updateInitialStreamWindow(HTTP2Session.this, initialWindow, true);
break;
}
default: {
break;
}
}
}
@Override
public void succeeded() {
switch (frame.getType()) {
case HEADERS: {
HeadersFrame headersFrame = (HeadersFrame) frame;
if (stream.updateClose(headersFrame.isEndStream(), true))
removeStream(stream);
break;
}
case RST_STREAM: {
if (stream != null) {
stream.close();
removeStream(stream);
}
break;
}
case PUSH_PROMISE: {
// Pushed streams are implicitly remotely closed.
// They are closed when sending an end-stream DATA frame.
stream.updateClose(true, false);
break;
}
case GO_AWAY: {
// We just sent a GO_AWAY, only shutdown the
// output without closing yet, to allow reads.
getEndPoint().shutdownOutput();
break;
}
case WINDOW_UPDATE: {
flowControl.windowUpdate(HTTP2Session.this, stream, (WindowUpdateFrame) frame);
break;
}
case DISCONNECT: {
terminate();
break;
}
default: {
break;
}
}
super.succeeded();
}
}
private class DataEntry extends HTTP2Flusher.Entry {
private int remaining;
private int generated;
private DataEntry(DataFrame frame, StreamSPI stream, Callback callback) {
super(frame, stream, callback);
// We don't do any padding, so the flow control length is
// always the data remaining. This simplifies the handling
// of data frames that cannot be completely written due to
// the flow control window exhausting, since in that case
// we would have to count the padding only once.
remaining = frame.remaining();
}
@Override
public int dataRemaining() {
return remaining;
}
protected boolean generate(Queue buffers) {
int toWrite = dataRemaining();
int sessionSendWindow = getSendWindow();
int streamSendWindow = stream.updateSendWindow(0);
int window = Math.min(streamSendWindow, sessionSendWindow);
if (window <= 0 && toWrite > 0)
return false;
int length = Math.min(toWrite, window);
Pair> pair = generator.data((DataFrame) frame, length);
buffers.addAll(pair.second);
int generated = pair.first;
log.debug("Generated {}, length/window/data={}/{}/{}", (DataFrame) frame, generated, window, toWrite);
this.generated += generated;
this.remaining -= generated;
flowControl.onDataSending(stream, generated);
return true;
}
@Override
public void succeeded() {
flowControl.onDataSent(stream, generated);
generated = 0;
// Do we have more to send ?
DataFrame dataFrame = (DataFrame) frame;
if (dataRemaining() <= 0) {
// Only now we can update the close state
// and eventually remove the stream.
if (stream.updateClose(dataFrame.isEndStream(), true))
removeStream(stream);
super.succeeded();
}
}
}
private static class PromiseCallback implements Callback {
private final Promise promise;
private final C value;
private PromiseCallback(Promise promise, C value) {
this.promise = promise;
this.value = value;
}
@Override
public void succeeded() {
promise.succeeded(value);
}
@Override
public void failed(Throwable x) {
promise.failed(x);
}
@Override
public boolean isNonBlocking() {
return false;
}
}
}