ControlService.swift 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  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(fullyQualifiedService: "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(fullyQualifiedService: "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(fullyQualifiedService: "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(fullyQualifiedService: "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(fullyQualifiedService: "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. router.registerHandler(
  64. forMethod: MethodDescriptor(fullyQualifiedService: "Control", method: "PeerInfo"),
  65. deserializer: JSONDeserializer<String>(),
  66. serializer: JSONSerializer<String>()
  67. ) { request, context in
  68. return StreamingServerResponse { response in
  69. let info = try await self.peerInfo(context: context)
  70. try await response.write(info)
  71. return [:]
  72. }
  73. }
  74. }
  75. }
  76. extension ControlService {
  77. private func waitForCancellation(
  78. request: ServerRequest<CancellationKind>,
  79. context: ServerContext
  80. ) async throws -> StreamingServerResponse<Empty> {
  81. switch request.message {
  82. case .awaitCancelled:
  83. return StreamingServerResponse { _ in
  84. try await context.cancellation.cancelled
  85. return [:]
  86. }
  87. case .withCancellationHandler:
  88. let signal = AsyncStream.makeStream(of: Void.self)
  89. return StreamingServerResponse { _ in
  90. await withRPCCancellationHandler {
  91. for await _ in signal.stream {}
  92. return [:]
  93. } onCancelRPC: {
  94. signal.continuation.finish()
  95. }
  96. }
  97. }
  98. }
  99. private func peerInfo(context: ServerContext) async throws -> String {
  100. return context.peer
  101. }
  102. private func handle(
  103. request: StreamingServerRequest<ControlInput>
  104. ) async throws -> StreamingServerResponse<ControlOutput> {
  105. var iterator = request.messages.makeAsyncIterator()
  106. guard let message = try await iterator.next() else {
  107. // Empty input stream, empty output stream.
  108. return StreamingServerResponse { _ in [:] }
  109. }
  110. // Check if the request is for a trailers-only response.
  111. if let status = message.status, message.isTrailersOnly {
  112. var trailers = message.echoMetadataInTrailers ? request.metadata.echo() : [:]
  113. for (key, value) in message.trailingMetadataToAdd {
  114. trailers.addString(value, forKey: key)
  115. }
  116. let code = Status.Code(rawValue: status.code.rawValue).flatMap { RPCError.Code($0) }
  117. if let code = code {
  118. throw RPCError(code: code, message: status.message, metadata: trailers)
  119. } else {
  120. // Invalid code, the request is invalid, so throw an appropriate error.
  121. throw RPCError(
  122. code: .invalidArgument,
  123. message: "Trailers only response must use a non-OK status code"
  124. )
  125. }
  126. }
  127. // Not a trailers-only response. Should the metadata be echo'd back?
  128. var metadata = message.echoMetadataInHeaders ? request.metadata.echo() : [:]
  129. for (key, value) in message.initialMetadataToAdd {
  130. metadata.addString(value, forKey: key)
  131. }
  132. // The iterator needs to be transferred into the response. This is okay: we won't touch the
  133. // iterator again from the current concurrency domain.
  134. let transfer = UnsafeTransfer(iterator)
  135. return StreamingServerResponse(metadata: metadata) { writer in
  136. // Finish dealing with the first message.
  137. switch try await self.processMessage(message, metadata: request.metadata, writer: writer) {
  138. case .return(let metadata):
  139. return metadata
  140. case .continue:
  141. ()
  142. }
  143. var iterator = transfer.wrappedValue
  144. // Process the rest of the messages.
  145. while let message = try await iterator.next() {
  146. switch try await self.processMessage(message, metadata: request.metadata, writer: writer) {
  147. case .return(let metadata):
  148. return metadata
  149. case .continue:
  150. ()
  151. }
  152. }
  153. // Input stream finished without explicitly setting a status; finish the RPC cleanly.
  154. return [:]
  155. }
  156. }
  157. private enum NextProcessingStep {
  158. case `return`(Metadata)
  159. case `continue`
  160. }
  161. private func processMessage(
  162. _ input: ControlInput,
  163. metadata: Metadata,
  164. writer: RPCWriter<ControlOutput>
  165. ) async throws -> NextProcessingStep {
  166. // If messages were requested, build a response and send them back.
  167. if input.numberOfMessages > 0 {
  168. let output = ControlOutput(
  169. payload: Data(
  170. repeating: input.payloadParameters.content,
  171. count: input.payloadParameters.size
  172. )
  173. )
  174. for _ in 0 ..< input.numberOfMessages {
  175. try await writer.write(output)
  176. }
  177. }
  178. // Check whether the RPC should be finished (i.e. the input `hasStatus`).
  179. guard let status = input.status else {
  180. if input.echoMetadataInTrailers || !input.trailingMetadataToAdd.isEmpty {
  181. // There was no status in the input, but echo metadata in trailers was set. This is an
  182. // implicit 'ok' status.
  183. var trailers = input.echoMetadataInTrailers ? metadata.echo() : [:]
  184. for (key, value) in input.trailingMetadataToAdd {
  185. trailers.addString(value, forKey: key)
  186. }
  187. return .return(trailers)
  188. } else {
  189. // No status, and not echoing back metadata. Continue consuming the input stream.
  190. return .continue
  191. }
  192. }
  193. // Build the trailers.
  194. var trailers = input.echoMetadataInTrailers ? metadata.echo() : [:]
  195. for (key, value) in input.trailingMetadataToAdd {
  196. trailers.addString(value, forKey: key)
  197. }
  198. if status.code == .ok {
  199. return .return(trailers)
  200. }
  201. // Non-OK status code, throw an error.
  202. let code = RPCError.Code(status.code)
  203. if let code = code {
  204. // Valid error code, throw it.
  205. throw RPCError(code: code, message: status.message, metadata: trailers)
  206. } else {
  207. // Invalid error code, throw an appropriate error.
  208. throw RPCError(
  209. code: .invalidArgument,
  210. message: "Invalid error code '\(status.code)'"
  211. )
  212. }
  213. }
  214. }
  215. extension Metadata {
  216. fileprivate func echo() -> Self {
  217. var copy = Metadata()
  218. copy.reserveCapacity(self.count)
  219. for (key, value) in self {
  220. // Header field names mustn't contain ":".
  221. let key = "echo-" + key.replacingOccurrences(of: ":", with: "")
  222. switch value {
  223. case .string(let stringValue):
  224. copy.addString(stringValue, forKey: key)
  225. case .binary(let binaryValue):
  226. copy.addBinary(binaryValue, forKey: key)
  227. }
  228. }
  229. return copy
  230. }
  231. }
  232. private struct UnsafeTransfer<Wrapped> {
  233. var wrappedValue: Wrapped
  234. init(_ wrappedValue: Wrapped) {
  235. self.wrappedValue = wrappedValue
  236. }
  237. }
  238. extension UnsafeTransfer: @unchecked Sendable {}