server-session-bidistreaming.swift 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  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: @escaping ()->()) 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: @escaping ()->()) throws {
  38. try handler.sendResponse(message:response.serializedData()) {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) {
  45. sem.signal()
  46. }
  47. _ = sem.wait(timeout: DispatchTime.distantFuture)
  48. }
  49. /// Run the session. Internal.
  50. func run(queue:DispatchQueue) throws {
  51. try self.handler.sendMetadata(initialMetadata:initialMetadata) {
  52. queue.async {
  53. do {
  54. try self.provider.{{ method|methodDescriptorName|lowercase }}(session:self)
  55. } catch (let error) {
  56. print("error \(error)")
  57. }
  58. }
  59. }
  60. }
  61. }
  62. //-{% if generateTestStubs %}
  63. /// Simple fake implementation of {{ .|session:file,service,method }} that returns a previously-defined set of results
  64. /// and stores sent values for later verification.
  65. class {{ .|session:file,service,method }}TestStub : {{ .|service:file,service }}SessionTestStub, {{ .|session:file,service,method }} {
  66. var inputs: [{{ method|input }}] = []
  67. var outputs: [{{ method|output }}] = []
  68. func receive() throws -> {{ method|input }} {
  69. if let input = inputs.first {
  70. inputs.removeFirst()
  71. return input
  72. } else {
  73. throw {{ .|clienterror:file,service }}.endOfStream
  74. }
  75. }
  76. func send(_ response: {{ method|output }}, completion: @escaping ()->()) throws {
  77. outputs.append(response)
  78. }
  79. func close() throws { }
  80. }
  81. //-{% endif %}