You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Motivation:
Users of the HTTPClientResponseDelegate expect that the event loop
futures returned from didReceiveHead and didReceiveBodyPart can be used
to exert backpressure. To be fair to them, they somewhat can. However,
the TaskHandler has a bit of a misunderstanding about how NIO
backpressure works, and does not correctly manage the buffer of inbound
data.
The result of this misunderstanding is that multiple calls to
didReceiveBodyPart and didReceiveHead can be outstanding at once. This
would likely lead to severe bugs in most delegates, as they do not
expect it.
We should make things work the way delegate implementers believe it
works.
Modifications:
- Added a buffer to the TaskHandler to avoid delivering data that the
delegate is not ready for.
- Added a new "pending close" state that keeps track of a state where
the TaskHandler has received .end but not yet delivered it to the
delegate. This allows better error management.
- Added some more tests.
- Documented our backpressure commitments.
Result:
Better respect for backpressure.
Resolves#348
Copy file name to clipboardexpand all lines: Sources/AsyncHTTPClient/HTTPHandler.swift
+104-28
Original file line number
Diff line number
Diff line change
@@ -419,7 +419,26 @@ public class ResponseAccumulator: HTTPClientResponseDelegate {
419
419
/// `HTTPClientResponseDelegate` allows an implementation to receive notifications about request processing and to control how response parts are processed.
420
420
/// You can implement this protocol if you need fine-grained control over an HTTP request/response, for example, if you want to inspect the response
421
421
/// headers before deciding whether to accept a response body, or if you want to stream your request body. Pass an instance of your conforming
422
-
/// class to the `HTTPClient.execute()` method and this package will call each delegate method appropriately as the request takes place.
422
+
/// class to the `HTTPClient.execute()` method and this package will call each delegate method appropriately as the request takes place./
423
+
///
424
+
/// ### Backpressure
425
+
///
426
+
/// A `HTTPClientResponseDelegate` can be used to exert backpressure on the server response. This is achieved by way of the futures returned from
427
+
/// `didReceiveHead` and `didReceiveBodyPart`. The following functions are part of the "backpressure system" in the delegate:
428
+
///
429
+
/// - `didReceiveHead`
430
+
/// - `didReceiveBodyPart`
431
+
/// - `didFinishRequest`
432
+
/// - `didReceiveError`
433
+
///
434
+
/// The first three methods are strictly _exclusive_, with that exclusivity managed by the futures returned by `didReceiveHead` and
435
+
/// `didReceiveBodyPart`. What this means is that until the returned future is completed, none of these three methods will be called
436
+
/// again. This allows delegates to rate limit the server to a capacity it can manage. `didFinishRequest` does not return a future,
437
+
/// as we are expecting no more data from the server at this time.
438
+
///
439
+
/// `didReceiveError` is somewhat special: it signals the end of this regime. `didRecieveError` is not exclusive: it may be called at
440
+
/// any time, even if a returned future is not yet completed. `didReceiveError` is terminal, meaning that once it has been called none
441
+
/// of these four methods will be called again. This can be used as a signal to abandon all outstanding work.
423
442
///
424
443
/// - note: This delegate is strongly held by the `HTTPTaskHandler`
425
444
/// for the duration of the `Request` processing and will be
@@ -463,6 +482,11 @@ public protocol HTTPClientResponseDelegate: AnyObject {
463
482
/// You must return an `EventLoopFuture<Void>` that you complete when you have finished processing the body part.
464
483
/// You can create an already succeeded future by calling `task.eventLoop.makeSucceededFuture(())`.
465
484
///
485
+
/// This function will not be called until the future returned by `didReceiveHead` has completed.
486
+
///
487
+
/// This function will not be called for subsequent body parts until the previous future returned by a
488
+
/// call to this function completes.
489
+
///
466
490
/// - parameters:
467
491
/// - task: Current request context.
468
492
/// - buffer: Received body `Part`.
@@ -471,13 +495,20 @@ public protocol HTTPClientResponseDelegate: AnyObject {
471
495
472
496
/// Called when error was thrown during request execution. Will be called zero or one time only. Request processing will be stopped after that.
473
497
///
498
+
/// This function may be called at any time: it does not respect the backpressure exerted by `didReceiveHead` and `didReceiveBodyPart`.
499
+
/// All outstanding work may be cancelled when this is received. Once called, no further calls will be made to `didReceiveHead`, `didReceiveBodyPart`,
500
+
/// or `didFinishRequest`.
501
+
///
474
502
/// - parameters:
475
503
/// - task: Current request context.
476
504
/// - error: Error that occured during response processing.
/// Called when the complete HTTP request is finished. You must return an instance of your `Response` associated type. Will be called once, except if an error occurred.
480
508
///
509
+
/// This function will not be called until all futures returned by `didReceiveHead` and `didReceiveBodyPart` have completed. Once called,
510
+
/// no further calls will be made to `didReceiveHead`, `didReceiveBodyPart`, or `didReceiveError`.
511
+
///
481
512
/// - parameters:
482
513
/// - task: Current request context.
483
514
/// - returns: Result of processing.
@@ -678,6 +709,7 @@ internal class TaskHandler<Delegate: HTTPClientResponseDelegate>: RemovableChann
678
709
case bodySentWaitingResponseHead
679
710
case bodySentResponseHeadReceived
680
711
case redirected(HTTPResponseHead,URL)
712
+
case bufferedEnd
681
713
case endOrError
682
714
}
683
715
@@ -688,10 +720,11 @@ internal class TaskHandler<Delegate: HTTPClientResponseDelegate>: RemovableChann
688
720
letlogger:Logger // We are okay to store the logger here because a TaskHandler is just for one request.
0 commit comments