| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246 |
- /*
- * Copyright 2021, gRPC Authors All rights reserved.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- import SwiftProtobuf
- import SwiftProtobufPluginLibrary
- // MARK: - Client protocol
- extension Generator {
- internal func printAsyncServiceClientProtocol() {
- let comments = self.service.protoSourceComments()
- if !comments.isEmpty {
- // Source comments already have the leading '///'
- self.println(comments, newline: false)
- }
- self.printAvailabilityForAsyncAwait()
- self.println("\(self.access) protocol \(self.asyncClientProtocolName): GRPCClient {")
- self.withIndentation {
- self.println("static var serviceDescriptor: GRPCServiceDescriptor { get }")
- self.println("var interceptors: \(self.clientInterceptorProtocolName)? { get }")
- for method in service.methods {
- self.println()
- self.method = method
- let rpcType = streamingType(self.method)
- let callType = Types.call(for: rpcType)
- let arguments: [String]
- switch rpcType {
- case .unary, .serverStreaming:
- arguments = [
- "_ request: \(self.methodInputName)",
- "callOptions: \(Types.clientCallOptions)?",
- ]
- case .clientStreaming, .bidirectionalStreaming:
- arguments = [
- "callOptions: \(Types.clientCallOptions)?"
- ]
- }
- self.printFunction(
- name: self.methodMakeFunctionCallName,
- arguments: arguments,
- returnType: "\(callType)<\(self.methodInputName), \(self.methodOutputName)>",
- bodyBuilder: nil
- )
- }
- }
- self.println("}") // protocol
- }
- }
- // MARK: - Client protocol default implementation: Calls
- extension Generator {
- internal func printAsyncClientProtocolExtension() {
- self.printAvailabilityForAsyncAwait()
- self.withIndentation("extension \(self.asyncClientProtocolName)", braces: .curly) {
- // Service descriptor.
- self.withIndentation(
- "\(self.access) static var serviceDescriptor: GRPCServiceDescriptor",
- braces: .curly
- ) {
- self.println("return \(self.serviceClientMetadata).serviceDescriptor")
- }
- self.println()
- // Interceptor factory.
- self.withIndentation(
- "\(self.access) var interceptors: \(self.clientInterceptorProtocolName)?",
- braces: .curly
- ) {
- self.println("return nil")
- }
- // 'Unsafe' calls.
- for method in self.service.methods {
- self.println()
- self.method = method
- let rpcType = streamingType(self.method)
- let callType = Types.call(for: rpcType)
- let callTypeWithoutPrefix = Types.call(for: rpcType, withGRPCPrefix: false)
- switch rpcType {
- case .unary, .serverStreaming:
- self.printFunction(
- name: self.methodMakeFunctionCallName,
- arguments: [
- "_ request: \(self.methodInputName)",
- "callOptions: \(Types.clientCallOptions)? = nil",
- ],
- returnType: "\(callType)<\(self.methodInputName), \(self.methodOutputName)>",
- access: self.access
- ) {
- self.withIndentation("return self.make\(callTypeWithoutPrefix)", braces: .round) {
- self.println("path: \(self.methodPathUsingClientMetadata),")
- self.println("request: request,")
- self.println("callOptions: callOptions ?? self.defaultCallOptions,")
- self.println(
- "interceptors: self.interceptors?.\(self.methodInterceptorFactoryName)() ?? []"
- )
- }
- }
- case .clientStreaming, .bidirectionalStreaming:
- self.printFunction(
- name: self.methodMakeFunctionCallName,
- arguments: ["callOptions: \(Types.clientCallOptions)? = nil"],
- returnType: "\(callType)<\(self.methodInputName), \(self.methodOutputName)>",
- access: self.access
- ) {
- self.withIndentation("return self.make\(callTypeWithoutPrefix)", braces: .round) {
- self.println("path: \(self.methodPathUsingClientMetadata),")
- self.println("callOptions: callOptions ?? self.defaultCallOptions,")
- self.println(
- "interceptors: self.interceptors?.\(self.methodInterceptorFactoryName)() ?? []"
- )
- }
- }
- }
- }
- }
- }
- }
- // MARK: - Client protocol extension: "Simple, but safe" call wrappers.
- extension Generator {
- internal func printAsyncClientProtocolSafeWrappersExtension() {
- self.printAvailabilityForAsyncAwait()
- self.withIndentation("extension \(self.asyncClientProtocolName)", braces: .curly) {
- for (i, method) in self.service.methods.enumerated() {
- self.method = method
- let rpcType = streamingType(self.method)
- let callTypeWithoutPrefix = Types.call(for: rpcType, withGRPCPrefix: false)
- let streamsResponses = [.serverStreaming, .bidirectionalStreaming].contains(rpcType)
- let streamsRequests = [.clientStreaming, .bidirectionalStreaming].contains(rpcType)
- // (protocol, requires sendable)
- let sequenceProtocols: [(String, Bool)?] =
- streamsRequests
- ? [("Sequence", false), ("AsyncSequence", true)]
- : [nil]
- for (j, sequenceProtocol) in sequenceProtocols.enumerated() {
- // Print a new line if this is not the first function in the extension.
- if i > 0 || j > 0 {
- self.println()
- }
- let functionName =
- streamsRequests
- ? "\(self.methodFunctionName)<RequestStream>"
- : self.methodFunctionName
- let requestParamName = streamsRequests ? "requests" : "request"
- let requestParamType = streamsRequests ? "RequestStream" : self.methodInputName
- let returnType =
- streamsResponses
- ? Types.responseStream(of: self.methodOutputName)
- : self.methodOutputName
- let maybeWhereClause = sequenceProtocol.map { protocolName, mustBeSendable -> String in
- let constraints = [
- "RequestStream: \(protocolName)" + (mustBeSendable ? " & Sendable" : ""),
- "RequestStream.Element == \(self.methodInputName)",
- ]
- return "where " + constraints.joined(separator: ", ")
- }
- self.printFunction(
- name: functionName,
- arguments: [
- "_ \(requestParamName): \(requestParamType)",
- "callOptions: \(Types.clientCallOptions)? = nil",
- ],
- returnType: returnType,
- access: self.access,
- async: !streamsResponses,
- throws: !streamsResponses,
- genericWhereClause: maybeWhereClause
- ) {
- self.withIndentation(
- "return\(!streamsResponses ? " try await" : "") self.perform\(callTypeWithoutPrefix)",
- braces: .round
- ) {
- self.println("path: \(self.methodPathUsingClientMetadata),")
- self.println("\(requestParamName): \(requestParamName),")
- self.println("callOptions: callOptions ?? self.defaultCallOptions,")
- self.println(
- "interceptors: self.interceptors?.\(self.methodInterceptorFactoryName)() ?? []"
- )
- }
- }
- }
- }
- }
- }
- }
- // MARK: - Client protocol implementation
- extension Generator {
- internal func printAsyncServiceClientImplementation() {
- self.printAvailabilityForAsyncAwait()
- self.withIndentation(
- "\(self.access) struct \(self.asyncClientStructName): \(self.asyncClientProtocolName)",
- braces: .curly
- ) {
- self.println("\(self.access) var channel: GRPCChannel")
- self.println("\(self.access) var defaultCallOptions: CallOptions")
- self.println("\(self.access) var interceptors: \(self.clientInterceptorProtocolName)?")
- self.println()
- self.println("\(self.access) init(")
- self.withIndentation {
- self.println("channel: GRPCChannel,")
- self.println("defaultCallOptions: CallOptions = CallOptions(),")
- self.println("interceptors: \(self.clientInterceptorProtocolName)? = nil")
- }
- self.println(") {")
- self.withIndentation {
- self.println("self.channel = channel")
- self.println("self.defaultCallOptions = defaultCallOptions")
- self.println("self.interceptors = interceptors")
- }
- self.println("}")
- }
- }
- }
|