GRPCAsyncServerHandler.swift 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861
  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 DequeModule
  17. import Logging
  18. import NIOCore
  19. import NIOHPACK
  20. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  21. public struct GRPCAsyncServerHandler<
  22. Serializer: MessageSerializer,
  23. Deserializer: MessageDeserializer,
  24. Request: Sendable,
  25. Response: Sendable
  26. >: GRPCServerHandlerProtocol where Serializer.Input == Response, Deserializer.Output == Request {
  27. @usableFromInline
  28. internal let _handler: AsyncServerHandler<Serializer, Deserializer, Request, Response>
  29. public func receiveMetadata(_ metadata: HPACKHeaders) {
  30. self._handler.receiveMetadata(metadata)
  31. }
  32. public func receiveMessage(_ bytes: ByteBuffer) {
  33. self._handler.receiveMessage(bytes)
  34. }
  35. public func receiveEnd() {
  36. self._handler.receiveEnd()
  37. }
  38. public func receiveError(_ error: Error) {
  39. self._handler.receiveError(error)
  40. }
  41. public func finish() {
  42. self._handler.finish()
  43. }
  44. }
  45. // MARK: - RPC Adapters
  46. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  47. extension GRPCAsyncServerHandler {
  48. public typealias Request = Deserializer.Output
  49. public typealias Response = Serializer.Input
  50. @inlinable
  51. public init(
  52. context: CallHandlerContext,
  53. requestDeserializer: Deserializer,
  54. responseSerializer: Serializer,
  55. interceptors: [ServerInterceptor<Request, Response>],
  56. wrapping unary: @escaping @Sendable (Request, GRPCAsyncServerCallContext) async throws
  57. -> Response
  58. ) {
  59. self._handler = .init(
  60. context: context,
  61. requestDeserializer: requestDeserializer,
  62. responseSerializer: responseSerializer,
  63. callType: .unary,
  64. interceptors: interceptors,
  65. userHandler: { requestStream, responseStreamWriter, context in
  66. var iterator = requestStream.makeAsyncIterator()
  67. guard let request = try await iterator.next(), try await iterator.next() == nil else {
  68. throw GRPCError.ProtocolViolation("Unary RPC expects exactly one request")
  69. }
  70. let response = try await unary(request, context)
  71. try await responseStreamWriter.send(response)
  72. }
  73. )
  74. }
  75. @inlinable
  76. public init(
  77. context: CallHandlerContext,
  78. requestDeserializer: Deserializer,
  79. responseSerializer: Serializer,
  80. interceptors: [ServerInterceptor<Request, Response>],
  81. wrapping clientStreaming: @escaping @Sendable (
  82. GRPCAsyncRequestStream<Request>,
  83. GRPCAsyncServerCallContext
  84. ) async throws -> Response
  85. ) {
  86. self._handler = .init(
  87. context: context,
  88. requestDeserializer: requestDeserializer,
  89. responseSerializer: responseSerializer,
  90. callType: .clientStreaming,
  91. interceptors: interceptors,
  92. userHandler: { requestStream, responseStreamWriter, context in
  93. let response = try await clientStreaming(requestStream, context)
  94. try await responseStreamWriter.send(response)
  95. }
  96. )
  97. }
  98. @inlinable
  99. public init(
  100. context: CallHandlerContext,
  101. requestDeserializer: Deserializer,
  102. responseSerializer: Serializer,
  103. interceptors: [ServerInterceptor<Request, Response>],
  104. wrapping serverStreaming: @escaping @Sendable (
  105. Request,
  106. GRPCAsyncResponseStreamWriter<Response>,
  107. GRPCAsyncServerCallContext
  108. ) async throws -> Void
  109. ) {
  110. self._handler = .init(
  111. context: context,
  112. requestDeserializer: requestDeserializer,
  113. responseSerializer: responseSerializer,
  114. callType: .serverStreaming,
  115. interceptors: interceptors,
  116. userHandler: { requestStream, responseStreamWriter, context in
  117. var iterator = requestStream.makeAsyncIterator()
  118. guard let request = try await iterator.next(), try await iterator.next() == nil else {
  119. throw GRPCError.ProtocolViolation("Server-streaming RPC expects exactly one request")
  120. }
  121. try await serverStreaming(request, responseStreamWriter, context)
  122. }
  123. )
  124. }
  125. @inlinable
  126. public init(
  127. context: CallHandlerContext,
  128. requestDeserializer: Deserializer,
  129. responseSerializer: Serializer,
  130. interceptors: [ServerInterceptor<Request, Response>],
  131. wrapping bidirectional: @escaping @Sendable (
  132. GRPCAsyncRequestStream<Request>,
  133. GRPCAsyncResponseStreamWriter<Response>,
  134. GRPCAsyncServerCallContext
  135. ) async throws -> Void
  136. ) {
  137. self._handler = .init(
  138. context: context,
  139. requestDeserializer: requestDeserializer,
  140. responseSerializer: responseSerializer,
  141. callType: .bidirectionalStreaming,
  142. interceptors: interceptors,
  143. userHandler: bidirectional
  144. )
  145. }
  146. }
  147. // MARK: - Server Handler
  148. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  149. @usableFromInline
  150. internal final class AsyncServerHandler<
  151. Serializer: MessageSerializer,
  152. Deserializer: MessageDeserializer,
  153. Request: Sendable,
  154. Response: Sendable
  155. >: GRPCServerHandlerProtocol where Serializer.Input == Response, Deserializer.Output == Request {
  156. /// A response serializer.
  157. @usableFromInline
  158. internal let serializer: Serializer
  159. /// A request deserializer.
  160. @usableFromInline
  161. internal let deserializer: Deserializer
  162. /// The event loop that this handler executes on.
  163. @usableFromInline
  164. internal let eventLoop: EventLoop
  165. /// A `ByteBuffer` allocator provided by the underlying `Channel`.
  166. @usableFromInline
  167. internal let allocator: ByteBufferAllocator
  168. /// A user-provided error delegate which, if provided, is used to transform errors and potentially
  169. /// pack errors into trailers.
  170. @usableFromInline
  171. internal let errorDelegate: ServerErrorDelegate?
  172. /// A logger.
  173. @usableFromInline
  174. internal let logger: Logger
  175. /// A reference to the user info. This is shared with the interceptor pipeline and may be accessed
  176. /// from the async call context. `UserInfo` is _not_ `Sendable` and must always be accessed from
  177. /// an appropriate event loop.
  178. @usableFromInline
  179. internal let userInfoRef: Ref<UserInfo>
  180. /// Whether compression is enabled on the server and an algorithm has been negotiated with
  181. /// the client
  182. @usableFromInline
  183. internal let compressionEnabledOnRPC: Bool
  184. /// Whether the RPC method would like to compress responses (if possible). Defaults to true.
  185. @usableFromInline
  186. internal var compressResponsesIfPossible: Bool
  187. /// A state machine for the interceptor pipeline.
  188. @usableFromInline
  189. internal private(set) var interceptorStateMachine: ServerInterceptorStateMachine
  190. /// The interceptor pipeline.
  191. @usableFromInline
  192. internal private(set) var interceptors: Optional<ServerInterceptorPipeline<Request, Response>>
  193. /// An object for writing intercepted responses to the channel.
  194. @usableFromInline
  195. internal private(set) var responseWriter: Optional<GRPCServerResponseWriter>
  196. /// A state machine for the user implemented function.
  197. @usableFromInline
  198. internal private(set) var handlerStateMachine: ServerHandlerStateMachine
  199. /// A bag of components used by the user handler.
  200. @usableFromInline
  201. internal private(set) var handlerComponents: Optional<ServerHandlerComponents<
  202. Request,
  203. Response,
  204. GRPCAsyncWriterSinkDelegate<(Response, Compression)>
  205. >>
  206. /// The user provided function to execute.
  207. @usableFromInline
  208. internal let userHandler: @Sendable (
  209. GRPCAsyncRequestStream<Request>,
  210. GRPCAsyncResponseStreamWriter<Response>,
  211. GRPCAsyncServerCallContext
  212. ) async throws -> Void
  213. @usableFromInline
  214. internal typealias AsyncSequenceProducer = NIOThrowingAsyncSequenceProducer<
  215. Request,
  216. Error,
  217. NIOAsyncSequenceProducerBackPressureStrategies.HighLowWatermark,
  218. GRPCAsyncSequenceProducerDelegate
  219. >
  220. @inlinable
  221. internal init(
  222. context: CallHandlerContext,
  223. requestDeserializer: Deserializer,
  224. responseSerializer: Serializer,
  225. callType: GRPCCallType,
  226. interceptors: [ServerInterceptor<Request, Response>],
  227. userHandler: @escaping @Sendable (
  228. GRPCAsyncRequestStream<Request>,
  229. GRPCAsyncResponseStreamWriter<Response>,
  230. GRPCAsyncServerCallContext
  231. ) async throws -> Void
  232. ) {
  233. self.serializer = responseSerializer
  234. self.deserializer = requestDeserializer
  235. self.eventLoop = context.eventLoop
  236. self.allocator = context.allocator
  237. self.responseWriter = context.responseWriter
  238. self.errorDelegate = context.errorDelegate
  239. self.compressionEnabledOnRPC = context.encoding.isEnabled
  240. self.compressResponsesIfPossible = true
  241. self.logger = context.logger
  242. self.userInfoRef = Ref(UserInfo())
  243. self.handlerStateMachine = .init()
  244. self.handlerComponents = nil
  245. self.userHandler = userHandler
  246. self.interceptorStateMachine = .init()
  247. self.interceptors = nil
  248. self.interceptors = ServerInterceptorPipeline(
  249. logger: context.logger,
  250. eventLoop: context.eventLoop,
  251. path: context.path,
  252. callType: callType,
  253. remoteAddress: context.remoteAddress,
  254. userInfoRef: self.userInfoRef,
  255. closeFuture: context.closeFuture,
  256. interceptors: interceptors,
  257. onRequestPart: self.receiveInterceptedPart(_:),
  258. onResponsePart: self.sendInterceptedPart(_:promise:)
  259. )
  260. }
  261. // MARK: - GRPCServerHandlerProtocol conformance
  262. @inlinable
  263. internal func receiveMetadata(_ headers: HPACKHeaders) {
  264. switch self.interceptorStateMachine.interceptRequestMetadata() {
  265. case .intercept:
  266. self.interceptors?.receive(.metadata(headers))
  267. case .cancel:
  268. self.cancel(error: nil)
  269. case .drop:
  270. ()
  271. }
  272. }
  273. @inlinable
  274. internal func receiveMessage(_ bytes: ByteBuffer) {
  275. let request: Request
  276. do {
  277. request = try self.deserializer.deserialize(byteBuffer: bytes)
  278. } catch {
  279. return self.cancel(error: error)
  280. }
  281. switch self.interceptorStateMachine.interceptRequestMessage() {
  282. case .intercept:
  283. self.interceptors?.receive(.message(request))
  284. case .cancel:
  285. self.cancel(error: nil)
  286. case .drop:
  287. ()
  288. }
  289. }
  290. @inlinable
  291. internal func receiveEnd() {
  292. switch self.interceptorStateMachine.interceptRequestEnd() {
  293. case .intercept:
  294. self.interceptors?.receive(.end)
  295. case .cancel:
  296. self.cancel(error: nil)
  297. case .drop:
  298. ()
  299. }
  300. }
  301. @inlinable
  302. internal func receiveError(_ error: Error) {
  303. self.cancel(error: error)
  304. }
  305. @inlinable
  306. internal func finish() {
  307. self.cancel(error: nil)
  308. }
  309. @usableFromInline
  310. internal func cancel(error: Error?) {
  311. self.eventLoop.assertInEventLoop()
  312. switch self.handlerStateMachine.cancel() {
  313. case .cancelAndNilOutHandlerComponents:
  314. // Cancel handler related things (task, response writer).
  315. self.handlerComponents?.cancel()
  316. self.handlerComponents = nil
  317. // We don't distinguish between having sent the status or not; we just tell the interceptor
  318. // state machine that we want to send a response status. It will inform us whether to
  319. // generate and send one or not.
  320. switch self.interceptorStateMachine.interceptedResponseStatus() {
  321. case .forward:
  322. let error = error ?? GRPCStatus.processingError
  323. let (status, trailers) = ServerErrorProcessor.processLibraryError(
  324. error,
  325. delegate: self.errorDelegate
  326. )
  327. self.responseWriter?.sendEnd(status: status, trailers: trailers, promise: nil)
  328. case .drop, .cancel:
  329. ()
  330. }
  331. case .none:
  332. ()
  333. }
  334. switch self.interceptorStateMachine.cancel() {
  335. case .sendStatusThenNilOutInterceptorPipeline:
  336. self.responseWriter?.sendEnd(status: .processingError, trailers: [:], promise: nil)
  337. fallthrough
  338. case .nilOutInterceptorPipeline:
  339. self.interceptors = nil
  340. self.responseWriter = nil
  341. case .none:
  342. ()
  343. }
  344. }
  345. // MARK: - Interceptors to User Function
  346. @inlinable
  347. internal func receiveInterceptedPart(_ part: GRPCServerRequestPart<Request>) {
  348. switch part {
  349. case let .metadata(headers):
  350. self.receiveInterceptedMetadata(headers)
  351. case let .message(message):
  352. self.receiveInterceptedMessage(message)
  353. case .end:
  354. self.receiveInterceptedEnd()
  355. }
  356. }
  357. @inlinable
  358. internal func receiveInterceptedMetadata(_ headers: HPACKHeaders) {
  359. switch self.interceptorStateMachine.interceptedRequestMetadata() {
  360. case .forward:
  361. () // continue
  362. case .cancel:
  363. return self.cancel(error: nil)
  364. case .drop:
  365. return
  366. }
  367. switch self.handlerStateMachine.handleMetadata() {
  368. case .invokeHandler:
  369. // We're going to invoke the handler. We need to create a handful of things in order to do
  370. // that:
  371. //
  372. // - A context which allows the handler to set response headers/trailers and provides them
  373. // with a logger amongst other things.
  374. // - A request source; we push request messages into this which the handler consumes via
  375. // an async sequence.
  376. // - An async writer and delegate. The delegate calls us back with responses. The writer is
  377. // passed to the handler.
  378. //
  379. // All of these components are held in a bundle ("handler components") outside of the state
  380. // machine. We release these when we eventually call cancel (either when we `self.cancel()`
  381. // as a result of an error or when `self.finish()` is called).
  382. let handlerContext = GRPCAsyncServerCallContext(
  383. headers: headers,
  384. logger: self.logger,
  385. contextProvider: self
  386. )
  387. let backpressureStrategy = NIOAsyncSequenceProducerBackPressureStrategies.HighLowWatermark(
  388. lowWatermark: 10,
  389. highWatermark: 50
  390. )
  391. let requestSequenceProducer = NIOThrowingAsyncSequenceProducer.makeSequence(
  392. elementType: Request.self,
  393. failureType: Error.self,
  394. backPressureStrategy: backpressureStrategy,
  395. delegate: GRPCAsyncSequenceProducerDelegate()
  396. )
  397. let responseWriter = NIOAsyncWriter.makeWriter(
  398. isWritable: true,
  399. delegate: GRPCAsyncWriterSinkDelegate<(Response, Compression)>(
  400. didYield: self.interceptResponseMessages,
  401. didTerminate: { error in
  402. self.interceptTermination(error)
  403. }
  404. )
  405. )
  406. // Update our state before invoke the handler.
  407. self.handlerStateMachine.handlerInvoked(requestHeaders: headers)
  408. self.handlerComponents = ServerHandlerComponents<
  409. Request,
  410. Response,
  411. GRPCAsyncWriterSinkDelegate<(Response, Compression)>
  412. >(
  413. requestSource: requestSequenceProducer.source,
  414. responseWriterSink: responseWriter.sink,
  415. task: Task {
  416. // We don't have a task cancellation handler here: we do it in `self.cancel()`.
  417. await self.invokeUserHandler(
  418. requestSequence: requestSequenceProducer,
  419. responseWriter: responseWriter.writer,
  420. callContext: handlerContext
  421. )
  422. }
  423. )
  424. case .cancel:
  425. self.cancel(error: nil)
  426. }
  427. }
  428. @Sendable
  429. @usableFromInline
  430. internal func invokeUserHandler(
  431. requestSequence: AsyncSequenceProducer.NewSequence,
  432. responseWriter: NIOAsyncWriter<
  433. (Response, Compression),
  434. GRPCAsyncWriterSinkDelegate<(Response, Compression)>
  435. >,
  436. callContext: GRPCAsyncServerCallContext
  437. ) async {
  438. defer {
  439. // It's possible the user handler completed before the end of the request stream. We
  440. // explicitly finish it to drop any unconsumed inbound messages.
  441. requestSequence.source.finish()
  442. }
  443. do {
  444. let grpcRequestStream = GRPCAsyncRequestStream(requestSequence.sequence)
  445. let grpcResponseStreamWriter = GRPCAsyncResponseStreamWriter(wrapping: responseWriter)
  446. try await self.userHandler(grpcRequestStream, grpcResponseStreamWriter, callContext)
  447. responseWriter.finish()
  448. } catch {
  449. responseWriter.finish(error: error)
  450. }
  451. }
  452. @inlinable
  453. internal func receiveInterceptedMessage(_ request: Request) {
  454. switch self.interceptorStateMachine.interceptedRequestMessage() {
  455. case .forward:
  456. switch self.handlerStateMachine.handleMessage() {
  457. case .forward:
  458. _ = self.handlerComponents?.requestSource.yield(request)
  459. case .cancel:
  460. self.cancel(error: nil)
  461. }
  462. case .cancel:
  463. self.cancel(error: nil)
  464. case .drop:
  465. ()
  466. }
  467. }
  468. @inlinable
  469. internal func receiveInterceptedEnd() {
  470. switch self.interceptorStateMachine.interceptedRequestEnd() {
  471. case .forward:
  472. switch self.handlerStateMachine.handleEnd() {
  473. case .forward:
  474. self.handlerComponents?.requestSource.finish()
  475. case .cancel:
  476. self.cancel(error: nil)
  477. }
  478. case .cancel:
  479. self.cancel(error: nil)
  480. case .drop:
  481. ()
  482. }
  483. }
  484. // MARK: - User Function To Interceptors
  485. @inlinable
  486. internal func _interceptResponseMessage(_ response: Response, compression: Compression) {
  487. self.eventLoop.assertInEventLoop()
  488. switch self.handlerStateMachine.sendMessage() {
  489. case let .intercept(.some(headers)):
  490. switch self.interceptorStateMachine.interceptResponseMetadata() {
  491. case .intercept:
  492. self.interceptors?.send(.metadata(headers), promise: nil)
  493. case .cancel:
  494. return self.cancel(error: nil)
  495. case .drop:
  496. ()
  497. }
  498. // Fall through to the next case to send the response message.
  499. fallthrough
  500. case .intercept(.none):
  501. switch self.interceptorStateMachine.interceptResponseMessage() {
  502. case .intercept:
  503. let senderWantsCompression = compression.isEnabled(
  504. callDefault: self.compressResponsesIfPossible
  505. )
  506. let compress = self.compressionEnabledOnRPC && senderWantsCompression
  507. let metadata = MessageMetadata(compress: compress, flush: true)
  508. self.interceptors?.send(.message(response, metadata), promise: nil)
  509. case .cancel:
  510. return self.cancel(error: nil)
  511. case .drop:
  512. ()
  513. }
  514. case .drop:
  515. ()
  516. }
  517. }
  518. @Sendable
  519. @inlinable
  520. internal func interceptResponseMessages(_ messages: Deque<(Response, Compression)>) {
  521. if self.eventLoop.inEventLoop {
  522. for message in messages {
  523. self._interceptResponseMessage(message.0, compression: message.1)
  524. }
  525. } else {
  526. self.eventLoop.execute {
  527. for message in messages {
  528. self._interceptResponseMessage(message.0, compression: message.1)
  529. }
  530. }
  531. }
  532. }
  533. @inlinable
  534. internal func _interceptTermination(_ error: Error?) {
  535. self.eventLoop.assertInEventLoop()
  536. let processedError: Error?
  537. if let thrownStatus = error as? GRPCStatus, thrownStatus.isOk {
  538. processedError = GRPCStatus(
  539. code: .unknown,
  540. message: "Handler threw error with status code 'ok'."
  541. )
  542. } else {
  543. processedError = error
  544. }
  545. switch self.handlerStateMachine.sendStatus() {
  546. case let .intercept(requestHeaders, trailers):
  547. let status: GRPCStatus
  548. let processedTrailers: HPACKHeaders
  549. if let processedError = processedError {
  550. (status, processedTrailers) = ServerErrorProcessor.processObserverError(
  551. processedError,
  552. headers: requestHeaders,
  553. trailers: trailers,
  554. delegate: self.errorDelegate
  555. )
  556. } else {
  557. status = GRPCStatus.ok
  558. processedTrailers = trailers
  559. }
  560. switch self.interceptorStateMachine.interceptResponseStatus() {
  561. case .intercept:
  562. self.interceptors?.send(.end(status, processedTrailers), promise: nil)
  563. case .cancel:
  564. return self.cancel(error: nil)
  565. case .drop:
  566. ()
  567. }
  568. case .drop:
  569. ()
  570. }
  571. }
  572. @Sendable
  573. @inlinable
  574. internal func interceptTermination(_ status: Error?) {
  575. if self.eventLoop.inEventLoop {
  576. self._interceptTermination(status)
  577. } else {
  578. self.eventLoop.execute {
  579. self._interceptTermination(status)
  580. }
  581. }
  582. }
  583. @inlinable
  584. internal func sendInterceptedPart(
  585. _ part: GRPCServerResponsePart<Response>,
  586. promise: EventLoopPromise<Void>?
  587. ) {
  588. switch part {
  589. case let .metadata(headers):
  590. self.sendInterceptedMetadata(headers, promise: promise)
  591. case let .message(message, metadata):
  592. do {
  593. let bytes = try self.serializer.serialize(message, allocator: ByteBufferAllocator())
  594. self.sendInterceptedResponse(bytes, metadata: metadata, promise: promise)
  595. } catch {
  596. promise?.fail(error)
  597. self.cancel(error: error)
  598. }
  599. case let .end(status, trailers):
  600. self.sendInterceptedStatus(status, metadata: trailers, promise: promise)
  601. }
  602. }
  603. @inlinable
  604. internal func sendInterceptedMetadata(
  605. _ metadata: HPACKHeaders,
  606. promise: EventLoopPromise<Void>?
  607. ) {
  608. switch self.interceptorStateMachine.interceptedResponseMetadata() {
  609. case .forward:
  610. if let responseWriter = self.responseWriter {
  611. responseWriter.sendMetadata(metadata, flush: false, promise: promise)
  612. } else if let promise = promise {
  613. promise.fail(GRPCStatus.processingError)
  614. }
  615. case .cancel:
  616. self.cancel(error: nil)
  617. case .drop:
  618. ()
  619. }
  620. }
  621. @inlinable
  622. internal func sendInterceptedResponse(
  623. _ bytes: ByteBuffer,
  624. metadata: MessageMetadata,
  625. promise: EventLoopPromise<Void>?
  626. ) {
  627. switch self.interceptorStateMachine.interceptedResponseMessage() {
  628. case .forward:
  629. if let responseWriter = self.responseWriter {
  630. responseWriter.sendMessage(bytes, metadata: metadata, promise: promise)
  631. } else if let promise = promise {
  632. promise.fail(GRPCStatus.processingError)
  633. }
  634. case .cancel:
  635. self.cancel(error: nil)
  636. case .drop:
  637. ()
  638. }
  639. }
  640. @inlinable
  641. internal func sendInterceptedStatus(
  642. _ status: GRPCStatus,
  643. metadata: HPACKHeaders,
  644. promise: EventLoopPromise<Void>?
  645. ) {
  646. switch self.interceptorStateMachine.interceptedResponseStatus() {
  647. case .forward:
  648. if let responseWriter = self.responseWriter {
  649. responseWriter.sendEnd(status: status, trailers: metadata, promise: promise)
  650. } else if let promise = promise {
  651. promise.fail(GRPCStatus.processingError)
  652. }
  653. case .cancel:
  654. self.cancel(error: nil)
  655. case .drop:
  656. ()
  657. }
  658. }
  659. }
  660. // Sendability is unchecked as all mutable state is accessed/modified from an appropriate event
  661. // loop.
  662. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  663. extension AsyncServerHandler: @unchecked Sendable {}
  664. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  665. extension AsyncServerHandler: AsyncServerCallContextProvider {
  666. @usableFromInline
  667. internal func setResponseHeaders(_ headers: HPACKHeaders) async throws {
  668. let completed = self.eventLoop.submit {
  669. self.handlerStateMachine.setResponseHeaders(headers)
  670. }
  671. try await completed.get()
  672. }
  673. @usableFromInline
  674. internal func setResponseTrailers(_ headers: HPACKHeaders) async throws {
  675. let completed = self.eventLoop.submit {
  676. self.handlerStateMachine.setResponseTrailers(headers)
  677. }
  678. try await completed.get()
  679. }
  680. @usableFromInline
  681. internal func setResponseCompression(_ enabled: Bool) async throws {
  682. let completed = self.eventLoop.submit {
  683. self.compressResponsesIfPossible = enabled
  684. }
  685. try await completed.get()
  686. }
  687. @usableFromInline
  688. func withUserInfo<Result: Sendable>(
  689. _ modify: @Sendable @escaping (UserInfo) throws -> Result
  690. ) async throws -> Result {
  691. let result = self.eventLoop.submit {
  692. try modify(self.userInfoRef.value)
  693. }
  694. return try await result.get()
  695. }
  696. @usableFromInline
  697. func withMutableUserInfo<Result: Sendable>(
  698. _ modify: @Sendable @escaping (inout UserInfo) throws -> Result
  699. ) async throws -> Result {
  700. let result = self.eventLoop.submit {
  701. try modify(&self.userInfoRef.value)
  702. }
  703. return try await result.get()
  704. }
  705. }
  706. /// This protocol exists so that the generic server handler can be erased from the
  707. /// `GRPCAsyncServerCallContext`.
  708. ///
  709. /// It provides methods which update context on the async handler by first executing onto the
  710. /// correct event loop.
  711. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  712. @usableFromInline
  713. protocol AsyncServerCallContextProvider: Sendable {
  714. func setResponseHeaders(_ headers: HPACKHeaders) async throws
  715. func setResponseTrailers(_ trailers: HPACKHeaders) async throws
  716. func setResponseCompression(_ enabled: Bool) async throws
  717. func withUserInfo<Result: Sendable>(
  718. _ modify: @Sendable @escaping (UserInfo) throws -> Result
  719. ) async throws -> Result
  720. func withMutableUserInfo<Result: Sendable>(
  721. _ modify: @Sendable @escaping (inout UserInfo) throws -> Result
  722. ) async throws -> Result
  723. }
  724. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  725. @usableFromInline
  726. internal struct ServerHandlerComponents<
  727. Request: Sendable,
  728. Response: Sendable,
  729. Delegate: NIOAsyncWriterSinkDelegate
  730. > where Delegate.Element == (Response, Compression) {
  731. @usableFromInline
  732. internal typealias AsyncWriterSink = NIOAsyncWriter<(Response, Compression), Delegate>.Sink
  733. @usableFromInline
  734. internal typealias AsyncSequenceSource = NIOThrowingAsyncSequenceProducer<
  735. Request,
  736. Error,
  737. NIOAsyncSequenceProducerBackPressureStrategies.HighLowWatermark,
  738. GRPCAsyncSequenceProducerDelegate
  739. >.Source
  740. @usableFromInline
  741. internal let task: Task<Void, Never>
  742. @usableFromInline
  743. internal let responseWriterSink: AsyncWriterSink
  744. @usableFromInline
  745. internal let requestSource: AsyncSequenceSource
  746. @inlinable
  747. init(
  748. requestSource: AsyncSequenceSource,
  749. responseWriterSink: AsyncWriterSink,
  750. task: Task<Void, Never>
  751. ) {
  752. self.task = task
  753. self.responseWriterSink = responseWriterSink
  754. self.requestSource = requestSource
  755. }
  756. func cancel() {
  757. // Cancel the request and response streams.
  758. //
  759. // The user handler is encouraged to check for cancellation, however, we should assume
  760. // they do not. Finishing the request source stops any more requests from being delivered
  761. // to the request stream, and finishing the writer sink will ensure no more responses are
  762. // written. This should reduce how long the user handler runs for as it can no longer do
  763. // anything useful.
  764. self.requestSource.finish()
  765. self.responseWriterSink.finish(error: CancellationError())
  766. self.task.cancel()
  767. }
  768. }