GRPCTests.swift 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. /*
  2. * Copyright 2017, 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 Dispatch
  17. import Foundation
  18. @testable import gRPC
  19. import XCTest
  20. class gRPCTests: XCTestCase {
  21. func testConnectivity() {
  22. runTest(useSSL: false)
  23. }
  24. func testConnectivitySecure() {
  25. runTest(useSSL: true)
  26. }
  27. static var allTests: [(String, (gRPCTests) -> () throws -> Void)] {
  28. return [
  29. ("testConnectivity", testConnectivity),
  30. ("testConnectivitySecure", testConnectivitySecure)
  31. ]
  32. }
  33. }
  34. let address = "localhost:8085"
  35. let host = "example.com"
  36. let clientText = "hello, server!"
  37. let serverText = "hello, client!"
  38. let initialClientMetadata =
  39. [
  40. "x": "xylophone",
  41. "y": "yu",
  42. "z": "zither"
  43. ]
  44. let initialServerMetadata =
  45. [
  46. "a": "Apple",
  47. "b": "Banana",
  48. "c": "Cherry"
  49. ]
  50. let trailingServerMetadata =
  51. [
  52. // We have more than ten entries here to ensure that even large metadata entries work
  53. // and aren't limited by e.g. a fixed-size entry buffer.
  54. "0": "zero",
  55. "1": "one",
  56. "2": "two",
  57. "3": "three",
  58. "4": "four",
  59. "5": "five",
  60. "6": "six",
  61. "7": "seven",
  62. "8": "eight",
  63. "9": "nine",
  64. "10": "ten",
  65. "11": "eleven",
  66. "12": "twelve"
  67. ]
  68. let steps = 10
  69. let hello = "/hello"
  70. let statusCode = StatusCode.ok
  71. let statusMessage = "OK"
  72. func runTest(useSSL: Bool) {
  73. gRPC.initialize()
  74. let serverRunningSemaphore = DispatchSemaphore(value: 0)
  75. // create the server
  76. let server: Server
  77. if useSSL {
  78. let certificateURL = URL(fileURLWithPath: "Tests/ssl.crt")
  79. let keyURL = URL(fileURLWithPath: "Tests/ssl.key")
  80. guard
  81. let certificate = try? String(contentsOf: certificateURL, encoding: .utf8),
  82. let key = try? String(contentsOf: keyURL, encoding: .utf8)
  83. else {
  84. // FIXME: We don't want tests to silently pass just because the certificates can't be loaded.
  85. return
  86. }
  87. server = Server(address: address,
  88. key: key,
  89. certs: certificate)
  90. } else {
  91. server = Server(address: address)
  92. }
  93. // start the server
  94. DispatchQueue.global().async {
  95. do {
  96. try runServer(server: server)
  97. } catch (let error) {
  98. XCTFail("server error \(error)")
  99. }
  100. serverRunningSemaphore.signal() // when the server exits, the test is finished
  101. }
  102. // run the client
  103. do {
  104. try runClient(useSSL: useSSL)
  105. } catch (let error) {
  106. XCTFail("client error \(error)")
  107. }
  108. // stop the server
  109. server.stop()
  110. // wait until the server has shut down
  111. _ = serverRunningSemaphore.wait()
  112. }
  113. func verify_metadata(_ metadata: Metadata, expected: [String: String], file: StaticString = #file, line: UInt = #line) {
  114. XCTAssertGreaterThanOrEqual(metadata.count(), expected.count)
  115. var allPresentKeys = Set<String>()
  116. for i in 0..<metadata.count() {
  117. guard let expectedValue = expected[metadata.key(i)!]
  118. else { continue }
  119. allPresentKeys.insert(metadata.key(i)!)
  120. XCTAssertEqual(metadata.value(i), expectedValue, file: file, line: line)
  121. }
  122. XCTAssertEqual(allPresentKeys.sorted(), expected.keys.sorted(), file: file, line: line)
  123. }
  124. func runClient(useSSL: Bool) throws {
  125. let message = clientText.data(using: .utf8)
  126. let channel: Channel
  127. if useSSL {
  128. let certificateURL = URL(fileURLWithPath: "Tests/ssl.crt")
  129. guard
  130. let certificates = try? String(contentsOf: certificateURL, encoding: .utf8)
  131. else {
  132. return
  133. }
  134. let host = "example.com"
  135. channel = Channel(address: address, certificates: certificates, host: host)
  136. } else {
  137. channel = Channel(address: address, secure: false)
  138. }
  139. channel.host = host
  140. for _ in 0..<steps {
  141. let sem = DispatchSemaphore(value: 0)
  142. let method = hello
  143. let call = channel.makeCall(method)
  144. let metadata = Metadata(initialClientMetadata)
  145. try call.start(.unary, metadata: metadata, message: message) {
  146. response in
  147. // verify the basic response from the server
  148. XCTAssertEqual(response.statusCode, statusCode)
  149. XCTAssertEqual(response.statusMessage, statusMessage)
  150. // verify the message from the server
  151. let resultData = response.resultData!
  152. let messageString = String(data: resultData, encoding: .utf8)
  153. XCTAssertEqual(messageString, serverText)
  154. // verify the initial metadata from the server
  155. let initialMetadata = response.initialMetadata!
  156. verify_metadata(initialMetadata, expected: initialServerMetadata)
  157. // verify the trailing metadata from the server
  158. let trailingMetadata = response.trailingMetadata!
  159. verify_metadata(trailingMetadata, expected: trailingServerMetadata)
  160. // report completion
  161. sem.signal()
  162. }
  163. // wait for the call to complete
  164. _ = sem.wait()
  165. }
  166. }
  167. func runServer(server: Server) throws {
  168. var requestCount = 0
  169. let sem = DispatchSemaphore(value: 0)
  170. server.run { requestHandler in
  171. do {
  172. requestCount += 1
  173. XCTAssertEqual(requestHandler.host, host)
  174. XCTAssertEqual(requestHandler.method, hello)
  175. let initialMetadata = requestHandler.requestMetadata
  176. verify_metadata(initialMetadata, expected: initialClientMetadata)
  177. let initialMetadataToSend = Metadata(initialServerMetadata)
  178. try requestHandler.receiveMessage(initialMetadata: initialMetadataToSend) { messageData in
  179. let messageString = String(data: messageData!, encoding: .utf8)
  180. XCTAssertEqual(messageString, clientText)
  181. }
  182. let replyMessage = serverText
  183. let trailingMetadataToSend = Metadata(trailingServerMetadata)
  184. try requestHandler.sendResponse(message: replyMessage.data(using: .utf8)!,
  185. statusCode: statusCode,
  186. statusMessage: statusMessage,
  187. trailingMetadata: trailingMetadataToSend)
  188. } catch (let error) {
  189. XCTFail("error \(error)")
  190. }
  191. }
  192. server.onCompletion = {
  193. // return from runServer()
  194. sem.signal()
  195. }
  196. // wait for the server to exit
  197. _ = sem.wait()
  198. }