ClientTransportFactory.swift 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. /*
  2. * Copyright 2020, 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 NIOCore
  17. import NIOHTTP2
  18. import protocol SwiftProtobuf.Message
  19. /// A `ClientTransport` factory for an RPC.
  20. @usableFromInline
  21. internal struct ClientTransportFactory<Request, Response> {
  22. /// The underlying transport factory.
  23. private var factory: Factory<Request, Response>
  24. @usableFromInline
  25. internal enum Factory<Request, Response> {
  26. case http2(HTTP2ClientTransportFactory<Request, Response>)
  27. case fake(FakeClientTransportFactory<Request, Response>)
  28. }
  29. private init(_ http2: HTTP2ClientTransportFactory<Request, Response>) {
  30. self.factory = .http2(http2)
  31. }
  32. private init(_ fake: FakeClientTransportFactory<Request, Response>) {
  33. self.factory = .fake(fake)
  34. }
  35. /// Create a transport factory for HTTP/2 based transport with `SwiftProtobuf.Message` messages.
  36. /// - Parameters:
  37. /// - multiplexer: The multiplexer used to create an HTTP/2 stream for the RPC.
  38. /// - host: The value of the ":authority" pseudo header.
  39. /// - scheme: The value of the ":scheme" pseudo header.
  40. /// - errorDelegate: A client error delegate.
  41. /// - Returns: A factory for making and configuring HTTP/2 based transport.
  42. @usableFromInline
  43. internal static func http2<Request: SwiftProtobuf.Message, Response: SwiftProtobuf.Message>(
  44. channel: EventLoopFuture<Channel>,
  45. authority: String,
  46. scheme: String,
  47. maximumReceiveMessageLength: Int,
  48. errorDelegate: ClientErrorDelegate?
  49. ) -> ClientTransportFactory<Request, Response> {
  50. let http2 = HTTP2ClientTransportFactory<Request, Response>(
  51. streamChannel: channel,
  52. scheme: scheme,
  53. authority: authority,
  54. serializer: ProtobufSerializer(),
  55. deserializer: ProtobufDeserializer(),
  56. maximumReceiveMessageLength: maximumReceiveMessageLength,
  57. errorDelegate: errorDelegate
  58. )
  59. return .init(http2)
  60. }
  61. /// Create a transport factory for HTTP/2 based transport with `GRPCPayload` messages.
  62. /// - Parameters:
  63. /// - multiplexer: The multiplexer used to create an HTTP/2 stream for the RPC.
  64. /// - host: The value of the ":authority" pseudo header.
  65. /// - scheme: The value of the ":scheme" pseudo header.
  66. /// - errorDelegate: A client error delegate.
  67. /// - Returns: A factory for making and configuring HTTP/2 based transport.
  68. @usableFromInline
  69. internal static func http2<Request: GRPCPayload, Response: GRPCPayload>(
  70. channel: EventLoopFuture<Channel>,
  71. authority: String,
  72. scheme: String,
  73. maximumReceiveMessageLength: Int,
  74. errorDelegate: ClientErrorDelegate?
  75. ) -> ClientTransportFactory<Request, Response> {
  76. let http2 = HTTP2ClientTransportFactory<Request, Response>(
  77. streamChannel: channel,
  78. scheme: scheme,
  79. authority: authority,
  80. serializer: AnySerializer(wrapping: GRPCPayloadSerializer()),
  81. deserializer: AnyDeserializer(wrapping: GRPCPayloadDeserializer()),
  82. maximumReceiveMessageLength: maximumReceiveMessageLength,
  83. errorDelegate: errorDelegate
  84. )
  85. return .init(http2)
  86. }
  87. /// Make a factory for 'fake' transport.
  88. /// - Parameter fakeResponse: The fake response stream.
  89. /// - Returns: A factory for making and configuring fake transport.
  90. @usableFromInline
  91. internal static func fake<Request: SwiftProtobuf.Message, Response: SwiftProtobuf.Message>(
  92. _ fakeResponse: _FakeResponseStream<Request, Response>?
  93. ) -> ClientTransportFactory<Request, Response> {
  94. let factory = FakeClientTransportFactory(
  95. fakeResponse,
  96. requestSerializer: ProtobufSerializer(),
  97. requestDeserializer: ProtobufDeserializer(),
  98. responseSerializer: ProtobufSerializer(),
  99. responseDeserializer: ProtobufDeserializer()
  100. )
  101. return .init(factory)
  102. }
  103. /// Make a factory for 'fake' transport.
  104. /// - Parameter fakeResponse: The fake response stream.
  105. /// - Returns: A factory for making and configuring fake transport.
  106. @usableFromInline
  107. internal static func fake<Request: GRPCPayload, Response: GRPCPayload>(
  108. _ fakeResponse: _FakeResponseStream<Request, Response>?
  109. ) -> ClientTransportFactory<Request, Response> {
  110. let factory = FakeClientTransportFactory(
  111. fakeResponse,
  112. requestSerializer: GRPCPayloadSerializer(),
  113. requestDeserializer: GRPCPayloadDeserializer(),
  114. responseSerializer: GRPCPayloadSerializer(),
  115. responseDeserializer: GRPCPayloadDeserializer()
  116. )
  117. return .init(factory)
  118. }
  119. /// Makes a configured `ClientTransport`.
  120. /// - Parameters:
  121. /// - path: The path of the RPC, e.g. "/echo.Echo/Get".
  122. /// - type: The type of RPC, e.g. `.unary`.
  123. /// - options: Options for the RPC.
  124. /// - interceptors: Interceptors to use for the RPC.
  125. /// - onError: A callback invoked when an error is received.
  126. /// - onResponsePart: A closure called for each response part received.
  127. /// - Returns: A configured transport.
  128. internal func makeConfiguredTransport(
  129. to path: String,
  130. for type: GRPCCallType,
  131. withOptions options: CallOptions,
  132. onEventLoop eventLoop: EventLoop,
  133. interceptedBy interceptors: [ClientInterceptor<Request, Response>],
  134. onStart: @escaping () -> Void,
  135. onError: @escaping (Error) -> Void,
  136. onResponsePart: @escaping (GRPCClientResponsePart<Response>) -> Void
  137. ) -> ClientTransport<Request, Response> {
  138. switch self.factory {
  139. case let .http2(factory):
  140. let transport = factory.makeTransport(
  141. to: path,
  142. for: type,
  143. withOptions: options,
  144. onEventLoop: eventLoop,
  145. interceptedBy: interceptors,
  146. onStart: onStart,
  147. onError: onError,
  148. onResponsePart: onResponsePart
  149. )
  150. factory.configure(transport)
  151. return transport
  152. case let .fake(factory):
  153. let transport = factory.makeTransport(
  154. to: path,
  155. for: type,
  156. withOptions: options,
  157. onEventLoop: eventLoop,
  158. interceptedBy: interceptors,
  159. onError: onError,
  160. onResponsePart
  161. )
  162. factory.configure(transport)
  163. return transport
  164. }
  165. }
  166. }
  167. @usableFromInline
  168. internal struct HTTP2ClientTransportFactory<Request, Response> {
  169. /// The multiplexer providing an HTTP/2 stream for the call.
  170. private var streamChannel: EventLoopFuture<Channel>
  171. /// The ":authority" pseudo-header.
  172. private var authority: String
  173. /// The ":scheme" pseudo-header.
  174. private var scheme: String
  175. /// An error delegate.
  176. private var errorDelegate: ClientErrorDelegate?
  177. /// The request serializer.
  178. private let serializer: AnySerializer<Request>
  179. /// The response deserializer.
  180. private let deserializer: AnyDeserializer<Response>
  181. /// Maximum allowed length of a received message.
  182. private let maximumReceiveMessageLength: Int
  183. @usableFromInline
  184. internal init<Serializer: MessageSerializer, Deserializer: MessageDeserializer>(
  185. streamChannel: EventLoopFuture<Channel>,
  186. scheme: String,
  187. authority: String,
  188. serializer: Serializer,
  189. deserializer: Deserializer,
  190. maximumReceiveMessageLength: Int,
  191. errorDelegate: ClientErrorDelegate?
  192. ) where Serializer.Input == Request, Deserializer.Output == Response {
  193. self.streamChannel = streamChannel
  194. self.scheme = scheme
  195. self.authority = authority
  196. self.serializer = AnySerializer(wrapping: serializer)
  197. self.deserializer = AnyDeserializer(wrapping: deserializer)
  198. self.maximumReceiveMessageLength = maximumReceiveMessageLength
  199. self.errorDelegate = errorDelegate
  200. }
  201. fileprivate func makeTransport(
  202. to path: String,
  203. for type: GRPCCallType,
  204. withOptions options: CallOptions,
  205. onEventLoop eventLoop: EventLoop,
  206. interceptedBy interceptors: [ClientInterceptor<Request, Response>],
  207. onStart: @escaping () -> Void,
  208. onError: @escaping (Error) -> Void,
  209. onResponsePart: @escaping (GRPCClientResponsePart<Response>) -> Void
  210. ) -> ClientTransport<Request, Response> {
  211. return ClientTransport(
  212. details: self.makeCallDetails(type: type, path: path, options: options),
  213. eventLoop: eventLoop,
  214. interceptors: interceptors,
  215. serializer: self.serializer,
  216. deserializer: self.deserializer,
  217. errorDelegate: self.errorDelegate,
  218. onStart: onStart,
  219. onError: onError,
  220. onResponsePart: onResponsePart
  221. )
  222. }
  223. fileprivate func configure<Request, Response>(_ transport: ClientTransport<Request, Response>) {
  224. transport.configure { _ in
  225. self.streamChannel.flatMapThrowing { channel in
  226. // This initializer will always occur on the appropriate event loop, sync operations are
  227. // fine here.
  228. let syncOperations = channel.pipeline.syncOperations
  229. do {
  230. let clientHandler = GRPCClientChannelHandler(
  231. callType: transport.callDetails.type,
  232. maximumReceiveMessageLength: self.maximumReceiveMessageLength,
  233. logger: transport.logger
  234. )
  235. try syncOperations.addHandler(clientHandler)
  236. try syncOperations.addHandler(transport)
  237. }
  238. }
  239. }
  240. }
  241. private func makeCallDetails(
  242. type: GRPCCallType,
  243. path: String,
  244. options: CallOptions
  245. ) -> CallDetails {
  246. return .init(
  247. type: type,
  248. path: path,
  249. authority: self.authority,
  250. scheme: self.scheme,
  251. options: options
  252. )
  253. }
  254. }
  255. @usableFromInline
  256. internal struct FakeClientTransportFactory<Request, Response> {
  257. /// The fake response stream for the call. This can be `nil` if the user did not correctly
  258. /// configure their client. The result will be a transport which immediately fails.
  259. private var fakeResponseStream: _FakeResponseStream<Request, Response>?
  260. /// The request serializer.
  261. private let requestSerializer: AnySerializer<Request>
  262. /// The response deserializer.
  263. private let responseDeserializer: AnyDeserializer<Response>
  264. /// A codec for deserializing requests and serializing responses.
  265. private let codec: ChannelHandler
  266. @usableFromInline
  267. internal init<
  268. RequestSerializer: MessageSerializer,
  269. RequestDeserializer: MessageDeserializer,
  270. ResponseSerializer: MessageSerializer,
  271. ResponseDeserializer: MessageDeserializer
  272. >(
  273. _ fakeResponseStream: _FakeResponseStream<Request, Response>?,
  274. requestSerializer: RequestSerializer,
  275. requestDeserializer: RequestDeserializer,
  276. responseSerializer: ResponseSerializer,
  277. responseDeserializer: ResponseDeserializer
  278. ) where RequestSerializer.Input == Request,
  279. RequestDeserializer.Output == Request,
  280. ResponseSerializer.Input == Response,
  281. ResponseDeserializer.Output == Response {
  282. self.fakeResponseStream = fakeResponseStream
  283. self.requestSerializer = AnySerializer(wrapping: requestSerializer)
  284. self.responseDeserializer = AnyDeserializer(wrapping: responseDeserializer)
  285. self.codec = GRPCClientReverseCodecHandler(
  286. serializer: responseSerializer,
  287. deserializer: requestDeserializer
  288. )
  289. }
  290. fileprivate func makeTransport(
  291. to path: String,
  292. for type: GRPCCallType,
  293. withOptions options: CallOptions,
  294. onEventLoop eventLoop: EventLoop,
  295. interceptedBy interceptors: [ClientInterceptor<Request, Response>],
  296. onError: @escaping (Error) -> Void,
  297. _ onResponsePart: @escaping (GRPCClientResponsePart<Response>) -> Void
  298. ) -> ClientTransport<Request, Response> {
  299. return ClientTransport(
  300. details: CallDetails(
  301. type: type,
  302. path: path,
  303. authority: "localhost",
  304. scheme: "http",
  305. options: options
  306. ),
  307. eventLoop: eventLoop,
  308. interceptors: interceptors,
  309. serializer: self.requestSerializer,
  310. deserializer: self.responseDeserializer,
  311. errorDelegate: nil,
  312. onStart: {},
  313. onError: onError,
  314. onResponsePart: onResponsePart
  315. )
  316. }
  317. fileprivate func configure<Request, Response>(_ transport: ClientTransport<Request, Response>) {
  318. transport.configure { handler in
  319. if let fakeResponse = self.fakeResponseStream {
  320. return fakeResponse.channel.pipeline.addHandlers(self.codec, handler).always { result in
  321. switch result {
  322. case .success:
  323. fakeResponse.activate()
  324. case .failure:
  325. ()
  326. }
  327. }
  328. } else {
  329. return transport.callEventLoop
  330. .makeFailedFuture(GRPCStatus(code: .unavailable, message: nil))
  331. }
  332. }
  333. }
  334. }