BidirectionalStreamingServerHandler.swift 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. /*
  2. * Copyright 2021, gRPC Authors All rights reserved.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. import NIOCore
  17. import NIOHPACK
  18. public final class BidirectionalStreamingServerHandler<
  19. Serializer: MessageSerializer,
  20. Deserializer: MessageDeserializer
  21. >: GRPCServerHandlerProtocol {
  22. public typealias Request = Deserializer.Output
  23. public typealias Response = Serializer.Input
  24. /// A response serializer.
  25. @usableFromInline
  26. internal let serializer: Serializer
  27. /// A request deserializer.
  28. @usableFromInline
  29. internal let deserializer: Deserializer
  30. /// A pipeline of user provided interceptors.
  31. @usableFromInline
  32. internal var interceptors: ServerInterceptorPipeline<Request, Response>!
  33. /// Stream events which have arrived before the stream observer future has been resolved.
  34. @usableFromInline
  35. internal var requestBuffer: CircularBuffer<StreamEvent<Request>> = CircularBuffer()
  36. /// The context required in order create the function.
  37. @usableFromInline
  38. internal let context: CallHandlerContext
  39. /// A reference to a `UserInfo`.
  40. @usableFromInline
  41. internal let userInfoRef: Ref<UserInfo>
  42. /// The user provided function to execute.
  43. @usableFromInline
  44. internal let observerFactory: (_StreamingResponseCallContext<Request, Response>)
  45. -> EventLoopFuture<(StreamEvent<Request>) -> Void>
  46. /// The state of the handler.
  47. @usableFromInline
  48. internal var state: State = .idle
  49. @usableFromInline
  50. internal enum State {
  51. // No headers have been received.
  52. case idle
  53. // Headers have been received, a context has been created and the user code has been called to
  54. // make a stream observer with. The observer is yet to see any messages.
  55. case creatingObserver(_StreamingResponseCallContext<Request, Response>)
  56. // The observer future has resolved and the observer may have seen messages.
  57. case observing((StreamEvent<Request>) -> Void, _StreamingResponseCallContext<Request, Response>)
  58. // The observer has completed by completing the status promise.
  59. case completed
  60. }
  61. @inlinable
  62. public init(
  63. context: CallHandlerContext,
  64. requestDeserializer: Deserializer,
  65. responseSerializer: Serializer,
  66. interceptors: [ServerInterceptor<Request, Response>],
  67. observerFactory: @escaping (StreamingResponseCallContext<Response>)
  68. -> EventLoopFuture<(StreamEvent<Request>) -> Void>
  69. ) {
  70. self.serializer = responseSerializer
  71. self.deserializer = requestDeserializer
  72. self.context = context
  73. self.observerFactory = observerFactory
  74. let userInfoRef = Ref(UserInfo())
  75. self.userInfoRef = userInfoRef
  76. self.interceptors = ServerInterceptorPipeline(
  77. logger: context.logger,
  78. eventLoop: context.eventLoop,
  79. path: context.path,
  80. callType: .bidirectionalStreaming,
  81. remoteAddress: context.remoteAddress,
  82. userInfoRef: userInfoRef,
  83. closeFuture: context.closeFuture,
  84. interceptors: interceptors,
  85. onRequestPart: self.receiveInterceptedPart(_:),
  86. onResponsePart: self.sendInterceptedPart(_:promise:)
  87. )
  88. }
  89. // MARK: - Public API: gRPC to Handler
  90. @inlinable
  91. public func receiveMetadata(_ headers: HPACKHeaders) {
  92. self.interceptors.receive(.metadata(headers))
  93. }
  94. @inlinable
  95. public func receiveMessage(_ bytes: ByteBuffer) {
  96. do {
  97. let message = try self.deserializer.deserialize(byteBuffer: bytes)
  98. self.interceptors.receive(.message(message))
  99. } catch {
  100. self.handleError(error)
  101. }
  102. }
  103. @inlinable
  104. public func receiveEnd() {
  105. self.interceptors.receive(.end)
  106. }
  107. @inlinable
  108. public func receiveError(_ error: Error) {
  109. self.handleError(error)
  110. self.finish()
  111. }
  112. @inlinable
  113. public func finish() {
  114. switch self.state {
  115. case .idle:
  116. self.interceptors = nil
  117. self.state = .completed
  118. case let .creatingObserver(context),
  119. let .observing(_, context):
  120. context.statusPromise.fail(GRPCStatus(code: .unavailable, message: nil))
  121. case .completed:
  122. self.interceptors = nil
  123. }
  124. }
  125. // MARK: - Interceptors to User Function
  126. @inlinable
  127. internal func receiveInterceptedPart(_ part: GRPCServerRequestPart<Request>) {
  128. switch part {
  129. case let .metadata(headers):
  130. self.receiveInterceptedMetadata(headers)
  131. case let .message(message):
  132. self.receiveInterceptedMessage(message)
  133. case .end:
  134. self.receiveInterceptedEnd()
  135. }
  136. }
  137. @inlinable
  138. internal func receiveInterceptedMetadata(_ headers: HPACKHeaders) {
  139. switch self.state {
  140. case .idle:
  141. // Make a context to invoke the observer block factory with.
  142. let context = _StreamingResponseCallContext<Request, Response>(
  143. eventLoop: self.context.eventLoop,
  144. headers: headers,
  145. logger: self.context.logger,
  146. userInfoRef: self.userInfoRef,
  147. compressionIsEnabled: self.context.encoding.isEnabled,
  148. closeFuture: self.context.closeFuture,
  149. sendResponse: self.interceptResponse(_:metadata:promise:)
  150. )
  151. // Move to the next state.
  152. self.state = .creatingObserver(context)
  153. // Send response headers back via the interceptors.
  154. self.interceptors.send(.metadata([:]), promise: nil)
  155. // Register callbacks on the status future.
  156. context.statusPromise.futureResult.whenComplete(self.userFunctionStatusResolved(_:))
  157. // Make an observer block and register a completion block.
  158. self.observerFactory(context).whenComplete(self.userFunctionResolvedWithResult(_:))
  159. case .creatingObserver, .observing:
  160. self.handleError(GRPCError.ProtocolViolation("Multiple header blocks received on RPC"))
  161. case .completed:
  162. // We may receive headers from the interceptor pipeline if we have already finished (i.e. due
  163. // to an error or otherwise) and an interceptor doing some async work later emitting headers.
  164. // Dropping them is fine.
  165. ()
  166. }
  167. }
  168. @inlinable
  169. internal func receiveInterceptedMessage(_ request: Request) {
  170. switch self.state {
  171. case .idle:
  172. self.handleError(GRPCError.ProtocolViolation("Message received before headers"))
  173. case .creatingObserver:
  174. self.requestBuffer.append(.message(request))
  175. case let .observing(observer, _):
  176. observer(.message(request))
  177. case .completed:
  178. // We received a message but we're already done: this may happen if we terminate the RPC
  179. // due to a channel error, for example.
  180. ()
  181. }
  182. }
  183. @inlinable
  184. internal func receiveInterceptedEnd() {
  185. switch self.state {
  186. case .idle:
  187. self.handleError(GRPCError.ProtocolViolation("End of stream received before headers"))
  188. case .creatingObserver:
  189. self.requestBuffer.append(.end)
  190. case let .observing(observer, _):
  191. observer(.end)
  192. case .completed:
  193. // We received a message but we're already done: this may happen if we terminate the RPC
  194. // due to a channel error, for example.
  195. ()
  196. }
  197. }
  198. // MARK: - User Function To Interceptors
  199. @inlinable
  200. internal func userFunctionResolvedWithResult(
  201. _ result: Result<(StreamEvent<Request>) -> Void, Error>
  202. ) {
  203. switch self.state {
  204. case .idle, .observing:
  205. // The observer block can't resolve if it hasn't been created ('idle') and it can't be
  206. // resolved more than once ('observing').
  207. preconditionFailure()
  208. case let .creatingObserver(context):
  209. switch result {
  210. case let .success(observer):
  211. // We have an observer block now; unbuffer any requests.
  212. self.state = .observing(observer, context)
  213. while let request = self.requestBuffer.popFirst() {
  214. observer(request)
  215. }
  216. case let .failure(error):
  217. self.handleError(error, thrownFromHandler: true)
  218. }
  219. case .completed:
  220. // We've already completed. That's fine.
  221. ()
  222. }
  223. }
  224. @inlinable
  225. internal func interceptResponse(
  226. _ response: Response,
  227. metadata: MessageMetadata,
  228. promise: EventLoopPromise<Void>?
  229. ) {
  230. switch self.state {
  231. case .idle:
  232. // The observer block can't end responses if it doesn't exist!
  233. preconditionFailure()
  234. case .creatingObserver, .observing:
  235. // The user has access to the response context before returning a future observer,
  236. // so 'creatingObserver' is valid here (if a little strange).
  237. self.interceptors.send(.message(response, metadata), promise: promise)
  238. case .completed:
  239. promise?.fail(GRPCError.AlreadyComplete())
  240. }
  241. }
  242. @inlinable
  243. internal func userFunctionStatusResolved(_ result: Result<GRPCStatus, Error>) {
  244. switch self.state {
  245. case .idle:
  246. // The promise can't fail before we create it.
  247. preconditionFailure()
  248. // Making is possible, the user can complete the status before returning a stream handler.
  249. case let .creatingObserver(context), let .observing(_, context):
  250. switch result {
  251. case let .success(status):
  252. // We're sending end back, we're done.
  253. self.state = .completed
  254. self.interceptors.send(.end(status, context.trailers), promise: nil)
  255. case let .failure(error):
  256. self.handleError(error, thrownFromHandler: true)
  257. }
  258. case .completed:
  259. ()
  260. }
  261. }
  262. @inlinable
  263. internal func handleError(_ error: Error, thrownFromHandler isHandlerError: Bool = false) {
  264. switch self.state {
  265. case .idle:
  266. assert(!isHandlerError)
  267. self.state = .completed
  268. // We don't have a promise to fail. Just send back end.
  269. let (status, trailers) = ServerErrorProcessor.processLibraryError(
  270. error,
  271. delegate: self.context.errorDelegate
  272. )
  273. self.interceptors.send(.end(status, trailers), promise: nil)
  274. case let .creatingObserver(context),
  275. let .observing(_, context):
  276. // We don't have a promise to fail. Just send back end.
  277. self.state = .completed
  278. let status: GRPCStatus
  279. let trailers: HPACKHeaders
  280. if isHandlerError {
  281. (status, trailers) = ServerErrorProcessor.processObserverError(
  282. error,
  283. headers: context.headers,
  284. trailers: context.trailers,
  285. delegate: self.context.errorDelegate
  286. )
  287. } else {
  288. (status, trailers) = ServerErrorProcessor.processLibraryError(
  289. error,
  290. delegate: self.context.errorDelegate
  291. )
  292. }
  293. self.interceptors.send(.end(status, trailers), promise: nil)
  294. // We're already in the 'completed' state so failing the promise will be a no-op in the
  295. // callback to 'userHandlerCompleted' (but we also need to avoid leaking the promise.)
  296. context.statusPromise.fail(error)
  297. case .completed:
  298. ()
  299. }
  300. }
  301. @inlinable
  302. internal func sendInterceptedPart(
  303. _ part: GRPCServerResponsePart<Response>,
  304. promise: EventLoopPromise<Void>?
  305. ) {
  306. switch part {
  307. case let .metadata(headers):
  308. self.context.responseWriter.sendMetadata(headers, flush: true, promise: promise)
  309. case let .message(message, metadata):
  310. do {
  311. let bytes = try self.serializer.serialize(message, allocator: ByteBufferAllocator())
  312. self.context.responseWriter.sendMessage(bytes, metadata: metadata, promise: promise)
  313. } catch {
  314. // Serialization failed: fail the promise and send end.
  315. promise?.fail(error)
  316. let (status, trailers) = ServerErrorProcessor.processLibraryError(
  317. error,
  318. delegate: self.context.errorDelegate
  319. )
  320. // Loop back via the interceptors.
  321. self.interceptors.send(.end(status, trailers), promise: nil)
  322. }
  323. case let .end(status, trailers):
  324. self.context.responseWriter.sendEnd(status: status, trailers: trailers, promise: promise)
  325. }
  326. }
  327. }