All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.larksuite.oapi.okhttp3_14.internal.http2.Http2Connection Maven / Gradle / Ivy

/*
 * Copyright (C) 2011 The Android Open Source Project
 *
 * Licensed 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 com.larksuite.oapi.okhttp3_14.internal.http2;

import java.io.Closeable;
import java.io.IOException;
import java.io.InterruptedIOException;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.net.SocketAddress;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.RejectedExecutionException;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.SynchronousQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import javax.annotation.Nullable;

import com.larksuite.oapi.okhttp3_14.Headers;
import com.larksuite.oapi.okhttp3_14.internal.NamedRunnable;
import com.larksuite.oapi.okhttp3_14.internal.Util;
import com.larksuite.oapi.okhttp3_14.internal.platform.Platform;
import com.larksuite.oapi.okio1_17.Buffer;
import com.larksuite.oapi.okio1_17.BufferedSink;
import com.larksuite.oapi.okio1_17.BufferedSource;
import com.larksuite.oapi.okio1_17.ByteString;
import com.larksuite.oapi.okio1_17.Okio;

import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static com.larksuite.oapi.okhttp3_14.internal.http2.ErrorCode.REFUSED_STREAM;
import static com.larksuite.oapi.okhttp3_14.internal.http2.Settings.DEFAULT_INITIAL_WINDOW_SIZE;
import static com.larksuite.oapi.okhttp3_14.internal.platform.Platform.INFO;

/**
 * A socket connection to a remote peer. A connection hosts streams which can send and receive
 * data.
 *
 * 

Many methods in this API are synchronous: the call is completed before the * method returns. This is typical for Java but atypical for HTTP/2. This is motivated by exception * transparency: an IOException that was triggered by a certain caller can be caught and handled by * that caller. */ public final class Http2Connection implements Closeable { // Internal state of this connection is guarded by 'this'. No blocking // operations may be performed while holding this lock! // // Socket writes are guarded by frameWriter. // // Socket reads are unguarded but are only made by the reader thread. // // Certain operations (like SYN_STREAM) need to synchronize on both the // frameWriter (to do blocking I/O) and this (to create streams). Such // operations must synchronize on 'this' last. This ensures that we never // wait for a blocking operation while holding 'this'. static final int OKHTTP_CLIENT_WINDOW_SIZE = 16 * 1024 * 1024; static final int INTERVAL_PING = 1; static final int DEGRADED_PING = 2; static final int AWAIT_PING = 3; static final long DEGRADED_PONG_TIMEOUT_NS = 1_000_000_000L; // 1 second. /** * Shared executor to send notifications of incoming streams. This executor requires multiple * threads because listeners are not required to return promptly. */ private static final ExecutorService listenerExecutor = new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60, TimeUnit.SECONDS, new SynchronousQueue<>(), Util.threadFactory("OkHttp Http2Connection", true)); /** True if this peer initiated the connection. */ final boolean client; /** * User code to run in response to incoming streams or settings. Calls to this are always invoked * on {@link #listenerExecutor}. */ final Listener listener; final Map streams = new LinkedHashMap<>(); final String connectionName; int lastGoodStreamId; int nextStreamId; private boolean shutdown; /** Asynchronously writes frames to the outgoing socket. */ private final ScheduledExecutorService writerExecutor; /** Ensures push promise callbacks events are sent in order per stream. */ private final ExecutorService pushExecutor; /** User code to run in response to push promise events. */ final PushObserver pushObserver; // Total number of pings send and received of the corresponding types. All guarded by this. private long intervalPingsSent = 0L; private long intervalPongsReceived = 0L; private long degradedPingsSent = 0L; private long degradedPongsReceived = 0L; private long awaitPingsSent = 0L; private long awaitPongsReceived = 0L; /** Consider this connection to be unhealthy if a degraded pong isn't received by this time. */ private long degradedPongDeadlineNs = 0L; /** * The total number of bytes consumed by the application, but not yet acknowledged by sending a * {@code WINDOW_UPDATE} frame on this connection. */ // Visible for testing long unacknowledgedBytesRead = 0; /** * Count of bytes that can be written on the connection before receiving a window update. */ // Visible for testing long bytesLeftInWriteWindow; /** Settings we communicate to the peer. */ Settings okHttpSettings = new Settings(); /** Settings we receive from the peer. */ // TODO: MWS will need to guard on this setting before attempting to push. final Settings peerSettings = new Settings(); final Socket socket; final Http2Writer writer; // Visible for testing final ReaderRunnable readerRunnable; Http2Connection(Builder builder) { pushObserver = builder.pushObserver; client = builder.client; listener = builder.listener; // http://tools.ietf.org/html/draft-ietf-httpbis-http2-17#section-5.1.1 nextStreamId = builder.client ? 1 : 2; if (builder.client) { nextStreamId += 2; // In HTTP/2, 1 on client is reserved for Upgrade. } // Flow control was designed more for servers, or proxies than edge clients. // If we are a client, set the flow control window to 16MiB. This avoids // thrashing window updates every 64KiB, yet small enough to avoid blowing // up the heap. if (builder.client) { okHttpSettings.set(Settings.INITIAL_WINDOW_SIZE, OKHTTP_CLIENT_WINDOW_SIZE); } connectionName = builder.connectionName; writerExecutor = new ScheduledThreadPoolExecutor(1, Util.threadFactory(Util.format("OkHttp %s Writer", connectionName), false)); if (builder.pingIntervalMillis != 0) { writerExecutor.scheduleAtFixedRate(new IntervalPingRunnable(), builder.pingIntervalMillis, builder.pingIntervalMillis, MILLISECONDS); } // Like newSingleThreadExecutor, except lazy creates the thread. pushExecutor = new ThreadPoolExecutor(0, 1, 60, TimeUnit.SECONDS, new LinkedBlockingQueue<>(), Util.threadFactory(Util.format("OkHttp %s Push Observer", connectionName), true)); peerSettings.set(Settings.INITIAL_WINDOW_SIZE, DEFAULT_INITIAL_WINDOW_SIZE); peerSettings.set(Settings.MAX_FRAME_SIZE, Http2.INITIAL_MAX_FRAME_SIZE); bytesLeftInWriteWindow = peerSettings.getInitialWindowSize(); socket = builder.socket; writer = new Http2Writer(builder.sink, client); readerRunnable = new ReaderRunnable(new Http2Reader(builder.source, client)); } /** * Returns the number of {@link Http2Stream#isOpen() open streams} on this connection. */ public synchronized int openStreamCount() { return streams.size(); } synchronized Http2Stream getStream(int id) { return streams.get(id); } synchronized Http2Stream removeStream(int streamId) { Http2Stream stream = streams.remove(streamId); notifyAll(); // The removed stream may be blocked on a connection-wide window update. return stream; } public synchronized int maxConcurrentStreams() { return peerSettings.getMaxConcurrentStreams(Integer.MAX_VALUE); } synchronized void updateConnectionFlowControl(long read) { unacknowledgedBytesRead += read; if (unacknowledgedBytesRead >= okHttpSettings.getInitialWindowSize() / 2) { writeWindowUpdateLater(0, unacknowledgedBytesRead); unacknowledgedBytesRead = 0; } } /** * Returns a new server-initiated stream. * * @param associatedStreamId the stream that triggered the sender to create this stream. * @param out true to create an output stream that we can use to send data to the remote peer. * Corresponds to {@code FLAG_FIN}. */ public Http2Stream pushStream(int associatedStreamId, List

requestHeaders, boolean out) throws IOException { if (client) throw new IllegalStateException("Client cannot push requests."); return newStream(associatedStreamId, requestHeaders, out); } /** * Returns a new locally-initiated stream. * @param out true to create an output stream that we can use to send data to the remote peer. * Corresponds to {@code FLAG_FIN}. */ public Http2Stream newStream(List
requestHeaders, boolean out) throws IOException { return newStream(0, requestHeaders, out); } private Http2Stream newStream( int associatedStreamId, List
requestHeaders, boolean out) throws IOException { boolean outFinished = !out; boolean inFinished = false; boolean flushHeaders; Http2Stream stream; int streamId; synchronized (writer) { synchronized (this) { if (nextStreamId > Integer.MAX_VALUE / 2) { shutdown(REFUSED_STREAM); } if (shutdown) { throw new ConnectionShutdownException(); } streamId = nextStreamId; nextStreamId += 2; stream = new Http2Stream(streamId, this, outFinished, inFinished, null); flushHeaders = !out || bytesLeftInWriteWindow == 0L || stream.bytesLeftInWriteWindow == 0L; if (stream.isOpen()) { streams.put(streamId, stream); } } if (associatedStreamId == 0) { writer.headers(outFinished, streamId, requestHeaders); } else if (client) { throw new IllegalArgumentException("client streams shouldn't have associated stream IDs"); } else { // HTTP/2 has a PUSH_PROMISE frame. writer.pushPromise(associatedStreamId, streamId, requestHeaders); } } if (flushHeaders) { writer.flush(); } return stream; } void writeHeaders(int streamId, boolean outFinished, List
alternating) throws IOException { writer.headers(outFinished, streamId, alternating); } /** * Callers of this method are not thread safe, and sometimes on application threads. Most often, * this method will be called to send a buffer worth of data to the peer. * *

Writes are subject to the write window of the stream and the connection. Until there is a * window sufficient to send {@code byteCount}, the caller will block. For example, a user of * {@code HttpURLConnection} who flushes more bytes to the output stream than the connection's * write window will block. * *

Zero {@code byteCount} writes are not subject to flow control and will not block. The only * use case for zero {@code byteCount} is closing a flushed output stream. */ public void writeData(int streamId, boolean outFinished, Buffer buffer, long byteCount) throws IOException { if (byteCount == 0) { // Empty data frames are not flow-controlled. writer.data(outFinished, streamId, buffer, 0); return; } while (byteCount > 0) { int toWrite; synchronized (Http2Connection.this) { try { while (bytesLeftInWriteWindow <= 0) { // Before blocking, confirm that the stream we're writing is still open. It's possible // that the stream has since been closed (such as if this write timed out.) if (!streams.containsKey(streamId)) { throw new IOException("stream closed"); } Http2Connection.this.wait(); // Wait until we receive a WINDOW_UPDATE. } } catch (InterruptedException e) { Thread.currentThread().interrupt(); // Retain interrupted status. throw new InterruptedIOException(); } toWrite = (int) Math.min(byteCount, bytesLeftInWriteWindow); toWrite = Math.min(toWrite, writer.maxDataLength()); bytesLeftInWriteWindow -= toWrite; } byteCount -= toWrite; writer.data(outFinished && byteCount == 0, streamId, buffer, toWrite); } } void writeSynResetLater(final int streamId, final ErrorCode errorCode) { try { writerExecutor.execute(new NamedRunnable("OkHttp %s stream %d", connectionName, streamId) { @Override public void execute() { try { writeSynReset(streamId, errorCode); } catch (IOException e) { failConnection(e); } } }); } catch (RejectedExecutionException ignored) { // This connection has been closed. } } void writeSynReset(int streamId, ErrorCode statusCode) throws IOException { writer.rstStream(streamId, statusCode); } void writeWindowUpdateLater(final int streamId, final long unacknowledgedBytesRead) { try { writerExecutor.execute( new NamedRunnable("OkHttp Window Update %s stream %d", connectionName, streamId) { @Override public void execute() { try { writer.windowUpdate(streamId, unacknowledgedBytesRead); } catch (IOException e) { failConnection(e); } } }); } catch (RejectedExecutionException ignored) { // This connection has been closed. } } final class PingRunnable extends NamedRunnable { final boolean reply; final int payload1; final int payload2; PingRunnable(boolean reply, int payload1, int payload2) { super("OkHttp %s ping %08x%08x", connectionName, payload1, payload2); this.reply = reply; this.payload1 = payload1; this.payload2 = payload2; } @Override public void execute() { writePing(reply, payload1, payload2); } } final class IntervalPingRunnable extends NamedRunnable { IntervalPingRunnable() { super("OkHttp %s ping", connectionName); } @Override public void execute() { boolean failDueToMissingPong; synchronized (Http2Connection.this) { if (intervalPongsReceived < intervalPingsSent) { failDueToMissingPong = true; } else { intervalPingsSent++; failDueToMissingPong = false; } } if (failDueToMissingPong) { failConnection(null); } else { writePing(false, INTERVAL_PING, 0); } } } void writePing(boolean reply, int payload1, int payload2) { try { writer.ping(reply, payload1, payload2); } catch (IOException e) { failConnection(e); } } /** For testing: sends a ping and waits for a pong. */ void writePingAndAwaitPong() throws InterruptedException { writePing(); awaitPong(); } /** For testing: sends a ping to be awaited with {@link #awaitPong}. */ void writePing() { synchronized (this) { awaitPingsSent++; } writePing(false, AWAIT_PING, 0x4f4b6f6b /* "OKok" */); } /** For testing: awaits a pong. */ synchronized void awaitPong() throws InterruptedException { while (awaitPongsReceived < awaitPingsSent) { wait(); } } public void flush() throws IOException { writer.flush(); } /** * Degrades this connection such that new streams can neither be created locally, nor accepted * from the remote peer. Existing streams are not impacted. This is intended to permit an endpoint * to gracefully stop accepting new requests without harming previously established streams. */ public void shutdown(ErrorCode statusCode) throws IOException { synchronized (writer) { int lastGoodStreamId; synchronized (this) { if (shutdown) { return; } shutdown = true; lastGoodStreamId = this.lastGoodStreamId; } // TODO: propagate exception message into debugData. // TODO: configure a timeout on the reader so that it doesn’t block forever. writer.goAway(lastGoodStreamId, statusCode, Util.EMPTY_BYTE_ARRAY); } } /** * Closes this connection. This cancels all open streams and unanswered pings. It closes the * underlying input and output streams and shuts down internal executor services. */ @Override public void close() { close(ErrorCode.NO_ERROR, ErrorCode.CANCEL, null); } void close(ErrorCode connectionCode, ErrorCode streamCode, @Nullable IOException cause) { assert (!Thread.holdsLock(this)); try { shutdown(connectionCode); } catch (IOException ignored) { } Http2Stream[] streamsToClose = null; synchronized (this) { if (!streams.isEmpty()) { streamsToClose = streams.values().toArray(new Http2Stream[streams.size()]); streams.clear(); } } if (streamsToClose != null) { for (Http2Stream stream : streamsToClose) { try { stream.close(streamCode, cause); } catch (IOException ignored) { } } } // Close the writer to release its resources (such as deflaters). try { writer.close(); } catch (IOException ignored) { } // Close the socket to break out the reader thread, which will clean up after itself. try { socket.close(); } catch (IOException ignored) { } // Release the threads. writerExecutor.shutdown(); pushExecutor.shutdown(); } private void failConnection(@Nullable IOException e) { close(ErrorCode.PROTOCOL_ERROR, ErrorCode.PROTOCOL_ERROR, e); } /** * Sends any initial frames and starts reading frames from the remote peer. This should be called * after {@link Builder#build} for all new connections. */ public void start() throws IOException { start(true); } /** * @param sendConnectionPreface true to send connection preface frames. This should always be true * except for in tests that don't check for a connection preface. */ void start(boolean sendConnectionPreface) throws IOException { if (sendConnectionPreface) { writer.connectionPreface(); writer.settings(okHttpSettings); int windowSize = okHttpSettings.getInitialWindowSize(); if (windowSize != Settings.DEFAULT_INITIAL_WINDOW_SIZE) { writer.windowUpdate(0, windowSize - Settings.DEFAULT_INITIAL_WINDOW_SIZE); } } new Thread(readerRunnable).start(); // Not a daemon thread. } /** Merges {@code settings} into this peer's settings and sends them to the remote peer. */ public void setSettings(Settings settings) throws IOException { synchronized (writer) { synchronized (this) { if (shutdown) { throw new ConnectionShutdownException(); } okHttpSettings.merge(settings); } writer.settings(settings); } } public synchronized boolean isHealthy(long nowNs) { if (shutdown) return false; // A degraded pong is overdue. if (degradedPongsReceived < degradedPingsSent && nowNs >= degradedPongDeadlineNs) return false; return true; } /** * HTTP/2 can have both stream timeouts (due to a problem with a single stream) and connection * timeouts (due to a problem with the transport). When a stream times out we don't know whether * the problem impacts just one stream or the entire connection. * *

To differentiate the two cases we ping the server when a stream times out. If the overall * connection is fine the ping will receive a pong; otherwise it won't. * *

The deadline to respond to this ping attempts to limit the cost of being wrong. If it is too * long, streams created while we await the pong will reuse broken connections and inevitably * fail. If it is too short, slow connections will be marked as failed and extra TCP and TLS * handshakes will be required. * *

The deadline is currently hardcoded. We may make this configurable in the future! */ void sendDegradedPingLater() { synchronized (this) { if (degradedPongsReceived < degradedPingsSent) return; // Already awaiting a degraded pong. degradedPingsSent++; degradedPongDeadlineNs = System.nanoTime() + DEGRADED_PONG_TIMEOUT_NS; } try { writerExecutor.execute(new NamedRunnable("OkHttp %s ping", connectionName) { @Override public void execute() { writePing(false, DEGRADED_PING, 0); } }); } catch (RejectedExecutionException ignored) { // This connection has been closed. } } public static class Builder { Socket socket; String connectionName; BufferedSource source; BufferedSink sink; Listener listener = Listener.REFUSE_INCOMING_STREAMS; PushObserver pushObserver = PushObserver.CANCEL; boolean client; int pingIntervalMillis; /** * @param client true if this peer initiated the connection; false if this peer accepted the * connection. */ public Builder(boolean client) { this.client = client; } public Builder socket(Socket socket) throws IOException { SocketAddress remoteSocketAddress = socket.getRemoteSocketAddress(); String connectionName = remoteSocketAddress instanceof InetSocketAddress ? ((InetSocketAddress) remoteSocketAddress).getHostName() : remoteSocketAddress.toString(); return socket(socket, connectionName, Okio.buffer(Okio.source(socket)), Okio.buffer(Okio.sink(socket))); } public Builder socket( Socket socket, String connectionName, BufferedSource source, BufferedSink sink) { this.socket = socket; this.connectionName = connectionName; this.source = source; this.sink = sink; return this; } public Builder listener(Listener listener) { this.listener = listener; return this; } public Builder pushObserver(PushObserver pushObserver) { this.pushObserver = pushObserver; return this; } public Builder pingIntervalMillis(int pingIntervalMillis) { this.pingIntervalMillis = pingIntervalMillis; return this; } public Http2Connection build() { return new Http2Connection(this); } } /** * Methods in this class must not lock FrameWriter. If a method needs to write a frame, create an * async task to do so. */ class ReaderRunnable extends NamedRunnable implements Http2Reader.Handler { final Http2Reader reader; ReaderRunnable(Http2Reader reader) { super("OkHttp %s", connectionName); this.reader = reader; } @Override protected void execute() { ErrorCode connectionErrorCode = ErrorCode.INTERNAL_ERROR; ErrorCode streamErrorCode = ErrorCode.INTERNAL_ERROR; IOException errorException = null; try { reader.readConnectionPreface(this); while (reader.nextFrame(false, this)) { } connectionErrorCode = ErrorCode.NO_ERROR; streamErrorCode = ErrorCode.CANCEL; } catch (IOException e) { errorException = e; connectionErrorCode = ErrorCode.PROTOCOL_ERROR; streamErrorCode = ErrorCode.PROTOCOL_ERROR; } finally { close(connectionErrorCode, streamErrorCode, errorException); Util.closeQuietly(reader); } } @Override public void data(boolean inFinished, int streamId, BufferedSource source, int length) throws IOException { if (pushedStream(streamId)) { pushDataLater(streamId, source, length, inFinished); return; } Http2Stream dataStream = getStream(streamId); if (dataStream == null) { writeSynResetLater(streamId, ErrorCode.PROTOCOL_ERROR); updateConnectionFlowControl(length); source.skip(length); return; } dataStream.receiveData(source, length); if (inFinished) { dataStream.receiveHeaders(Util.EMPTY_HEADERS, true); } } @Override public void headers(boolean inFinished, int streamId, int associatedStreamId, List

headerBlock) { if (pushedStream(streamId)) { pushHeadersLater(streamId, headerBlock, inFinished); return; } Http2Stream stream; synchronized (Http2Connection.this) { stream = getStream(streamId); if (stream == null) { // If we're shutdown, don't bother with this stream. if (shutdown) return; // If the stream ID is less than the last created ID, assume it's already closed. if (streamId <= lastGoodStreamId) return; // If the stream ID is in the client's namespace, assume it's already closed. if (streamId % 2 == nextStreamId % 2) return; // Create a stream. Headers headers = Util.toHeaders(headerBlock); final Http2Stream newStream = new Http2Stream(streamId, Http2Connection.this, false, inFinished, headers); lastGoodStreamId = streamId; streams.put(streamId, newStream); listenerExecutor.execute(new NamedRunnable( "OkHttp %s stream %d", connectionName, streamId) { @Override public void execute() { try { listener.onStream(newStream); } catch (IOException e) { Platform.get().log( INFO, "Http2Connection.Listener failure for " + connectionName, e); try { newStream.close(ErrorCode.PROTOCOL_ERROR, e); } catch (IOException ignored) { } } } }); return; } } // Update an existing stream. stream.receiveHeaders(Util.toHeaders(headerBlock), inFinished); } @Override public void rstStream(int streamId, ErrorCode errorCode) { if (pushedStream(streamId)) { pushResetLater(streamId, errorCode); return; } Http2Stream rstStream = removeStream(streamId); if (rstStream != null) { rstStream.receiveRstStream(errorCode); } } @Override public void settings(boolean clearPrevious, Settings settings) { try { writerExecutor.execute(new NamedRunnable("OkHttp %s ACK Settings", connectionName) { @Override public void execute() { applyAndAckSettings(clearPrevious, settings); } }); } catch (RejectedExecutionException ignored) { // This connection has been closed. } } void applyAndAckSettings(boolean clearPrevious, Settings settings) { long delta = 0; Http2Stream[] streamsToNotify = null; synchronized (writer) { synchronized (Http2Connection.this) { int priorWriteWindowSize = peerSettings.getInitialWindowSize(); if (clearPrevious) peerSettings.clear(); peerSettings.merge(settings); int peerInitialWindowSize = peerSettings.getInitialWindowSize(); if (peerInitialWindowSize != -1 && peerInitialWindowSize != priorWriteWindowSize) { delta = peerInitialWindowSize - priorWriteWindowSize; streamsToNotify = !streams.isEmpty() ? streams.values().toArray(new Http2Stream[streams.size()]) : null; } } try { writer.applyAndAckSettings(peerSettings); } catch (IOException e) { failConnection(e); } } if (streamsToNotify != null) { for (Http2Stream stream : streamsToNotify) { synchronized (stream) { stream.addBytesToWriteWindow(delta); } } } listenerExecutor.execute(new NamedRunnable("OkHttp %s settings", connectionName) { @Override public void execute() { listener.onSettings(Http2Connection.this); } }); } @Override public void ackSettings() { // TODO: If we don't get this callback after sending settings to the peer, SETTINGS_TIMEOUT. } @Override public void ping(boolean reply, int payload1, int payload2) { if (reply) { synchronized (Http2Connection.this) { if (payload1 == INTERVAL_PING) { intervalPongsReceived++; } else if (payload1 == DEGRADED_PING) { degradedPongsReceived++; } else if (payload1 == AWAIT_PING) { awaitPongsReceived++; Http2Connection.this.notifyAll(); } } } else { try { // Send a reply to a client ping if this is a server and vice versa. writerExecutor.execute(new PingRunnable(true, payload1, payload2)); } catch (RejectedExecutionException ignored) { // This connection has been closed. } } } @Override public void goAway(int lastGoodStreamId, ErrorCode errorCode, ByteString debugData) { if (debugData.size() > 0) { // TODO: log the debugData } // Copy the streams first. We don't want to hold a lock when we call receiveRstStream(). Http2Stream[] streamsCopy; synchronized (Http2Connection.this) { streamsCopy = streams.values().toArray(new Http2Stream[streams.size()]); shutdown = true; } // Fail all streams created after the last good stream ID. for (Http2Stream http2Stream : streamsCopy) { if (http2Stream.getId() > lastGoodStreamId && http2Stream.isLocallyInitiated()) { http2Stream.receiveRstStream(REFUSED_STREAM); removeStream(http2Stream.getId()); } } } @Override public void windowUpdate(int streamId, long windowSizeIncrement) { if (streamId == 0) { synchronized (Http2Connection.this) { bytesLeftInWriteWindow += windowSizeIncrement; Http2Connection.this.notifyAll(); } } else { Http2Stream stream = getStream(streamId); if (stream != null) { synchronized (stream) { stream.addBytesToWriteWindow(windowSizeIncrement); } } } } @Override public void priority(int streamId, int streamDependency, int weight, boolean exclusive) { // TODO: honor priority. } @Override public void pushPromise(int streamId, int promisedStreamId, List
requestHeaders) { pushRequestLater(promisedStreamId, requestHeaders); } @Override public void alternateService(int streamId, String origin, ByteString protocol, String host, int port, long maxAge) { // TODO: register alternate service. } } /** Even, positive numbered streams are pushed streams in HTTP/2. */ boolean pushedStream(int streamId) { return streamId != 0 && (streamId & 1) == 0; } // Guarded by this. final Set currentPushRequests = new LinkedHashSet<>(); void pushRequestLater(final int streamId, final List
requestHeaders) { synchronized (this) { if (currentPushRequests.contains(streamId)) { writeSynResetLater(streamId, ErrorCode.PROTOCOL_ERROR); return; } currentPushRequests.add(streamId); } try { pushExecutorExecute(new NamedRunnable( "OkHttp %s Push Request[%s]", connectionName, streamId) { @Override public void execute() { boolean cancel = pushObserver.onRequest(streamId, requestHeaders); try { if (cancel) { writer.rstStream(streamId, ErrorCode.CANCEL); synchronized (Http2Connection.this) { currentPushRequests.remove(streamId); } } } catch (IOException ignored) { } } }); } catch (RejectedExecutionException ignored) { // This connection has been closed. } } void pushHeadersLater(final int streamId, final List
requestHeaders, final boolean inFinished) { try { pushExecutorExecute(new NamedRunnable( "OkHttp %s Push Headers[%s]", connectionName, streamId) { @Override public void execute() { boolean cancel = pushObserver.onHeaders(streamId, requestHeaders, inFinished); try { if (cancel) writer.rstStream(streamId, ErrorCode.CANCEL); if (cancel || inFinished) { synchronized (Http2Connection.this) { currentPushRequests.remove(streamId); } } } catch (IOException ignored) { } } }); } catch (RejectedExecutionException ignored) { // This connection has been closed. } } /** * Eagerly reads {@code byteCount} bytes from the source before launching a background task to * process the data. This avoids corrupting the stream. */ void pushDataLater(final int streamId, final BufferedSource source, final int byteCount, final boolean inFinished) throws IOException { final Buffer buffer = new Buffer(); source.require(byteCount); // Eagerly read the frame before firing client thread. source.read(buffer, byteCount); if (buffer.size() != byteCount) throw new IOException(buffer.size() + " != " + byteCount); pushExecutorExecute(new NamedRunnable("OkHttp %s Push Data[%s]", connectionName, streamId) { @Override public void execute() { try { boolean cancel = pushObserver.onData(streamId, buffer, byteCount, inFinished); if (cancel) writer.rstStream(streamId, ErrorCode.CANCEL); if (cancel || inFinished) { synchronized (Http2Connection.this) { currentPushRequests.remove(streamId); } } } catch (IOException ignored) { } } }); } void pushResetLater(final int streamId, final ErrorCode errorCode) { pushExecutorExecute(new NamedRunnable("OkHttp %s Push Reset[%s]", connectionName, streamId) { @Override public void execute() { pushObserver.onReset(streamId, errorCode); synchronized (Http2Connection.this) { currentPushRequests.remove(streamId); } } }); } private synchronized void pushExecutorExecute(NamedRunnable namedRunnable) { if (!shutdown) { pushExecutor.execute(namedRunnable); } } /** Listener of streams and settings initiated by the peer. */ public abstract static class Listener { public static final Listener REFUSE_INCOMING_STREAMS = new Listener() { @Override public void onStream(Http2Stream stream) throws IOException { stream.close(REFUSED_STREAM, null); } }; /** * Handle a new stream from this connection's peer. Implementations should respond by either * {@linkplain Http2Stream#writeHeaders replying to the stream} or {@linkplain * Http2Stream#close closing it}. This response does not need to be synchronous. */ public abstract void onStream(Http2Stream stream) throws IOException; /** * Notification that the connection's peer's settings may have changed. Implementations should * take appropriate action to handle the updated settings. * *

It is the implementation's responsibility to handle concurrent calls to this method. A * remote peer that sends multiple settings frames will trigger multiple calls to this method, * and those calls are not necessarily serialized. */ public void onSettings(Http2Connection connection) { } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy