ControlService.swift 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. /*
  2. * Copyright 2024, 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 Foundation
  17. import GRPCCore
  18. struct ControlService: RegistrableRPCService {
  19. func registerMethods(with router: inout RPCRouter) {
  20. router.registerHandler(
  21. forMethod: MethodDescriptor(service: "Control", method: "Unary"),
  22. deserializer: JSONDeserializer<ControlInput>(),
  23. serializer: JSONSerializer<ControlOutput>(),
  24. handler: { request, context in
  25. return try await self.handle(request: request)
  26. }
  27. )
  28. router.registerHandler(
  29. forMethod: MethodDescriptor(service: "Control", method: "ServerStream"),
  30. deserializer: JSONDeserializer<ControlInput>(),
  31. serializer: JSONSerializer<ControlOutput>(),
  32. handler: { request, context in
  33. return try await self.handle(request: request)
  34. }
  35. )
  36. router.registerHandler(
  37. forMethod: MethodDescriptor(service: "Control", method: "ClientStream"),
  38. deserializer: JSONDeserializer<ControlInput>(),
  39. serializer: JSONSerializer<ControlOutput>(),
  40. handler: { request, context in
  41. return try await self.handle(request: request)
  42. }
  43. )
  44. router.registerHandler(
  45. forMethod: MethodDescriptor(service: "Control", method: "BidiStream"),
  46. deserializer: JSONDeserializer<ControlInput>(),
  47. serializer: JSONSerializer<ControlOutput>(),
  48. handler: { request, context in
  49. return try await self.handle(request: request)
  50. }
  51. )
  52. router.registerHandler(
  53. forMethod: MethodDescriptor(service: "Control", method: "WaitForCancellation"),
  54. deserializer: JSONDeserializer<CancellationKind>(),
  55. serializer: JSONSerializer<Empty>(),
  56. handler: { request, context in
  57. return try await self.waitForCancellation(
  58. request: ServerRequest(stream: request),
  59. context: context
  60. )
  61. }
  62. )
  63. }
  64. }
  65. extension ControlService {
  66. private func waitForCancellation(
  67. request: ServerRequest<CancellationKind>,
  68. context: ServerContext
  69. ) async throws -> StreamingServerResponse<Empty> {
  70. switch request.message {
  71. case .awaitCancelled:
  72. return StreamingServerResponse { _ in
  73. try await context.cancellation.cancelled
  74. return [:]
  75. }
  76. case .withCancellationHandler:
  77. let signal = AsyncStream.makeStream(of: Void.self)
  78. return StreamingServerResponse { _ in
  79. await withRPCCancellationHandler {
  80. for await _ in signal.stream {}
  81. return [:]
  82. } onCancelRPC: {
  83. signal.continuation.finish()
  84. }
  85. }
  86. }
  87. }
  88. private func handle(
  89. request: StreamingServerRequest<ControlInput>
  90. ) async throws -> StreamingServerResponse<ControlOutput> {
  91. var iterator = request.messages.makeAsyncIterator()
  92. guard let message = try await iterator.next() else {
  93. // Empty input stream, empty output stream.
  94. return StreamingServerResponse { _ in [:] }
  95. }
  96. // Check if the request is for a trailers-only response.
  97. if let status = message.status, message.isTrailersOnly {
  98. let trailers = message.echoMetadataInTrailers ? request.metadata.echo() : [:]
  99. let code = Status.Code(rawValue: status.code.rawValue).flatMap { RPCError.Code($0) }
  100. if let code = code {
  101. throw RPCError(code: code, message: status.message, metadata: trailers)
  102. } else {
  103. // Invalid code, the request is invalid, so throw an appropriate error.
  104. throw RPCError(
  105. code: .invalidArgument,
  106. message: "Trailers only response must use a non-OK status code"
  107. )
  108. }
  109. }
  110. // Not a trailers-only response. Should the metadata be echo'd back?
  111. let metadata = message.echoMetadataInHeaders ? request.metadata.echo() : [:]
  112. // The iterator needs to be transferred into the response. This is okay: we won't touch the
  113. // iterator again from the current concurrency domain.
  114. let transfer = UnsafeTransfer(iterator)
  115. return StreamingServerResponse(metadata: metadata) { writer in
  116. // Finish dealing with the first message.
  117. switch try await self.processMessage(message, metadata: request.metadata, writer: writer) {
  118. case .return(let metadata):
  119. return metadata
  120. case .continue:
  121. ()
  122. }
  123. var iterator = transfer.wrappedValue
  124. // Process the rest of the messages.
  125. while let message = try await iterator.next() {
  126. switch try await self.processMessage(message, metadata: request.metadata, writer: writer) {
  127. case .return(let metadata):
  128. return metadata
  129. case .continue:
  130. ()
  131. }
  132. }
  133. // Input stream finished without explicitly setting a status; finish the RPC cleanly.
  134. return [:]
  135. }
  136. }
  137. private enum NextProcessingStep {
  138. case `return`(Metadata)
  139. case `continue`
  140. }
  141. private func processMessage(
  142. _ input: ControlInput,
  143. metadata: Metadata,
  144. writer: RPCWriter<ControlOutput>
  145. ) async throws -> NextProcessingStep {
  146. // If messages were requested, build a response and send them back.
  147. if input.numberOfMessages > 0 {
  148. let output = ControlOutput(
  149. payload: Data(
  150. repeating: input.payloadParameters.content,
  151. count: input.payloadParameters.size
  152. )
  153. )
  154. for _ in 0 ..< input.numberOfMessages {
  155. try await writer.write(output)
  156. }
  157. }
  158. // Check whether the RPC should be finished (i.e. the input `hasStatus`).
  159. guard let status = input.status else {
  160. if input.echoMetadataInTrailers {
  161. // There was no status in the input, but echo metadata in trailers was set. This is an
  162. // implicit 'ok' status.
  163. let trailers = input.echoMetadataInTrailers ? metadata.echo() : [:]
  164. return .return(trailers)
  165. } else {
  166. // No status, and not echoing back metadata. Continue consuming the input stream.
  167. return .continue
  168. }
  169. }
  170. // Build the trailers.
  171. let trailers = input.echoMetadataInTrailers ? metadata.echo() : [:]
  172. if status.code == .ok {
  173. return .return(trailers)
  174. }
  175. // Non-OK status code, throw an error.
  176. let code = RPCError.Code(status.code)
  177. if let code = code {
  178. // Valid error code, throw it.
  179. throw RPCError(code: code, message: status.message, metadata: trailers)
  180. } else {
  181. // Invalid error code, throw an appropriate error.
  182. throw RPCError(
  183. code: .invalidArgument,
  184. message: "Invalid error code '\(status.code)'"
  185. )
  186. }
  187. }
  188. }
  189. extension Metadata {
  190. fileprivate func echo() -> Self {
  191. var copy = Metadata()
  192. copy.reserveCapacity(self.count)
  193. for (key, value) in self {
  194. // Header field names mustn't contain ":".
  195. let key = "echo-" + key.replacingOccurrences(of: ":", with: "")
  196. switch value {
  197. case .string(let stringValue):
  198. copy.addString(stringValue, forKey: key)
  199. case .binary(let binaryValue):
  200. copy.addBinary(binaryValue, forKey: key)
  201. }
  202. }
  203. return copy
  204. }
  205. }
  206. private struct UnsafeTransfer<Wrapped> {
  207. var wrappedValue: Wrapped
  208. init(_ wrappedValue: Wrapped) {
  209. self.wrappedValue = wrappedValue
  210. }
  211. }
  212. extension UnsafeTransfer: @unchecked Sendable {}