ClientStreamingServerHandler.swift 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365
  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 ClientStreamingServerHandler<
  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 handlerFactory: (UnaryResponseCallContext<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. // Nothing has happened yet.
  52. case idle
  53. // Headers have been received, a context has been created and the user code has been called to
  54. // make an observer with. The observer future hasn't completed yet and, as such, the observer
  55. // is yet to see any events.
  56. case creatingObserver(UnaryResponseCallContext<Response>)
  57. // The observer future has succeeded, messages may have been delivered to it.
  58. case observing((StreamEvent<Request>) -> Void, UnaryResponseCallContext<Response>)
  59. // The observer has completed by completing the status promise.
  60. case completed
  61. }
  62. @inlinable
  63. public init(
  64. context: CallHandlerContext,
  65. requestDeserializer: Deserializer,
  66. responseSerializer: Serializer,
  67. interceptors: [ServerInterceptor<Request, Response>],
  68. observerFactory: @escaping (UnaryResponseCallContext<Response>)
  69. -> EventLoopFuture<(StreamEvent<Request>) -> Void>
  70. ) {
  71. self.serializer = responseSerializer
  72. self.deserializer = requestDeserializer
  73. self.context = context
  74. self.handlerFactory = observerFactory
  75. let userInfoRef = Ref(UserInfo())
  76. self.userInfoRef = userInfoRef
  77. self.interceptors = ServerInterceptorPipeline(
  78. logger: context.logger,
  79. eventLoop: context.eventLoop,
  80. path: context.path,
  81. callType: .clientStreaming,
  82. remoteAddress: context.remoteAddress,
  83. userInfoRef: userInfoRef,
  84. closeFuture: context.closeFuture,
  85. interceptors: interceptors,
  86. onRequestPart: self.receiveInterceptedPart(_:),
  87. onResponsePart: self.sendInterceptedPart(_:promise:)
  88. )
  89. }
  90. // MARK: Public API; gRPC to Handler
  91. @inlinable
  92. public func receiveMetadata(_ headers: HPACKHeaders) {
  93. self.interceptors.receive(.metadata(headers))
  94. }
  95. @inlinable
  96. public func receiveMessage(_ bytes: ByteBuffer) {
  97. do {
  98. let message = try self.deserializer.deserialize(byteBuffer: bytes)
  99. self.interceptors.receive(.message(message))
  100. } catch {
  101. self.handleError(error)
  102. }
  103. }
  104. @inlinable
  105. public func receiveEnd() {
  106. self.interceptors.receive(.end)
  107. }
  108. @inlinable
  109. public func receiveError(_ error: Error) {
  110. self.handleError(error)
  111. self.finish()
  112. }
  113. @inlinable
  114. public func finish() {
  115. switch self.state {
  116. case .idle:
  117. self.interceptors = nil
  118. self.state = .completed
  119. case let .creatingObserver(context),
  120. let .observing(_, context):
  121. context.responsePromise.fail(GRPCStatus(code: .unavailable, message: nil))
  122. case .completed:
  123. self.interceptors = nil
  124. }
  125. }
  126. // MARK: Interceptors to User Function
  127. @inlinable
  128. internal func receiveInterceptedPart(_ part: GRPCServerRequestPart<Request>) {
  129. switch part {
  130. case let .metadata(headers):
  131. self.receiveInterceptedMetadata(headers)
  132. case let .message(message):
  133. self.receiveInterceptedMessage(message)
  134. case .end:
  135. self.receiveInterceptedEnd()
  136. }
  137. }
  138. @inlinable
  139. internal func receiveInterceptedMetadata(_ headers: HPACKHeaders) {
  140. switch self.state {
  141. case .idle:
  142. // Make a context to invoke the observer block factory with.
  143. let context = UnaryResponseCallContext<Response>(
  144. eventLoop: self.context.eventLoop,
  145. headers: headers,
  146. logger: self.context.logger,
  147. userInfoRef: self.userInfoRef,
  148. closeFuture: self.context.closeFuture
  149. )
  150. // Move to the next state.
  151. self.state = .creatingObserver(context)
  152. // Register a callback on the response future.
  153. context.responsePromise.futureResult.whenComplete(self.userFunctionCompletedWithResult(_:))
  154. // Make an observer block and register a completion block.
  155. self.handlerFactory(context).whenComplete(self.userFunctionResolved(_:))
  156. // Send response headers back via the interceptors.
  157. self.interceptors.send(.metadata([:]), promise: nil)
  158. case .creatingObserver, .observing:
  159. self.handleError(GRPCError.ProtocolViolation("Multiple header blocks received"))
  160. case .completed:
  161. // We may receive headers from the interceptor pipeline if we have already finished (i.e. due
  162. // to an error or otherwise) and an interceptor doing some async work later emitting headers.
  163. // Dropping them is fine.
  164. ()
  165. }
  166. }
  167. @inlinable
  168. internal func receiveInterceptedMessage(_ request: Request) {
  169. switch self.state {
  170. case .idle:
  171. self.handleError(GRPCError.ProtocolViolation("Message received before headers"))
  172. case .creatingObserver:
  173. self.requestBuffer.append(.message(request))
  174. case let .observing(observer, _):
  175. observer(.message(request))
  176. case .completed:
  177. // We received a message but we're already done: this may happen if we terminate the RPC
  178. // due to a channel error, for example.
  179. ()
  180. }
  181. }
  182. @inlinable
  183. internal func receiveInterceptedEnd() {
  184. switch self.state {
  185. case .idle:
  186. self.handleError(GRPCError.ProtocolViolation("end received before headers"))
  187. case .creatingObserver:
  188. self.requestBuffer.append(.end)
  189. case let .observing(observer, _):
  190. observer(.end)
  191. case .completed:
  192. // We received a message but we're already done: this may happen if we terminate the RPC
  193. // due to a channel error, for example.
  194. ()
  195. }
  196. }
  197. // MARK: User Function to Interceptors
  198. @inlinable
  199. internal func userFunctionResolved(_ result: Result<(StreamEvent<Request>) -> Void, Error>) {
  200. switch self.state {
  201. case .idle, .observing:
  202. // The observer block can't resolve if it hasn't been created ('idle') and it can't be
  203. // resolved more than once ('created').
  204. preconditionFailure()
  205. case let .creatingObserver(context):
  206. switch result {
  207. case let .success(observer):
  208. // We have an observer block now; unbuffer any requests.
  209. self.state = .observing(observer, context)
  210. while let request = self.requestBuffer.popFirst() {
  211. observer(request)
  212. }
  213. case let .failure(error):
  214. self.handleError(error, thrownFromHandler: true)
  215. }
  216. case .completed:
  217. // We've already completed. That's fine.
  218. ()
  219. }
  220. }
  221. @inlinable
  222. internal func userFunctionCompletedWithResult(_ result: Result<Response, Error>) {
  223. switch self.state {
  224. case .idle:
  225. // Invalid state: the user function can only complete if it exists..
  226. preconditionFailure()
  227. case let .creatingObserver(context),
  228. let .observing(_, context):
  229. switch result {
  230. case let .success(response):
  231. // Complete when we send end.
  232. self.state = .completed
  233. // Compression depends on whether it's enabled on the server and the setting in the caller
  234. // context.
  235. let compress = self.context.encoding.isEnabled && context.compressionEnabled
  236. let metadata = MessageMetadata(compress: compress, flush: false)
  237. self.interceptors.send(.message(response, metadata), promise: nil)
  238. self.interceptors.send(.end(context.responseStatus, context.trailers), promise: nil)
  239. case let .failure(error):
  240. self.handleError(error, thrownFromHandler: true)
  241. }
  242. case .completed:
  243. // We've already completed. Ignore this.
  244. ()
  245. }
  246. }
  247. @inlinable
  248. internal func handleError(_ error: Error, thrownFromHandler isHandlerError: Bool = false) {
  249. switch self.state {
  250. case .idle:
  251. assert(!isHandlerError)
  252. self.state = .completed
  253. // We don't have a promise to fail. Just send back end.
  254. let (status, trailers) = ServerErrorProcessor.processLibraryError(
  255. error,
  256. delegate: self.context.errorDelegate
  257. )
  258. self.interceptors.send(.end(status, trailers), promise: nil)
  259. case let .creatingObserver(context),
  260. let .observing(_, context):
  261. // We don't have a promise to fail. Just send back end.
  262. self.state = .completed
  263. let status: GRPCStatus
  264. let trailers: HPACKHeaders
  265. if isHandlerError {
  266. (status, trailers) = ServerErrorProcessor.processObserverError(
  267. error,
  268. headers: context.headers,
  269. trailers: context.trailers,
  270. delegate: self.context.errorDelegate
  271. )
  272. } else {
  273. (status, trailers) = ServerErrorProcessor.processLibraryError(
  274. error,
  275. delegate: self.context.errorDelegate
  276. )
  277. }
  278. self.interceptors.send(.end(status, trailers), promise: nil)
  279. // We're already in the 'completed' state so failing the promise will be a no-op in the
  280. // callback to 'userFunctionCompletedWithResult' (but we also need to avoid leaking the
  281. // promise.)
  282. context.responsePromise.fail(error)
  283. case .completed:
  284. ()
  285. }
  286. }
  287. // MARK: Interceptor Glue
  288. @inlinable
  289. internal func sendInterceptedPart(
  290. _ part: GRPCServerResponsePart<Response>,
  291. promise: EventLoopPromise<Void>?
  292. ) {
  293. switch part {
  294. case let .metadata(headers):
  295. self.context.responseWriter.sendMetadata(headers, flush: true, promise: promise)
  296. case let .message(message, metadata):
  297. do {
  298. let bytes = try self.serializer.serialize(message, allocator: ByteBufferAllocator())
  299. self.context.responseWriter.sendMessage(bytes, metadata: metadata, promise: promise)
  300. } catch {
  301. // Serialization failed: fail the promise and send end.
  302. promise?.fail(error)
  303. let (status, trailers) = ServerErrorProcessor.processLibraryError(
  304. error,
  305. delegate: self.context.errorDelegate
  306. )
  307. // Loop back via the interceptors.
  308. self.interceptors.send(.end(status, trailers), promise: nil)
  309. }
  310. case let .end(status, trailers):
  311. self.context.responseWriter.sendEnd(status: status, trailers: trailers, promise: promise)
  312. }
  313. }
  314. }