-
Notifications
You must be signed in to change notification settings - Fork 35
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for streaming on recent Darwin platforms #24
Add support for streaming on recent Darwin platforms #24
Conversation
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.
Looks great on my first pass, left a few questions. I'll take one more look once the NIO changes landed and this is finalized.
...PIURLSession/URLSessionBidirectionalStreaming/BidirectionalStreamingURLSessionDelegate.swift
Outdated
Show resolved
Hide resolved
...PIURLSession/URLSessionBidirectionalStreaming/BidirectionalStreamingURLSessionDelegate.swift
Outdated
Show resolved
Hide resolved
...PIURLSession/URLSessionBidirectionalStreaming/BidirectionalStreamingURLSessionDelegate.swift
Outdated
Show resolved
Hide resolved
...PIURLSession/URLSessionBidirectionalStreaming/BidirectionalStreamingURLSessionDelegate.swift
Outdated
Show resolved
Hide resolved
...PIURLSession/URLSessionBidirectionalStreaming/BidirectionalStreamingURLSessionDelegate.swift
Outdated
Show resolved
Hide resolved
Sources/OpenAPIURLSession/URLSessionBidirectionalStreaming/HTTPBodyOutputStreamBridge.swift
Outdated
Show resolved
Hide resolved
Sources/OpenAPIURLSession/URLSessionBidirectionalStreaming/HTTPBodyOutputStreamBridge.swift
Outdated
Show resolved
Hide resolved
825d2b3
to
daec73b
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.
Ok lgtm
daec73b
to
279b738
Compare
Sources/OpenAPIURLSession/URLSessionBidirectionalStreaming/HTTPBodyOutputStreamBridge.swift
Outdated
Show resolved
Hide resolved
|
||
enum Implementation { | ||
case buffering | ||
case streaming(requestBodyStreamBufferSize: Int, responseBodyStreamWatermarks: (low: Int, high: 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.
My preference is to not expose requestBodyStreamBufferSize
. It is an extremely confusing parameter that does not reflect buffering done in CFNetwork and Network framework. We could have gotten away with not using bounded stream pair if NSStream subclassing were easier in Swift, and it will likely be unnecessary in the future.
We'd like to retain the ability to fine tune these values on our platforms without needing developers to change their code, e.g. reducing the buffer size on low memory devices like Apple Watch, or even adjusting it dynamically based on memory pressure / app state. Most developers are not equipped to decide the best buffer size to use.
Would they be better as environment variables for advanced use cases that we don't need to commit as APIs?
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.
My preference is to not expose
requestBodyStreamBufferSize
. It is an extremely confusing parameter that does not reflect buffering done in CFNetwork and Network framework.
...
We'd like to retain the ability to fine tune these values on our platforms without needing developers to change their code, e.g. reducing the buffer size on low memory devices like Apple Watch, or even adjusting it dynamically based on memory pressure / app state. Most developers are not equipped to decide the best buffer size to use.
This is not exposed. This enum Implementation
is an internal type, only used in the internal init(session:implementation:)
. The only public initializer is public init(session:)
which calls internal init(session:implementation:)
with Implementation.platformDefault
, which has some hard-coded buffer sizes and is also internal, which could allow us to tweak on a per-platform basis.
We could have gotten away with not using bounded stream pair if NSStream subclassing were easier in Swift, and it will likely be unnecessary in the future.
Oh, that's interesting to know. I'll watch this space 👀
Would they be better as environment variables for advanced use cases that we don't need to commit as APIs?
Hadn't considered an environment variable based approach. I think we could incorporate that too, for advanced users.
One thing to call out here that we were considering exposing was a way to force buffering. Right now, when providing an HTTPBody
with iterationBehavior == .single
, the observed behaviour is different when faced with a HTTP redirect. Specifically, it will succeed when using buffering, but fail when using streaming, because we cannot rewind the request body stream.
We have other ideas on how we could solve this—e.g. wrapping the provided AsyncSequence
in another that buffers the first N bytes and might have a better chance of replaying on redirect—but in general we may want to expose the ability to just turn off streaming.
Curious if you have any thoughts on this?
//cc @dnadoba @czechboy0
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 let's start with the parameter being internal, and only make it public if a genuine use case comes up.
I'm not a fan of libraries checking environment variables, it makes them difficult to test and reason about.
...PIURLSession/URLSessionBidirectionalStreaming/BidirectionalStreamingURLSessionDelegate.swift
Outdated
Show resolved
Hide resolved
if hasAlreadyIteratedRequestBody { | ||
guard requestBody!.iterationBehavior == .multiple else { | ||
debug("Task delegate: Cannot rewind request body, cancelling task") | ||
task.cancel() |
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.
Returning nil is sufficient, no need to do both
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.
Hm, I just tried removing task.cancel()
here and testHTTPRedirect_singleIterationBehavior_fails
never completes.
Without the task.cancel()
, the needNewBodyStreamForTask
callback is called in a loop.
Motivation
Recently, the
ClientTransport
protocol was updated to be in terms ofHTTPBody
, which is anAsyncSequence<ArraySlice<UInt8>>
. Until now, the URLSession transport has buffered the request and response bodies in memory. This PR seeks to bring streaming support to recent Darwin platforms.Modifications
On new enough Darwin platforms1, use URLSession delegate to perform bidirectional streaming with backpressure. Request body backpressure is provided by the use of
StreamDelegate
with a bound stream pair. Response body backpressure is provided by the use ofAsyncBackpressuredStream
which allows for a high and low watermark to suspend and resume the URLSession task.In more detail:
AsyncBackpressuredStream
implementation from SE-0406.HTTPBodyOutputStreamBridge
which provides theOutputStreamDelegate
required to bridge theAsyncSequence
-basedHTTPBody
to anOutputStream
.BidirectionalStreamingURLSessionDelegate
which makes use ofHTTPBodyOutputStreamBridge
for streaming request bodies and also uses the delegate callbacks to bridge the response body toHTTPBody
.URLSession.bidirectionalStreamingRequest(...) async throws
, which provides a high-level URLSession-like API for setting up the appropriate URLSession task and delegate.URLSession.bufferedRequest(...) async throws
to provide a similar interfaces on platforms that don't support the streaming APIs on which we depend.enum URLSessionTransport.Configuration.Implementation
to control whether to use buffered or streaming implementation.Result
Test Plan
However, it's worth noting that our CI only runs on Linux, so we won't be testing the majority of this new feature in CI.
Footnotes
macOS 12, iOS 15, tvOS 15, watchOS 8 ↩ ↩2