-
Notifications
You must be signed in to change notification settings - Fork 9.2k
New issue
Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? # to your account
Add a socket abstraction #8410
base: master
Are you sure you want to change the base?
Add a socket abstraction #8410
Conversation
8c9dd28
to
37f5924
Compare
@@ -107,7 +107,7 @@ public final class mockwebserver3/MockWebServer : java/io/Closeable { | |||
public final fun getProtocolNegotiationEnabled ()Z | |||
public final fun getProtocols ()Ljava/util/List; | |||
public final fun getRequestCount ()I | |||
public final fun getServerSocketFactory ()Ljavax/net/ServerSocketFactory; | |||
public final fun getServerSocketFactory ()Lokhttp3/internal/socket/OkioServerSocketFactory; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How do we feel about breaking changes to MockWebServer public methods?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Changing this new mockwebserver seems fine, we should avoid changes to okhttp3/mockwebserver/* in the deprecated package.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But I guess these APIs in the non deprecated API become experimental APIs, is that ok?
@@ -409,6 +409,7 @@ public final class okhttp3/ConnectionSpec { | |||
public fun equals (Ljava/lang/Object;)Z | |||
public fun hashCode ()I | |||
public final fun isCompatible (Ljavax/net/ssl/SSLSocket;)Z | |||
public final fun isCompatible (Lokhttp3/internal/socket/OkioSslSocket;)Z |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note: I tossed a @ExperimentalOkHttpApi
on here
587f76c
to
353c888
Compare
353c888
to
7266ab7
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I like it, but it's disruptive enough including for legacy 4.x APIs that you probably want @swankjesse input before addressing any of my suggestions.
@@ -107,7 +107,7 @@ public final class mockwebserver3/MockWebServer : java/io/Closeable { | |||
public final fun getProtocolNegotiationEnabled ()Z | |||
public final fun getProtocols ()Ljava/util/List; | |||
public final fun getRequestCount ()I | |||
public final fun getServerSocketFactory ()Ljavax/net/ServerSocketFactory; | |||
public final fun getServerSocketFactory ()Lokhttp3/internal/socket/OkioServerSocketFactory; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Changing this new mockwebserver seems fine, we should avoid changes to okhttp3/mockwebserver/* in the deprecated package.
@@ -107,7 +107,7 @@ public final class mockwebserver3/MockWebServer : java/io/Closeable { | |||
public final fun getProtocolNegotiationEnabled ()Z | |||
public final fun getProtocols ()Ljava/util/List; | |||
public final fun getRequestCount ()I | |||
public final fun getServerSocketFactory ()Ljavax/net/ServerSocketFactory; | |||
public final fun getServerSocketFactory ()Lokhttp3/internal/socket/OkioServerSocketFactory; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But I guess these APIs in the non deprecated API become experimental APIs, is that ok?
@@ -69,7 +69,7 @@ public final class okhttp3/mockwebserver/MockWebServer : org/junit/rules/Externa | |||
public final fun -deprecated_protocols ()Ljava/util/List; | |||
public final fun -deprecated_protocols (Ljava/util/List;)V | |||
public final fun -deprecated_requestCount ()I | |||
public final fun -deprecated_serverSocketFactory (Ljavax/net/ServerSocketFactory;)V | |||
public final fun -deprecated_serverSocketFactory (Lokhttp3/internal/socket/OkioServerSocketFactory;)V |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we hide this in this deprecated API, keep it still in terms of javax.net?
@@ -37,7 +37,7 @@ class MockWebServer : ExternalResource(), Closeable { | |||
|
|||
var bodyLimit: Long by delegate::bodyLimit | |||
|
|||
var serverSocketFactory: ServerSocketFactory? by delegate::serverSocketFactory | |||
var serverSocketFactory: OkioServerSocketFactory? by delegate::serverSocketFactory |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess I'm wondering whether dropping down to the native implementation should be a feature, and so here we could grab the real ServerSocketFactory?
@@ -107,7 +109,7 @@ class MockWebServer : Closeable { | |||
private val taskRunner = TaskRunner(taskRunnerBackend) | |||
private val requestQueue = LinkedBlockingQueue<RecordedRequest>() | |||
private val openClientSockets = | |||
Collections.newSetFromMap(ConcurrentHashMap<Socket, Boolean>()) | |||
Collections.newSetFromMap(ConcurrentHashMap<OkioSocket, Boolean>()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Given OkioSocket is an interface, does this weaken our safe assumptions about equals/hashcode being object reference ones?
Not sure if this means
a) documenting the equals/hashcode behaviour
b) introducing some Socket identifier?
c) changing this to a IdentityHashMap?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a great question. My instinct is to document that equals/hashcode are referenced-based, because I think that's the only reasonable way to think about socket equality. I think I would be surprised if sockets were considered equal based on anything than their unique state, at least by default.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe, although a socket is a 4-tuple of host:port, host:port.
And equality could break with decoration used this way.
Probably documenting reference equality is right, just bouncing ideas.
interface OkioSocketFactory { | ||
fun createSocket(): OkioSocket | ||
|
||
fun createSocket(proxy: Proxy): OkioSocket |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I wonder if we want this here? Could we say that a proxy is a different OkioSocketFactory? What would we lose here?
connectTimeout: Int, | ||
) | ||
|
||
fun shutdownOutput() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would we lose much by expecting clients to do sink.close()
?
|
||
fun connect( | ||
address: InetSocketAddress, | ||
connectTimeout: Int, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should connectTimeout be a property of the OkioSocket instead? or handled via Okio Timeout?
} | ||
} | ||
RealOkioServerSocketFactory( | ||
object : DelegatingServerSocketFactory(getDefault()) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
possibly a TODO? It would be nice to not need DelegatingServerSocketFactory
and friends because of your beautiful new API.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not suggesting interceptors :) More that decorating could be made simpler?
@@ -34,7 +35,9 @@ public void run() throws Exception { | |||
socketFile.delete(); // Clean up from previous runs. | |||
|
|||
MockWebServer server = new MockWebServer(); | |||
server.setServerSocketFactory(new UnixDomainServerSocketFactory(socketFile)); | |||
RealOkioServerSocketFactory serverSocketFactory = | |||
new RealOkioServerSocketFactory(new UnixDomainServerSocketFactory(socketFile)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
possibly worth a TODO? Feels like this new API supporting other socket types would be a win?
What
This PR adds a bunch of abstractions over
Socket
and related factories. We named the abstraction classes things likeOkioSocket
which telegraphs our intent to eventually upstream these into Okio. For now they'll just live inokhttp3.internal
.Why
This is the first step toward making our test infrastructure better. This will allow us to create fake sockets that integrate with TaskFaker which yield on read/write so that other tasks can get a shot.
As a motivating use case, this will allow us to fix the flaky
ConnectPoolTest.connectionPreWarmingHttp2
.What's next
TaskFaker
and use them inMockHttp2Peer
,MockWebServer
, etcOkioSslSocket