options.swift 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. /*
  2. * Copyright 2017, 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 SwiftProtobufPluginLibrary
  18. enum GenerationError: Error {
  19. /// Raised when parsing the parameter string and found an unknown key
  20. case unknownParameter(name: String)
  21. /// Raised when a parameter was giving an invalid value
  22. case invalidParameterValue(name: String, value: String)
  23. /// Raised to wrap another error but provide a context message.
  24. case wrappedError(message: String, error: Error)
  25. var localizedDescription: String {
  26. switch self {
  27. case let .unknownParameter(name):
  28. return "Unknown generation parameter '\(name)'"
  29. case let .invalidParameterValue(name, value):
  30. return "Unknown value for generation parameter '\(name)': '\(value)'"
  31. case let .wrappedError(message, error):
  32. return "\(message): \(error.localizedDescription)"
  33. }
  34. }
  35. }
  36. final class GeneratorOptions {
  37. enum Visibility: String {
  38. case `internal` = "Internal"
  39. case `public` = "Public"
  40. case `package` = "Package"
  41. var sourceSnippet: String {
  42. switch self {
  43. case .internal:
  44. return "internal"
  45. case .public:
  46. return "public"
  47. case .package:
  48. return "package"
  49. }
  50. }
  51. }
  52. private(set) var visibility = Visibility.internal
  53. private(set) var generateServer = true
  54. private(set) var generateClient = true
  55. private(set) var generateTestClient = false
  56. private(set) var keepMethodCasing = false
  57. private(set) var protoToModuleMappings = ProtoFileToModuleMappings()
  58. private(set) var fileNaming = FileNaming.FullPath
  59. private(set) var extraModuleImports: [String] = []
  60. private(set) var gRPCModuleName = "GRPC"
  61. private(set) var swiftProtobufModuleName = "SwiftProtobuf"
  62. private(set) var generateReflectionData = false
  63. #if compiler(>=6.0)
  64. private(set) var v2 = false
  65. #endif
  66. init(parameter: String?) throws {
  67. for pair in GeneratorOptions.parseParameter(string: parameter) {
  68. switch pair.key {
  69. case "Visibility":
  70. if let value = Visibility(rawValue: pair.value) {
  71. self.visibility = value
  72. } else {
  73. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  74. }
  75. case "Server":
  76. if let value = Bool(pair.value) {
  77. self.generateServer = value
  78. } else {
  79. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  80. }
  81. case "Client":
  82. if let value = Bool(pair.value) {
  83. self.generateClient = value
  84. } else {
  85. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  86. }
  87. case "TestClient":
  88. if let value = Bool(pair.value) {
  89. self.generateTestClient = value
  90. } else {
  91. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  92. }
  93. case "KeepMethodCasing":
  94. if let value = Bool(pair.value) {
  95. self.keepMethodCasing = value
  96. } else {
  97. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  98. }
  99. case "ProtoPathModuleMappings":
  100. if !pair.value.isEmpty {
  101. do {
  102. self.protoToModuleMappings = try ProtoFileToModuleMappings(path: pair.value)
  103. } catch let e {
  104. throw GenerationError.wrappedError(
  105. message: "Parameter 'ProtoPathModuleMappings=\(pair.value)'",
  106. error: e
  107. )
  108. }
  109. }
  110. case "FileNaming":
  111. if let value = FileNaming(rawValue: pair.value) {
  112. self.fileNaming = value
  113. } else {
  114. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  115. }
  116. case "ExtraModuleImports":
  117. if !pair.value.isEmpty {
  118. self.extraModuleImports.append(pair.value)
  119. } else {
  120. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  121. }
  122. case "GRPCModuleName":
  123. if !pair.value.isEmpty {
  124. self.gRPCModuleName = pair.value
  125. } else {
  126. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  127. }
  128. case "SwiftProtobufModuleName":
  129. if !pair.value.isEmpty {
  130. self.swiftProtobufModuleName = pair.value
  131. } else {
  132. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  133. }
  134. case "ReflectionData":
  135. if let value = Bool(pair.value) {
  136. self.generateReflectionData = value
  137. } else {
  138. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  139. }
  140. #if compiler(>=6.0)
  141. case "_V2":
  142. if let value = Bool(pair.value) {
  143. self.v2 = value
  144. } else {
  145. throw GenerationError.invalidParameterValue(name: pair.key, value: pair.value)
  146. }
  147. #endif
  148. default:
  149. throw GenerationError.unknownParameter(name: pair.key)
  150. }
  151. }
  152. }
  153. static func parseParameter(string: String?) -> [(key: String, value: String)] {
  154. guard let string = string, !string.isEmpty else {
  155. return []
  156. }
  157. let parts = string.components(separatedBy: ",")
  158. // Partitions the string into the section before the = and after the =
  159. let result = parts.map { string -> (key: String, value: String) in
  160. // Finds the equal sign and exits early if none
  161. guard let index = string.range(of: "=")?.lowerBound else {
  162. return (string, "")
  163. }
  164. // Creates key/value pair and trims whitespace
  165. let key = string[..<index]
  166. .trimmingCharacters(in: .whitespacesAndNewlines)
  167. let value = string[string.index(after: index)...]
  168. .trimmingCharacters(in: .whitespacesAndNewlines)
  169. return (key: key, value: value)
  170. }
  171. return result
  172. }
  173. }