GRPCServerCodec.swift 2.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  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. ctx.fireErrorCaught(GRPCServerError.requestProtoParseFailure)
  33. }
  34. case .end:
  35. ctx.fireChannelRead(self.wrapInboundOut(.end))
  36. }
  37. }
  38. }
  39. extension GRPCServerCodec: ChannelOutboundHandler {
  40. public typealias OutboundIn = GRPCServerResponsePart<ResponseMessage>
  41. public typealias OutboundOut = RawGRPCServerResponsePart
  42. public func write(ctx: ChannelHandlerContext, data: NIOAny, promise: EventLoopPromise<Void>?) {
  43. let responsePart = self.unwrapOutboundIn(data)
  44. switch responsePart {
  45. case .headers(let headers):
  46. ctx.write(self.wrapOutboundOut(.headers(headers)), promise: promise)
  47. case .message(let message):
  48. do {
  49. let messageData = try message.serializedData()
  50. var responseBuffer = ctx.channel.allocator.buffer(capacity: messageData.count)
  51. responseBuffer.write(bytes: messageData)
  52. ctx.write(self.wrapOutboundOut(.message(responseBuffer)), promise: promise)
  53. } catch {
  54. let error = GRPCServerError.responseProtoSerializationFailure
  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. }