IDLToStructuredSwiftTranslatorSnippetBasedTests.swift 20 KB

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