main.swift 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  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 SwiftProtobuf
  18. import SwiftProtobufPluginLibrary
  19. func Log(_ message: String) {
  20. FileHandle.standardError.write((message + "\n").data(using: .utf8)!)
  21. }
  22. // from apple/swift-protobuf/Sources/protoc-gen-swift/StringUtils.swift
  23. func splitPath(pathname: String) -> (dir: String, base: String, suffix: String) {
  24. var dir = ""
  25. var base = ""
  26. var suffix = ""
  27. #if swift(>=3.2)
  28. let pathnameChars = pathname
  29. #else
  30. let pathnameChars = pathname.characters
  31. #endif
  32. for c in pathnameChars {
  33. if c == "/" {
  34. dir += base + suffix + String(c)
  35. base = ""
  36. suffix = ""
  37. } else if c == "." {
  38. base += suffix
  39. suffix = String(c)
  40. } else {
  41. suffix += String(c)
  42. }
  43. }
  44. #if swift(>=3.2)
  45. let validSuffix = suffix.isEmpty || suffix.first == "."
  46. #else
  47. let validSuffix = suffix.isEmpty || suffix.characters.first == "."
  48. #endif
  49. if !validSuffix {
  50. base += suffix
  51. suffix = ""
  52. }
  53. return (dir: dir, base: base, suffix: suffix)
  54. }
  55. enum FileNaming: String {
  56. case FullPath
  57. case PathToUnderscores
  58. case DropPath
  59. }
  60. func outputFileName(component: String, fileDescriptor: FileDescriptor,
  61. fileNamingOption: FileNaming) -> String {
  62. let ext = "." + component + ".swift"
  63. let pathParts = splitPath(pathname: fileDescriptor.name)
  64. switch fileNamingOption {
  65. case .FullPath:
  66. return pathParts.dir + pathParts.base + ext
  67. case .PathToUnderscores:
  68. let dirWithUnderscores =
  69. pathParts.dir.replacingOccurrences(of: "/", with: "_")
  70. return dirWithUnderscores + pathParts.base + ext
  71. case .DropPath:
  72. return pathParts.base + ext
  73. }
  74. }
  75. var generatedFiles: [String: Int] = [:]
  76. func uniqueOutputFileName(
  77. component: String,
  78. fileDescriptor: FileDescriptor,
  79. fileNamingOption: FileNaming
  80. ) -> String {
  81. let defaultName = outputFileName(
  82. component: component,
  83. fileDescriptor: fileDescriptor,
  84. fileNamingOption: fileNamingOption
  85. )
  86. if let count = generatedFiles[defaultName] {
  87. generatedFiles[defaultName] = count + 1
  88. return outputFileName(
  89. component: "\(count)." + component,
  90. fileDescriptor: fileDescriptor,
  91. fileNamingOption: fileNamingOption
  92. )
  93. } else {
  94. generatedFiles[defaultName] = 1
  95. return defaultName
  96. }
  97. }
  98. func main() throws {
  99. // initialize responses
  100. var response = Google_Protobuf_Compiler_CodeGeneratorResponse(
  101. files: [],
  102. supportedFeatures: [.proto3Optional]
  103. )
  104. // read plugin input
  105. let rawRequest = try Stdin.readall()
  106. let request = try Google_Protobuf_Compiler_CodeGeneratorRequest(serializedData: rawRequest)
  107. let options = try GeneratorOptions(parameter: request.parameter)
  108. // Build the SwiftProtobufPluginLibrary model of the plugin input
  109. let descriptorSet = DescriptorSet(protos: request.protoFile)
  110. // Only generate output for services.
  111. for name in request.fileToGenerate {
  112. let fileDescriptor = descriptorSet.lookupFileDescriptor(protoName: name)
  113. if !fileDescriptor.services.isEmpty {
  114. let grpcFileName = uniqueOutputFileName(
  115. component: "grpc",
  116. fileDescriptor: fileDescriptor,
  117. fileNamingOption: options.fileNaming
  118. )
  119. let grpcGenerator = Generator(fileDescriptor, options: options)
  120. var grpcFile = Google_Protobuf_Compiler_CodeGeneratorResponse.File()
  121. grpcFile.name = grpcFileName
  122. grpcFile.content = grpcGenerator.code
  123. response.file.append(grpcFile)
  124. }
  125. }
  126. // return everything to the caller
  127. let serializedResponse = try response.serializedData()
  128. Stdout.write(bytes: serializedResponse)
  129. }
  130. do {
  131. try main()
  132. } catch {
  133. Log("ERROR: \(error)")
  134. }