main.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. package main
  2. import (
  3. "flag"
  4. "fmt"
  5. "os"
  6. "runtime/debug"
  7. "strings"
  8. "github.com/grpc-ecosystem/grpc-gateway/v2/internal/codegenerator"
  9. "github.com/grpc-ecosystem/grpc-gateway/v2/internal/descriptor"
  10. "github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-openapiv2/internal/genopenapi"
  11. "github.com/grpc-ecosystem/grpc-gateway/v2/utilities"
  12. "google.golang.org/grpc/grpclog"
  13. "google.golang.org/protobuf/proto"
  14. "google.golang.org/protobuf/types/pluginpb"
  15. )
  16. var (
  17. importPrefix = flag.String("import_prefix", "", "prefix to be added to go package paths for imported proto files")
  18. file = flag.String("file", "-", "where to load data from")
  19. allowDeleteBody = flag.Bool("allow_delete_body", false, "unless set, HTTP DELETE methods may not have a body")
  20. grpcAPIConfiguration = flag.String("grpc_api_configuration", "", "path to file which describes the gRPC API Configuration in YAML format")
  21. allowMerge = flag.Bool("allow_merge", false, "if set, generation one OpenAPI file out of multiple protos")
  22. mergeFileName = flag.String("merge_file_name", "apidocs", "target OpenAPI file name prefix after merge")
  23. useJSONNamesForFields = flag.Bool("json_names_for_fields", true, "if disabled, the original proto name will be used for generating OpenAPI definitions")
  24. repeatedPathParamSeparator = flag.String("repeated_path_param_separator", "csv", "configures how repeated fields should be split. Allowed values are `csv`, `pipes`, `ssv` and `tsv`")
  25. versionFlag = flag.Bool("version", false, "print the current version")
  26. _ = flag.Bool("allow_repeated_fields_in_body", true, "allows to use repeated field in `body` and `response_body` field of `google.api.http` annotation option. DEPRECATED: the value is ignored and always behaves as `true`.")
  27. includePackageInTags = flag.Bool("include_package_in_tags", false, "if unset, the gRPC service name is added to the `Tags` field of each operation. If set and the `package` directive is shown in the proto file, the package name will be prepended to the service name")
  28. useFQNForOpenAPIName = flag.Bool("fqn_for_openapi_name", false, "if set, the object's OpenAPI names will use the fully qualified names from the proto definition (ie my.package.MyMessage.MyInnerMessage). DEPRECATED: prefer `openapi_naming_strategy=fqn`")
  29. openAPINamingStrategy = flag.String("openapi_naming_strategy", "", "use the given OpenAPI naming strategy. Allowed values are `legacy`, `fqn`, `simple`, `package`. If unset, either `legacy` or `fqn` are selected, depending on the value of the `fqn_for_openapi_name` flag")
  30. useGoTemplate = flag.Bool("use_go_templates", false, "if set, you can use Go templates in protofile comments")
  31. goTemplateArgs = utilities.StringArrayFlag(flag.CommandLine, "go_template_args", "provide a custom value that can override a key in the Go template. Requires the `use_go_templates` option to be set")
  32. ignoreComments = flag.Bool("ignore_comments", false, "if set, all protofile comments are excluded from output")
  33. removeInternalComments = flag.Bool("remove_internal_comments", false, "if set, removes all substrings in comments that start with `(--` and end with `--)` as specified in https://google.aip.dev/192#internal-comments")
  34. disableDefaultErrors = flag.Bool("disable_default_errors", false, "if set, disables generation of default errors. This is useful if you have defined custom error handling")
  35. enumsAsInts = flag.Bool("enums_as_ints", false, "whether to render enum values as integers, as opposed to string values")
  36. simpleOperationIDs = flag.Bool("simple_operation_ids", false, "whether to remove the service prefix in the operationID generation. Can introduce duplicate operationIDs, use with caution.")
  37. proto3OptionalNullable = flag.Bool("proto3_optional_nullable", false, "whether Proto3 Optional fields should be marked as x-nullable")
  38. openAPIConfiguration = flag.String("openapi_configuration", "", "path to file which describes the OpenAPI Configuration in YAML format")
  39. generateUnboundMethods = flag.Bool("generate_unbound_methods", false, "generate swagger metadata even for RPC methods that have no HttpRule annotation")
  40. recursiveDepth = flag.Int("recursive-depth", 1000, "maximum recursion count allowed for a field type")
  41. omitEnumDefaultValue = flag.Bool("omit_enum_default_value", false, "if set, omit default enum value")
  42. outputFormat = flag.String("output_format", string(genopenapi.FormatJSON), fmt.Sprintf("output content format. Allowed values are: `%s`, `%s`", genopenapi.FormatJSON, genopenapi.FormatYAML))
  43. visibilityRestrictionSelectors = utilities.StringArrayFlag(flag.CommandLine, "visibility_restriction_selectors", "list of `google.api.VisibilityRule` visibility labels to include in the generated output when a visibility annotation is defined. Repeat this option to supply multiple values. Elements without visibility annotations are unaffected by this setting.")
  44. disableServiceTags = flag.Bool("disable_service_tags", false, "if set, disables generation of service tags. This is useful if you do not want to expose the names of your backend grpc services.")
  45. disableDefaultResponses = flag.Bool("disable_default_responses", false, "if set, disables generation of default responses. Useful if you have to support custom response codes that are not 200.")
  46. useAllOfForRefs = flag.Bool("use_allof_for_refs", false, "if set, will use allOf as container for $ref to preserve same-level properties.")
  47. allowPatchFeature = flag.Bool("allow_patch_feature", true, "whether to hide update_mask fields in PATCH requests from the generated swagger file.")
  48. preserveRPCOrder = flag.Bool("preserve_rpc_order", false, "if true, will ensure the order of paths emitted in openapi swagger files mirror the order of RPC methods found in proto files. If false, emitted paths will be ordered alphabetically.")
  49. enableRpcDeprecation = flag.Bool("enable_rpc_deprecation", false, "whether to process grpc method's deprecated option.")
  50. expandSlashedPathPatterns = flag.Bool("expand_slashed_path_patterns", false, "if set, expands path parameters with URI sub-paths into the URI. For example, \"/v1/{name=projects/*}/resource\" becomes \"/v1/projects/{project}/resource\".")
  51. _ = flag.Bool("logtostderr", false, "Legacy glog compatibility. This flag is a no-op, you can safely remove it")
  52. )
  53. // Variables set by goreleaser at build time
  54. var (
  55. version = "dev"
  56. commit = "unknown"
  57. date = "unknown"
  58. )
  59. /*
  60. *
  61. swagger json 文件中
  62. 在请求地址中添加参数
  63. "parameters": [
  64. {
  65. "name": "r",
  66. "in": "query",
  67. "required": true,
  68. "style": "form",
  69. "example": "abcdefg",
  70. "description": "接口地址"
  71. }
  72. ]
  73. */
  74. func main() {
  75. flag.Parse()
  76. if *versionFlag {
  77. if commit == "unknown" {
  78. buildInfo, ok := debug.ReadBuildInfo()
  79. if ok {
  80. version = buildInfo.Main.Version
  81. for _, setting := range buildInfo.Settings {
  82. if setting.Key == "vcs.revision" {
  83. commit = setting.Value
  84. }
  85. if setting.Key == "vcs.time" {
  86. date = setting.Value
  87. }
  88. }
  89. }
  90. }
  91. fmt.Printf("Version %v, commit %v, built at %v\n", version, commit, date)
  92. os.Exit(0)
  93. }
  94. reg := descriptor.NewRegistry()
  95. if grpclog.V(1) {
  96. grpclog.Info("Processing code generator request")
  97. }
  98. f := os.Stdin
  99. if *file != "-" {
  100. var err error
  101. f, err = os.Open(*file)
  102. if err != nil {
  103. grpclog.Fatal(err)
  104. }
  105. }
  106. if grpclog.V(1) {
  107. grpclog.Info("Parsing code generator request")
  108. }
  109. req, err := codegenerator.ParseRequest(f)
  110. if err != nil {
  111. grpclog.Fatal(err)
  112. }
  113. if grpclog.V(1) {
  114. grpclog.Info("Parsed code generator request")
  115. }
  116. pkgMap := make(map[string]string)
  117. if req.Parameter != nil {
  118. if err := parseReqParam(req.GetParameter(), flag.CommandLine, pkgMap); err != nil {
  119. grpclog.Fatalf("Error parsing flags: %v", err)
  120. }
  121. }
  122. reg.SetPrefix(*importPrefix)
  123. reg.SetAllowDeleteBody(*allowDeleteBody)
  124. reg.SetAllowMerge(*allowMerge)
  125. reg.SetMergeFileName(*mergeFileName)
  126. reg.SetUseJSONNamesForFields(*useJSONNamesForFields)
  127. flag.Visit(func(f *flag.Flag) {
  128. if f.Name == "allow_repeated_fields_in_body" {
  129. grpclog.Warning("The `allow_repeated_fields_in_body` flag is deprecated and will always behave as `true`.")
  130. }
  131. })
  132. reg.SetIncludePackageInTags(*includePackageInTags)
  133. reg.SetUseFQNForOpenAPIName(*useFQNForOpenAPIName)
  134. // Set the naming strategy either directly from the flag, or via the value of the legacy fqn_for_openapi_name
  135. // flag.
  136. namingStrategy := *openAPINamingStrategy
  137. if *useFQNForOpenAPIName {
  138. if namingStrategy != "" {
  139. grpclog.Fatal("The deprecated `fqn_for_openapi_name` flag must remain unset if `openapi_naming_strategy` is set.")
  140. }
  141. grpclog.Warning("The `fqn_for_openapi_name` flag is deprecated. Please use `openapi_naming_strategy=fqn` instead.")
  142. namingStrategy = "fqn"
  143. } else if namingStrategy == "" {
  144. namingStrategy = "legacy"
  145. }
  146. if strategyFn := genopenapi.LookupNamingStrategy(namingStrategy); strategyFn == nil {
  147. emitError(fmt.Errorf("invalid naming strategy %q", namingStrategy))
  148. return
  149. }
  150. if *useGoTemplate && *ignoreComments {
  151. emitError(fmt.Errorf("`ignore_comments` and `use_go_templates` are mutually exclusive and cannot be enabled at the same time"))
  152. return
  153. }
  154. reg.SetUseGoTemplate(*useGoTemplate)
  155. reg.SetIgnoreComments(*ignoreComments)
  156. reg.SetRemoveInternalComments(*removeInternalComments)
  157. if len(*goTemplateArgs) > 0 && !*useGoTemplate {
  158. emitError(fmt.Errorf("`go_template_args` requires `use_go_templates` to be enabled"))
  159. return
  160. }
  161. reg.SetGoTemplateArgs(*goTemplateArgs)
  162. reg.SetOpenAPINamingStrategy(namingStrategy)
  163. reg.SetEnumsAsInts(*enumsAsInts)
  164. reg.SetDisableDefaultErrors(*disableDefaultErrors)
  165. reg.SetSimpleOperationIDs(*simpleOperationIDs)
  166. reg.SetProto3OptionalNullable(*proto3OptionalNullable)
  167. reg.SetGenerateUnboundMethods(*generateUnboundMethods)
  168. reg.SetRecursiveDepth(*recursiveDepth)
  169. reg.SetOmitEnumDefaultValue(*omitEnumDefaultValue)
  170. reg.SetVisibilityRestrictionSelectors(*visibilityRestrictionSelectors)
  171. reg.SetDisableServiceTags(*disableServiceTags)
  172. reg.SetDisableDefaultResponses(*disableDefaultResponses)
  173. reg.SetUseAllOfForRefs(*useAllOfForRefs)
  174. reg.SetAllowPatchFeature(*allowPatchFeature)
  175. reg.SetPreserveRPCOrder(*preserveRPCOrder)
  176. reg.SetEnableRpcDeprecation(*enableRpcDeprecation)
  177. reg.SetExpandSlashedPathPatterns(*expandSlashedPathPatterns)
  178. if err := reg.SetRepeatedPathParamSeparator(*repeatedPathParamSeparator); err != nil {
  179. emitError(err)
  180. return
  181. }
  182. for k, v := range pkgMap {
  183. reg.AddPkgMap(k, v)
  184. }
  185. if *grpcAPIConfiguration != "" {
  186. if err := reg.LoadGrpcAPIServiceFromYAML(*grpcAPIConfiguration); err != nil {
  187. emitError(err)
  188. return
  189. }
  190. }
  191. format := genopenapi.Format(*outputFormat)
  192. if err := format.Validate(); err != nil {
  193. emitError(err)
  194. return
  195. }
  196. g := genopenapi.New(reg, format)
  197. if err := genopenapi.AddErrorDefs(reg); err != nil {
  198. emitError(err)
  199. return
  200. }
  201. if err := reg.Load(req); err != nil {
  202. emitError(err)
  203. return
  204. }
  205. if *openAPIConfiguration != "" {
  206. if err := reg.LoadOpenAPIConfigFromYAML(*openAPIConfiguration); err != nil {
  207. emitError(err)
  208. return
  209. }
  210. }
  211. targets := make([]*descriptor.File, 0, len(req.FileToGenerate))
  212. for _, target := range req.FileToGenerate {
  213. f, err := reg.LookupFile(target)
  214. if err != nil {
  215. grpclog.Fatal(err)
  216. }
  217. targets = append(targets, f)
  218. }
  219. out, err := g.Generate(targets)
  220. if grpclog.V(1) {
  221. grpclog.Info("Processed code generator request")
  222. }
  223. if err != nil {
  224. emitError(err)
  225. return
  226. }
  227. emitFiles(out)
  228. }
  229. func emitFiles(out []*descriptor.ResponseFile) {
  230. files := make([]*pluginpb.CodeGeneratorResponse_File, len(out))
  231. for idx, item := range out {
  232. files[idx] = item.CodeGeneratorResponse_File
  233. }
  234. resp := &pluginpb.CodeGeneratorResponse{File: files}
  235. codegenerator.SetSupportedFeaturesOnCodeGeneratorResponse(resp)
  236. emitResp(resp)
  237. }
  238. func emitError(err error) {
  239. emitResp(&pluginpb.CodeGeneratorResponse{Error: proto.String(err.Error())})
  240. }
  241. func emitResp(resp *pluginpb.CodeGeneratorResponse) {
  242. buf, err := proto.Marshal(resp)
  243. if err != nil {
  244. grpclog.Fatal(err)
  245. }
  246. if _, err := os.Stdout.Write(buf); err != nil {
  247. grpclog.Fatal(err)
  248. }
  249. }
  250. // parseReqParam parses a CodeGeneratorRequest parameter and adds the
  251. // extracted values to the given FlagSet and pkgMap. Returns a non-nil
  252. // error if setting a flag failed.
  253. func parseReqParam(param string, f *flag.FlagSet, pkgMap map[string]string) error {
  254. if param == "" {
  255. return nil
  256. }
  257. for _, p := range strings.Split(param, ",") {
  258. spec := strings.SplitN(p, "=", 2)
  259. if len(spec) == 1 {
  260. switch spec[0] {
  261. case "allow_delete_body":
  262. if err := f.Set(spec[0], "true"); err != nil {
  263. return fmt.Errorf("cannot set flag %s: %w", p, err)
  264. }
  265. continue
  266. case "allow_merge":
  267. if err := f.Set(spec[0], "true"); err != nil {
  268. return fmt.Errorf("cannot set flag %s: %w", p, err)
  269. }
  270. continue
  271. case "allow_repeated_fields_in_body":
  272. if err := f.Set(spec[0], "true"); err != nil {
  273. return fmt.Errorf("cannot set flag %s: %w", p, err)
  274. }
  275. continue
  276. case "include_package_in_tags":
  277. if err := f.Set(spec[0], "true"); err != nil {
  278. return fmt.Errorf("cannot set flag %s: %w", p, err)
  279. }
  280. continue
  281. }
  282. if err := f.Set(spec[0], ""); err != nil {
  283. return fmt.Errorf("cannot set flag %s: %w", p, err)
  284. }
  285. continue
  286. }
  287. name, value := spec[0], spec[1]
  288. if strings.HasPrefix(name, "M") {
  289. pkgMap[name[1:]] = value
  290. continue
  291. }
  292. if err := f.Set(name, value); err != nil {
  293. return fmt.Errorf("cannot set flag %s: %w", p, err)
  294. }
  295. }
  296. return nil
  297. }