ClientTransportFactory.swift 12 KB

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