GRPCServerCodec.swift 2.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
  1. import Foundation
  2. import SwiftProtobuf
  3. import NIO
  4. import NIOFoundationCompat
  5. import NIOHTTP1
  6. /// Incoming gRPC package with a fixed message type.
  7. public enum GRPCServerRequestPart<MessageType: Message> {
  8. case head(HTTPRequestHead)
  9. case message(MessageType)
  10. case end
  11. }
  12. /// Outgoing gRPC package with a fixed message type.
  13. public enum GRPCServerResponsePart<MessageType: Message> {
  14. case headers(HTTPHeaders)
  15. case message(MessageType)
  16. case status(GRPCStatus)
  17. }
  18. /// A simple channel handler that translates raw gRPC packets into decoded protobuf messages, and vice versa.
  19. public final class GRPCServerCodec<RequestMessage: Message, ResponseMessage: Message> { }
  20. extension GRPCServerCodec: ChannelInboundHandler {
  21. public typealias InboundIn = RawGRPCServerRequestPart
  22. public typealias InboundOut = GRPCServerRequestPart<RequestMessage>
  23. public func channelRead(ctx: ChannelHandlerContext, data: NIOAny) {
  24. switch self.unwrapInboundIn(data) {
  25. case .head(let requestHead):
  26. ctx.fireChannelRead(self.wrapInboundOut(.head(requestHead)))
  27. case .message(var message):
  28. let messageAsData = message.readData(length: message.readableBytes)!
  29. do {
  30. ctx.fireChannelRead(self.wrapInboundOut(.message(try RequestMessage(serializedData: messageAsData))))
  31. } catch {
  32. //! FIXME: Ensure that the last handler in the pipeline returns `.dataLoss` here?
  33. ctx.fireErrorCaught(error)
  34. }
  35. case .end:
  36. ctx.fireChannelRead(self.wrapInboundOut(.end))
  37. }
  38. }
  39. }
  40. extension GRPCServerCodec: ChannelOutboundHandler {
  41. public typealias OutboundIn = GRPCServerResponsePart<ResponseMessage>
  42. public typealias OutboundOut = RawGRPCServerResponsePart
  43. public func write(ctx: ChannelHandlerContext, data: NIOAny, promise: EventLoopPromise<Void>?) {
  44. let responsePart = self.unwrapOutboundIn(data)
  45. switch responsePart {
  46. case .headers(let headers):
  47. ctx.write(self.wrapOutboundOut(.headers(headers)), promise: promise)
  48. case .message(let message):
  49. do {
  50. let messageData = try message.serializedData()
  51. var responseBuffer = ctx.channel.allocator.buffer(capacity: messageData.count)
  52. responseBuffer.write(bytes: messageData)
  53. ctx.write(self.wrapOutboundOut(.message(responseBuffer)), promise: promise)
  54. } catch {
  55. promise?.fail(error: error)
  56. ctx.fireErrorCaught(error)
  57. }
  58. case .status(let status):
  59. ctx.write(self.wrapOutboundOut(.status(status)), promise: promise)
  60. }
  61. }
  62. }