defs.bzl 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. """Generated an open-api spec for a grpc api spec.
  2. Reads the api spec in protobuf format and generate an open-api spec.
  3. Optionally applies settings from the grpc-service configuration.
  4. """
  5. load("@rules_proto//proto:defs.bzl", "ProtoInfo")
  6. # TODO(yannic): Replace with |proto_common.direct_source_infos| when
  7. # https://github.com/bazelbuild/rules_proto/pull/22 lands.
  8. def _direct_source_infos(proto_info, provided_sources = []):
  9. """Returns sequence of `ProtoFileInfo` for `proto_info`'s direct sources.
  10. Files that are both in `proto_info`'s direct sources and in
  11. `provided_sources` are skipped. This is useful, e.g., for well-known
  12. protos that are already provided by the Protobuf runtime.
  13. Args:
  14. proto_info: An instance of `ProtoInfo`.
  15. provided_sources: Optional. A sequence of files to ignore.
  16. Usually, these files are already provided by the
  17. Protocol Buffer runtime (e.g. Well-Known protos).
  18. Returns: A sequence of `ProtoFileInfo` containing information about
  19. `proto_info`'s direct sources.
  20. """
  21. source_root = proto_info.proto_source_root
  22. if "." == source_root:
  23. return [struct(file = src, import_path = src.path) for src in proto_info.check_deps_sources.to_list()]
  24. offset = len(source_root) + 1 # + '/'.
  25. infos = []
  26. for src in proto_info.check_deps_sources.to_list():
  27. # TODO(yannic): Remove this hack when we drop support for Bazel < 1.0.
  28. local_offset = offset
  29. if src.root.path and not source_root.startswith(src.root.path):
  30. # Before Bazel 1.0, `proto_source_root` wasn't guaranteed to be a
  31. # prefix of `src.path`. This could happened, e.g., if `file` was
  32. # generated (https://github.com/bazelbuild/bazel/issues/9215).
  33. local_offset += len(src.root.path) + 1 # + '/'.
  34. infos.append(struct(file = src, import_path = src.path[local_offset:]))
  35. return infos
  36. def _run_proto_gen_openapi(
  37. actions,
  38. proto_info,
  39. target_name,
  40. transitive_proto_srcs,
  41. protoc,
  42. protoc_gen_openapiv2,
  43. single_output,
  44. allow_delete_body,
  45. grpc_api_configuration,
  46. json_names_for_fields,
  47. repeated_path_param_separator,
  48. include_package_in_tags,
  49. fqn_for_openapi_name,
  50. openapi_naming_strategy,
  51. use_go_templates,
  52. go_template_args,
  53. ignore_comments,
  54. remove_internal_comments,
  55. disable_default_errors,
  56. disable_service_tags,
  57. enums_as_ints,
  58. omit_enum_default_value,
  59. output_format,
  60. simple_operation_ids,
  61. proto3_optional_nullable,
  62. openapi_configuration,
  63. generate_unbound_methods,
  64. visibility_restriction_selectors,
  65. use_allof_for_refs,
  66. disable_default_responses,
  67. enable_rpc_deprecation,
  68. expand_slashed_path_patterns):
  69. args = actions.args()
  70. args.add("--plugin", "protoc-gen-openapiv2=%s" % protoc_gen_openapiv2.path)
  71. extra_inputs = []
  72. if grpc_api_configuration:
  73. extra_inputs.append(grpc_api_configuration)
  74. args.add("--openapiv2_opt", "grpc_api_configuration=%s" % grpc_api_configuration.path)
  75. if openapi_configuration:
  76. extra_inputs.append(openapi_configuration)
  77. args.add("--openapiv2_opt", "openapi_configuration=%s" % openapi_configuration.path)
  78. if not json_names_for_fields:
  79. args.add("--openapiv2_opt", "json_names_for_fields=false")
  80. if fqn_for_openapi_name:
  81. args.add("--openapiv2_opt", "fqn_for_openapi_name=true")
  82. if openapi_naming_strategy:
  83. args.add("--openapiv2_opt", "openapi_naming_strategy=%s" % openapi_naming_strategy)
  84. if generate_unbound_methods:
  85. args.add("--openapiv2_opt", "generate_unbound_methods=true")
  86. if simple_operation_ids:
  87. args.add("--openapiv2_opt", "simple_operation_ids=true")
  88. if allow_delete_body:
  89. args.add("--openapiv2_opt", "allow_delete_body=true")
  90. if include_package_in_tags:
  91. args.add("--openapiv2_opt", "include_package_in_tags=true")
  92. if use_go_templates:
  93. args.add("--openapiv2_opt", "use_go_templates=true")
  94. for go_template_arg in go_template_args:
  95. args.add("--openapiv2_opt", "go_template_args=%s" % go_template_arg)
  96. if ignore_comments:
  97. args.add("--openapiv2_opt", "ignore_comments=true")
  98. if remove_internal_comments:
  99. args.add("--openapiv2_opt", "remove_internal_comments=true")
  100. if disable_default_errors:
  101. args.add("--openapiv2_opt", "disable_default_errors=true")
  102. if disable_service_tags:
  103. args.add("--openapiv2_opt", "disable_service_tags=true")
  104. if enums_as_ints:
  105. args.add("--openapiv2_opt", "enums_as_ints=true")
  106. if omit_enum_default_value:
  107. args.add("--openapiv2_opt", "omit_enum_default_value=true")
  108. if output_format:
  109. args.add("--openapiv2_opt", "output_format=%s" % output_format)
  110. if proto3_optional_nullable:
  111. args.add("--openapiv2_opt", "proto3_optional_nullable=true")
  112. for visibility_restriction_selector in visibility_restriction_selectors:
  113. args.add("--openapiv2_opt", "visibility_restriction_selectors=%s" % visibility_restriction_selector)
  114. if use_allof_for_refs:
  115. args.add("--openapiv2_opt", "use_allof_for_refs=true")
  116. if disable_default_responses:
  117. args.add("--openapiv2_opt", "disable_default_responses=true")
  118. if enable_rpc_deprecation:
  119. args.add("--openapiv2_opt", "enable_rpc_deprecation=true")
  120. if expand_slashed_path_patterns:
  121. args.add("--openapiv2_opt", "expand_slashed_path_patterns=true")
  122. args.add("--openapiv2_opt", "repeated_path_param_separator=%s" % repeated_path_param_separator)
  123. proto_file_infos = _direct_source_infos(proto_info)
  124. # TODO(yannic): Use |proto_info.transitive_descriptor_sets| when
  125. # https://github.com/bazelbuild/bazel/issues/9337 is fixed.
  126. args.add_all(proto_info.transitive_proto_path, format_each = "--proto_path=%s")
  127. if single_output:
  128. args.add("--openapiv2_opt", "allow_merge=true")
  129. args.add("--openapiv2_opt", "merge_file_name=%s" % target_name)
  130. openapi_file = actions.declare_file("%s.swagger.json" % target_name)
  131. args.add("--openapiv2_out", openapi_file.dirname)
  132. args.add_all([f.import_path for f in proto_file_infos])
  133. actions.run(
  134. executable = protoc,
  135. tools = [protoc_gen_openapiv2],
  136. inputs = depset(
  137. direct = extra_inputs,
  138. transitive = [transitive_proto_srcs],
  139. ),
  140. outputs = [openapi_file],
  141. arguments = [args],
  142. )
  143. return [openapi_file]
  144. # TODO(yannic): We may be able to generate all files in a single action,
  145. # but that will change at least the semantics of `use_go_template.proto`.
  146. openapi_files = []
  147. for proto_file_info in proto_file_infos:
  148. # TODO(yannic): This probably doesn't work as expected: we only add this
  149. # option after we have seen it, so `.proto` sources that happen to be
  150. # in the list of `.proto` files before `use_go_template.proto` will be
  151. # compiled without this option, and all sources that get compiled after
  152. # `use_go_template.proto` will have this option on.
  153. if proto_file_info.file.basename == "use_go_template.proto":
  154. args.add("--openapiv2_opt", "use_go_templates=true")
  155. file_name = "%s.swagger.json" % proto_file_info.import_path[:-len(".proto")]
  156. openapi_file = actions.declare_file(
  157. "_virtual_imports/%s/%s" % (target_name, file_name),
  158. )
  159. file_args = actions.args()
  160. offset = len(file_name) + 1 # + '/'.
  161. file_args.add("--openapiv2_out", openapi_file.path[:-offset])
  162. file_args.add(proto_file_info.import_path)
  163. actions.run(
  164. executable = protoc,
  165. tools = [protoc_gen_openapiv2],
  166. inputs = depset(
  167. direct = extra_inputs,
  168. transitive = [transitive_proto_srcs],
  169. ),
  170. outputs = [openapi_file],
  171. arguments = [args, file_args],
  172. )
  173. openapi_files.append(openapi_file)
  174. return openapi_files
  175. def _proto_gen_openapi_impl(ctx):
  176. proto = ctx.attr.proto[ProtoInfo]
  177. return [
  178. DefaultInfo(
  179. files = depset(
  180. _run_proto_gen_openapi(
  181. actions = ctx.actions,
  182. proto_info = proto,
  183. target_name = ctx.attr.name,
  184. transitive_proto_srcs = depset(
  185. direct = ctx.files._well_known_protos,
  186. transitive = [proto.transitive_sources],
  187. ),
  188. protoc = ctx.executable._protoc,
  189. protoc_gen_openapiv2 = ctx.executable._protoc_gen_openapi,
  190. single_output = ctx.attr.single_output,
  191. allow_delete_body = ctx.attr.allow_delete_body,
  192. grpc_api_configuration = ctx.file.grpc_api_configuration,
  193. json_names_for_fields = ctx.attr.json_names_for_fields,
  194. repeated_path_param_separator = ctx.attr.repeated_path_param_separator,
  195. include_package_in_tags = ctx.attr.include_package_in_tags,
  196. fqn_for_openapi_name = ctx.attr.fqn_for_openapi_name,
  197. openapi_naming_strategy = ctx.attr.openapi_naming_strategy,
  198. use_go_templates = ctx.attr.use_go_templates,
  199. go_template_args = ctx.attr.go_template_args,
  200. ignore_comments = ctx.attr.ignore_comments,
  201. remove_internal_comments = ctx.attr.remove_internal_comments,
  202. disable_default_errors = ctx.attr.disable_default_errors,
  203. disable_service_tags = ctx.attr.disable_service_tags,
  204. enums_as_ints = ctx.attr.enums_as_ints,
  205. omit_enum_default_value = ctx.attr.omit_enum_default_value,
  206. output_format = ctx.attr.output_format,
  207. simple_operation_ids = ctx.attr.simple_operation_ids,
  208. proto3_optional_nullable = ctx.attr.proto3_optional_nullable,
  209. openapi_configuration = ctx.file.openapi_configuration,
  210. generate_unbound_methods = ctx.attr.generate_unbound_methods,
  211. visibility_restriction_selectors = ctx.attr.visibility_restriction_selectors,
  212. use_allof_for_refs = ctx.attr.use_allof_for_refs,
  213. disable_default_responses = ctx.attr.disable_default_responses,
  214. enable_rpc_deprecation = ctx.attr.enable_rpc_deprecation,
  215. expand_slashed_path_patterns = ctx.attr.expand_slashed_path_patterns,
  216. ),
  217. ),
  218. ),
  219. ]
  220. protoc_gen_openapiv2 = rule(
  221. attrs = {
  222. "proto": attr.label(
  223. mandatory = True,
  224. providers = [ProtoInfo],
  225. ),
  226. "single_output": attr.bool(
  227. default = False,
  228. mandatory = False,
  229. doc = "if set, the rule will generate a single OpenAPI file",
  230. ),
  231. "allow_delete_body": attr.bool(
  232. default = False,
  233. mandatory = False,
  234. doc = "unless set, HTTP DELETE methods may not have a body",
  235. ),
  236. "grpc_api_configuration": attr.label(
  237. allow_single_file = True,
  238. mandatory = False,
  239. doc = "path to file which describes the gRPC API Configuration in YAML format",
  240. ),
  241. "json_names_for_fields": attr.bool(
  242. default = True,
  243. mandatory = False,
  244. doc = "if disabled, the original proto name will be used for generating OpenAPI definitions",
  245. ),
  246. "repeated_path_param_separator": attr.string(
  247. default = "csv",
  248. mandatory = False,
  249. values = ["csv", "pipes", "ssv", "tsv"],
  250. doc = "configures how repeated fields should be split." +
  251. " Allowed values are `csv`, `pipes`, `ssv` and `tsv`",
  252. ),
  253. "include_package_in_tags": attr.bool(
  254. default = False,
  255. mandatory = False,
  256. doc = "if unset, the gRPC service name is added to the `Tags`" +
  257. " field of each operation. If set and the `package` directive" +
  258. " is shown in the proto file, the package name will be " +
  259. " prepended to the service name",
  260. ),
  261. "fqn_for_openapi_name": attr.bool(
  262. default = False,
  263. mandatory = False,
  264. doc = "if set, the object's OpenAPI names will use the fully" +
  265. " qualified names from the proto definition" +
  266. " (ie my.package.MyMessage.MyInnerMessage",
  267. ),
  268. "openapi_naming_strategy": attr.string(
  269. default = "",
  270. mandatory = False,
  271. values = ["", "simple", "package", "legacy", "fqn"],
  272. doc = "configures how OpenAPI names are determined." +
  273. " Allowed values are `` (empty), `simple`, `package`, `legacy` and `fqn`." +
  274. " If unset, either `legacy` or `fqn` are selected, depending" +
  275. " on the value of the `fqn_for_openapi_name` setting",
  276. ),
  277. "use_go_templates": attr.bool(
  278. default = False,
  279. mandatory = False,
  280. doc = "if set, you can use Go templates in protofile comments",
  281. ),
  282. "go_template_args": attr.string_list(
  283. mandatory = False,
  284. doc = "specify a key value pair as inputs to the Go template of the protofile" +
  285. " comments. Repeat this option to specify multiple template arguments." +
  286. " Requires the `use_go_templates` option to be set.",
  287. ),
  288. "ignore_comments": attr.bool(
  289. default = False,
  290. mandatory = False,
  291. doc = "if set, all protofile comments are excluded from output",
  292. ),
  293. "remove_internal_comments": attr.bool(
  294. default = False,
  295. mandatory = False,
  296. doc = "if set, removes all substrings in comments that start with " +
  297. "`(--` and end with `--)` as specified in " +
  298. "https://google.aip.dev/192#internal-comments",
  299. ),
  300. "disable_default_errors": attr.bool(
  301. default = False,
  302. mandatory = False,
  303. doc = "if set, disables generation of default errors." +
  304. " This is useful if you have defined custom error handling",
  305. ),
  306. "disable_service_tags": attr.bool(
  307. default = False,
  308. mandatory = False,
  309. doc = "if set, disables generation of service tags." +
  310. " This is useful if you do not want to expose the names of your backend grpc services.",
  311. ),
  312. "enums_as_ints": attr.bool(
  313. default = False,
  314. mandatory = False,
  315. doc = "whether to render enum values as integers, as opposed to string values",
  316. ),
  317. "omit_enum_default_value": attr.bool(
  318. default = False,
  319. mandatory = False,
  320. doc = "if set, omit default enum value",
  321. ),
  322. "output_format": attr.string(
  323. default = "json",
  324. mandatory = False,
  325. values = ["json", "yaml"],
  326. doc = "output content format. Allowed values are: `json`, `yaml`",
  327. ),
  328. "simple_operation_ids": attr.bool(
  329. default = False,
  330. mandatory = False,
  331. doc = "whether to remove the service prefix in the operationID" +
  332. " generation. Can introduce duplicate operationIDs, use with caution.",
  333. ),
  334. "proto3_optional_nullable": attr.bool(
  335. default = False,
  336. mandatory = False,
  337. doc = "whether Proto3 Optional fields should be marked as x-nullable",
  338. ),
  339. "openapi_configuration": attr.label(
  340. allow_single_file = True,
  341. mandatory = False,
  342. doc = "path to file which describes the OpenAPI Configuration in YAML format",
  343. ),
  344. "generate_unbound_methods": attr.bool(
  345. default = False,
  346. mandatory = False,
  347. doc = "generate swagger metadata even for RPC methods that have" +
  348. " no HttpRule annotation",
  349. ),
  350. "visibility_restriction_selectors": attr.string_list(
  351. mandatory = False,
  352. doc = "list of `google.api.VisibilityRule` visibility labels to include" +
  353. " in the generated output when a visibility annotation is defined." +
  354. " Repeat this option to supply multiple values. Elements without" +
  355. " visibility annotations are unaffected by this setting.",
  356. ),
  357. "use_allof_for_refs": attr.bool(
  358. default = False,
  359. mandatory = False,
  360. doc = "if set, will use allOf as container for $ref to preserve" +
  361. " same-level properties.",
  362. ),
  363. "disable_default_responses": attr.bool(
  364. default = False,
  365. mandatory = False,
  366. doc = "if set, disables generation of default responses. Useful" +
  367. " if you have to support custom response codes that are" +
  368. " not 200.",
  369. ),
  370. "enable_rpc_deprecation": attr.bool(
  371. default = False,
  372. mandatory = False,
  373. doc = "whether to process grpc method's deprecated option.",
  374. ),
  375. "expand_slashed_path_patterns": attr.bool(
  376. default = False,
  377. mandatory = False,
  378. doc = "if set, expands path patterns containing slashes into URI." +
  379. " It also creates a new path parameter for each wildcard in " +
  380. " the path pattern.",
  381. ),
  382. "_protoc": attr.label(
  383. default = "@com_google_protobuf//:protoc",
  384. executable = True,
  385. cfg = "exec",
  386. ),
  387. "_well_known_protos": attr.label(
  388. default = "@com_google_protobuf//:well_known_type_protos",
  389. allow_files = True,
  390. ),
  391. "_protoc_gen_openapi": attr.label(
  392. default = Label("//protoc-gen-openapiv2:protoc-gen-openapiv2"),
  393. executable = True,
  394. cfg = "exec",
  395. ),
  396. },
  397. implementation = _proto_gen_openapi_impl,
  398. )