| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576 |
- /*
- * Copyright 2024, 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.
- */
- #if os(macOS) || os(Linux) // swift-format doesn't like canImport(Foundation.Process)
- import XCTest
- @testable import GRPCCodeGen
- final class IDLToStructuredSwiftTranslatorSnippetBasedTests: XCTestCase {
- typealias MethodDescriptor = GRPCCodeGen.CodeGenerationRequest.ServiceDescriptor.MethodDescriptor
- typealias ServiceDescriptor = GRPCCodeGen.CodeGenerationRequest.ServiceDescriptor
- typealias Name = GRPCCodeGen.CodeGenerationRequest.Name
- func testImports() throws {
- var dependencies = [CodeGenerationRequest.Dependency]()
- dependencies.append(CodeGenerationRequest.Dependency(module: "Foo"))
- dependencies.append(
- CodeGenerationRequest.Dependency(item: .init(kind: .typealias, name: "Bar"), module: "Foo")
- )
- dependencies.append(
- CodeGenerationRequest.Dependency(item: .init(kind: .struct, name: "Baz"), module: "Foo")
- )
- dependencies.append(
- CodeGenerationRequest.Dependency(item: .init(kind: .class, name: "Bac"), module: "Foo")
- )
- dependencies.append(
- CodeGenerationRequest.Dependency(item: .init(kind: .enum, name: "Bap"), module: "Foo")
- )
- dependencies.append(
- CodeGenerationRequest.Dependency(item: .init(kind: .protocol, name: "Bat"), module: "Foo")
- )
- dependencies.append(
- CodeGenerationRequest.Dependency(item: .init(kind: .let, name: "Baq"), module: "Foo")
- )
- dependencies.append(
- CodeGenerationRequest.Dependency(item: .init(kind: .var, name: "Bag"), module: "Foo")
- )
- dependencies.append(
- CodeGenerationRequest.Dependency(item: .init(kind: .func, name: "Bak"), module: "Foo")
- )
- let expectedSwift =
- """
- /// Some really exciting license header 2023.
- import GRPCCore
- import Foo
- import typealias Foo.Bar
- import struct Foo.Baz
- import class Foo.Bac
- import enum Foo.Bap
- import protocol Foo.Bat
- import let Foo.Baq
- import var Foo.Bag
- import func Foo.Bak
- """
- try self.assertIDLToStructuredSwiftTranslation(
- codeGenerationRequest: makeCodeGenerationRequest(dependencies: dependencies),
- expectedSwift: expectedSwift,
- accessLevel: .public
- )
- }
- func testPreconcurrencyImports() throws {
- var dependencies = [CodeGenerationRequest.Dependency]()
- dependencies.append(CodeGenerationRequest.Dependency(module: "Foo", preconcurrency: .required))
- dependencies.append(
- CodeGenerationRequest.Dependency(
- item: .init(kind: .enum, name: "Bar"),
- module: "Foo",
- preconcurrency: .required
- )
- )
- dependencies.append(
- CodeGenerationRequest.Dependency(
- module: "Baz",
- preconcurrency: .requiredOnOS(["Deq", "Der"])
- )
- )
- let expectedSwift =
- """
- /// Some really exciting license header 2023.
- import GRPCCore
- @preconcurrency import Foo
- @preconcurrency import enum Foo.Bar
- #if os(Deq) || os(Der)
- @preconcurrency import Baz
- #else
- import Baz
- #endif
- """
- try self.assertIDLToStructuredSwiftTranslation(
- codeGenerationRequest: makeCodeGenerationRequest(dependencies: dependencies),
- expectedSwift: expectedSwift,
- accessLevel: .public
- )
- }
- func testSPIImports() throws {
- var dependencies = [CodeGenerationRequest.Dependency]()
- dependencies.append(CodeGenerationRequest.Dependency(module: "Foo", spi: "Secret"))
- dependencies.append(
- CodeGenerationRequest.Dependency(
- item: .init(kind: .enum, name: "Bar"),
- module: "Foo",
- spi: "Secret"
- )
- )
- let expectedSwift =
- """
- /// Some really exciting license header 2023.
- import GRPCCore
- @_spi(Secret) import Foo
- @_spi(Secret) import enum Foo.Bar
- """
- try self.assertIDLToStructuredSwiftTranslation(
- codeGenerationRequest: makeCodeGenerationRequest(dependencies: dependencies),
- expectedSwift: expectedSwift,
- accessLevel: .public
- )
- }
- func testGeneration() throws {
- var dependencies = [CodeGenerationRequest.Dependency]()
- dependencies.append(CodeGenerationRequest.Dependency(module: "Foo", spi: "Secret"))
- dependencies.append(
- CodeGenerationRequest.Dependency(
- item: .init(kind: .enum, name: "Bar"),
- module: "Foo",
- spi: "Secret"
- )
- )
- let serviceA = ServiceDescriptor(
- documentation: "/// Documentation for AService\n",
- name: Name(base: "ServiceA", generatedUpperCase: "ServiceA", generatedLowerCase: "serviceA"),
- namespace: Name(
- base: "namespaceA",
- generatedUpperCase: "NamespaceA",
- generatedLowerCase: "namespaceA"
- ),
- methods: []
- )
- let expectedSwift =
- """
- /// Some really exciting license header 2023.
- import GRPCCore
- @_spi(Secret) import Foo
- @_spi(Secret) import enum Foo.Bar
- public enum NamespaceA {
- public enum ServiceA {
- public enum Methods {}
- public static let methods: [MethodDescriptor] = []
- public typealias StreamingServiceProtocol = NamespaceA_ServiceAServiceStreamingProtocol
- public typealias ServiceProtocol = NamespaceA_ServiceAServiceProtocol
- }
- }
- /// Documentation for AService
- public protocol NamespaceA_ServiceAStreamingServiceProtocol: GRPCCore.RegistrableRPCService {}
- /// Conformance to `GRPCCore.RegistrableRPCService`.
- extension NamespaceA.ServiceA.StreamingServiceProtocol {
- public func registerMethods(with router: inout GRPCCore.RPCRouter) {}
- }
- /// Documentation for AService
- public protocol NamespaceA_ServiceAServiceProtocol: NamespaceA.ServiceA.StreamingServiceProtocol {}
- /// Partial conformance to `NamespaceA_ServiceAStreamingServiceProtocol`.
- extension NamespaceA.ServiceA.ServiceProtocol {
- }
- """
- try self.assertIDLToStructuredSwiftTranslation(
- codeGenerationRequest: makeCodeGenerationRequest(
- services: [serviceA],
- dependencies: dependencies
- ),
- expectedSwift: expectedSwift,
- accessLevel: .public,
- server: true
- )
- }
- private func assertIDLToStructuredSwiftTranslation(
- codeGenerationRequest: CodeGenerationRequest,
- expectedSwift: String,
- accessLevel: SourceGenerator.Configuration.AccessLevel,
- server: Bool = false
- ) throws {
- let translator = IDLToStructuredSwiftTranslator()
- let structuredSwift = try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: accessLevel,
- client: false,
- server: server
- )
- let renderer = TextBasedRenderer.default
- let sourceFile = try renderer.render(structured: structuredSwift)
- let contents = sourceFile.contents
- try XCTAssertEqualWithDiff(contents, expectedSwift)
- }
- func testSameNameServicesNoNamespaceError() throws {
- let serviceA = ServiceDescriptor(
- documentation: "Documentation for AService",
- name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(base: "", generatedUpperCase: "", generatedLowerCase: ""),
- methods: []
- )
- let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceA])
- let translator = IDLToStructuredSwiftTranslator()
- XCTAssertThrowsError(
- ofType: CodeGenError.self,
- try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: .public,
- client: true,
- server: true
- )
- ) {
- error in
- XCTAssertEqual(
- error as CodeGenError,
- CodeGenError(
- code: .nonUniqueServiceName,
- message: """
- Services must have unique descriptors. \
- AService is the descriptor of at least two different services.
- """
- )
- )
- }
- }
- func testSameDescriptorsServicesNoNamespaceError() throws {
- let serviceA = ServiceDescriptor(
- documentation: "Documentation for AService",
- name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(base: "", generatedUpperCase: "", generatedLowerCase: ""),
- methods: []
- )
- let serviceB = ServiceDescriptor(
- documentation: "Documentation for BService",
- name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(base: "", generatedUpperCase: "", generatedLowerCase: ""),
- methods: []
- )
- let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceB])
- let translator = IDLToStructuredSwiftTranslator()
- XCTAssertThrowsError(
- ofType: CodeGenError.self,
- try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: .public,
- client: true,
- server: true
- )
- ) {
- error in
- XCTAssertEqual(
- error as CodeGenError,
- CodeGenError(
- code: .nonUniqueServiceName,
- message: """
- Services must have unique descriptors. AService is the descriptor of at least two different services.
- """
- )
- )
- }
- }
- func testSameDescriptorsSameNamespaceError() throws {
- let serviceA = ServiceDescriptor(
- documentation: "Documentation for AService",
- name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(
- base: "namespacea",
- generatedUpperCase: "NamespaceA",
- generatedLowerCase: "namespacea"
- ),
- methods: []
- )
- let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceA])
- let translator = IDLToStructuredSwiftTranslator()
- XCTAssertThrowsError(
- ofType: CodeGenError.self,
- try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: .public,
- client: true,
- server: true
- )
- ) {
- error in
- XCTAssertEqual(
- error as CodeGenError,
- CodeGenError(
- code: .nonUniqueServiceName,
- message: """
- Services must have unique descriptors. \
- namespacea.AService is the descriptor of at least two different services.
- """
- )
- )
- }
- }
- func testSameGeneratedNameServicesSameNamespaceError() throws {
- let serviceA = ServiceDescriptor(
- documentation: "/// Documentation for AService\n",
- name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(
- base: "namespacea",
- generatedUpperCase: "NamespaceA",
- generatedLowerCase: "namespacea"
- ),
- methods: []
- )
- let serviceB = ServiceDescriptor(
- documentation: "/// Documentation for BService\n",
- name: Name(base: "BService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(
- base: "namespacea",
- generatedUpperCase: "NamespaceA",
- generatedLowerCase: "namespacea"
- ),
- methods: []
- )
- let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceB])
- let translator = IDLToStructuredSwiftTranslator()
- XCTAssertThrowsError(
- ofType: CodeGenError.self,
- try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: .internal,
- client: true,
- server: true
- )
- ) {
- error in
- XCTAssertEqual(
- error as CodeGenError,
- CodeGenError(
- code: .nonUniqueServiceName,
- message: """
- Services within the same namespace must have unique generated upper case names. \
- AService is used as a generated upper case name for multiple services in the namespacea namespace.
- """
- )
- )
- }
- }
- func testSameBaseNameMethodsSameServiceError() throws {
- let methodA = MethodDescriptor(
- documentation: "Documentation for MethodA",
- name: Name(base: "MethodA", generatedUpperCase: "MethodA", generatedLowerCase: "methodA"),
- isInputStreaming: false,
- isOutputStreaming: false,
- inputType: "NamespaceA_ServiceARequest",
- outputType: "NamespaceA_ServiceAResponse"
- )
- let service = ServiceDescriptor(
- documentation: "Documentation for AService",
- name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(
- base: "namespacea",
- generatedUpperCase: "NamespaceA",
- generatedLowerCase: "namespacea"
- ),
- methods: [methodA, methodA]
- )
- let codeGenerationRequest = makeCodeGenerationRequest(services: [service])
- let translator = IDLToStructuredSwiftTranslator()
- XCTAssertThrowsError(
- ofType: CodeGenError.self,
- try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: .public,
- client: true,
- server: true
- )
- ) {
- error in
- XCTAssertEqual(
- error as CodeGenError,
- CodeGenError(
- code: .nonUniqueMethodName,
- message: """
- Methods of a service must have unique base names. \
- MethodA is used as a base name for multiple methods of the AService service.
- """
- )
- )
- }
- }
- func testSameGeneratedUpperCaseNameMethodsSameServiceError() throws {
- let methodA = MethodDescriptor(
- documentation: "Documentation for MethodA",
- name: Name(base: "MethodA", generatedUpperCase: "MethodA", generatedLowerCase: "methodA"),
- isInputStreaming: false,
- isOutputStreaming: false,
- inputType: "NamespaceA_ServiceARequest",
- outputType: "NamespaceA_ServiceAResponse"
- )
- let methodB = MethodDescriptor(
- documentation: "Documentation for MethodA",
- name: Name(base: "MethodB", generatedUpperCase: "MethodA", generatedLowerCase: "methodA"),
- isInputStreaming: false,
- isOutputStreaming: false,
- inputType: "NamespaceA_ServiceARequest",
- outputType: "NamespaceA_ServiceAResponse"
- )
- let service = ServiceDescriptor(
- documentation: "Documentation for AService",
- name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(
- base: "namespacea",
- generatedUpperCase: "NamespaceA",
- generatedLowerCase: "namespacea"
- ),
- methods: [methodA, methodB]
- )
- let codeGenerationRequest = makeCodeGenerationRequest(services: [service])
- let translator = IDLToStructuredSwiftTranslator()
- XCTAssertThrowsError(
- ofType: CodeGenError.self,
- try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: .public,
- client: true,
- server: true
- )
- ) {
- error in
- XCTAssertEqual(
- error as CodeGenError,
- CodeGenError(
- code: .nonUniqueMethodName,
- message: """
- Methods of a service must have unique generated upper case names. \
- MethodA is used as a generated upper case name for multiple methods of the AService service.
- """
- )
- )
- }
- }
- func testSameLowerCaseNameMethodsSameServiceError() throws {
- let methodA = MethodDescriptor(
- documentation: "Documentation for MethodA",
- name: Name(base: "MethodA", generatedUpperCase: "MethodA", generatedLowerCase: "methodA"),
- isInputStreaming: false,
- isOutputStreaming: false,
- inputType: "NamespaceA_ServiceARequest",
- outputType: "NamespaceA_ServiceAResponse"
- )
- let methodB = MethodDescriptor(
- documentation: "Documentation for MethodA",
- name: Name(base: "MethodB", generatedUpperCase: "MethodB", generatedLowerCase: "methodA"),
- isInputStreaming: false,
- isOutputStreaming: false,
- inputType: "NamespaceA_ServiceARequest",
- outputType: "NamespaceA_ServiceAResponse"
- )
- let service = ServiceDescriptor(
- documentation: "Documentation for AService",
- name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
- namespace: Name(
- base: "namespacea",
- generatedUpperCase: "NamespaceA",
- generatedLowerCase: "namespacea"
- ),
- methods: [methodA, methodB]
- )
- let codeGenerationRequest = makeCodeGenerationRequest(services: [service])
- let translator = IDLToStructuredSwiftTranslator()
- XCTAssertThrowsError(
- ofType: CodeGenError.self,
- try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: .public,
- client: true,
- server: true
- )
- ) {
- error in
- XCTAssertEqual(
- error as CodeGenError,
- CodeGenError(
- code: .nonUniqueMethodName,
- message: """
- Methods of a service must have unique lower case names. \
- methodA is used as a signature name for multiple methods of the AService service.
- """
- )
- )
- }
- }
- func testSameGeneratedNameNoNamespaceServiceAndNamespaceError() throws {
- let serviceA = ServiceDescriptor(
- documentation: "Documentation for SameName service with no namespace",
- name: Name(base: "SameName", generatedUpperCase: "SameName", generatedLowerCase: "sameName"),
- namespace: Name(base: "", generatedUpperCase: "", generatedLowerCase: ""),
- methods: []
- )
- let serviceB = ServiceDescriptor(
- documentation: "Documentation for BService",
- name: Name(base: "BService", generatedUpperCase: "BService", generatedLowerCase: "bService"),
- namespace: Name(
- base: "sameName",
- generatedUpperCase: "SameName",
- generatedLowerCase: "sameName"
- ),
- methods: []
- )
- let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceB])
- let translator = IDLToStructuredSwiftTranslator()
- XCTAssertThrowsError(
- ofType: CodeGenError.self,
- try translator.translate(
- codeGenerationRequest: codeGenerationRequest,
- accessLevel: .public,
- client: true,
- server: true
- )
- ) {
- error in
- XCTAssertEqual(
- error as CodeGenError,
- CodeGenError(
- code: .nonUniqueServiceName,
- message: """
- Services with no namespace must not have the same generated upper case names as the namespaces. \
- SameName is used as a generated upper case name for a service with no namespace and a namespace.
- """
- )
- )
- }
- }
- }
- #endif // os(macOS) || os(Linux)
|