FakeChannel.swift 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  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 Logging
  17. import NIO
  18. import SwiftProtobuf
  19. /// A fake channel for use with generated test clients.
  20. ///
  21. /// The `FakeChannel` provides factories for calls which avoid most of the gRPC stack and don't do
  22. /// real networking. Each call relies on either a `FakeUnaryResponse` or a `FakeStreamingResponse`
  23. /// to get responses or errors. The fake response of each type should be registered with the channel
  24. /// prior to making a call via `makeFakeUnaryResponse` or `makeFakeStreamingResponse` respectively.
  25. ///
  26. /// Users will typically not be required to interact with the channel directly, instead they should
  27. /// do so via a generated test client.
  28. public class FakeChannel: GRPCChannel {
  29. /// Fake response streams keyed by their path.
  30. private var responseStreams: [String: CircularBuffer<Any>]
  31. /// A logger.
  32. public let logger: Logger
  33. public init(logger: Logger = Logger(label: "io.grpc", factory: { _ in
  34. SwiftLogNoOpLogHandler()
  35. })) {
  36. self.responseStreams = [:]
  37. self.logger = logger
  38. }
  39. /// Make and store a fake unary response for the given path. Users should prefer making a response
  40. /// stream for their RPC directly via the appropriate method on their generated test client.
  41. public func makeFakeUnaryResponse<Request, Response>(
  42. path: String,
  43. requestHandler: @escaping (FakeRequestPart<Request>) -> Void
  44. ) -> FakeUnaryResponse<Request, Response> {
  45. let proxy = FakeUnaryResponse<Request, Response>(requestHandler: requestHandler)
  46. self.responseStreams[path, default: []].append(proxy)
  47. return proxy
  48. }
  49. /// Make and store a fake streaming response for the given path. Users should prefer making a
  50. /// response stream for their RPC directly via the appropriate method on their generated test
  51. /// client.
  52. public func makeFakeStreamingResponse<Request, Response>(
  53. path: String,
  54. requestHandler: @escaping (FakeRequestPart<Request>) -> Void
  55. ) -> FakeStreamingResponse<Request, Response> {
  56. let proxy = FakeStreamingResponse<Request, Response>(requestHandler: requestHandler)
  57. self.responseStreams[path, default: []].append(proxy)
  58. return proxy
  59. }
  60. /// Returns true if there are fake responses enqueued for the given path.
  61. public func hasFakeResponseEnqueued(forPath path: String) -> Bool {
  62. guard let noStreamsForPath = self.responseStreams[path]?.isEmpty else {
  63. return false
  64. }
  65. return !noStreamsForPath
  66. }
  67. // (Docs inherited from `GRPCChannel`)
  68. public func makeUnaryCall<Request: SwiftProtobuf.Message, Response: SwiftProtobuf.Message>(
  69. path: String,
  70. request: Request,
  71. callOptions: CallOptions
  72. ) -> UnaryCall<Request, Response> {
  73. return self.makeUnaryCall(
  74. serializer: ProtobufSerializer(),
  75. deserializer: ProtobufDeserializer(),
  76. path: path,
  77. request: request,
  78. callOptions: callOptions
  79. )
  80. }
  81. // (Docs inherited from `GRPCChannel`)
  82. public func makeUnaryCall<Request: GRPCPayload, Response: GRPCPayload>(
  83. path: String,
  84. request: Request,
  85. callOptions: CallOptions
  86. ) -> UnaryCall<Request, Response> {
  87. return self.makeUnaryCall(
  88. serializer: GRPCPayloadSerializer(),
  89. deserializer: GRPCPayloadDeserializer(),
  90. path: path,
  91. request: request,
  92. callOptions: callOptions
  93. )
  94. }
  95. // (Docs inherited from `GRPCChannel`)
  96. public func makeServerStreamingCall<
  97. Request: SwiftProtobuf.Message,
  98. Response: SwiftProtobuf.Message
  99. >(
  100. path: String,
  101. request: Request,
  102. callOptions: CallOptions,
  103. handler: @escaping (Response) -> Void
  104. ) -> ServerStreamingCall<Request, Response> {
  105. return self.makeServerStreamingCall(
  106. serializer: ProtobufSerializer(),
  107. deserializer: ProtobufDeserializer(),
  108. path: path,
  109. request: request,
  110. callOptions: callOptions,
  111. handler: handler
  112. )
  113. }
  114. // (Docs inherited from `GRPCChannel`)
  115. public func makeServerStreamingCall<Request: GRPCPayload, Response: GRPCPayload>(
  116. path: String,
  117. request: Request,
  118. callOptions: CallOptions,
  119. handler: @escaping (Response) -> Void
  120. ) -> ServerStreamingCall<Request, Response> {
  121. return self.makeServerStreamingCall(
  122. serializer: GRPCPayloadSerializer(),
  123. deserializer: GRPCPayloadDeserializer(),
  124. path: path,
  125. request: request,
  126. callOptions: callOptions,
  127. handler: handler
  128. )
  129. }
  130. // (Docs inherited from `GRPCChannel`)
  131. public func makeClientStreamingCall<
  132. Request: SwiftProtobuf.Message,
  133. Response: SwiftProtobuf.Message
  134. >(
  135. path: String,
  136. callOptions: CallOptions
  137. ) -> ClientStreamingCall<Request, Response> {
  138. return self.makeClientStreamingCall(
  139. serializer: ProtobufSerializer(),
  140. deserializer: ProtobufDeserializer(),
  141. path: path,
  142. callOptions: callOptions
  143. )
  144. }
  145. // (Docs inherited from `GRPCChannel`)
  146. public func makeClientStreamingCall<Request: GRPCPayload, Response: GRPCPayload>(
  147. path: String,
  148. callOptions: CallOptions
  149. ) -> ClientStreamingCall<Request, Response> {
  150. return self.makeClientStreamingCall(
  151. serializer: GRPCPayloadSerializer(),
  152. deserializer: GRPCPayloadDeserializer(),
  153. path: path,
  154. callOptions: callOptions
  155. )
  156. }
  157. // (Docs inherited from `GRPCChannel`)
  158. public func makeBidirectionalStreamingCall<
  159. Request: SwiftProtobuf.Message,
  160. Response: SwiftProtobuf.Message
  161. >(
  162. path: String,
  163. callOptions: CallOptions,
  164. handler: @escaping (Response) -> Void
  165. ) -> BidirectionalStreamingCall<Request, Response> {
  166. return self.makeBidirectionalStreamingCall(
  167. serializer: ProtobufSerializer(),
  168. deserializer: ProtobufDeserializer(),
  169. path: path,
  170. callOptions: callOptions,
  171. handler: handler
  172. )
  173. }
  174. // (Docs inherited from `GRPCChannel`)
  175. public func makeBidirectionalStreamingCall<Request: GRPCPayload, Response: GRPCPayload>(
  176. path: String,
  177. callOptions: CallOptions,
  178. handler: @escaping (Response) -> Void
  179. ) -> BidirectionalStreamingCall<Request, Response> {
  180. return self.makeBidirectionalStreamingCall(
  181. serializer: GRPCPayloadSerializer(),
  182. deserializer: GRPCPayloadDeserializer(),
  183. path: path,
  184. callOptions: callOptions,
  185. handler: handler
  186. )
  187. }
  188. public func close() -> EventLoopFuture<Void> {
  189. // We don't have anything to close.
  190. return EmbeddedEventLoop().makeSucceededFuture(())
  191. }
  192. }
  193. extension FakeChannel {
  194. /// Dequeue a proxy for the given path and casts it to the given type, if one exists.
  195. private func dequeueResponseStream<Stream>(
  196. forPath path: String,
  197. as: Stream.Type = Stream.self
  198. ) -> Stream? {
  199. guard var streams = self.responseStreams[path], !streams.isEmpty else {
  200. return nil
  201. }
  202. // This is fine: we know we're non-empty.
  203. let first = streams.removeFirst()
  204. self.responseStreams.updateValue(streams, forKey: path)
  205. return first as? Stream
  206. }
  207. private func makeRequestHead(path: String, callOptions: CallOptions) -> _GRPCRequestHead {
  208. return _GRPCRequestHead(
  209. scheme: "http",
  210. path: path,
  211. host: "localhost",
  212. options: callOptions,
  213. requestID: nil
  214. )
  215. }
  216. }
  217. extension FakeChannel {
  218. private func makeUnaryCall<Serializer: MessageSerializer, Deserializer: MessageDeserializer>(
  219. serializer: Serializer,
  220. deserializer: Deserializer,
  221. path: String,
  222. request: Serializer.Input,
  223. callOptions: CallOptions
  224. ) -> UnaryCall<Serializer.Input, Deserializer.Output> {
  225. let call = UnaryCall<Serializer.Input, Deserializer.Output>.make(
  226. serializer: serializer,
  227. deserializer: deserializer,
  228. fakeResponse: self.dequeueResponseStream(forPath: path),
  229. callOptions: callOptions,
  230. logger: self.logger
  231. )
  232. call.send(self.makeRequestHead(path: path, callOptions: callOptions), request: request)
  233. return call
  234. }
  235. private func makeClientStreamingCall<
  236. Serializer: MessageSerializer,
  237. Deserializer: MessageDeserializer
  238. >(
  239. serializer: Serializer,
  240. deserializer: Deserializer,
  241. path: String,
  242. callOptions: CallOptions
  243. ) -> ClientStreamingCall<Serializer.Input, Deserializer.Output> {
  244. let call = ClientStreamingCall<Serializer.Input, Deserializer.Output>.make(
  245. serializer: serializer,
  246. deserializer: deserializer,
  247. fakeResponse: self.dequeueResponseStream(forPath: path),
  248. callOptions: callOptions,
  249. logger: self.logger
  250. )
  251. call.sendHead(self.makeRequestHead(path: path, callOptions: callOptions))
  252. return call
  253. }
  254. private func makeServerStreamingCall<
  255. Serializer: MessageSerializer,
  256. Deserializer: MessageDeserializer
  257. >(
  258. serializer: Serializer,
  259. deserializer: Deserializer,
  260. path: String,
  261. request: Serializer.Input,
  262. callOptions: CallOptions,
  263. handler: @escaping (Deserializer.Output) -> Void
  264. ) -> ServerStreamingCall<Serializer.Input, Deserializer.Output> {
  265. let call = ServerStreamingCall<Serializer.Input, Deserializer.Output>.make(
  266. serializer: serializer,
  267. deserializer: deserializer,
  268. fakeResponse: self.dequeueResponseStream(forPath: path),
  269. callOptions: callOptions,
  270. logger: self.logger,
  271. responseHandler: handler
  272. )
  273. call.send(self.makeRequestHead(path: path, callOptions: callOptions), request: request)
  274. return call
  275. }
  276. private func makeBidirectionalStreamingCall<
  277. Serializer: MessageSerializer,
  278. Deserializer: MessageDeserializer
  279. >(
  280. serializer: Serializer,
  281. deserializer: Deserializer,
  282. path: String,
  283. callOptions: CallOptions,
  284. handler: @escaping (Deserializer.Output) -> Void
  285. ) -> BidirectionalStreamingCall<Serializer.Input, Deserializer.Output> {
  286. let call = BidirectionalStreamingCall<Serializer.Input, Deserializer.Output>.make(
  287. serializer: serializer,
  288. deserializer: deserializer,
  289. fakeResponse: self.dequeueResponseStream(forPath: path),
  290. callOptions: callOptions,
  291. logger: self.logger,
  292. responseHandler: handler
  293. )
  294. call.sendHead(self.makeRequestHead(path: path, callOptions: callOptions))
  295. return call
  296. }
  297. }