server-session-bidistreaming.swift 3.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. // {{ method|methodDescriptorName }} (Bidirectional Streaming)
  2. {{ access }} protocol {{ .|session:file,service,method }} : {{ .|service:file,service }}Session {
  3. /// Receive a message. Blocks until a message is received or the client closes the connection.
  4. func receive() throws -> {{ method|input }}
  5. /// Send a message. Nonblocking.
  6. func send(_ response: {{ method|output }}, completion: ((Bool)->())?) throws
  7. /// Close a connection. Blocks until the connection is closed.
  8. func close() throws
  9. }
  10. fileprivate final class {{ .|session:file,service,method }}Impl : {{ .|service:file,service }}SessionImpl, {{ .|session:file,service,method }} {
  11. private var provider : {{ .|provider:file,service }}
  12. /// Create a session.
  13. init(handler:Handler, provider: {{ .|provider:file,service }}) {
  14. self.provider = provider
  15. super.init(handler:handler)
  16. }
  17. func receive() throws -> {{ method|input }} {
  18. let sem = DispatchSemaphore(value: 0)
  19. var requestMessage : {{ method|input }}?
  20. try self.handler.receiveMessage() {(requestData) in
  21. if let requestData = requestData {
  22. do {
  23. requestMessage = try {{ method|input }}(serializedData:requestData)
  24. } catch (let error) {
  25. print("error \(error)")
  26. }
  27. }
  28. sem.signal()
  29. }
  30. _ = sem.wait(timeout: DispatchTime.distantFuture)
  31. if let requestMessage = requestMessage {
  32. return requestMessage
  33. } else {
  34. throw {{ .|servererror:file,service }}.endOfStream
  35. }
  36. }
  37. func send(_ response: {{ method|output }}, completion: ((Bool)->())?) throws {
  38. try handler.sendResponse(message:response.serializedData(), completion: completion)
  39. }
  40. func close() throws {
  41. let sem = DispatchSemaphore(value: 0)
  42. try self.handler.sendStatus(statusCode:self.statusCode,
  43. statusMessage:self.statusMessage,
  44. trailingMetadata:self.trailingMetadata) { _ in sem.signal() }
  45. _ = sem.wait(timeout: DispatchTime.distantFuture)
  46. }
  47. /// Run the session. Internal.
  48. func run(queue:DispatchQueue) throws {
  49. try self.handler.sendMetadata(initialMetadata:initialMetadata) { _ in
  50. queue.async {
  51. do {
  52. try self.provider.{{ method|methodDescriptorName|lowercase }}(session:self)
  53. } catch (let error) {
  54. print("error \(error)")
  55. }
  56. }
  57. }
  58. }
  59. }
  60. //-{% if generateTestStubs %}
  61. /// Simple fake implementation of {{ .|session:file,service,method }} that returns a previously-defined set of results
  62. /// and stores sent values for later verification.
  63. class {{ .|session:file,service,method }}TestStub : {{ .|service:file,service }}SessionTestStub, {{ .|session:file,service,method }} {
  64. var inputs: [{{ method|input }}] = []
  65. var outputs: [{{ method|output }}] = []
  66. func receive() throws -> {{ method|input }} {
  67. if let input = inputs.first {
  68. inputs.removeFirst()
  69. return input
  70. } else {
  71. throw {{ .|clienterror:file,service }}.endOfStream
  72. }
  73. }
  74. func send(_ response: {{ method|output }}, completion: ((Bool)->())?) throws {
  75. outputs.append(response)
  76. }
  77. func close() throws { }
  78. }
  79. //-{% endif %}