IDLToStructuredSwiftTranslatorSnippetBasedTests.swift 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579
  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. #if os(macOS) || os(Linux) // swift-format doesn't like canImport(Foundation.Process)
  17. import XCTest
  18. @testable import GRPCCodeGen
  19. final class IDLToStructuredSwiftTranslatorSnippetBasedTests: XCTestCase {
  20. typealias MethodDescriptor = GRPCCodeGen.CodeGenerationRequest.ServiceDescriptor.MethodDescriptor
  21. typealias ServiceDescriptor = GRPCCodeGen.CodeGenerationRequest.ServiceDescriptor
  22. typealias Name = GRPCCodeGen.CodeGenerationRequest.Name
  23. func testImports() throws {
  24. var dependencies = [CodeGenerationRequest.Dependency]()
  25. dependencies.append(CodeGenerationRequest.Dependency(module: "Foo"))
  26. dependencies.append(
  27. CodeGenerationRequest.Dependency(item: .init(kind: .typealias, name: "Bar"), module: "Foo")
  28. )
  29. dependencies.append(
  30. CodeGenerationRequest.Dependency(item: .init(kind: .struct, name: "Baz"), module: "Foo")
  31. )
  32. dependencies.append(
  33. CodeGenerationRequest.Dependency(item: .init(kind: .class, name: "Bac"), module: "Foo")
  34. )
  35. dependencies.append(
  36. CodeGenerationRequest.Dependency(item: .init(kind: .enum, name: "Bap"), module: "Foo")
  37. )
  38. dependencies.append(
  39. CodeGenerationRequest.Dependency(item: .init(kind: .protocol, name: "Bat"), module: "Foo")
  40. )
  41. dependencies.append(
  42. CodeGenerationRequest.Dependency(item: .init(kind: .let, name: "Baq"), module: "Foo")
  43. )
  44. dependencies.append(
  45. CodeGenerationRequest.Dependency(item: .init(kind: .var, name: "Bag"), module: "Foo")
  46. )
  47. dependencies.append(
  48. CodeGenerationRequest.Dependency(item: .init(kind: .func, name: "Bak"), module: "Foo")
  49. )
  50. let expectedSwift =
  51. """
  52. /// Some really exciting license header 2023.
  53. import GRPCCore
  54. import Foo
  55. import typealias Foo.Bar
  56. import struct Foo.Baz
  57. import class Foo.Bac
  58. import enum Foo.Bap
  59. import protocol Foo.Bat
  60. import let Foo.Baq
  61. import var Foo.Bag
  62. import func Foo.Bak
  63. """
  64. try self.assertIDLToStructuredSwiftTranslation(
  65. codeGenerationRequest: makeCodeGenerationRequest(dependencies: dependencies),
  66. expectedSwift: expectedSwift,
  67. accessLevel: .public
  68. )
  69. }
  70. func testPreconcurrencyImports() throws {
  71. var dependencies = [CodeGenerationRequest.Dependency]()
  72. dependencies.append(CodeGenerationRequest.Dependency(module: "Foo", preconcurrency: .required))
  73. dependencies.append(
  74. CodeGenerationRequest.Dependency(
  75. item: .init(kind: .enum, name: "Bar"),
  76. module: "Foo",
  77. preconcurrency: .required
  78. )
  79. )
  80. dependencies.append(
  81. CodeGenerationRequest.Dependency(
  82. module: "Baz",
  83. preconcurrency: .requiredOnOS(["Deq", "Der"])
  84. )
  85. )
  86. let expectedSwift =
  87. """
  88. /// Some really exciting license header 2023.
  89. import GRPCCore
  90. @preconcurrency import Foo
  91. @preconcurrency import enum Foo.Bar
  92. #if os(Deq) || os(Der)
  93. @preconcurrency import Baz
  94. #else
  95. import Baz
  96. #endif
  97. """
  98. try self.assertIDLToStructuredSwiftTranslation(
  99. codeGenerationRequest: makeCodeGenerationRequest(dependencies: dependencies),
  100. expectedSwift: expectedSwift,
  101. accessLevel: .public
  102. )
  103. }
  104. func testSPIImports() throws {
  105. var dependencies = [CodeGenerationRequest.Dependency]()
  106. dependencies.append(CodeGenerationRequest.Dependency(module: "Foo", spi: "Secret"))
  107. dependencies.append(
  108. CodeGenerationRequest.Dependency(
  109. item: .init(kind: .enum, name: "Bar"),
  110. module: "Foo",
  111. spi: "Secret"
  112. )
  113. )
  114. let expectedSwift =
  115. """
  116. /// Some really exciting license header 2023.
  117. import GRPCCore
  118. @_spi(Secret) import Foo
  119. @_spi(Secret) import enum Foo.Bar
  120. """
  121. try self.assertIDLToStructuredSwiftTranslation(
  122. codeGenerationRequest: makeCodeGenerationRequest(dependencies: dependencies),
  123. expectedSwift: expectedSwift,
  124. accessLevel: .public
  125. )
  126. }
  127. func testGeneration() throws {
  128. var dependencies = [CodeGenerationRequest.Dependency]()
  129. dependencies.append(CodeGenerationRequest.Dependency(module: "Foo", spi: "Secret"))
  130. dependencies.append(
  131. CodeGenerationRequest.Dependency(
  132. item: .init(kind: .enum, name: "Bar"),
  133. module: "Foo",
  134. spi: "Secret"
  135. )
  136. )
  137. let serviceA = ServiceDescriptor(
  138. documentation: "/// Documentation for AService\n",
  139. name: Name(base: "ServiceA", generatedUpperCase: "ServiceA", generatedLowerCase: "serviceA"),
  140. namespace: Name(
  141. base: "namespaceA",
  142. generatedUpperCase: "NamespaceA",
  143. generatedLowerCase: "namespaceA"
  144. ),
  145. methods: []
  146. )
  147. let expectedSwift =
  148. """
  149. /// Some really exciting license header 2023.
  150. import GRPCCore
  151. @_spi(Secret) import Foo
  152. @_spi(Secret) import enum Foo.Bar
  153. public enum NamespaceA {
  154. public enum ServiceA {
  155. public enum Method {
  156. public static let descriptors: [MethodDescriptor] = []
  157. }
  158. public typealias StreamingServiceProtocol = NamespaceA_ServiceAStreamingServiceProtocol
  159. public typealias ServiceProtocol = NamespaceA_ServiceAServiceProtocol
  160. }
  161. }
  162. /// Documentation for AService
  163. @available(macOS 13.0, iOS 16.0, watchOS 9.0, tvOS 16.0, *)
  164. public protocol NamespaceA_ServiceAStreamingServiceProtocol: GRPCCore.RegistrableRPCService {}
  165. /// Conformance to `GRPCCore.RegistrableRPCService`.
  166. extension NamespaceA.ServiceA.StreamingServiceProtocol {
  167. @available(macOS 13.0, iOS 16.0, watchOS 9.0, tvOS 16.0, *)
  168. public func registerMethods(with router: inout GRPCCore.RPCRouter) {}
  169. }
  170. /// Documentation for AService
  171. public protocol NamespaceA_ServiceAServiceProtocol: NamespaceA.ServiceA.StreamingServiceProtocol {}
  172. /// Partial conformance to `NamespaceA_ServiceAStreamingServiceProtocol`.
  173. extension NamespaceA.ServiceA.ServiceProtocol {
  174. }
  175. """
  176. try self.assertIDLToStructuredSwiftTranslation(
  177. codeGenerationRequest: makeCodeGenerationRequest(
  178. services: [serviceA],
  179. dependencies: dependencies
  180. ),
  181. expectedSwift: expectedSwift,
  182. accessLevel: .public,
  183. server: true
  184. )
  185. }
  186. private func assertIDLToStructuredSwiftTranslation(
  187. codeGenerationRequest: CodeGenerationRequest,
  188. expectedSwift: String,
  189. accessLevel: SourceGenerator.Configuration.AccessLevel,
  190. server: Bool = false
  191. ) throws {
  192. let translator = IDLToStructuredSwiftTranslator()
  193. let structuredSwift = try translator.translate(
  194. codeGenerationRequest: codeGenerationRequest,
  195. accessLevel: accessLevel,
  196. client: false,
  197. server: server
  198. )
  199. let renderer = TextBasedRenderer.default
  200. let sourceFile = try renderer.render(structured: structuredSwift)
  201. let contents = sourceFile.contents
  202. try XCTAssertEqualWithDiff(contents, expectedSwift)
  203. }
  204. func testSameNameServicesNoNamespaceError() throws {
  205. let serviceA = ServiceDescriptor(
  206. documentation: "Documentation for AService",
  207. name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  208. namespace: Name(base: "", generatedUpperCase: "", generatedLowerCase: ""),
  209. methods: []
  210. )
  211. let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceA])
  212. let translator = IDLToStructuredSwiftTranslator()
  213. XCTAssertThrowsError(
  214. ofType: CodeGenError.self,
  215. try translator.translate(
  216. codeGenerationRequest: codeGenerationRequest,
  217. accessLevel: .public,
  218. client: true,
  219. server: true
  220. )
  221. ) {
  222. error in
  223. XCTAssertEqual(
  224. error as CodeGenError,
  225. CodeGenError(
  226. code: .nonUniqueServiceName,
  227. message: """
  228. Services must have unique descriptors. \
  229. AService is the descriptor of at least two different services.
  230. """
  231. )
  232. )
  233. }
  234. }
  235. func testSameDescriptorsServicesNoNamespaceError() throws {
  236. let serviceA = ServiceDescriptor(
  237. documentation: "Documentation for AService",
  238. name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  239. namespace: Name(base: "", generatedUpperCase: "", generatedLowerCase: ""),
  240. methods: []
  241. )
  242. let serviceB = ServiceDescriptor(
  243. documentation: "Documentation for BService",
  244. name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  245. namespace: Name(base: "", generatedUpperCase: "", generatedLowerCase: ""),
  246. methods: []
  247. )
  248. let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceB])
  249. let translator = IDLToStructuredSwiftTranslator()
  250. XCTAssertThrowsError(
  251. ofType: CodeGenError.self,
  252. try translator.translate(
  253. codeGenerationRequest: codeGenerationRequest,
  254. accessLevel: .public,
  255. client: true,
  256. server: true
  257. )
  258. ) {
  259. error in
  260. XCTAssertEqual(
  261. error as CodeGenError,
  262. CodeGenError(
  263. code: .nonUniqueServiceName,
  264. message: """
  265. Services must have unique descriptors. AService is the descriptor of at least two different services.
  266. """
  267. )
  268. )
  269. }
  270. }
  271. func testSameDescriptorsSameNamespaceError() throws {
  272. let serviceA = ServiceDescriptor(
  273. documentation: "Documentation for AService",
  274. name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  275. namespace: Name(
  276. base: "namespacea",
  277. generatedUpperCase: "NamespaceA",
  278. generatedLowerCase: "namespacea"
  279. ),
  280. methods: []
  281. )
  282. let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceA])
  283. let translator = IDLToStructuredSwiftTranslator()
  284. XCTAssertThrowsError(
  285. ofType: CodeGenError.self,
  286. try translator.translate(
  287. codeGenerationRequest: codeGenerationRequest,
  288. accessLevel: .public,
  289. client: true,
  290. server: true
  291. )
  292. ) {
  293. error in
  294. XCTAssertEqual(
  295. error as CodeGenError,
  296. CodeGenError(
  297. code: .nonUniqueServiceName,
  298. message: """
  299. Services must have unique descriptors. \
  300. namespacea.AService is the descriptor of at least two different services.
  301. """
  302. )
  303. )
  304. }
  305. }
  306. func testSameGeneratedNameServicesSameNamespaceError() throws {
  307. let serviceA = ServiceDescriptor(
  308. documentation: "/// Documentation for AService\n",
  309. name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  310. namespace: Name(
  311. base: "namespacea",
  312. generatedUpperCase: "NamespaceA",
  313. generatedLowerCase: "namespacea"
  314. ),
  315. methods: []
  316. )
  317. let serviceB = ServiceDescriptor(
  318. documentation: "/// Documentation for BService\n",
  319. name: Name(base: "BService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  320. namespace: Name(
  321. base: "namespacea",
  322. generatedUpperCase: "NamespaceA",
  323. generatedLowerCase: "namespacea"
  324. ),
  325. methods: []
  326. )
  327. let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceB])
  328. let translator = IDLToStructuredSwiftTranslator()
  329. XCTAssertThrowsError(
  330. ofType: CodeGenError.self,
  331. try translator.translate(
  332. codeGenerationRequest: codeGenerationRequest,
  333. accessLevel: .internal,
  334. client: true,
  335. server: true
  336. )
  337. ) {
  338. error in
  339. XCTAssertEqual(
  340. error as CodeGenError,
  341. CodeGenError(
  342. code: .nonUniqueServiceName,
  343. message: """
  344. Services within the same namespace must have unique generated upper case names. \
  345. AService is used as a generated upper case name for multiple services in the namespacea namespace.
  346. """
  347. )
  348. )
  349. }
  350. }
  351. func testSameBaseNameMethodsSameServiceError() throws {
  352. let methodA = MethodDescriptor(
  353. documentation: "Documentation for MethodA",
  354. name: Name(base: "MethodA", generatedUpperCase: "MethodA", generatedLowerCase: "methodA"),
  355. isInputStreaming: false,
  356. isOutputStreaming: false,
  357. inputType: "NamespaceA_ServiceARequest",
  358. outputType: "NamespaceA_ServiceAResponse"
  359. )
  360. let service = ServiceDescriptor(
  361. documentation: "Documentation for AService",
  362. name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  363. namespace: Name(
  364. base: "namespacea",
  365. generatedUpperCase: "NamespaceA",
  366. generatedLowerCase: "namespacea"
  367. ),
  368. methods: [methodA, methodA]
  369. )
  370. let codeGenerationRequest = makeCodeGenerationRequest(services: [service])
  371. let translator = IDLToStructuredSwiftTranslator()
  372. XCTAssertThrowsError(
  373. ofType: CodeGenError.self,
  374. try translator.translate(
  375. codeGenerationRequest: codeGenerationRequest,
  376. accessLevel: .public,
  377. client: true,
  378. server: true
  379. )
  380. ) {
  381. error in
  382. XCTAssertEqual(
  383. error as CodeGenError,
  384. CodeGenError(
  385. code: .nonUniqueMethodName,
  386. message: """
  387. Methods of a service must have unique base names. \
  388. MethodA is used as a base name for multiple methods of the AService service.
  389. """
  390. )
  391. )
  392. }
  393. }
  394. func testSameGeneratedUpperCaseNameMethodsSameServiceError() throws {
  395. let methodA = MethodDescriptor(
  396. documentation: "Documentation for MethodA",
  397. name: Name(base: "MethodA", generatedUpperCase: "MethodA", generatedLowerCase: "methodA"),
  398. isInputStreaming: false,
  399. isOutputStreaming: false,
  400. inputType: "NamespaceA_ServiceARequest",
  401. outputType: "NamespaceA_ServiceAResponse"
  402. )
  403. let methodB = MethodDescriptor(
  404. documentation: "Documentation for MethodA",
  405. name: Name(base: "MethodB", generatedUpperCase: "MethodA", generatedLowerCase: "methodA"),
  406. isInputStreaming: false,
  407. isOutputStreaming: false,
  408. inputType: "NamespaceA_ServiceARequest",
  409. outputType: "NamespaceA_ServiceAResponse"
  410. )
  411. let service = ServiceDescriptor(
  412. documentation: "Documentation for AService",
  413. name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  414. namespace: Name(
  415. base: "namespacea",
  416. generatedUpperCase: "NamespaceA",
  417. generatedLowerCase: "namespacea"
  418. ),
  419. methods: [methodA, methodB]
  420. )
  421. let codeGenerationRequest = makeCodeGenerationRequest(services: [service])
  422. let translator = IDLToStructuredSwiftTranslator()
  423. XCTAssertThrowsError(
  424. ofType: CodeGenError.self,
  425. try translator.translate(
  426. codeGenerationRequest: codeGenerationRequest,
  427. accessLevel: .public,
  428. client: true,
  429. server: true
  430. )
  431. ) {
  432. error in
  433. XCTAssertEqual(
  434. error as CodeGenError,
  435. CodeGenError(
  436. code: .nonUniqueMethodName,
  437. message: """
  438. Methods of a service must have unique generated upper case names. \
  439. MethodA is used as a generated upper case name for multiple methods of the AService service.
  440. """
  441. )
  442. )
  443. }
  444. }
  445. func testSameLowerCaseNameMethodsSameServiceError() throws {
  446. let methodA = MethodDescriptor(
  447. documentation: "Documentation for MethodA",
  448. name: Name(base: "MethodA", generatedUpperCase: "MethodA", generatedLowerCase: "methodA"),
  449. isInputStreaming: false,
  450. isOutputStreaming: false,
  451. inputType: "NamespaceA_ServiceARequest",
  452. outputType: "NamespaceA_ServiceAResponse"
  453. )
  454. let methodB = MethodDescriptor(
  455. documentation: "Documentation for MethodA",
  456. name: Name(base: "MethodB", generatedUpperCase: "MethodB", generatedLowerCase: "methodA"),
  457. isInputStreaming: false,
  458. isOutputStreaming: false,
  459. inputType: "NamespaceA_ServiceARequest",
  460. outputType: "NamespaceA_ServiceAResponse"
  461. )
  462. let service = ServiceDescriptor(
  463. documentation: "Documentation for AService",
  464. name: Name(base: "AService", generatedUpperCase: "AService", generatedLowerCase: "aService"),
  465. namespace: Name(
  466. base: "namespacea",
  467. generatedUpperCase: "NamespaceA",
  468. generatedLowerCase: "namespacea"
  469. ),
  470. methods: [methodA, methodB]
  471. )
  472. let codeGenerationRequest = makeCodeGenerationRequest(services: [service])
  473. let translator = IDLToStructuredSwiftTranslator()
  474. XCTAssertThrowsError(
  475. ofType: CodeGenError.self,
  476. try translator.translate(
  477. codeGenerationRequest: codeGenerationRequest,
  478. accessLevel: .public,
  479. client: true,
  480. server: true
  481. )
  482. ) {
  483. error in
  484. XCTAssertEqual(
  485. error as CodeGenError,
  486. CodeGenError(
  487. code: .nonUniqueMethodName,
  488. message: """
  489. Methods of a service must have unique lower case names. \
  490. methodA is used as a signature name for multiple methods of the AService service.
  491. """
  492. )
  493. )
  494. }
  495. }
  496. func testSameGeneratedNameNoNamespaceServiceAndNamespaceError() throws {
  497. let serviceA = ServiceDescriptor(
  498. documentation: "Documentation for SameName service with no namespace",
  499. name: Name(base: "SameName", generatedUpperCase: "SameName", generatedLowerCase: "sameName"),
  500. namespace: Name(base: "", generatedUpperCase: "", generatedLowerCase: ""),
  501. methods: []
  502. )
  503. let serviceB = ServiceDescriptor(
  504. documentation: "Documentation for BService",
  505. name: Name(base: "BService", generatedUpperCase: "BService", generatedLowerCase: "bService"),
  506. namespace: Name(
  507. base: "sameName",
  508. generatedUpperCase: "SameName",
  509. generatedLowerCase: "sameName"
  510. ),
  511. methods: []
  512. )
  513. let codeGenerationRequest = makeCodeGenerationRequest(services: [serviceA, serviceB])
  514. let translator = IDLToStructuredSwiftTranslator()
  515. XCTAssertThrowsError(
  516. ofType: CodeGenError.self,
  517. try translator.translate(
  518. codeGenerationRequest: codeGenerationRequest,
  519. accessLevel: .public,
  520. client: true,
  521. server: true
  522. )
  523. ) {
  524. error in
  525. XCTAssertEqual(
  526. error as CodeGenError,
  527. CodeGenError(
  528. code: .nonUniqueServiceName,
  529. message: """
  530. Services with no namespace must not have the same generated upper case names as the namespaces. \
  531. SameName is used as a generated upper case name for a service with no namespace and a namespace.
  532. """
  533. )
  534. )
  535. }
  536. }
  537. }
  538. #endif // os(macOS) || os(Linux)