UnaryCallHandler.swift 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  1. /*
  2. * Copyright 2019, 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 Foundation
  17. import Logging
  18. import NIO
  19. import NIOHPACK
  20. import NIOHTTP1
  21. import SwiftProtobuf
  22. /// Handles unary calls. Calls the observer block with the request message.
  23. ///
  24. /// - The observer block is implemented by the framework user and returns a future containing the call result.
  25. /// - To return a response to the client, the framework user should complete that future
  26. /// (similar to e.g. serving regular HTTP requests in frameworks such as Vapor).
  27. public final class UnaryCallHandler<
  28. RequestDeserializer: MessageDeserializer,
  29. ResponseSerializer: MessageSerializer
  30. >: _BaseCallHandler<RequestDeserializer, ResponseSerializer> {
  31. private typealias Context = UnaryResponseCallContext<ResponsePayload>
  32. private typealias Observer = (RequestPayload) -> EventLoopFuture<ResponsePayload>
  33. private var state: State
  34. private enum State {
  35. // We don't have the following states (which we do have in the main state machine):
  36. // - 'requestOpenResponseIdle',
  37. // - 'requestClosedResponseIdle'
  38. //
  39. // We'll send headers back when we transition away from 'requestIdleResponseIdle' so the
  40. // response stream can never be less idle than the request stream.
  41. /// Fully idle, we haven't seen the request headers yet and we haven't made an event observer
  42. /// yet.
  43. case requestIdleResponseIdle((Context) -> Observer)
  44. /// Received the request headers, created an observer and have sent back response headers.
  45. /// We may or may not have observer the request message yet.
  46. case requestOpenResponseOpen(Context, ObserverState)
  47. /// Received the request headers, a message and the end of the request stream. The observer has
  48. /// been invoked but it hasn't yet finished processing the request.
  49. ///
  50. /// Note: we know we've received a message if we're in this state, if we had seen the request
  51. /// headers followed by end we'd fully close.
  52. case requestClosedResponseOpen(Context)
  53. /// We're done.
  54. case requestClosedResponseClosed
  55. /// The state of the event observer.
  56. enum ObserverState {
  57. /// We have an event observer, but haven't yet received a request.
  58. case notObserved(Observer)
  59. /// We've invoked the event observer with a request.
  60. case observed
  61. }
  62. }
  63. internal init(
  64. serializer: ResponseSerializer,
  65. deserializer: RequestDeserializer,
  66. callHandlerContext: CallHandlerContext,
  67. interceptors: [ServerInterceptor<RequestDeserializer.Output, ResponseSerializer.Input>],
  68. eventObserverFactory: @escaping (UnaryResponseCallContext<ResponsePayload>)
  69. -> (RequestPayload) -> EventLoopFuture<ResponsePayload>
  70. ) {
  71. self.state = .requestIdleResponseIdle(eventObserverFactory)
  72. super.init(
  73. callHandlerContext: callHandlerContext,
  74. requestDeserializr: deserializer,
  75. responseSerializer: serializer,
  76. callType: .unary,
  77. interceptors: interceptors
  78. )
  79. }
  80. override public func channelInactive(context: ChannelHandlerContext) {
  81. super.channelInactive(context: context)
  82. // Fail any remaining promise.
  83. switch self.state {
  84. case .requestIdleResponseIdle,
  85. .requestClosedResponseClosed:
  86. self.state = .requestClosedResponseClosed
  87. case let .requestOpenResponseOpen(context, _),
  88. let .requestClosedResponseOpen(context):
  89. self.state = .requestClosedResponseClosed
  90. context.responsePromise.fail(GRPCError.AlreadyComplete())
  91. }
  92. }
  93. /// Handle an error from the event observer.
  94. private func handleObserverError(_ error: Error) {
  95. switch self.state {
  96. case .requestIdleResponseIdle:
  97. preconditionFailure("Invalid state: request observer hasn't been created")
  98. case let .requestOpenResponseOpen(context, _),
  99. let .requestClosedResponseOpen(context):
  100. let (status, trailers) = self.processObserverError(
  101. error,
  102. headers: context.headers,
  103. trailers: context.trailers
  104. )
  105. // This will handle the response promise as well.
  106. self.sendEnd(status: status, trailers: trailers)
  107. case .requestClosedResponseClosed:
  108. // We hit an error, but we're already closed (i.e. we hit a library error first). Ignore
  109. // the error.
  110. ()
  111. }
  112. }
  113. /// Handle a 'library' error, i.e. an error emanating from the `Channel`.
  114. private func handleLibraryError(_ error: Error) {
  115. switch self.state {
  116. case .requestIdleResponseIdle,
  117. .requestOpenResponseOpen(_, .notObserved):
  118. // We haven't seen a message, we'll send end to close the stream.
  119. let (status, trailers) = self.processLibraryError(error)
  120. self.sendEnd(status: status, trailers: trailers)
  121. case .requestOpenResponseOpen(_, .observed),
  122. .requestClosedResponseOpen:
  123. // We've seen a message, the observer is in flight, we'll let it play out.
  124. ()
  125. case .requestClosedResponseClosed:
  126. // We're already closed, we can just ignore this.
  127. ()
  128. }
  129. }
  130. // MARK: - Inbound
  131. override internal func observeLibraryError(_ error: Error) {
  132. self.handleLibraryError(error)
  133. }
  134. override internal func observeHeaders(_ headers: HPACKHeaders) {
  135. switch self.state {
  136. case let .requestIdleResponseIdle(factory):
  137. // This allocates a promise, but the observer is provided with 'StatusOnlyCallContext' and
  138. // doesn't get access to the promise. The observer must return a response future instead
  139. // which we cascade to this promise. We can avoid this extra allocation by using a different
  140. // context here.
  141. //
  142. // TODO: provide a new context without a promise.
  143. let context = UnaryResponseCallContext<ResponsePayload>(
  144. eventLoop: self.eventLoop,
  145. headers: headers,
  146. logger: self.logger,
  147. userInfoRef: self.userInfoRef
  148. )
  149. let observer = factory(context)
  150. // We're fully open now (we'll send the response headers back in a moment).
  151. self.state = .requestOpenResponseOpen(context, .notObserved(observer))
  152. // Register callbacks for the response promise.
  153. context.responsePromise.futureResult.whenComplete { result in
  154. switch result {
  155. case let .success(response):
  156. self.sendResponse(response)
  157. case let .failure(error):
  158. self.handleObserverError(error)
  159. }
  160. }
  161. // Write back the response headers.
  162. self.sendResponsePartFromObserver(.metadata([:]), promise: nil)
  163. // The main state machine guards against these states.
  164. case .requestOpenResponseOpen,
  165. .requestClosedResponseOpen,
  166. .requestClosedResponseClosed:
  167. preconditionFailure("Invalid state: request headers already received")
  168. }
  169. }
  170. override internal func observeRequest(_ message: RequestPayload) {
  171. switch self.state {
  172. case .requestIdleResponseIdle:
  173. preconditionFailure("Invalid state: request received before headers")
  174. case let .requestOpenResponseOpen(context, request):
  175. switch request {
  176. case .observed:
  177. // We've already observed the request message. The main state machine doesn't guard against
  178. // too many messages for unary streams. Assuming downstream handlers protect against this
  179. // then this must be an errant interceptor, we'll ignore it.
  180. ()
  181. case let .notObserved(observer):
  182. self.state = .requestOpenResponseOpen(context, .observed)
  183. // Complete the promise with the observer block.
  184. context.responsePromise.completeWith(observer(message))
  185. }
  186. case .requestClosedResponseOpen,
  187. .requestClosedResponseClosed:
  188. preconditionFailure("Invalid state: the request stream has already been closed")
  189. }
  190. }
  191. override internal func observeEnd() {
  192. switch self.state {
  193. case .requestIdleResponseIdle:
  194. preconditionFailure("Invalid state: no request headers received")
  195. case let .requestOpenResponseOpen(context, request):
  196. switch request {
  197. case .observed:
  198. // Close the request stream.
  199. self.state = .requestClosedResponseOpen(context)
  200. case .notObserved:
  201. // We haven't received a request: this is an empty stream, the observer will never be
  202. // invoked.
  203. context.responsePromise.fail(GRPCError.StreamCardinalityViolation.request)
  204. }
  205. case .requestClosedResponseOpen,
  206. .requestClosedResponseClosed:
  207. preconditionFailure("Invalid state: request stream is already closed")
  208. }
  209. }
  210. // MARK: - Outbound
  211. private func sendResponse(_ message: ResponsePayload) {
  212. switch self.state {
  213. case .requestIdleResponseIdle:
  214. preconditionFailure("Invalid state: can't send response before receiving headers and request")
  215. case .requestOpenResponseOpen(_, .notObserved):
  216. preconditionFailure("Invalid state: can't send response before receiving request")
  217. case let .requestOpenResponseOpen(context, .observed),
  218. let .requestClosedResponseOpen(context):
  219. self.state = .requestClosedResponseClosed
  220. self.sendResponsePartFromObserver(
  221. .message(message, .init(compress: context.compressionEnabled, flush: false)),
  222. promise: nil
  223. )
  224. self.sendResponsePartFromObserver(
  225. .end(context.responseStatus, context.trailers),
  226. promise: nil
  227. )
  228. case .requestClosedResponseClosed:
  229. // Already closed, do nothing.
  230. ()
  231. }
  232. }
  233. private func sendEnd(status: GRPCStatus, trailers: HPACKHeaders) {
  234. switch self.state {
  235. case .requestIdleResponseIdle,
  236. .requestClosedResponseOpen:
  237. self.state = .requestClosedResponseClosed
  238. self.sendResponsePartFromObserver(.end(status, trailers), promise: nil)
  239. case let .requestOpenResponseOpen(context, _):
  240. self.state = .requestClosedResponseClosed
  241. self.sendResponsePartFromObserver(.end(status, trailers), promise: nil)
  242. // Fail the promise.
  243. context.responsePromise.fail(status)
  244. case .requestClosedResponseClosed:
  245. // Already closed, do nothing.
  246. ()
  247. }
  248. }
  249. }