Skip to content

Commit

Permalink
Fix parent channel read() call on HTTP2StreamChannel initializati…
Browse files Browse the repository at this point in the history
…on causing incorrect order of inbound `HTTP2Frame`s

Motivation:

Make sure the order of `HTTP2Frame`s that are fired through the pipeline of a `HTTP2StreamChannel` by `HTTP2CommonInboundStreamMultiplexer` is correct when a `read()` call on the parent channel synchronously causes further frames to be processed.

Modifications:

Reorder calls in `HTTP2CommonInboundStreamMultiplexer.receivedFrame(frame:context:multiplexer)` so that initial header frame is buffered and processed first.

Result:

Resolves #410 and makes newly added test case `HTTP2StreamMultiplexerTests.testMultiplexerFiresInitialFramesInCorrectOrder()` pass.
  • Loading branch information
qusc committed Jul 26, 2023
1 parent 7e4dd14 commit 3a1b63c
Showing 2 changed files with 51 additions and 1 deletion.
5 changes: 4 additions & 1 deletion Sources/NIOHTTP2/HTTP2CommonInboundStreamMultiplexer.swift
Original file line number Diff line number Diff line change
@@ -109,8 +109,11 @@ extension HTTP2CommonInboundStreamMultiplexer {
// does no actual work.
self.streamChannelContinuation?.yield(channel: channel.baseChannel)

channel.configureInboundStream(initializer: self.inboundStreamStateInitializer)
// Note: Firing the initial (header) frame before calling `HTTP2StreamChannel.configureInboundStream(initializer:)`
// is crucial to preserve frame order, since the initialization process might trigger another read on the parent
// channel which in turn might cause further frames to be processed synchronously.
channel.receiveInboundFrame(frame)
channel.configureInboundStream(initializer: self.inboundStreamStateInitializer)

if !channel.inList {
self.didReadChannels.append(channel)
47 changes: 47 additions & 0 deletions Tests/NIOHTTP2Tests/HTTP2StreamMultiplexerTests.swift
Original file line number Diff line number Diff line change
@@ -1927,4 +1927,51 @@ final class HTTP2StreamMultiplexerTests: XCTestCase {
XCTAssertNoThrow(try channel.finish(acceptAlreadyClosed: false))
}

@available(*, deprecated, message: "Deprecated so deprecated functionality can be tested without warnings")
func testMultiplexerFiresInitialFramesInCorrectOrder() throws {
final class BufferingChannelHandler<Element>: ChannelDuplexHandler {
typealias InboundIn = Element
typealias InboundOut = Element
typealias OutboundIn = Element
typealias OutboundOut = Element

let elementsToBufferCount: Int = 1
var bufferedElements: CircularBuffer<Element> = []

func read(context: ChannelHandlerContext) {
while let bufferedElement = bufferedElements.popFirst() {
context.fireChannelRead(wrapInboundOut(bufferedElement))
}

context.read()
}

func channelRead(context: ChannelHandlerContext, data: NIOAny) {
bufferedElements.append(unwrapInboundIn(data))

if bufferedElements.count > elementsToBufferCount {
context.fireChannelRead(wrapInboundOut(bufferedElements.removeFirst()))
}
}
}

XCTAssertNoThrow(try self.channel.connect(to: SocketAddress(unixDomainSocketPath: "/whatever"), promise: nil))
let streamID = HTTP2StreamID(1)
let headerFrame = HTTP2Frame(streamID: streamID, payload: .headers(.init(headers: HPACKHeaders())))
let payloadBuffer = channel.allocator.buffer(capacity: 0)
let dataFrame = HTTP2Frame(streamID: streamID, payload: .data(.init(data: .byteBuffer(payloadBuffer))))

let multiplexer = HTTP2StreamMultiplexer(mode: .server, channel: self.channel) { (channel, _) in
channel.pipeline.addHandler(FrameExpecter(expectedFrames: [headerFrame, dataFrame]))
}

XCTAssertNoThrow(try self.channel.pipeline.addHandler(BufferingChannelHandler<HTTP2Frame>()).wait())
XCTAssertNoThrow(try self.channel.pipeline.addHandler(multiplexer).wait())

XCTAssertNoThrow(try self.channel.writeInbound(headerFrame))
XCTAssertNoThrow(try self.channel.writeInbound(dataFrame))
self.activateStream(streamID)
(self.channel.eventLoop as! EmbeddedEventLoop).run()
XCTAssertNoThrow(try self.channel.finish())
}
}

0 comments on commit 3a1b63c

Please sign in to comment.