GRPCAsyncServerHandler.swift 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863
  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. #if compiler(>=5.6)
  17. import DequeModule
  18. import Logging
  19. import NIOCore
  20. import NIOHPACK
  21. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  22. public struct GRPCAsyncServerHandler<
  23. Serializer: MessageSerializer,
  24. Deserializer: MessageDeserializer,
  25. Request: Sendable,
  26. Response: Sendable
  27. >: GRPCServerHandlerProtocol where Serializer.Input == Response, Deserializer.Output == Request {
  28. @usableFromInline
  29. internal let _handler: AsyncServerHandler<Serializer, Deserializer, Request, Response>
  30. public func receiveMetadata(_ metadata: HPACKHeaders) {
  31. self._handler.receiveMetadata(metadata)
  32. }
  33. public func receiveMessage(_ bytes: ByteBuffer) {
  34. self._handler.receiveMessage(bytes)
  35. }
  36. public func receiveEnd() {
  37. self._handler.receiveEnd()
  38. }
  39. public func receiveError(_ error: Error) {
  40. self._handler.receiveError(error)
  41. }
  42. public func finish() {
  43. self._handler.finish()
  44. }
  45. }
  46. // MARK: - RPC Adapters
  47. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  48. extension GRPCAsyncServerHandler {
  49. public typealias Request = Deserializer.Output
  50. public typealias Response = Serializer.Input
  51. @inlinable
  52. public init(
  53. context: CallHandlerContext,
  54. requestDeserializer: Deserializer,
  55. responseSerializer: Serializer,
  56. interceptors: [ServerInterceptor<Request, Response>],
  57. wrapping unary: @escaping @Sendable (Request, GRPCAsyncServerCallContext) async throws
  58. -> Response
  59. ) {
  60. self._handler = .init(
  61. context: context,
  62. requestDeserializer: requestDeserializer,
  63. responseSerializer: responseSerializer,
  64. callType: .unary,
  65. interceptors: interceptors,
  66. userHandler: { requestStream, responseStreamWriter, context in
  67. var iterator = requestStream.makeAsyncIterator()
  68. guard let request = try await iterator.next(), try await iterator.next() == nil else {
  69. throw GRPCError.ProtocolViolation("Unary RPC expects exactly one request")
  70. }
  71. let response = try await unary(request, context)
  72. try await responseStreamWriter.send(response)
  73. }
  74. )
  75. }
  76. @inlinable
  77. public init(
  78. context: CallHandlerContext,
  79. requestDeserializer: Deserializer,
  80. responseSerializer: Serializer,
  81. interceptors: [ServerInterceptor<Request, Response>],
  82. wrapping clientStreaming: @escaping @Sendable (
  83. GRPCAsyncRequestStream<Request>,
  84. GRPCAsyncServerCallContext
  85. ) async throws -> Response
  86. ) {
  87. self._handler = .init(
  88. context: context,
  89. requestDeserializer: requestDeserializer,
  90. responseSerializer: responseSerializer,
  91. callType: .clientStreaming,
  92. interceptors: interceptors,
  93. userHandler: { requestStream, responseStreamWriter, context in
  94. let response = try await clientStreaming(requestStream, context)
  95. try await responseStreamWriter.send(response)
  96. }
  97. )
  98. }
  99. @inlinable
  100. public init(
  101. context: CallHandlerContext,
  102. requestDeserializer: Deserializer,
  103. responseSerializer: Serializer,
  104. interceptors: [ServerInterceptor<Request, Response>],
  105. wrapping serverStreaming: @escaping @Sendable (
  106. Request,
  107. GRPCAsyncResponseStreamWriter<Response>,
  108. GRPCAsyncServerCallContext
  109. ) async throws -> Void
  110. ) {
  111. self._handler = .init(
  112. context: context,
  113. requestDeserializer: requestDeserializer,
  114. responseSerializer: responseSerializer,
  115. callType: .serverStreaming,
  116. interceptors: interceptors,
  117. userHandler: { requestStream, responseStreamWriter, context in
  118. var iterator = requestStream.makeAsyncIterator()
  119. guard let request = try await iterator.next(), try await iterator.next() == nil else {
  120. throw GRPCError.ProtocolViolation("Server-streaming RPC expects exactly one request")
  121. }
  122. try await serverStreaming(request, responseStreamWriter, context)
  123. }
  124. )
  125. }
  126. @inlinable
  127. public init(
  128. context: CallHandlerContext,
  129. requestDeserializer: Deserializer,
  130. responseSerializer: Serializer,
  131. interceptors: [ServerInterceptor<Request, Response>],
  132. wrapping bidirectional: @escaping @Sendable (
  133. GRPCAsyncRequestStream<Request>,
  134. GRPCAsyncResponseStreamWriter<Response>,
  135. GRPCAsyncServerCallContext
  136. ) async throws -> Void
  137. ) {
  138. self._handler = .init(
  139. context: context,
  140. requestDeserializer: requestDeserializer,
  141. responseSerializer: responseSerializer,
  142. callType: .bidirectionalStreaming,
  143. interceptors: interceptors,
  144. userHandler: bidirectional
  145. )
  146. }
  147. }
  148. // MARK: - Server Handler
  149. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  150. @usableFromInline
  151. internal final class AsyncServerHandler<
  152. Serializer: MessageSerializer,
  153. Deserializer: MessageDeserializer,
  154. Request: Sendable,
  155. Response: Sendable
  156. >: GRPCServerHandlerProtocol where Serializer.Input == Response, Deserializer.Output == Request {
  157. /// A response serializer.
  158. @usableFromInline
  159. internal let serializer: Serializer
  160. /// A request deserializer.
  161. @usableFromInline
  162. internal let deserializer: Deserializer
  163. /// The event loop that this handler executes on.
  164. @usableFromInline
  165. internal let eventLoop: EventLoop
  166. /// A `ByteBuffer` allocator provided by the underlying `Channel`.
  167. @usableFromInline
  168. internal let allocator: ByteBufferAllocator
  169. /// A user-provided error delegate which, if provided, is used to transform errors and potentially
  170. /// pack errors into trailers.
  171. @usableFromInline
  172. internal let errorDelegate: ServerErrorDelegate?
  173. /// A logger.
  174. @usableFromInline
  175. internal let logger: Logger
  176. /// A reference to the user info. This is shared with the interceptor pipeline and may be accessed
  177. /// from the async call context. `UserInfo` is _not_ `Sendable` and must always be accessed from
  178. /// an appropriate event loop.
  179. @usableFromInline
  180. internal let userInfoRef: Ref<UserInfo>
  181. /// Whether compression is enabled on the server and an algorithm has been negotiated with
  182. /// the client
  183. @usableFromInline
  184. internal let compressionEnabledOnRPC: Bool
  185. /// Whether the RPC method would like to compress responses (if possible). Defaults to true.
  186. @usableFromInline
  187. internal var compressResponsesIfPossible: Bool
  188. /// A state machine for the interceptor pipeline.
  189. @usableFromInline
  190. internal private(set) var interceptorStateMachine: ServerInterceptorStateMachine
  191. /// The interceptor pipeline.
  192. @usableFromInline
  193. internal private(set) var interceptors: Optional<ServerInterceptorPipeline<Request, Response>>
  194. /// An object for writing intercepted responses to the channel.
  195. @usableFromInline
  196. internal private(set) var responseWriter: Optional<GRPCServerResponseWriter>
  197. /// A state machine for the user implemented function.
  198. @usableFromInline
  199. internal private(set) var handlerStateMachine: ServerHandlerStateMachine
  200. /// A bag of components used by the user handler.
  201. @usableFromInline
  202. internal private(set) var handlerComponents: Optional<ServerHandlerComponents<
  203. Request,
  204. Response,
  205. GRPCAsyncWriterSinkDelegate<(Response, Compression)>
  206. >>
  207. /// The user provided function to execute.
  208. @usableFromInline
  209. internal let userHandler: @Sendable (
  210. GRPCAsyncRequestStream<Request>,
  211. GRPCAsyncResponseStreamWriter<Response>,
  212. GRPCAsyncServerCallContext
  213. ) async throws -> Void
  214. @usableFromInline
  215. internal typealias AsyncSequenceProducer = NIOThrowingAsyncSequenceProducer<
  216. Request,
  217. Error,
  218. NIOAsyncSequenceProducerBackPressureStrategies.HighLowWatermark,
  219. GRPCAsyncSequenceProducerDelegate
  220. >
  221. @inlinable
  222. internal init(
  223. context: CallHandlerContext,
  224. requestDeserializer: Deserializer,
  225. responseSerializer: Serializer,
  226. callType: GRPCCallType,
  227. interceptors: [ServerInterceptor<Request, Response>],
  228. userHandler: @escaping @Sendable (
  229. GRPCAsyncRequestStream<Request>,
  230. GRPCAsyncResponseStreamWriter<Response>,
  231. GRPCAsyncServerCallContext
  232. ) async throws -> Void
  233. ) {
  234. self.serializer = responseSerializer
  235. self.deserializer = requestDeserializer
  236. self.eventLoop = context.eventLoop
  237. self.allocator = context.allocator
  238. self.responseWriter = context.responseWriter
  239. self.errorDelegate = context.errorDelegate
  240. self.compressionEnabledOnRPC = context.encoding.isEnabled
  241. self.compressResponsesIfPossible = true
  242. self.logger = context.logger
  243. self.userInfoRef = Ref(UserInfo())
  244. self.handlerStateMachine = .init()
  245. self.handlerComponents = nil
  246. self.userHandler = userHandler
  247. self.interceptorStateMachine = .init()
  248. self.interceptors = nil
  249. self.interceptors = ServerInterceptorPipeline(
  250. logger: context.logger,
  251. eventLoop: context.eventLoop,
  252. path: context.path,
  253. callType: callType,
  254. remoteAddress: context.remoteAddress,
  255. userInfoRef: self.userInfoRef,
  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. }
  769. #endif // compiler(>=5.6)