template.go 111 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420
  1. package genopenapi
  2. import (
  3. "bytes"
  4. "errors"
  5. "fmt"
  6. "math"
  7. "net/textproto"
  8. "os"
  9. "reflect"
  10. "regexp"
  11. "slices"
  12. "sort"
  13. "strconv"
  14. "strings"
  15. "sync"
  16. "text/template"
  17. "time"
  18. "github.com/grpc-ecosystem/grpc-gateway/v2/internal/casing"
  19. "github.com/grpc-ecosystem/grpc-gateway/v2/internal/descriptor"
  20. openapi_options "github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-openapiv2/options"
  21. "google.golang.org/genproto/googleapis/api/annotations"
  22. "google.golang.org/genproto/googleapis/api/visibility"
  23. "google.golang.org/grpc/grpclog"
  24. "google.golang.org/protobuf/encoding/protojson"
  25. "google.golang.org/protobuf/proto"
  26. "google.golang.org/protobuf/types/descriptorpb"
  27. "google.golang.org/protobuf/types/known/structpb"
  28. )
  29. // The OpenAPI specification does not allow for more than one endpoint with the same HTTP method and path.
  30. // This prevents multiple gRPC service methods from sharing the same stripped version of the path and method.
  31. // For example: `GET /v1/{name=organizations/*}/roles` and `GET /v1/{name=users/*}/roles` both get stripped to `GET /v1/{name}/roles`.
  32. // We must make the URL unique by adding a suffix and an incrementing index to each path parameter
  33. // to differentiate the endpoints.
  34. // Since path parameter names do not affect the request contents (i.e. they're replaced in the path)
  35. // this will be hidden from the real grpc gateway consumer.
  36. const pathParamUniqueSuffixDeliminator = "_"
  37. const paragraphDeliminator = "\n\n"
  38. // wktSchemas are the schemas of well-known-types.
  39. // The schemas must match with the behavior of the JSON unmarshaler in
  40. // https://github.com/protocolbuffers/protobuf-go/blob/v1.25.0/encoding/protojson/well_known_types.go
  41. var wktSchemas = map[string]schemaCore{
  42. ".google.protobuf.FieldMask": {
  43. Type: "string",
  44. },
  45. ".google.protobuf.Timestamp": {
  46. Type: "string",
  47. Format: "date-time",
  48. },
  49. ".google.protobuf.Duration": {
  50. Type: "string",
  51. },
  52. ".google.protobuf.StringValue": {
  53. Type: "string",
  54. },
  55. ".google.protobuf.BytesValue": {
  56. Type: "string",
  57. Format: "byte",
  58. },
  59. ".google.protobuf.Int32Value": {
  60. Type: "integer",
  61. Format: "int32",
  62. },
  63. ".google.protobuf.UInt32Value": {
  64. Type: "integer",
  65. Format: "int64",
  66. },
  67. ".google.protobuf.Int64Value": {
  68. Type: "string",
  69. Format: "int64",
  70. },
  71. ".google.protobuf.UInt64Value": {
  72. Type: "string",
  73. Format: "uint64",
  74. },
  75. ".google.protobuf.FloatValue": {
  76. Type: "number",
  77. Format: "float",
  78. },
  79. ".google.protobuf.DoubleValue": {
  80. Type: "number",
  81. Format: "double",
  82. },
  83. ".google.protobuf.BoolValue": {
  84. Type: "boolean",
  85. },
  86. ".google.protobuf.Empty": {
  87. Type: "object",
  88. },
  89. ".google.protobuf.Struct": {
  90. Type: "object",
  91. },
  92. ".google.protobuf.Value": {},
  93. ".google.protobuf.ListValue": {
  94. Type: "array",
  95. Items: (*openapiItemsObject)(&openapiSchemaObject{
  96. schemaCore: schemaCore{
  97. Type: "object",
  98. },
  99. }),
  100. },
  101. ".google.protobuf.NullValue": {
  102. Type: "string",
  103. },
  104. }
  105. func listEnumNames(reg *descriptor.Registry, enum *descriptor.Enum) interface{} {
  106. var names []string
  107. for _, value := range enum.GetValue() {
  108. if !isVisible(getEnumValueVisibilityOption(value), reg) {
  109. continue
  110. }
  111. if reg.GetOmitEnumDefaultValue() && value.GetNumber() == 0 {
  112. continue
  113. }
  114. names = append(names, value.GetName())
  115. }
  116. if len(names) > 0 {
  117. return names
  118. }
  119. return nil
  120. }
  121. func listEnumNumbers(reg *descriptor.Registry, enum *descriptor.Enum) interface{} {
  122. var numbers []int
  123. for _, value := range enum.GetValue() {
  124. if reg.GetOmitEnumDefaultValue() && value.GetNumber() == 0 {
  125. continue
  126. }
  127. if !isVisible(getEnumValueVisibilityOption(value), reg) {
  128. continue
  129. }
  130. numbers = append(numbers, int(value.GetNumber()))
  131. }
  132. if len(numbers) > 0 {
  133. return numbers
  134. }
  135. return nil
  136. }
  137. func getEnumDefault(reg *descriptor.Registry, enum *descriptor.Enum) interface{} {
  138. if !reg.GetOmitEnumDefaultValue() {
  139. for _, value := range enum.GetValue() {
  140. if value.GetNumber() == 0 {
  141. if !isVisible(getEnumValueVisibilityOption(value), reg) {
  142. return nil
  143. }
  144. return value.GetName()
  145. }
  146. }
  147. }
  148. return nil
  149. }
  150. func getEnumDefaultNumber(reg *descriptor.Registry, enum *descriptor.Enum) interface{} {
  151. if !reg.GetOmitEnumDefaultValue() {
  152. for _, value := range enum.GetValue() {
  153. if value.GetNumber() == 0 {
  154. return int(value.GetNumber())
  155. }
  156. }
  157. }
  158. return nil
  159. }
  160. // messageToQueryParameters converts a message to a list of OpenAPI query parameters.
  161. func messageToQueryParameters(message *descriptor.Message, reg *descriptor.Registry, pathParams []descriptor.Parameter, body *descriptor.Body, httpMethod string) (params []openapiParameterObject, err error) {
  162. for _, field := range message.Fields {
  163. // When body is set to oneof field, we want to skip other fields in the oneof group.
  164. if isBodySameOneOf(body, field) {
  165. continue
  166. }
  167. if !isVisible(getFieldVisibilityOption(field), reg) {
  168. continue
  169. }
  170. if reg.GetAllowPatchFeature() && field.GetTypeName() == ".google.protobuf.FieldMask" && field.GetName() == "update_mask" && httpMethod == "PATCH" && len(body.FieldPath) != 0 {
  171. continue
  172. }
  173. p, err := queryParams(message, field, "", reg, pathParams, body, reg.GetRecursiveDepth())
  174. if err != nil {
  175. return nil, err
  176. }
  177. params = append(params, p...)
  178. }
  179. return params, nil
  180. }
  181. func isBodySameOneOf(body *descriptor.Body, field *descriptor.Field) bool {
  182. if field.OneofIndex == nil {
  183. return false
  184. }
  185. if body == nil || len(body.FieldPath) == 0 {
  186. return false
  187. }
  188. if body.FieldPath[0].Target.OneofIndex == nil {
  189. return false
  190. }
  191. return *body.FieldPath[0].Target.OneofIndex == *field.OneofIndex
  192. }
  193. // queryParams converts a field to a list of OpenAPI query parameters recursively through the use of nestedQueryParams.
  194. func queryParams(message *descriptor.Message, field *descriptor.Field, prefix string, reg *descriptor.Registry, pathParams []descriptor.Parameter, body *descriptor.Body, recursiveCount int) (params []openapiParameterObject, err error) {
  195. return nestedQueryParams(message, field, prefix, reg, pathParams, body, newCycleChecker(recursiveCount))
  196. }
  197. type cycleChecker struct {
  198. m map[string]int
  199. count int
  200. }
  201. func newCycleChecker(recursive int) *cycleChecker {
  202. return &cycleChecker{
  203. m: make(map[string]int),
  204. count: recursive,
  205. }
  206. }
  207. // Check returns whether name is still within recursion
  208. // toleration
  209. func (c *cycleChecker) Check(name string) bool {
  210. count, ok := c.m[name]
  211. count += 1
  212. isCycle := count > c.count
  213. if isCycle {
  214. return false
  215. }
  216. // provision map entry if not available
  217. if !ok {
  218. c.m[name] = 1
  219. return true
  220. }
  221. c.m[name] = count
  222. return true
  223. }
  224. func (c *cycleChecker) Branch() *cycleChecker {
  225. copy := &cycleChecker{
  226. count: c.count,
  227. m: make(map[string]int, len(c.m)),
  228. }
  229. for k, v := range c.m {
  230. copy.m[k] = v
  231. }
  232. return copy
  233. }
  234. // nestedQueryParams converts a field to a list of OpenAPI query parameters recursively.
  235. // This function is a helper function for queryParams, that keeps track of cyclical message references
  236. // through the use of
  237. //
  238. // touched map[string]int
  239. //
  240. // If a cycle is discovered, an error is returned, as cyclical data structures are dangerous
  241. // in query parameters.
  242. func nestedQueryParams(message *descriptor.Message, field *descriptor.Field, prefix string, reg *descriptor.Registry, pathParams []descriptor.Parameter, body *descriptor.Body, cycle *cycleChecker) (params []openapiParameterObject, err error) {
  243. // make sure the parameter is not already listed as a path parameter
  244. for _, pathParam := range pathParams {
  245. if pathParam.Target == field {
  246. return nil, nil
  247. }
  248. }
  249. // make sure the parameter is not already listed as a body parameter
  250. if body != nil {
  251. if body.FieldPath == nil {
  252. return nil, nil
  253. }
  254. for _, fieldPath := range body.FieldPath {
  255. if fieldPath.Target == field {
  256. return nil, nil
  257. }
  258. }
  259. }
  260. schema := schemaOfField(field, reg, nil)
  261. fieldType := field.GetTypeName()
  262. if message.File != nil {
  263. comments := fieldProtoComments(reg, message, field)
  264. if err := updateOpenAPIDataFromComments(reg, &schema, message, comments, false); err != nil {
  265. return nil, err
  266. }
  267. }
  268. isEnum := field.GetType() == descriptorpb.FieldDescriptorProto_TYPE_ENUM
  269. items := schema.Items
  270. if schema.Type != "" || isEnum {
  271. if schema.Type == "object" {
  272. location := ""
  273. if ix := strings.LastIndex(field.Message.FQMN(), "."); ix > 0 {
  274. location = field.Message.FQMN()[0:ix]
  275. }
  276. if m, err := reg.LookupMsg(location, field.GetTypeName()); err == nil {
  277. if opt := m.GetOptions(); opt != nil && opt.MapEntry != nil && *opt.MapEntry {
  278. k := m.GetField()[0]
  279. kType, err := getMapParamKey(k.GetType())
  280. if err != nil {
  281. return nil, err
  282. }
  283. // This will generate a query in the format map_name[key_type]
  284. fName := fmt.Sprintf("%s[%s]", *field.Name, kType)
  285. field.Name = proto.String(fName)
  286. schema.Type = schema.AdditionalProperties.schemaCore.Type
  287. schema.Description = `This is a request variable of the map type. The query format is "map_name[key]=value", e.g. If the map name is Age, the key type is string, and the value type is integer, the query parameter is expressed as Age["bob"]=18`
  288. }
  289. }
  290. }
  291. if items != nil && (items.Type == "" || items.Type == "object") && !isEnum {
  292. return nil, nil // TODO: currently, mapping object in query parameter is not supported
  293. }
  294. desc := mergeDescription(schema)
  295. // verify if the field is required
  296. required := false
  297. for _, fieldName := range schema.Required {
  298. if fieldName == reg.FieldName(field) {
  299. required = true
  300. break
  301. }
  302. }
  303. // verify if the field is required in message options
  304. if messageSchema, err := extractSchemaOptionFromMessageDescriptor(message.DescriptorProto); err == nil {
  305. for _, fieldName := range messageSchema.GetJsonSchema().GetRequired() {
  306. // Required fields can be field names or json_name values
  307. if fieldName == field.GetJsonName() || fieldName == field.GetName() {
  308. required = true
  309. break
  310. }
  311. }
  312. }
  313. param := openapiParameterObject{
  314. Description: desc,
  315. In: "query",
  316. Default: schema.Default,
  317. Type: schema.Type,
  318. Items: schema.Items,
  319. Format: schema.Format,
  320. Pattern: schema.Pattern,
  321. Required: required,
  322. UniqueItems: schema.UniqueItems,
  323. extensions: schema.extensions,
  324. Enum: schema.Enum,
  325. }
  326. if param.Type == "array" {
  327. param.CollectionFormat = "multi"
  328. }
  329. param.Name = prefix + reg.FieldName(field)
  330. if isEnum {
  331. enum, err := reg.LookupEnum("", fieldType)
  332. if err != nil {
  333. return nil, fmt.Errorf("unknown enum type %s", fieldType)
  334. }
  335. if items != nil { // array
  336. param.Items = &openapiItemsObject{
  337. schemaCore: schemaCore{
  338. Type: "string",
  339. Enum: listEnumNames(reg, enum),
  340. },
  341. }
  342. if reg.GetEnumsAsInts() {
  343. param.Items.Type = "integer"
  344. param.Items.Enum = listEnumNumbers(reg, enum)
  345. }
  346. } else {
  347. param.Type = "string"
  348. param.Enum = listEnumNames(reg, enum)
  349. param.Default = getEnumDefault(reg, enum)
  350. if reg.GetEnumsAsInts() {
  351. param.Type = "integer"
  352. param.Enum = listEnumNumbers(reg, enum)
  353. param.Default = getEnumDefaultNumber(reg, enum)
  354. }
  355. }
  356. valueComments := enumValueProtoComments(reg, enum)
  357. if valueComments != "" {
  358. param.Description = strings.TrimLeft(param.Description+"\n\n "+valueComments, "\n")
  359. }
  360. }
  361. return []openapiParameterObject{param}, nil
  362. }
  363. // nested type, recurse
  364. msg, err := reg.LookupMsg("", fieldType)
  365. if err != nil {
  366. return nil, fmt.Errorf("unknown message type %s", fieldType)
  367. }
  368. // Check for cyclical message reference:
  369. if ok := cycle.Check(*msg.Name); !ok {
  370. return nil, fmt.Errorf("exceeded recursive count (%d) for query parameter %q", cycle.count, fieldType)
  371. }
  372. // Construct a new map with the message name so a cycle further down the recursive path can be detected.
  373. // Do not keep anything in the original touched reference and do not pass that reference along. This will
  374. // prevent clobbering adjacent records while recursing.
  375. touchedOut := cycle.Branch()
  376. for _, nestedField := range msg.Fields {
  377. if !isVisible(getFieldVisibilityOption(nestedField), reg) {
  378. continue
  379. }
  380. fieldName := reg.FieldName(field)
  381. p, err := nestedQueryParams(msg, nestedField, prefix+fieldName+".", reg, pathParams, body, touchedOut)
  382. if err != nil {
  383. return nil, err
  384. }
  385. params = append(params, p...)
  386. }
  387. return params, nil
  388. }
  389. func getMapParamKey(t descriptorpb.FieldDescriptorProto_Type) (string, error) {
  390. tType, f, ok := primitiveSchema(t)
  391. if !ok || f == "byte" || f == "float" || f == "double" {
  392. return "", fmt.Errorf("unsupported type: %q", f)
  393. }
  394. return tType, nil
  395. }
  396. // findServicesMessagesAndEnumerations discovers all messages and enums defined in the RPC methods of the service.
  397. func findServicesMessagesAndEnumerations(s []*descriptor.Service, reg *descriptor.Registry, m messageMap, ms messageMap, e enumMap, refs refMap) {
  398. for _, svc := range s {
  399. if !isVisible(getServiceVisibilityOption(svc), reg) {
  400. continue
  401. }
  402. for _, meth := range svc.Methods {
  403. // Request may be fully included in query
  404. {
  405. if !isVisible(getMethodVisibilityOption(meth), reg) {
  406. continue
  407. }
  408. swgReqName, ok := fullyQualifiedNameToOpenAPIName(meth.RequestType.FQMN(), reg)
  409. if !ok {
  410. grpclog.Errorf("couldn't resolve OpenAPI name for FQMN %q", meth.RequestType.FQMN())
  411. continue
  412. }
  413. if _, ok := refs[fmt.Sprintf("#/definitions/%s", swgReqName)]; ok {
  414. if !skipRenderingRef(meth.RequestType.FQMN()) {
  415. m[swgReqName] = meth.RequestType
  416. }
  417. }
  418. }
  419. swgRspName, ok := fullyQualifiedNameToOpenAPIName(meth.ResponseType.FQMN(), reg)
  420. if !ok && !skipRenderingRef(meth.ResponseType.FQMN()) {
  421. grpclog.Errorf("couldn't resolve OpenAPI name for FQMN %q", meth.ResponseType.FQMN())
  422. continue
  423. }
  424. findNestedMessagesAndEnumerations(meth.RequestType, reg, m, e)
  425. if !skipRenderingRef(meth.ResponseType.FQMN()) {
  426. m[swgRspName] = meth.ResponseType
  427. }
  428. findNestedMessagesAndEnumerations(meth.ResponseType, reg, m, e)
  429. }
  430. }
  431. }
  432. // findNestedMessagesAndEnumerations those can be generated by the services.
  433. func findNestedMessagesAndEnumerations(message *descriptor.Message, reg *descriptor.Registry, m messageMap, e enumMap) {
  434. // Iterate over all the fields that
  435. for _, t := range message.Fields {
  436. if !isVisible(getFieldVisibilityOption(t), reg) {
  437. continue
  438. }
  439. fieldType := t.GetTypeName()
  440. // If the type is an empty string then it is a proto primitive
  441. if fieldType != "" {
  442. if _, ok := m[fieldType]; !ok {
  443. msg, err := reg.LookupMsg("", fieldType)
  444. if err != nil {
  445. enum, err := reg.LookupEnum("", fieldType)
  446. if err != nil {
  447. panic(err)
  448. }
  449. e[fieldType] = enum
  450. continue
  451. }
  452. m[fieldType] = msg
  453. findNestedMessagesAndEnumerations(msg, reg, m, e)
  454. }
  455. }
  456. }
  457. }
  458. func skipRenderingRef(refName string) bool {
  459. _, ok := wktSchemas[refName]
  460. return ok
  461. }
  462. func renderMessageAsDefinition(msg *descriptor.Message, reg *descriptor.Registry, customRefs refMap, pathParams []descriptor.Parameter) (openapiSchemaObject, error) {
  463. schema := openapiSchemaObject{
  464. schemaCore: schemaCore{
  465. Type: "object",
  466. },
  467. }
  468. msgComments := protoComments(reg, msg.File, msg.Outers, "MessageType", int32(msg.Index))
  469. if err := updateOpenAPIDataFromComments(reg, &schema, msg, msgComments, false); err != nil {
  470. return openapiSchemaObject{}, err
  471. }
  472. opts, err := getMessageOpenAPIOption(reg, msg)
  473. if err != nil {
  474. return openapiSchemaObject{}, err
  475. }
  476. if opts != nil {
  477. protoSchema := openapiSchemaFromProtoSchema(opts, reg, customRefs, msg)
  478. // Warning: Make sure not to overwrite any fields already set on the schema type.
  479. schema.ExternalDocs = protoSchema.ExternalDocs
  480. schema.ReadOnly = protoSchema.ReadOnly
  481. schema.MultipleOf = protoSchema.MultipleOf
  482. schema.Maximum = protoSchema.Maximum
  483. schema.ExclusiveMaximum = protoSchema.ExclusiveMaximum
  484. schema.Minimum = protoSchema.Minimum
  485. schema.ExclusiveMinimum = protoSchema.ExclusiveMinimum
  486. schema.MaxLength = protoSchema.MaxLength
  487. schema.MinLength = protoSchema.MinLength
  488. schema.Pattern = protoSchema.Pattern
  489. schema.Default = protoSchema.Default
  490. schema.MaxItems = protoSchema.MaxItems
  491. schema.MinItems = protoSchema.MinItems
  492. schema.UniqueItems = protoSchema.UniqueItems
  493. schema.MaxProperties = protoSchema.MaxProperties
  494. schema.MinProperties = protoSchema.MinProperties
  495. schema.Required = protoSchema.Required
  496. schema.XNullable = protoSchema.XNullable
  497. schema.extensions = protoSchema.extensions
  498. if protoSchema.schemaCore.Type != "" || protoSchema.schemaCore.Ref != "" {
  499. schema.schemaCore = protoSchema.schemaCore
  500. }
  501. if protoSchema.Title != "" {
  502. schema.Title = protoSchema.Title
  503. }
  504. if protoSchema.Description != "" {
  505. schema.Description = protoSchema.Description
  506. }
  507. if protoSchema.Example != nil {
  508. schema.Example = protoSchema.Example
  509. }
  510. }
  511. schema.Required = filterOutExcludedFields(schema.Required, pathParams)
  512. for _, f := range msg.Fields {
  513. if !isVisible(getFieldVisibilityOption(f), reg) {
  514. continue
  515. }
  516. if shouldExcludeField(f.GetName(), pathParams) {
  517. continue
  518. }
  519. subPathParams := subPathParams(f.GetName(), pathParams)
  520. fieldSchema, err := renderFieldAsDefinition(f, reg, customRefs, subPathParams)
  521. if err != nil {
  522. return openapiSchemaObject{}, err
  523. }
  524. comments := fieldProtoComments(reg, msg, f)
  525. if err := updateOpenAPIDataFromComments(reg, &fieldSchema, f, comments, false); err != nil {
  526. return openapiSchemaObject{}, err
  527. }
  528. if requiredIdx := find(schema.Required, *f.Name); requiredIdx != -1 && reg.GetUseJSONNamesForFields() {
  529. schema.Required[requiredIdx] = f.GetJsonName()
  530. }
  531. if fieldSchema.Required != nil {
  532. schema.Required = getUniqueFields(schema.Required, fieldSchema.Required)
  533. schema.Required = append(schema.Required, fieldSchema.Required...)
  534. // To avoid populating both the field schema require and message schema require, unset the field schema require.
  535. // See issue #2635.
  536. fieldSchema.Required = nil
  537. }
  538. if reg.GetUseAllOfForRefs() {
  539. if fieldSchema.Ref != "" {
  540. // Per the JSON Reference syntax: Any members other than "$ref" in a JSON Reference object SHALL be ignored.
  541. // https://tools.ietf.org/html/draft-pbryan-zyp-json-ref-03#section-3
  542. // However, use allOf to specify Title/Description/Example/readOnly fields.
  543. if fieldSchema.Title != "" || fieldSchema.Description != "" || len(fieldSchema.Example) > 0 || fieldSchema.ReadOnly {
  544. fieldSchema = openapiSchemaObject{
  545. Title: fieldSchema.Title,
  546. Description: fieldSchema.Description,
  547. schemaCore: schemaCore{
  548. Example: fieldSchema.Example,
  549. },
  550. ReadOnly: fieldSchema.ReadOnly,
  551. AllOf: []allOfEntry{{Ref: fieldSchema.Ref}},
  552. }
  553. } else {
  554. fieldSchema = openapiSchemaObject{schemaCore: schemaCore{Ref: fieldSchema.Ref}}
  555. }
  556. }
  557. }
  558. kv := keyVal{Value: fieldSchema}
  559. kv.Key = reg.FieldName(f)
  560. if schema.Properties == nil {
  561. schema.Properties = &openapiSchemaObjectProperties{}
  562. }
  563. *schema.Properties = append(*schema.Properties, kv)
  564. }
  565. if msg.FQMN() == ".google.protobuf.Any" {
  566. transformAnyForJSON(&schema, reg.GetUseJSONNamesForFields())
  567. }
  568. return schema, nil
  569. }
  570. func renderFieldAsDefinition(f *descriptor.Field, reg *descriptor.Registry, refs refMap, pathParams []descriptor.Parameter) (openapiSchemaObject, error) {
  571. if len(pathParams) == 0 {
  572. return schemaOfField(f, reg, refs), nil
  573. }
  574. location := ""
  575. if ix := strings.LastIndex(f.Message.FQMN(), "."); ix > 0 {
  576. location = f.Message.FQMN()[0:ix]
  577. }
  578. msg, err := reg.LookupMsg(location, f.GetTypeName())
  579. if err != nil {
  580. return openapiSchemaObject{}, err
  581. }
  582. schema, err := renderMessageAsDefinition(msg, reg, refs, pathParams)
  583. if err != nil {
  584. return openapiSchemaObject{}, err
  585. }
  586. comments := fieldProtoComments(reg, f.Message, f)
  587. if len(comments) > 0 {
  588. // Use title and description from field instead of nested message if present.
  589. paragraphs := strings.Split(comments, paragraphDeliminator)
  590. schema.Title = strings.TrimSpace(paragraphs[0])
  591. schema.Description = strings.TrimSpace(strings.Join(paragraphs[1:], paragraphDeliminator))
  592. }
  593. // to handle case where path param is present inside the field of descriptorpb.FieldDescriptorProto_TYPE_MESSAGE type
  594. // it still needs to consider the behaviour of the field which was being done by schemaOfField() in case there are no path params
  595. if j, err := getFieldBehaviorOption(reg, f); err == nil {
  596. updateSwaggerObjectFromFieldBehavior(&schema, j, reg, f)
  597. }
  598. return schema, nil
  599. }
  600. // transformAnyForJSON should be called when the schema object represents a google.protobuf.Any, and will replace the
  601. // Properties slice with a single value for '@type'. We mutate the incorrectly named field so that we inherit the same
  602. // documentation as specified on the original field in the protobuf descriptors.
  603. func transformAnyForJSON(schema *openapiSchemaObject, useJSONNames bool) {
  604. var typeFieldName string
  605. if useJSONNames {
  606. typeFieldName = "typeUrl"
  607. } else {
  608. typeFieldName = "type_url"
  609. }
  610. for _, property := range *schema.Properties {
  611. if property.Key == typeFieldName {
  612. schema.AdditionalProperties = &openapiSchemaObject{}
  613. schema.Properties = &openapiSchemaObjectProperties{keyVal{
  614. Key: "@type",
  615. Value: property.Value,
  616. }}
  617. break
  618. }
  619. }
  620. }
  621. func renderMessagesAsDefinition(messages messageMap, d openapiDefinitionsObject, reg *descriptor.Registry, customRefs refMap, pathParams []descriptor.Parameter) error {
  622. for name, msg := range messages {
  623. swgName, ok := fullyQualifiedNameToOpenAPIName(msg.FQMN(), reg)
  624. if !ok {
  625. return fmt.Errorf("can't resolve OpenAPI name from %q", msg.FQMN())
  626. }
  627. if skipRenderingRef(name) {
  628. continue
  629. }
  630. if opt := msg.GetOptions(); opt != nil && opt.MapEntry != nil && *opt.MapEntry {
  631. continue
  632. }
  633. var err error
  634. d[swgName], err = renderMessageAsDefinition(msg, reg, customRefs, pathParams)
  635. if err != nil {
  636. return err
  637. }
  638. }
  639. return nil
  640. }
  641. // isVisible checks if a field/RPC is visible based on the visibility restriction
  642. // combined with the `visibility_restriction_selectors`.
  643. // Elements with an overlap on `visibility_restriction_selectors` are visible, those without are not visible.
  644. // Elements without `google.api.VisibilityRule` annotations entirely are always visible.
  645. func isVisible(r *visibility.VisibilityRule, reg *descriptor.Registry) bool {
  646. if r == nil {
  647. return true
  648. }
  649. restrictions := strings.Split(strings.TrimSpace(r.Restriction), ",")
  650. // No restrictions results in the element always being visible
  651. if len(restrictions) == 0 {
  652. return true
  653. }
  654. for _, restriction := range restrictions {
  655. if reg.GetVisibilityRestrictionSelectors()[strings.TrimSpace(restriction)] {
  656. return true
  657. }
  658. }
  659. return false
  660. }
  661. func shouldExcludeField(name string, excluded []descriptor.Parameter) bool {
  662. for _, p := range excluded {
  663. if len(p.FieldPath) == 1 && name == p.FieldPath[0].Name {
  664. return true
  665. }
  666. }
  667. return false
  668. }
  669. func filterOutExcludedFields(fields []string, excluded []descriptor.Parameter) []string {
  670. var filtered []string
  671. for _, f := range fields {
  672. if !shouldExcludeField(f, excluded) {
  673. filtered = append(filtered, f)
  674. }
  675. }
  676. return filtered
  677. }
  678. // schemaOfField returns a OpenAPI Schema Object for a protobuf field.
  679. func schemaOfField(f *descriptor.Field, reg *descriptor.Registry, refs refMap) openapiSchemaObject {
  680. const (
  681. singular = 0
  682. array = 1
  683. object = 2
  684. )
  685. var (
  686. core schemaCore
  687. aggregate int
  688. )
  689. fd := f.FieldDescriptorProto
  690. location := ""
  691. if ix := strings.LastIndex(f.Message.FQMN(), "."); ix > 0 {
  692. location = f.Message.FQMN()[0:ix]
  693. }
  694. if m, err := reg.LookupMsg(location, f.GetTypeName()); err == nil {
  695. if opt := m.GetOptions(); opt != nil && opt.MapEntry != nil && *opt.MapEntry {
  696. fd = m.GetField()[1]
  697. aggregate = object
  698. }
  699. }
  700. if fd.GetLabel() == descriptorpb.FieldDescriptorProto_LABEL_REPEATED {
  701. aggregate = array
  702. }
  703. var props *openapiSchemaObjectProperties
  704. switch ft := fd.GetType(); ft {
  705. case descriptorpb.FieldDescriptorProto_TYPE_ENUM, descriptorpb.FieldDescriptorProto_TYPE_MESSAGE, descriptorpb.FieldDescriptorProto_TYPE_GROUP:
  706. if wktSchema, ok := wktSchemas[fd.GetTypeName()]; ok {
  707. core = wktSchema
  708. if fd.GetTypeName() == ".google.protobuf.Empty" {
  709. props = &openapiSchemaObjectProperties{}
  710. }
  711. } else {
  712. swgRef, ok := fullyQualifiedNameToOpenAPIName(fd.GetTypeName(), reg)
  713. if !ok {
  714. panic(fmt.Sprintf("can't resolve OpenAPI ref from typename %q", fd.GetTypeName()))
  715. }
  716. core = schemaCore{
  717. Ref: "#/definitions/" + swgRef,
  718. }
  719. if refs != nil {
  720. refs[fd.GetTypeName()] = struct{}{}
  721. }
  722. }
  723. default:
  724. ftype, format, ok := primitiveSchema(ft)
  725. if ok {
  726. core = schemaCore{Type: ftype, Format: format}
  727. } else {
  728. core = schemaCore{Type: ft.String(), Format: "UNKNOWN"}
  729. }
  730. }
  731. ret := openapiSchemaObject{}
  732. switch aggregate {
  733. case array:
  734. if _, ok := wktSchemas[fd.GetTypeName()]; !ok && fd.GetType() == descriptorpb.FieldDescriptorProto_TYPE_MESSAGE {
  735. core.Type = "object"
  736. }
  737. ret = openapiSchemaObject{
  738. schemaCore: schemaCore{
  739. Type: "array",
  740. Items: (*openapiItemsObject)(&openapiSchemaObject{schemaCore: core}),
  741. },
  742. }
  743. case object:
  744. ret = openapiSchemaObject{
  745. schemaCore: schemaCore{
  746. Type: "object",
  747. },
  748. AdditionalProperties: &openapiSchemaObject{Properties: props, schemaCore: core},
  749. }
  750. default:
  751. ret = openapiSchemaObject{
  752. schemaCore: core,
  753. Properties: props,
  754. }
  755. }
  756. if j, err := getFieldOpenAPIOption(reg, f); err == nil {
  757. updateswaggerObjectFromJSONSchema(&ret, j, reg, f)
  758. }
  759. if j, err := getFieldBehaviorOption(reg, f); err == nil {
  760. updateSwaggerObjectFromFieldBehavior(&ret, j, reg, f)
  761. }
  762. for i, required := range ret.Required {
  763. if required == f.GetName() {
  764. ret.Required[i] = reg.FieldName(f)
  765. }
  766. }
  767. if reg.GetProto3OptionalNullable() && f.GetProto3Optional() {
  768. ret.XNullable = true
  769. }
  770. return ret
  771. }
  772. // primitiveSchema returns a pair of "Type" and "Format" in JSON Schema for
  773. // the given primitive field type.
  774. // The last return parameter is true iff the field type is actually primitive.
  775. func primitiveSchema(t descriptorpb.FieldDescriptorProto_Type) (ftype, format string, ok bool) {
  776. switch t {
  777. case descriptorpb.FieldDescriptorProto_TYPE_DOUBLE:
  778. return "number", "double", true
  779. case descriptorpb.FieldDescriptorProto_TYPE_FLOAT:
  780. return "number", "float", true
  781. case descriptorpb.FieldDescriptorProto_TYPE_INT64:
  782. return "string", "int64", true
  783. case descriptorpb.FieldDescriptorProto_TYPE_UINT64:
  784. // 64bit integer types are marshaled as string in the default JSONPb marshaler.
  785. // TODO(yugui) Add an option to declare 64bit integers as int64.
  786. //
  787. // NOTE: uint64 is not a predefined format of integer type in OpenAPI spec.
  788. // So we cannot expect that uint64 is commonly supported by OpenAPI processor.
  789. return "string", "uint64", true
  790. case descriptorpb.FieldDescriptorProto_TYPE_INT32:
  791. return "integer", "int32", true
  792. case descriptorpb.FieldDescriptorProto_TYPE_FIXED64:
  793. // Ditto.
  794. return "string", "uint64", true
  795. case descriptorpb.FieldDescriptorProto_TYPE_FIXED32:
  796. // Ditto.
  797. return "integer", "int64", true
  798. case descriptorpb.FieldDescriptorProto_TYPE_BOOL:
  799. // NOTE: in OpenAPI specification, format should be empty on boolean type
  800. return "boolean", "", true
  801. case descriptorpb.FieldDescriptorProto_TYPE_STRING:
  802. // NOTE: in OpenAPI specification, can be empty on string type
  803. // see: https://swagger.io/specification/v2/#data-types
  804. return "string", "", true
  805. case descriptorpb.FieldDescriptorProto_TYPE_BYTES:
  806. return "string", "byte", true
  807. case descriptorpb.FieldDescriptorProto_TYPE_UINT32:
  808. // Ditto.
  809. return "integer", "int64", true
  810. case descriptorpb.FieldDescriptorProto_TYPE_SFIXED32:
  811. return "integer", "int32", true
  812. case descriptorpb.FieldDescriptorProto_TYPE_SFIXED64:
  813. return "string", "int64", true
  814. case descriptorpb.FieldDescriptorProto_TYPE_SINT32:
  815. return "integer", "int32", true
  816. case descriptorpb.FieldDescriptorProto_TYPE_SINT64:
  817. return "string", "int64", true
  818. default:
  819. return "", "", false
  820. }
  821. }
  822. // renderEnumerationsAsDefinition inserts enums into the definitions object.
  823. func renderEnumerationsAsDefinition(enums enumMap, d openapiDefinitionsObject, reg *descriptor.Registry, customRefs refMap) {
  824. for _, enum := range enums {
  825. swgName, ok := fullyQualifiedNameToOpenAPIName(enum.FQEN(), reg)
  826. if !ok {
  827. panic(fmt.Sprintf("can't resolve OpenAPI name from FQEN %q", enum.FQEN()))
  828. }
  829. enumComments := protoComments(reg, enum.File, enum.Outers, "EnumType", int32(enum.Index))
  830. // it may be necessary to sort the result of the GetValue function.
  831. enumNames := listEnumNames(reg, enum)
  832. defaultValue := getEnumDefault(reg, enum)
  833. valueComments := enumValueProtoComments(reg, enum)
  834. if valueComments != "" {
  835. enumComments = strings.TrimLeft(enumComments+"\n\n "+valueComments, "\n")
  836. }
  837. enumSchemaObject := openapiSchemaObject{
  838. schemaCore: schemaCore{
  839. Type: "string",
  840. Enum: enumNames,
  841. Default: defaultValue,
  842. },
  843. }
  844. if reg.GetEnumsAsInts() {
  845. enumSchemaObject.Type = "integer"
  846. enumSchemaObject.Format = "int32"
  847. enumSchemaObject.Default = getEnumDefaultNumber(reg, enum)
  848. enumSchemaObject.Enum = listEnumNumbers(reg, enum)
  849. }
  850. opts, err := getEnumOpenAPIOption(reg, enum)
  851. if err != nil {
  852. panic(err)
  853. }
  854. if opts != nil {
  855. protoSchema := openapiSchemaFromProtoEnumSchema(opts, reg, customRefs, enum)
  856. // Warning: Make sure not to overwrite any fields already set on the schema type.
  857. // This is only a subset of the fields from JsonSchema since most of them only apply to arrays or objects not enums
  858. enumSchemaObject.ExternalDocs = protoSchema.ExternalDocs
  859. enumSchemaObject.ReadOnly = protoSchema.ReadOnly
  860. enumSchemaObject.extensions = protoSchema.extensions
  861. if protoSchema.Type != "" || protoSchema.Ref != "" {
  862. enumSchemaObject.schemaCore = protoSchema.schemaCore
  863. }
  864. if protoSchema.Title != "" {
  865. enumSchemaObject.Title = protoSchema.Title
  866. }
  867. if protoSchema.Description != "" {
  868. enumSchemaObject.Description = protoSchema.Description
  869. }
  870. if protoSchema.Example != nil {
  871. enumSchemaObject.Example = protoSchema.Example
  872. }
  873. }
  874. if err := updateOpenAPIDataFromComments(reg, &enumSchemaObject, enum, enumComments, false); err != nil {
  875. panic(err)
  876. }
  877. d[swgName] = enumSchemaObject
  878. }
  879. }
  880. // Take in a FQMN or FQEN and return a OpenAPI safe version of the FQMN and
  881. // a boolean indicating if FQMN was properly resolved.
  882. func fullyQualifiedNameToOpenAPIName(fqn string, reg *descriptor.Registry) (string, bool) {
  883. registriesSeenMutex.Lock()
  884. defer registriesSeenMutex.Unlock()
  885. if mapping, present := registriesSeen[reg]; present {
  886. ret, ok := mapping[fqn]
  887. return ret, ok
  888. }
  889. mapping := resolveFullyQualifiedNameToOpenAPINames(append(reg.GetAllFQMNs(), append(reg.GetAllFQENs(), reg.GetAllFQMethNs()...)...), reg.GetOpenAPINamingStrategy())
  890. registriesSeen[reg] = mapping
  891. ret, ok := mapping[fqn]
  892. return ret, ok
  893. }
  894. // Lookup message type by location.name and return an openapiv2-safe version
  895. // of its FQMN.
  896. func lookupMsgAndOpenAPIName(location, name string, reg *descriptor.Registry) (*descriptor.Message, string, error) {
  897. msg, err := reg.LookupMsg(location, name)
  898. if err != nil {
  899. return nil, "", err
  900. }
  901. swgName, ok := fullyQualifiedNameToOpenAPIName(msg.FQMN(), reg)
  902. if !ok {
  903. return nil, "", fmt.Errorf("can't map OpenAPI name from FQMN %q", msg.FQMN())
  904. }
  905. return msg, swgName, nil
  906. }
  907. // registriesSeen is used to memoise calls to resolveFullyQualifiedNameToOpenAPINames so
  908. // we don't repeat it unnecessarily, since it can take some time.
  909. var (
  910. registriesSeen = map[*descriptor.Registry]map[string]string{}
  911. registriesSeenMutex sync.Mutex
  912. )
  913. // Take the names of every proto message and generate a unique reference for each, according to the given strategy.
  914. func resolveFullyQualifiedNameToOpenAPINames(messages []string, namingStrategy string) map[string]string {
  915. strategyFn := LookupNamingStrategy(namingStrategy)
  916. if strategyFn == nil {
  917. return nil
  918. }
  919. return strategyFn(messages)
  920. }
  921. var canRegexp = regexp.MustCompile("{([a-zA-Z][a-zA-Z0-9_.]*)([^}]*)}")
  922. // templateToParts splits a URL template into path segments for use by `partsToOpenAPIPath` and `partsToRegexpMap`.
  923. //
  924. // Parameters:
  925. // - path: The URL template as defined by https://github.com/googleapis/googleapis/blob/master/google/api/http.proto
  926. // - reg: The descriptor registry used to read compiler flags
  927. // - fields: The fields of the request message, only used when `useJSONNamesForFields` is true
  928. // - msgs: The Messages of the service binding, only used when `useJSONNamesForFields` is true
  929. //
  930. // Returns:
  931. //
  932. // The path segments of the URL template.
  933. func templateToParts(path string, reg *descriptor.Registry, fields []*descriptor.Field, msgs []*descriptor.Message) []string {
  934. // It seems like the right thing to do here is to just use
  935. // strings.Split(path, "/") but that breaks badly when you hit a url like
  936. // /{my_field=prefix/*}/ and end up with 2 sections representing my_field.
  937. // Instead do the right thing and write a small pushdown (counter) automata
  938. // for it.
  939. var parts []string
  940. depth := 0
  941. buffer := ""
  942. pathLoop:
  943. for i, char := range path {
  944. switch char {
  945. case '{':
  946. // Push on the stack
  947. depth++
  948. buffer += string(char)
  949. case '}':
  950. if depth == 0 {
  951. panic("Encountered } without matching { before it.")
  952. }
  953. // Pop from the stack
  954. depth--
  955. if !reg.GetUseJSONNamesForFields() {
  956. buffer += string(char)
  957. continue
  958. }
  959. paramNameProto := strings.SplitN(buffer[1:], "=", 2)[0]
  960. paramNameCamelCase := lowerCamelCase(paramNameProto, fields, msgs)
  961. buffer = strings.Join([]string{"{", paramNameCamelCase, buffer[len(paramNameProto)+1:], "}"}, "")
  962. case '/':
  963. if depth == 0 {
  964. parts = append(parts, buffer)
  965. buffer = ""
  966. // Since the stack was empty when we hit the '/' we are done with this
  967. // section.
  968. continue
  969. }
  970. buffer += string(char)
  971. case ':':
  972. if depth == 0 {
  973. // As soon as we find a ":" outside a variable,
  974. // everything following is a verb
  975. parts = append(parts, buffer)
  976. buffer = path[i:]
  977. break pathLoop
  978. }
  979. buffer += string(char)
  980. default:
  981. buffer += string(char)
  982. }
  983. }
  984. // Now append the last element to parts
  985. parts = append(parts, buffer)
  986. return parts
  987. }
  988. // partsToOpenAPIPath converts each path part of the form /path/{string_value=strprefix/*} which is defined in
  989. // https://github.com/googleapis/googleapis/blob/master/google/api/http.proto to the OpenAPI expected form /path/{string_value}.
  990. // For example this would replace the path segment of "{foo=bar/*}" with "{foo}" or "prefix{bang=bash/**}" with "prefix{bang}".
  991. // OpenAPI 2 only allows simple path parameters with the constraints on that parameter specified in the OpenAPI
  992. // schema's "pattern" instead of in the path parameter itself.
  993. func partsToOpenAPIPath(parts []string, overrides map[string]string) string {
  994. for index, part := range parts {
  995. part = canRegexp.ReplaceAllString(part, "{$1}")
  996. if override, ok := overrides[part]; ok {
  997. part = override
  998. }
  999. parts[index] = part
  1000. }
  1001. if last := len(parts) - 1; strings.HasPrefix(parts[last], ":") {
  1002. // Last item is a verb (":" LITERAL).
  1003. return strings.Join(parts[:last], "/") + parts[last]
  1004. }
  1005. return strings.Join(parts, "/")
  1006. }
  1007. // partsToRegexpMap returns a map of parameter name to ECMA 262 patterns
  1008. // which is what the "pattern" field on an OpenAPI parameter expects.
  1009. // See https://swagger.io/specification/v2/ (Parameter Object) and
  1010. // https://tools.ietf.org/html/draft-fge-json-schema-validation-00#section-5.2.3.
  1011. // The expression is generated based on expressions defined by https://github.com/googleapis/googleapis/blob/master/google/api/http.proto
  1012. // "Path Template Syntax" section which allow for a "param_name=foobar/*/bang/**" style expressions inside
  1013. // the path parameter placeholders that indicate constraints on the values of those parameters.
  1014. // This function will scan the split parts of a path template for parameters and
  1015. // outputs a map of the name of the parameter to a ECMA regular expression. See the http.proto file for descriptions
  1016. // of the supported syntax. This function will ignore any path parameters that don't contain a "=" after the
  1017. // parameter name. For supported parameters, we assume "*" represent all characters except "/" as it's
  1018. // intended to match a single path element and we assume "**" matches any character as it's intended to match multiple
  1019. // path elements.
  1020. // For example "{name=organizations/*/roles/*}" would produce the regular expression for the "name" parameter of
  1021. // "organizations/[^/]+/roles/[^/]+" or "{bar=bing/*/bang/**}" would produce the regular expression for the "bar"
  1022. // parameter of "bing/[^/]+/bang/.+".
  1023. //
  1024. // Note that OpenAPI does not actually support path parameters with "/", see https://github.com/OAI/OpenAPI-Specification/issues/892
  1025. func partsToRegexpMap(parts []string) map[string]string {
  1026. regExps := make(map[string]string)
  1027. for _, part := range parts {
  1028. if strings.Contains(part, "/") {
  1029. grpclog.Warningf("Path parameter %q contains '/', which is not supported in OpenAPI", part)
  1030. }
  1031. if submatch := canRegexp.FindStringSubmatch(part); len(submatch) > 2 {
  1032. if strings.HasPrefix(submatch[2], "=") { // this part matches the standard and should be made into a regular expression
  1033. // assume the string's characters other than "**" and "*" are literals (not necessarily a good assumption 100% of the times, but it will support most use cases)
  1034. regex := submatch[2][1:]
  1035. regex = strings.ReplaceAll(regex, "**", ".+") // ** implies any character including "/"
  1036. regex = strings.ReplaceAll(regex, "*", "[^/]+") // * implies any character except "/"
  1037. regExps[submatch[1]] = regex
  1038. }
  1039. }
  1040. }
  1041. return regExps
  1042. }
  1043. func renderServiceTags(services []*descriptor.Service, reg *descriptor.Registry) []openapiTagObject {
  1044. var tags []openapiTagObject
  1045. var lastFile *descriptor.File = nil
  1046. svcBaseIdx := 0
  1047. for svcIdx, svc := range services {
  1048. if svc.File != lastFile {
  1049. lastFile = svc.File
  1050. svcBaseIdx = svcIdx
  1051. }
  1052. if !isVisible(getServiceVisibilityOption(svc), reg) {
  1053. continue
  1054. }
  1055. tagName := svc.GetName()
  1056. serviceComments := protoComments(reg, svc.File, nil, "Service", int32(svcIdx-svcBaseIdx))
  1057. if serviceComments != "" {
  1058. tagName = serviceComments
  1059. }
  1060. if pkg := svc.File.GetPackage(); pkg != "" && reg.IsIncludePackageInTags() {
  1061. tagName = pkg + "." + tagName
  1062. }
  1063. tag := openapiTagObject{
  1064. Name: tagName,
  1065. }
  1066. opts, err := getServiceOpenAPIOption(reg, svc)
  1067. if err != nil {
  1068. grpclog.Error(err)
  1069. return nil
  1070. }
  1071. if opts != nil {
  1072. tag.Description = opts.Description
  1073. if reg.GetUseGoTemplate() {
  1074. tag.Description = goTemplateComments(tag.Description, svc, reg)
  1075. }
  1076. if opts.ExternalDocs != nil {
  1077. tag.ExternalDocs = &openapiExternalDocumentationObject{
  1078. Description: opts.ExternalDocs.Description,
  1079. URL: opts.ExternalDocs.Url,
  1080. }
  1081. if reg.GetUseGoTemplate() {
  1082. tag.ExternalDocs.Description = goTemplateComments(opts.ExternalDocs.Description, svc, reg)
  1083. }
  1084. }
  1085. if opts.GetName() != "" {
  1086. tag.Name = opts.GetName()
  1087. }
  1088. }
  1089. tags = append(tags, tag)
  1090. }
  1091. return tags
  1092. }
  1093. // expandPathPatterns searches the URI parts for path parameters with pattern and when the pattern contains a sub-path,
  1094. // it expands the pattern into the URI parts and adds the new path parameters to the pathParams slice.
  1095. //
  1096. // Parameters:
  1097. // - pathParts: the URI parts parsed from the path template with `templateToParts` function
  1098. // - pathParams: the path parameters of the service binding
  1099. //
  1100. // Returns:
  1101. //
  1102. // The modified pathParts and pathParams slice.
  1103. func expandPathPatterns(pathParts []string, pathParams []descriptor.Parameter, reg *descriptor.Registry) ([]string, []descriptor.Parameter) {
  1104. expandedPathParts := []string{}
  1105. modifiedPathParams := pathParams
  1106. for _, pathPart := range pathParts {
  1107. if !strings.HasPrefix(pathPart, "{") || !strings.HasSuffix(pathPart, "}") {
  1108. expandedPathParts = append(expandedPathParts, pathPart)
  1109. continue
  1110. }
  1111. woBraces := pathPart[1 : len(pathPart)-1]
  1112. paramPattern := strings.SplitN(woBraces, "=", 2)
  1113. if len(paramPattern) != 2 {
  1114. expandedPathParts = append(expandedPathParts, pathPart)
  1115. continue
  1116. }
  1117. paramName := paramPattern[0]
  1118. pattern := paramPattern[1]
  1119. if pattern == "*" {
  1120. expandedPathParts = append(expandedPathParts, pathPart)
  1121. continue
  1122. }
  1123. pathParamIndex := slices.IndexFunc(modifiedPathParams, func(p descriptor.Parameter) bool {
  1124. if !reg.GetUseJSONNamesForFields() {
  1125. return p.FieldPath.String() == paramName
  1126. }
  1127. fieldPath := casing.JSONCamelCase(p.FieldPath.String())
  1128. return fieldPath == paramName
  1129. })
  1130. if pathParamIndex == -1 {
  1131. panic(fmt.Sprintf("Path parameter %q not found in path parameters", paramName))
  1132. }
  1133. pathParam := modifiedPathParams[pathParamIndex]
  1134. patternParts := strings.Split(pattern, "/")
  1135. for _, patternPart := range patternParts {
  1136. if patternPart != "*" {
  1137. expandedPathParts = append(expandedPathParts, patternPart)
  1138. continue
  1139. }
  1140. lastPart := expandedPathParts[len(expandedPathParts)-1]
  1141. paramName := strings.TrimSuffix(lastPart, "s")
  1142. if reg.GetUseJSONNamesForFields() {
  1143. paramName = casing.JSONCamelCase(paramName)
  1144. }
  1145. expandedPathParts = append(expandedPathParts, "{"+paramName+"}")
  1146. newParam := descriptor.Parameter{
  1147. Target: &descriptor.Field{
  1148. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  1149. Name: proto.String(paramName),
  1150. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  1151. },
  1152. Message: pathParam.Target.Message,
  1153. FieldMessage: pathParam.Target.FieldMessage,
  1154. ForcePrefixedName: pathParam.Target.ForcePrefixedName,
  1155. },
  1156. FieldPath: []descriptor.FieldPathComponent{{
  1157. Name: paramName,
  1158. Target: nil,
  1159. }},
  1160. Method: nil,
  1161. }
  1162. modifiedPathParams = append(modifiedPathParams, newParam)
  1163. if pathParamIndex != -1 {
  1164. // the new parameter from the pattern replaces the old path parameter
  1165. modifiedPathParams = append(modifiedPathParams[:pathParamIndex], modifiedPathParams[pathParamIndex+1:]...)
  1166. pathParamIndex = -1
  1167. }
  1168. }
  1169. }
  1170. return expandedPathParts, modifiedPathParams
  1171. }
  1172. func renderServices(services []*descriptor.Service, paths *openapiPathsObject, reg *descriptor.Registry, requestResponseRefs, customRefs refMap, msgs []*descriptor.Message, defs openapiDefinitionsObject) error {
  1173. // Correctness of svcIdx and methIdx depends on 'services' containing the services in the same order as the 'file.Service' array.
  1174. svcBaseIdx := 0
  1175. var lastFile *descriptor.File = nil
  1176. for svcIdx, svc := range services {
  1177. if svc.File != lastFile {
  1178. lastFile = svc.File
  1179. svcBaseIdx = svcIdx
  1180. }
  1181. if !isVisible(getServiceVisibilityOption(svc), reg) {
  1182. continue
  1183. }
  1184. for methIdx, meth := range svc.Methods {
  1185. if !isVisible(getMethodVisibilityOption(meth), reg) {
  1186. continue
  1187. }
  1188. deprecated := reg.GetEnableRpcDeprecation() && meth.GetOptions().GetDeprecated()
  1189. for bIdx, b := range meth.Bindings {
  1190. operationFunc := operationForMethod(b.HTTPMethod)
  1191. // Iterate over all the OpenAPI parameters
  1192. parameters := openapiParametersObject{}
  1193. // split the path template into its parts
  1194. parts := templateToParts(b.PathTmpl.Template, reg, meth.RequestType.Fields, msgs)
  1195. pathParams := b.PathParams
  1196. if reg.GetExpandSlashedPathPatterns() {
  1197. parts, pathParams = expandPathPatterns(parts, pathParams, reg)
  1198. }
  1199. // extract any constraints specified in the path placeholders into ECMA regular expressions
  1200. pathParamRegexpMap := partsToRegexpMap(parts)
  1201. // Keep track of path parameter overrides
  1202. pathParamNames := make(map[string]string)
  1203. for _, parameter := range pathParams {
  1204. var paramType, paramFormat, desc, collectionFormat string
  1205. var defaultValue interface{}
  1206. var enumNames interface{}
  1207. var items *openapiItemsObject
  1208. var minItems *int
  1209. var extensions []extension
  1210. switch pt := parameter.Target.GetType(); pt {
  1211. case descriptorpb.FieldDescriptorProto_TYPE_GROUP, descriptorpb.FieldDescriptorProto_TYPE_MESSAGE:
  1212. if descriptor.IsWellKnownType(parameter.Target.GetTypeName()) {
  1213. if parameter.IsRepeated() {
  1214. return errors.New("only primitive and enum types are allowed in repeated path parameters")
  1215. }
  1216. schema := schemaOfField(parameter.Target, reg, customRefs)
  1217. paramType = schema.Type
  1218. paramFormat = schema.Format
  1219. desc = schema.Description
  1220. defaultValue = schema.Default
  1221. extensions = schema.extensions
  1222. } else {
  1223. return errors.New("only primitive and well-known types are allowed in path parameters")
  1224. }
  1225. case descriptorpb.FieldDescriptorProto_TYPE_ENUM:
  1226. enum, err := reg.LookupEnum("", parameter.Target.GetTypeName())
  1227. if err != nil {
  1228. return err
  1229. }
  1230. paramType = "string"
  1231. paramFormat = ""
  1232. enumNames = listEnumNames(reg, enum)
  1233. if reg.GetEnumsAsInts() {
  1234. paramType = "integer"
  1235. paramFormat = ""
  1236. enumNames = listEnumNumbers(reg, enum)
  1237. }
  1238. schema := schemaOfField(parameter.Target, reg, customRefs)
  1239. desc = schema.Description
  1240. defaultValue = schema.Default
  1241. extensions = schema.extensions
  1242. default:
  1243. var ok bool
  1244. paramType, paramFormat, ok = primitiveSchema(pt)
  1245. if !ok {
  1246. return fmt.Errorf("unknown field type %v", pt)
  1247. }
  1248. schema := schemaOfField(parameter.Target, reg, customRefs)
  1249. desc = schema.Description
  1250. defaultValue = schema.Default
  1251. extensions = schema.extensions
  1252. // If there is no mandatory format based on the field,
  1253. // allow it to be overridden by the user
  1254. if paramFormat == "" {
  1255. paramFormat = schema.Format
  1256. }
  1257. }
  1258. if parameter.IsRepeated() {
  1259. core := schemaCore{Type: paramType, Format: paramFormat}
  1260. if parameter.IsEnum() {
  1261. core.Enum = enumNames
  1262. enumNames = nil
  1263. }
  1264. items = (*openapiItemsObject)(&openapiSchemaObject{schemaCore: core})
  1265. paramType = "array"
  1266. paramFormat = ""
  1267. collectionFormat = reg.GetRepeatedPathParamSeparatorName()
  1268. minItems = new(int)
  1269. *minItems = 1
  1270. }
  1271. if desc == "" {
  1272. desc = fieldProtoComments(reg, parameter.Target.Message, parameter.Target)
  1273. }
  1274. parameterString := parameter.String()
  1275. if reg.GetUseJSONNamesForFields() {
  1276. parameterString = lowerCamelCase(parameterString, meth.RequestType.Fields, msgs)
  1277. }
  1278. var pattern string
  1279. if regExp, ok := pathParamRegexpMap[parameterString]; ok {
  1280. pattern = regExp
  1281. }
  1282. if fc := getFieldConfiguration(reg, parameter.Target); fc != nil {
  1283. pathParamName := fc.GetPathParamName()
  1284. if pathParamName != "" && pathParamName != parameterString {
  1285. pathParamNames["{"+parameterString+"}"] = "{" + pathParamName + "}"
  1286. parameterString, _, _ = strings.Cut(pathParamName, "=")
  1287. }
  1288. }
  1289. parameters = append(parameters, openapiParameterObject{
  1290. Name: parameterString,
  1291. Description: desc,
  1292. In: "path",
  1293. Required: true,
  1294. Default: defaultValue,
  1295. // Parameters in gRPC-Gateway can only be strings?
  1296. Type: paramType,
  1297. Format: paramFormat,
  1298. Enum: enumNames,
  1299. Items: items,
  1300. CollectionFormat: collectionFormat,
  1301. MinItems: minItems,
  1302. Pattern: pattern,
  1303. extensions: extensions,
  1304. })
  1305. }
  1306. // Now check if there is a body parameter
  1307. if b.Body != nil {
  1308. // Recursively render fields as definitions as long as they contain path parameters.
  1309. // Special case for top level body if we don't have a body field.
  1310. var schema openapiSchemaObject
  1311. desc := ""
  1312. var bodyFieldName string
  1313. schema = openapiSchemaObject{
  1314. schemaCore: schemaCore{},
  1315. }
  1316. if len(b.Body.FieldPath) == 0 {
  1317. // No field for body, use type.
  1318. bodyFieldName = "body"
  1319. wknSchemaCore, isWkn := wktSchemas[meth.RequestType.FQMN()]
  1320. if isWkn {
  1321. schema.schemaCore = wknSchemaCore
  1322. // Special workaround for Empty: it's well-known type but wknSchemas only returns schema.schemaCore; but we need to set schema.Properties which is a level higher.
  1323. if meth.RequestType.FQMN() == ".google.protobuf.Empty" {
  1324. schema.Properties = &openapiSchemaObjectProperties{}
  1325. }
  1326. } else {
  1327. messageSchema, err := renderMessageAsDefinition(meth.RequestType, reg, customRefs, b.PathParams)
  1328. if err != nil {
  1329. return err
  1330. }
  1331. if len(b.PathParams) == 0 {
  1332. if err := schema.setRefFromFQN(meth.RequestType.FQMN(), reg); err != nil {
  1333. return err
  1334. }
  1335. desc = messageSchema.Description
  1336. } else {
  1337. if meth.Name != nil {
  1338. methFQN, ok := fullyQualifiedNameToOpenAPIName(meth.FQMN(), reg)
  1339. if !ok {
  1340. panic(fmt.Errorf("failed to resolve method FQN: '%s'", meth.FQMN()))
  1341. }
  1342. defName := methFQN + "Body"
  1343. schema.Ref = fmt.Sprintf("#/definitions/%s", defName)
  1344. defs[defName] = messageSchema
  1345. } else {
  1346. schema = messageSchema
  1347. if schema.Properties == nil || len(*schema.Properties) == 0 {
  1348. grpclog.Warningf("created a body with 0 properties in the message, this might be unintended: %s", *meth.RequestType)
  1349. }
  1350. }
  1351. }
  1352. }
  1353. } else {
  1354. // Body field path is limited to one path component. From google.api.HttpRule.body:
  1355. // "NOTE: the referred field must be present at the top-level of the request message type."
  1356. // Ref: https://github.com/googleapis/googleapis/blob/b3397f5febbf21dfc69b875ddabaf76bee765058/google/api/http.proto#L350-L352
  1357. if len(b.Body.FieldPath) > 1 {
  1358. return fmt.Errorf("body of request %q is not a top level field: '%v'", meth.Service.GetName(), b.Body.FieldPath)
  1359. }
  1360. bodyField := b.Body.FieldPath[0]
  1361. if reg.GetUseJSONNamesForFields() {
  1362. bodyFieldName = lowerCamelCase(bodyField.Name, meth.RequestType.Fields, msgs)
  1363. } else {
  1364. bodyFieldName = bodyField.Name
  1365. }
  1366. // Align pathParams with body field path.
  1367. pathParams := subPathParams(bodyField.Name, b.PathParams)
  1368. var err error
  1369. schema, err = renderFieldAsDefinition(bodyField.Target, reg, customRefs, pathParams)
  1370. if err != nil {
  1371. return err
  1372. }
  1373. if schema.Title != "" {
  1374. desc = mergeDescription(schema)
  1375. } else {
  1376. desc = fieldProtoComments(reg, bodyField.Target.Message, bodyField.Target)
  1377. }
  1378. }
  1379. if meth.GetClientStreaming() {
  1380. desc += " (streaming inputs)"
  1381. }
  1382. parameters = append(parameters, openapiParameterObject{
  1383. Name: bodyFieldName,
  1384. Description: desc,
  1385. In: "body",
  1386. Required: true,
  1387. Schema: &schema,
  1388. })
  1389. }
  1390. // add the parameters to the query string
  1391. queryParams, err := messageToQueryParameters(meth.RequestType, reg, b.PathParams, b.Body, b.HTTPMethod)
  1392. if err != nil {
  1393. return err
  1394. }
  1395. parameters = append(parameters, queryParams...)
  1396. path := partsToOpenAPIPath(parts, pathParamNames)
  1397. pathItemObject, ok := getPathItemObject(*paths, path)
  1398. if !ok {
  1399. pathItemObject = openapiPathItemObject{}
  1400. } else {
  1401. // handle case where we have an existing mapping for the same path and method
  1402. existingOperationObject := operationFunc(&pathItemObject)
  1403. if existingOperationObject != nil {
  1404. var firstPathParameter *openapiParameterObject
  1405. var firstParamIndex int
  1406. for index, param := range parameters {
  1407. param := param
  1408. if param.In == "path" {
  1409. firstPathParameter = &param
  1410. firstParamIndex = index
  1411. break
  1412. }
  1413. }
  1414. if firstPathParameter == nil {
  1415. // Without a path parameter, there is nothing to vary to support multiple mappings of the same path/method.
  1416. // Previously this did not log an error and only overwrote the mapping, we now log the error but
  1417. // still overwrite the mapping
  1418. grpclog.Errorf("Duplicate mapping for path %s %s", b.HTTPMethod, path)
  1419. } else {
  1420. newPathCount := 0
  1421. var newPath string
  1422. var newPathElement string
  1423. // Iterate until there is not an existing operation that matches the same escaped path.
  1424. // Most of the time this will only be a single iteration, but a large API could technically have
  1425. // a pretty large amount of these if it used similar patterns for all its functions.
  1426. for existingOperationObject != nil {
  1427. newPathCount += 1
  1428. newPathElement = firstPathParameter.Name + pathParamUniqueSuffixDeliminator + strconv.Itoa(newPathCount)
  1429. newPath = strings.ReplaceAll(path, "{"+firstPathParameter.Name+"}", "{"+newPathElement+"}")
  1430. if newPathItemObject, ok := getPathItemObject(*paths, newPath); ok {
  1431. existingOperationObject = operationFunc(&newPathItemObject)
  1432. } else {
  1433. existingOperationObject = nil
  1434. }
  1435. }
  1436. // update the pathItemObject we are adding to with the new path
  1437. pathItemObject, _ = getPathItemObject(*paths, newPath)
  1438. firstPathParameter.Name = newPathElement
  1439. path = newPath
  1440. parameters[firstParamIndex] = *firstPathParameter
  1441. }
  1442. }
  1443. }
  1444. methProtoPath := protoPathIndex(reflect.TypeOf((*descriptorpb.ServiceDescriptorProto)(nil)), "Method")
  1445. desc := "A successful response."
  1446. var responseSchema openapiSchemaObject
  1447. if b.ResponseBody == nil || len(b.ResponseBody.FieldPath) == 0 {
  1448. responseSchema = openapiSchemaObject{
  1449. schemaCore: schemaCore{},
  1450. }
  1451. // Don't link to a full definition for
  1452. // empty; it's overly verbose.
  1453. // schema.Properties{} renders it as
  1454. // well, without a definition
  1455. wknSchemaCore, isWkn := wktSchemas[meth.ResponseType.FQMN()]
  1456. if !isWkn {
  1457. if err := responseSchema.setRefFromFQN(meth.ResponseType.FQMN(), reg); err != nil {
  1458. return err
  1459. }
  1460. } else {
  1461. responseSchema.schemaCore = wknSchemaCore
  1462. // Special workaround for Empty: it's well-known type but wknSchemas only returns schema.schemaCore; but we need to set schema.Properties which is a level higher.
  1463. if meth.ResponseType.FQMN() == ".google.protobuf.Empty" {
  1464. responseSchema.Properties = &openapiSchemaObjectProperties{}
  1465. }
  1466. }
  1467. } else {
  1468. // This is resolving the value of response_body in the google.api.HttpRule
  1469. lastField := b.ResponseBody.FieldPath[len(b.ResponseBody.FieldPath)-1]
  1470. responseSchema = schemaOfField(lastField.Target, reg, customRefs)
  1471. if responseSchema.Description != "" {
  1472. desc = responseSchema.Description
  1473. } else {
  1474. desc = fieldProtoComments(reg, lastField.Target.Message, lastField.Target)
  1475. }
  1476. }
  1477. if meth.GetServerStreaming() {
  1478. desc += "(streaming responses)"
  1479. responseSchema.Type = "object"
  1480. swgRef, _ := fullyQualifiedNameToOpenAPIName(meth.ResponseType.FQMN(), reg)
  1481. responseSchema.Title = "Stream result of " + swgRef
  1482. props := openapiSchemaObjectProperties{
  1483. keyVal{
  1484. Key: "result",
  1485. Value: openapiSchemaObject{
  1486. schemaCore: schemaCore{
  1487. Ref: responseSchema.Ref,
  1488. },
  1489. },
  1490. },
  1491. }
  1492. if !reg.GetDisableDefaultErrors() {
  1493. statusDef, hasStatus := fullyQualifiedNameToOpenAPIName(".google.rpc.Status", reg)
  1494. if hasStatus {
  1495. props = append(props, keyVal{
  1496. Key: "error",
  1497. Value: openapiSchemaObject{
  1498. schemaCore: schemaCore{
  1499. Ref: fmt.Sprintf("#/definitions/%s", statusDef),
  1500. },
  1501. },
  1502. })
  1503. }
  1504. }
  1505. // Special case HttpBody responses, they will be unformatted bytes
  1506. if meth.ResponseType.FQMN() == ".google.api.HttpBody" {
  1507. responseSchema.Type = "string"
  1508. responseSchema.Format = "binary"
  1509. responseSchema.Title = "Free form byte stream"
  1510. // The error response is still JSON, but technically the full response
  1511. // is still unformatted, so don't include the error response structure.
  1512. props = nil
  1513. }
  1514. responseSchema.Properties = &props
  1515. responseSchema.Ref = ""
  1516. }
  1517. // 添加固定的路径参数
  1518. parameters = append(parameters, openapiParameterObject{
  1519. In: "query",
  1520. Name: "r",
  1521. Required: true,
  1522. Description: "接口地址",
  1523. Example: strings.TrimLeft(path, "/"),
  1524. })
  1525. operationObject := &openapiOperationObject{
  1526. Parameters: parameters,
  1527. Responses: openapiResponsesObject{},
  1528. Deprecated: deprecated,
  1529. }
  1530. if !reg.GetDisableDefaultResponses() {
  1531. operationObject.Responses["200"] = openapiResponseObject{
  1532. Description: desc,
  1533. Schema: responseSchema,
  1534. Headers: openapiHeadersObject{},
  1535. }
  1536. }
  1537. if !reg.GetDisableServiceTags() {
  1538. // 获取service的注释
  1539. serviceComments := protoComments(reg, svc.File, nil, "Service", int32(svcIdx-svcBaseIdx))
  1540. tag := svc.GetName()
  1541. if serviceComments != "" {
  1542. tag = serviceComments
  1543. }
  1544. if pkg := svc.File.GetPackage(); pkg != "" && reg.IsIncludePackageInTags() {
  1545. tag = pkg + "." + tag
  1546. }
  1547. operationObject.Tags = []string{tag}
  1548. }
  1549. if !reg.GetDisableDefaultErrors() {
  1550. errDef, hasErrDef := fullyQualifiedNameToOpenAPIName(".google.rpc.Status", reg)
  1551. if hasErrDef {
  1552. // https://github.com/OAI/OpenAPI-Specification/blob/3.0.0/versions/2.0.md#responses-object
  1553. operationObject.Responses["default"] = openapiResponseObject{
  1554. Description: "An unexpected error response.",
  1555. Schema: openapiSchemaObject{
  1556. schemaCore: schemaCore{
  1557. Ref: fmt.Sprintf("#/definitions/%s", errDef),
  1558. },
  1559. },
  1560. }
  1561. }
  1562. }
  1563. operationObject.OperationID = fmt.Sprintf("%s_%s", svc.GetName(), meth.GetName())
  1564. if reg.GetSimpleOperationIDs() {
  1565. operationObject.OperationID = meth.GetName()
  1566. }
  1567. if bIdx != 0 {
  1568. // OperationID must be unique in an OpenAPI v2 definition.
  1569. operationObject.OperationID += strconv.Itoa(bIdx + 1)
  1570. }
  1571. // Fill reference map with referenced request messages
  1572. for _, param := range operationObject.Parameters {
  1573. if param.Schema != nil && param.Schema.Ref != "" {
  1574. requestResponseRefs[param.Schema.Ref] = struct{}{}
  1575. }
  1576. }
  1577. methComments := protoComments(reg, svc.File, nil, "Service", int32(svcIdx-svcBaseIdx), methProtoPath, int32(methIdx))
  1578. if err := updateOpenAPIDataFromComments(reg, operationObject, meth, methComments, false); err != nil {
  1579. panic(err)
  1580. }
  1581. svcOpts, err := getServiceOpenAPIOption(reg, svc)
  1582. if err != nil {
  1583. grpclog.Error(err)
  1584. return err
  1585. }
  1586. opts, err := getMethodOpenAPIOption(reg, meth)
  1587. if opts != nil {
  1588. if err != nil {
  1589. panic(err)
  1590. }
  1591. operationObject.ExternalDocs = protoExternalDocumentationToOpenAPIExternalDocumentation(opts.ExternalDocs, reg, meth)
  1592. if opts.Deprecated {
  1593. operationObject.Deprecated = true
  1594. }
  1595. if opts.Summary != "" {
  1596. operationObject.Summary = opts.Summary
  1597. }
  1598. if opts.Description != "" {
  1599. operationObject.Description = opts.Description
  1600. }
  1601. if len(opts.Tags) > 0 {
  1602. operationObject.Tags = make([]string, len(opts.Tags))
  1603. copy(operationObject.Tags, opts.Tags)
  1604. } else if svcOpts.GetName() != "" {
  1605. operationObject.Tags = []string{svcOpts.GetName()}
  1606. }
  1607. if opts.OperationId != "" {
  1608. operationObject.OperationID = opts.OperationId
  1609. }
  1610. if opts.Security != nil {
  1611. newSecurity := []openapiSecurityRequirementObject{}
  1612. if operationObject.Security != nil {
  1613. newSecurity = *operationObject.Security
  1614. }
  1615. for _, secReq := range opts.Security {
  1616. newSecReq := openapiSecurityRequirementObject{}
  1617. for secReqKey, secReqValue := range secReq.SecurityRequirement {
  1618. if secReqValue == nil {
  1619. continue
  1620. }
  1621. newSecReqValue := make([]string, len(secReqValue.Scope))
  1622. copy(newSecReqValue, secReqValue.Scope)
  1623. newSecReq[secReqKey] = newSecReqValue
  1624. }
  1625. if len(newSecReq) > 0 {
  1626. newSecurity = append(newSecurity, newSecReq)
  1627. }
  1628. }
  1629. operationObject.Security = &newSecurity
  1630. }
  1631. if opts.Responses != nil {
  1632. for name, resp := range opts.Responses {
  1633. // Merge response data into default response if available.
  1634. respObj := operationObject.Responses[name]
  1635. if resp.Description != "" {
  1636. respObj.Description = resp.Description
  1637. }
  1638. if resp.Schema != nil {
  1639. respObj.Schema = openapiSchemaFromProtoSchema(resp.Schema, reg, customRefs, meth)
  1640. }
  1641. if resp.Examples != nil {
  1642. respObj.Examples = openapiExamplesFromProtoExamples(resp.Examples)
  1643. }
  1644. if resp.Headers != nil {
  1645. hdrs, err := processHeaders(resp.Headers)
  1646. if err != nil {
  1647. return err
  1648. }
  1649. respObj.Headers = hdrs
  1650. }
  1651. if resp.Extensions != nil {
  1652. exts, err := processExtensions(resp.Extensions)
  1653. if err != nil {
  1654. return err
  1655. }
  1656. respObj.extensions = exts
  1657. }
  1658. operationObject.Responses[name] = respObj
  1659. }
  1660. }
  1661. if opts.Extensions != nil {
  1662. exts, err := processExtensions(opts.Extensions)
  1663. if err != nil {
  1664. return err
  1665. }
  1666. operationObject.extensions = exts
  1667. }
  1668. if len(opts.Consumes) > 0 {
  1669. operationObject.Consumes = make([]string, len(opts.Consumes))
  1670. copy(operationObject.Consumes, opts.Consumes)
  1671. }
  1672. if len(opts.Produces) > 0 {
  1673. operationObject.Produces = make([]string, len(opts.Produces))
  1674. copy(operationObject.Produces, opts.Produces)
  1675. }
  1676. if params := opts.Parameters; params != nil && len(params.Headers) > 0 {
  1677. for _, header := range params.Headers {
  1678. param := openapiParameterObject{
  1679. In: "header",
  1680. Name: header.Name,
  1681. Description: header.Description,
  1682. Required: header.Required,
  1683. Format: header.Format,
  1684. }
  1685. switch header.Type {
  1686. case openapi_options.HeaderParameter_STRING:
  1687. param.Type = "string"
  1688. case openapi_options.HeaderParameter_NUMBER:
  1689. param.Type = "number"
  1690. case openapi_options.HeaderParameter_INTEGER:
  1691. param.Type = "integer"
  1692. case openapi_options.HeaderParameter_BOOLEAN:
  1693. param.Type = "boolean"
  1694. default:
  1695. return fmt.Errorf("invalid header parameter type: %+v", header.Type)
  1696. }
  1697. operationObject.Parameters = append(operationObject.Parameters, param)
  1698. }
  1699. }
  1700. // TODO(ivucica): add remaining fields of operation object
  1701. }
  1702. switch b.HTTPMethod {
  1703. case "DELETE":
  1704. pathItemObject.Delete = operationObject
  1705. case "GET":
  1706. pathItemObject.Get = operationObject
  1707. case "POST":
  1708. pathItemObject.Post = operationObject
  1709. case "PUT":
  1710. pathItemObject.Put = operationObject
  1711. case "PATCH":
  1712. pathItemObject.Patch = operationObject
  1713. case "HEAD":
  1714. pathItemObject.Head = operationObject
  1715. case "OPTIONS":
  1716. pathItemObject.Options = operationObject
  1717. }
  1718. updatePaths(paths, path, pathItemObject)
  1719. }
  1720. }
  1721. }
  1722. // Success! return nil on the error object
  1723. return nil
  1724. }
  1725. // Returns the openapiPathItemObject associated with a path. If path is not present, returns
  1726. // empty openapiPathItemObject and false.
  1727. func getPathItemObject(paths openapiPathsObject, path string) (openapiPathItemObject, bool) {
  1728. for _, pathData := range paths {
  1729. if pathData.Path == path {
  1730. return pathData.PathItemObject, true
  1731. }
  1732. }
  1733. return openapiPathItemObject{}, false
  1734. }
  1735. // If a path already exists in openapiPathsObject, updates that path's openapiPathItemObject. If not,
  1736. // appends a new path and openapiPathItemObject to the openapiPathsObject.
  1737. func updatePaths(paths *openapiPathsObject, path string, pathItemObject openapiPathItemObject) {
  1738. for i, p := range *paths {
  1739. if p.Path == path {
  1740. (*paths)[i].PathItemObject = pathItemObject
  1741. return
  1742. }
  1743. }
  1744. *paths = append(*paths, pathData{
  1745. Path: path,
  1746. PathItemObject: pathItemObject,
  1747. })
  1748. }
  1749. func mergeDescription(schema openapiSchemaObject) string {
  1750. desc := schema.Description
  1751. if schema.Title != "" { // join title because title of parameter object will be ignored
  1752. desc = strings.TrimSpace(schema.Title + paragraphDeliminator + schema.Description)
  1753. }
  1754. return desc
  1755. }
  1756. func operationForMethod(httpMethod string) func(*openapiPathItemObject) *openapiOperationObject {
  1757. switch httpMethod {
  1758. case "GET":
  1759. return func(obj *openapiPathItemObject) *openapiOperationObject { return obj.Get }
  1760. case "POST":
  1761. return func(obj *openapiPathItemObject) *openapiOperationObject { return obj.Post }
  1762. case "PUT":
  1763. return func(obj *openapiPathItemObject) *openapiOperationObject { return obj.Put }
  1764. case "DELETE":
  1765. return func(obj *openapiPathItemObject) *openapiOperationObject { return obj.Delete }
  1766. case "PATCH":
  1767. return func(obj *openapiPathItemObject) *openapiOperationObject { return obj.Patch }
  1768. case "HEAD":
  1769. return func(obj *openapiPathItemObject) *openapiOperationObject { return obj.Head }
  1770. case "OPTIONS":
  1771. return func(obj *openapiPathItemObject) *openapiOperationObject { return obj.Options }
  1772. default:
  1773. return func(obj *openapiPathItemObject) *openapiOperationObject { return nil }
  1774. }
  1775. }
  1776. // This function is called with a param which contains the entire definition of a method.
  1777. func applyTemplate(p param) (*openapiSwaggerObject, error) {
  1778. // Create the basic template object. This is the object that everything is
  1779. // defined off of.
  1780. s := openapiSwaggerObject{
  1781. // OpenAPI 2.0 is the version of this document
  1782. Swagger: "2.0",
  1783. Consumes: []string{"application/json"},
  1784. Produces: []string{"application/json"},
  1785. Paths: openapiPathsObject{},
  1786. Definitions: make(openapiDefinitionsObject),
  1787. Info: openapiInfoObject{
  1788. Title: *p.File.Name,
  1789. Version: "version not set",
  1790. },
  1791. }
  1792. // Loops through all the services and their exposed GET/POST/PUT/DELETE definitions
  1793. // and create entries for all of them.
  1794. // Also adds custom user specified references to second map.
  1795. requestResponseRefs, customRefs := refMap{}, refMap{}
  1796. if err := renderServices(p.Services, &s.Paths, p.reg, requestResponseRefs, customRefs, p.Messages, s.Definitions); err != nil {
  1797. panic(err)
  1798. }
  1799. messages := messageMap{}
  1800. streamingMessages := messageMap{}
  1801. enums := enumMap{}
  1802. if !p.reg.GetDisableDefaultErrors() {
  1803. // Add the error type to the message map
  1804. runtimeError, swgRef, err := lookupMsgAndOpenAPIName("google.rpc", "Status", p.reg)
  1805. if err == nil {
  1806. messages[swgRef] = runtimeError
  1807. } else {
  1808. // just in case there is an error looking up runtimeError
  1809. grpclog.Error(err)
  1810. }
  1811. }
  1812. // Find all the service's messages and enumerations that are defined (recursively)
  1813. // and write request, response and other custom (but referenced) types out as definition objects.
  1814. findServicesMessagesAndEnumerations(p.Services, p.reg, messages, streamingMessages, enums, requestResponseRefs)
  1815. if err := renderMessagesAsDefinition(messages, s.Definitions, p.reg, customRefs, nil); err != nil {
  1816. return nil, err
  1817. }
  1818. renderEnumerationsAsDefinition(enums, s.Definitions, p.reg, requestResponseRefs)
  1819. // File itself might have some comments and metadata.
  1820. packageProtoPath := protoPathIndex(reflect.TypeOf((*descriptorpb.FileDescriptorProto)(nil)), "Package")
  1821. packageComments := protoComments(p.reg, p.File, nil, "Package", packageProtoPath)
  1822. if err := updateOpenAPIDataFromComments(p.reg, &s, p, packageComments, true); err != nil {
  1823. return nil, err
  1824. }
  1825. // There may be additional options in the OpenAPI option in the proto.
  1826. spb, err := getFileOpenAPIOption(p.reg, p.File)
  1827. if err != nil {
  1828. return nil, err
  1829. }
  1830. if spb != nil {
  1831. if spb.Swagger != "" {
  1832. s.Swagger = spb.Swagger
  1833. }
  1834. if spb.Info != nil {
  1835. if spb.Info.Title != "" {
  1836. s.Info.Title = spb.Info.Title
  1837. }
  1838. if spb.Info.Description != "" {
  1839. s.Info.Description = spb.Info.Description
  1840. }
  1841. if spb.Info.TermsOfService != "" {
  1842. s.Info.TermsOfService = spb.Info.TermsOfService
  1843. }
  1844. if spb.Info.Version != "" {
  1845. s.Info.Version = spb.Info.Version
  1846. }
  1847. if spb.Info.Contact != nil {
  1848. if s.Info.Contact == nil {
  1849. s.Info.Contact = &openapiContactObject{}
  1850. }
  1851. if spb.Info.Contact.Name != "" {
  1852. s.Info.Contact.Name = spb.Info.Contact.Name
  1853. }
  1854. if spb.Info.Contact.Url != "" {
  1855. s.Info.Contact.URL = spb.Info.Contact.Url
  1856. }
  1857. if spb.Info.Contact.Email != "" {
  1858. s.Info.Contact.Email = spb.Info.Contact.Email
  1859. }
  1860. }
  1861. if spb.Info.License != nil {
  1862. if s.Info.License == nil {
  1863. s.Info.License = &openapiLicenseObject{}
  1864. }
  1865. if spb.Info.License.Name != "" {
  1866. s.Info.License.Name = spb.Info.License.Name
  1867. }
  1868. if spb.Info.License.Url != "" {
  1869. s.Info.License.URL = spb.Info.License.Url
  1870. }
  1871. }
  1872. if spb.Info.Extensions != nil {
  1873. exts, err := processExtensions(spb.Info.Extensions)
  1874. if err != nil {
  1875. return nil, err
  1876. }
  1877. s.Info.extensions = exts
  1878. }
  1879. }
  1880. if spb.Host != "" {
  1881. s.Host = spb.Host
  1882. }
  1883. if spb.BasePath != "" {
  1884. s.BasePath = spb.BasePath
  1885. }
  1886. if len(spb.Schemes) > 0 {
  1887. s.Schemes = make([]string, len(spb.Schemes))
  1888. for i, scheme := range spb.Schemes {
  1889. s.Schemes[i] = strings.ToLower(scheme.String())
  1890. }
  1891. }
  1892. if len(spb.Consumes) > 0 {
  1893. s.Consumes = make([]string, len(spb.Consumes))
  1894. copy(s.Consumes, spb.Consumes)
  1895. }
  1896. if len(spb.Produces) > 0 {
  1897. s.Produces = make([]string, len(spb.Produces))
  1898. copy(s.Produces, spb.Produces)
  1899. }
  1900. if spb.SecurityDefinitions != nil && spb.SecurityDefinitions.Security != nil {
  1901. if s.SecurityDefinitions == nil {
  1902. s.SecurityDefinitions = openapiSecurityDefinitionsObject{}
  1903. }
  1904. for secDefKey, secDefValue := range spb.SecurityDefinitions.Security {
  1905. var newSecDefValue openapiSecuritySchemeObject
  1906. if oldSecDefValue, ok := s.SecurityDefinitions[secDefKey]; !ok {
  1907. newSecDefValue = openapiSecuritySchemeObject{}
  1908. } else {
  1909. newSecDefValue = oldSecDefValue
  1910. }
  1911. if secDefValue.Type != openapi_options.SecurityScheme_TYPE_INVALID {
  1912. switch secDefValue.Type {
  1913. case openapi_options.SecurityScheme_TYPE_BASIC:
  1914. newSecDefValue.Type = "basic"
  1915. case openapi_options.SecurityScheme_TYPE_API_KEY:
  1916. newSecDefValue.Type = "apiKey"
  1917. case openapi_options.SecurityScheme_TYPE_OAUTH2:
  1918. newSecDefValue.Type = "oauth2"
  1919. }
  1920. }
  1921. if secDefValue.Description != "" {
  1922. newSecDefValue.Description = secDefValue.Description
  1923. }
  1924. if secDefValue.Name != "" {
  1925. newSecDefValue.Name = secDefValue.Name
  1926. }
  1927. if secDefValue.In != openapi_options.SecurityScheme_IN_INVALID {
  1928. switch secDefValue.In {
  1929. case openapi_options.SecurityScheme_IN_QUERY:
  1930. newSecDefValue.In = "query"
  1931. case openapi_options.SecurityScheme_IN_HEADER:
  1932. newSecDefValue.In = "header"
  1933. }
  1934. }
  1935. if secDefValue.Flow != openapi_options.SecurityScheme_FLOW_INVALID {
  1936. switch secDefValue.Flow {
  1937. case openapi_options.SecurityScheme_FLOW_IMPLICIT:
  1938. newSecDefValue.Flow = "implicit"
  1939. case openapi_options.SecurityScheme_FLOW_PASSWORD:
  1940. newSecDefValue.Flow = "password"
  1941. case openapi_options.SecurityScheme_FLOW_APPLICATION:
  1942. newSecDefValue.Flow = "application"
  1943. case openapi_options.SecurityScheme_FLOW_ACCESS_CODE:
  1944. newSecDefValue.Flow = "accessCode"
  1945. }
  1946. }
  1947. if secDefValue.AuthorizationUrl != "" {
  1948. newSecDefValue.AuthorizationURL = secDefValue.AuthorizationUrl
  1949. }
  1950. if secDefValue.TokenUrl != "" {
  1951. newSecDefValue.TokenURL = secDefValue.TokenUrl
  1952. }
  1953. if secDefValue.Scopes != nil {
  1954. if newSecDefValue.Scopes == nil {
  1955. newSecDefValue.Scopes = openapiScopesObject{}
  1956. }
  1957. for scopeKey, scopeDesc := range secDefValue.Scopes.Scope {
  1958. newSecDefValue.Scopes[scopeKey] = scopeDesc
  1959. }
  1960. }
  1961. if secDefValue.Extensions != nil {
  1962. exts, err := processExtensions(secDefValue.Extensions)
  1963. if err != nil {
  1964. return nil, err
  1965. }
  1966. newSecDefValue.extensions = exts
  1967. }
  1968. s.SecurityDefinitions[secDefKey] = newSecDefValue
  1969. }
  1970. }
  1971. if spb.Security != nil {
  1972. var newSecurity []openapiSecurityRequirementObject
  1973. if s.Security != nil {
  1974. newSecurity = s.Security
  1975. }
  1976. for _, secReq := range spb.Security {
  1977. newSecReq := openapiSecurityRequirementObject{}
  1978. for secReqKey, secReqValue := range secReq.SecurityRequirement {
  1979. if secReqValue == nil {
  1980. return nil, fmt.Errorf("malformed security requirement spec for key %q; value is required", secReqKey)
  1981. }
  1982. newSecReqValue := make([]string, len(secReqValue.Scope))
  1983. copy(newSecReqValue, secReqValue.Scope)
  1984. newSecReq[secReqKey] = newSecReqValue
  1985. }
  1986. newSecurity = append(newSecurity, newSecReq)
  1987. }
  1988. s.Security = newSecurity
  1989. }
  1990. s.ExternalDocs = protoExternalDocumentationToOpenAPIExternalDocumentation(spb.ExternalDocs, p.reg, spb)
  1991. // Populate all Paths with Responses set at top level,
  1992. // preferring Responses already set over those at the top level.
  1993. if spb.Responses != nil {
  1994. for _, verbs := range s.Paths {
  1995. var maps []openapiResponsesObject
  1996. if verbs.PathItemObject.Delete != nil {
  1997. maps = append(maps, verbs.PathItemObject.Delete.Responses)
  1998. }
  1999. if verbs.PathItemObject.Get != nil {
  2000. maps = append(maps, verbs.PathItemObject.Get.Responses)
  2001. }
  2002. if verbs.PathItemObject.Post != nil {
  2003. maps = append(maps, verbs.PathItemObject.Post.Responses)
  2004. }
  2005. if verbs.PathItemObject.Put != nil {
  2006. maps = append(maps, verbs.PathItemObject.Put.Responses)
  2007. }
  2008. if verbs.PathItemObject.Patch != nil {
  2009. maps = append(maps, verbs.PathItemObject.Patch.Responses)
  2010. }
  2011. for k, v := range spb.Responses {
  2012. for _, respMap := range maps {
  2013. if _, ok := respMap[k]; ok {
  2014. // Don't overwrite already existing Responses
  2015. continue
  2016. }
  2017. respMap[k] = openapiResponseObject{
  2018. Description: v.Description,
  2019. Schema: openapiSchemaFromProtoSchema(v.Schema, p.reg, customRefs, nil),
  2020. Examples: openapiExamplesFromProtoExamples(v.Examples),
  2021. }
  2022. }
  2023. }
  2024. }
  2025. }
  2026. if spb.Extensions != nil {
  2027. exts, err := processExtensions(spb.Extensions)
  2028. if err != nil {
  2029. return nil, err
  2030. }
  2031. s.extensions = exts
  2032. }
  2033. if spb.Tags != nil {
  2034. for _, v := range spb.Tags {
  2035. newTag := openapiTagObject{}
  2036. newTag.Name = v.Name
  2037. newTag.Description = v.Description
  2038. if p.reg.GetUseGoTemplate() {
  2039. newTag.Description = goTemplateComments(newTag.Description, nil, p.reg)
  2040. }
  2041. if v.ExternalDocs != nil {
  2042. newTag.ExternalDocs = &openapiExternalDocumentationObject{
  2043. Description: v.ExternalDocs.Description,
  2044. URL: v.ExternalDocs.Url,
  2045. }
  2046. if p.reg.GetUseGoTemplate() {
  2047. newTag.ExternalDocs.Description = goTemplateComments(v.ExternalDocs.Description, nil, p.reg)
  2048. }
  2049. }
  2050. if v.Extensions != nil {
  2051. exts, err := processExtensions(v.Extensions)
  2052. if err != nil {
  2053. return nil, err
  2054. }
  2055. newTag.extensions = exts
  2056. }
  2057. s.Tags = append(s.Tags, newTag)
  2058. }
  2059. }
  2060. // Additional fields on the OpenAPI v2 spec's "OpenAPI" object
  2061. // should be added here, once supported in the proto.
  2062. }
  2063. if !p.reg.GetDisableServiceTags() {
  2064. s.Tags = mergeTags(s.Tags, renderServiceTags(p.Services, p.reg))
  2065. }
  2066. // Finally add any references added by users that aren't
  2067. // otherwise rendered.
  2068. if err := addCustomRefs(s.Definitions, p.reg, customRefs); err != nil {
  2069. return nil, err
  2070. }
  2071. return &s, nil
  2072. }
  2073. func mergeTags(existingTags []openapiTagObject, tags []openapiTagObject) []openapiTagObject {
  2074. for _, tag := range tags {
  2075. matched := false
  2076. for i, existingTag := range existingTags {
  2077. if existingTag.Name == tag.Name {
  2078. if existingTag.Description == "" {
  2079. existingTags[i].Description = tag.Description
  2080. }
  2081. if existingTag.ExternalDocs == nil {
  2082. existingTags[i].ExternalDocs = tag.ExternalDocs
  2083. } else if tag.ExternalDocs != nil {
  2084. if existingTag.ExternalDocs.Description == "" {
  2085. existingTags[i].ExternalDocs.Description = tag.ExternalDocs.Description
  2086. }
  2087. if existingTag.ExternalDocs.URL == "" {
  2088. existingTags[i].ExternalDocs.URL = tag.ExternalDocs.URL
  2089. }
  2090. }
  2091. if existingTag.extensions == nil {
  2092. existingTags[i].extensions = tag.extensions
  2093. } else if tag.extensions != nil {
  2094. for _, ext := range tag.extensions {
  2095. matchedExt := false
  2096. for _, existingExt := range existingTag.extensions {
  2097. if existingExt.key == ext.key {
  2098. matchedExt = true
  2099. break
  2100. }
  2101. }
  2102. if !matchedExt {
  2103. existingTags[i].extensions = append(existingTags[i].extensions, ext)
  2104. }
  2105. }
  2106. }
  2107. matched = true
  2108. break
  2109. }
  2110. }
  2111. if !matched {
  2112. existingTags = append(existingTags, tag)
  2113. }
  2114. }
  2115. return existingTags
  2116. }
  2117. func processExtensions(inputExts map[string]*structpb.Value) ([]extension, error) {
  2118. exts := make([]extension, 0, len(inputExts))
  2119. for k, v := range inputExts {
  2120. if !strings.HasPrefix(k, "x-") {
  2121. return nil, fmt.Errorf("extension keys need to start with \"x-\": %q", k)
  2122. }
  2123. ext, err := (&protojson.MarshalOptions{Indent: " "}).Marshal(v)
  2124. if err != nil {
  2125. return nil, err
  2126. }
  2127. exts = append(exts, extension{key: k, value: ext})
  2128. }
  2129. sort.Slice(exts, func(i, j int) bool { return exts[i].key < exts[j].key })
  2130. return exts, nil
  2131. }
  2132. func validateHeaderTypeAndFormat(headerType, format string) error {
  2133. // The type of the object. The value MUST be one of "string", "number", "integer", "boolean", or "array"
  2134. // See: https://github.com/OAI/OpenAPI-Specification/blob/3.0.0/versions/2.0.md#headerObject
  2135. // Note: currently not implementing array as we are only implementing this in the operation response context
  2136. switch headerType {
  2137. // the format property is an open string-valued property, and can have any value to support documentation needs
  2138. // primary check for format is to ensure that the number/integer formats are extensions of the specified type
  2139. // See: https://github.com/OAI/OpenAPI-Specification/blob/3.0.0/versions/2.0.md#dataTypeFormat
  2140. case "string":
  2141. return nil
  2142. case "number":
  2143. switch format {
  2144. case "uint",
  2145. "uint8",
  2146. "uint16",
  2147. "uint32",
  2148. "uint64",
  2149. "int",
  2150. "int8",
  2151. "int16",
  2152. "int32",
  2153. "int64",
  2154. "float",
  2155. "float32",
  2156. "float64",
  2157. "complex64",
  2158. "complex128",
  2159. "double",
  2160. "byte",
  2161. "rune",
  2162. "uintptr",
  2163. "":
  2164. return nil
  2165. default:
  2166. return fmt.Errorf("the provided format %q is not a valid extension of the type %q", format, headerType)
  2167. }
  2168. case "integer":
  2169. switch format {
  2170. case "uint",
  2171. "uint8",
  2172. "uint16",
  2173. "uint32",
  2174. "uint64",
  2175. "int",
  2176. "int8",
  2177. "int16",
  2178. "int32",
  2179. "int64",
  2180. "":
  2181. return nil
  2182. default:
  2183. return fmt.Errorf("the provided format %q is not a valid extension of the type %q", format, headerType)
  2184. }
  2185. case "boolean":
  2186. return nil
  2187. }
  2188. return fmt.Errorf("the provided header type %q is not supported", headerType)
  2189. }
  2190. func validateDefaultValueTypeAndFormat(headerType string, defaultValue string, format string) error {
  2191. switch headerType {
  2192. case "string":
  2193. if !isQuotedString(defaultValue) {
  2194. return fmt.Errorf("the provided default value %q does not match provider type %q, or is not properly quoted with escaped quotations", defaultValue, headerType)
  2195. }
  2196. switch format {
  2197. case "date-time":
  2198. unquoteTime := strings.Trim(defaultValue, `"`)
  2199. if _, err := time.Parse(time.RFC3339, unquoteTime); err != nil {
  2200. return fmt.Errorf("the provided default value %q is not a valid RFC3339 date-time string", defaultValue)
  2201. }
  2202. case "date":
  2203. const layoutRFC3339Date = "2006-01-02"
  2204. unquoteDate := strings.Trim(defaultValue, `"`)
  2205. if _, err := time.Parse(layoutRFC3339Date, unquoteDate); err != nil {
  2206. return fmt.Errorf("the provided default value %q is not a valid RFC3339 date-time string", defaultValue)
  2207. }
  2208. }
  2209. case "number":
  2210. if err := isJSONNumber(defaultValue, headerType); err != nil {
  2211. return err
  2212. }
  2213. case "integer":
  2214. switch format {
  2215. case "int32":
  2216. if _, err := strconv.ParseInt(defaultValue, 0, 32); err != nil {
  2217. return fmt.Errorf("the provided default value %q does not match provided format %q", defaultValue, format)
  2218. }
  2219. case "uint32":
  2220. if _, err := strconv.ParseUint(defaultValue, 0, 32); err != nil {
  2221. return fmt.Errorf("the provided default value %q does not match provided format %q", defaultValue, format)
  2222. }
  2223. case "int64":
  2224. if _, err := strconv.ParseInt(defaultValue, 0, 64); err != nil {
  2225. return fmt.Errorf("the provided default value %q does not match provided format %q", defaultValue, format)
  2226. }
  2227. case "uint64":
  2228. if _, err := strconv.ParseUint(defaultValue, 0, 64); err != nil {
  2229. return fmt.Errorf("the provided default value %q does not match provided format %q", defaultValue, format)
  2230. }
  2231. default:
  2232. if _, err := strconv.ParseInt(defaultValue, 0, 64); err != nil {
  2233. return fmt.Errorf("the provided default value %q does not match provided type %q", defaultValue, headerType)
  2234. }
  2235. }
  2236. case "boolean":
  2237. if !isBool(defaultValue) {
  2238. return fmt.Errorf("the provided default value %q does not match provider type %q", defaultValue, headerType)
  2239. }
  2240. }
  2241. return nil
  2242. }
  2243. func isQuotedString(s string) bool {
  2244. return len(s) >= 2 && s[0] == '"' && s[len(s)-1] == '"'
  2245. }
  2246. func isJSONNumber(s string, t string) error {
  2247. val, err := strconv.ParseFloat(s, 64)
  2248. if err != nil {
  2249. return fmt.Errorf("the provided default value %q does not match provider type %q", s, t)
  2250. }
  2251. // Floating point values that cannot be represented as sequences of digits (such as Infinity and NaN) are not permitted.
  2252. // See: https://tools.ietf.org/html/rfc4627#section-2.4
  2253. if math.IsInf(val, 0) || math.IsNaN(val) {
  2254. return fmt.Errorf("the provided number %q is not a valid JSON number", s)
  2255. }
  2256. return nil
  2257. }
  2258. func isBool(s string) bool {
  2259. // Unable to use strconv.ParseBool because it returns truthy values https://golang.org/pkg/strconv/#example_ParseBool
  2260. // per https://swagger.io/specification/v2/#data-types
  2261. // type: boolean represents two values: true and false. Note that truthy and falsy values such as "true", "", 0 or null are not considered boolean values.
  2262. return s == "true" || s == "false"
  2263. }
  2264. func processHeaders(inputHdrs map[string]*openapi_options.Header) (openapiHeadersObject, error) {
  2265. hdrs := make(map[string]openapiHeaderObject, len(inputHdrs))
  2266. for k, v := range inputHdrs {
  2267. header := textproto.CanonicalMIMEHeaderKey(k)
  2268. ret := openapiHeaderObject{
  2269. Description: v.Description,
  2270. Format: v.Format,
  2271. Pattern: v.Pattern,
  2272. }
  2273. if err := validateHeaderTypeAndFormat(v.Type, v.Format); err != nil {
  2274. return nil, err
  2275. }
  2276. ret.Type = v.Type
  2277. if v.Default != "" {
  2278. if err := validateDefaultValueTypeAndFormat(v.Type, v.Default, v.Format); err != nil {
  2279. return nil, err
  2280. }
  2281. ret.Default = RawExample(v.Default)
  2282. }
  2283. hdrs[header] = ret
  2284. }
  2285. return hdrs, nil
  2286. }
  2287. func removeInternalComments(comment string) string {
  2288. c := []string{}
  2289. for len(comment) > 0 {
  2290. open := strings.SplitN(comment, "(--", 2)
  2291. if len(open) == 1 {
  2292. c = append(c, open[0])
  2293. break
  2294. }
  2295. ex := strings.TrimRight(open[0], " \t")
  2296. // Trim only one line prior to all spaces
  2297. switch {
  2298. case strings.HasSuffix(ex, "\r\n"):
  2299. ex = strings.TrimSuffix(ex, "\r\n")
  2300. case strings.HasSuffix(ex, "\n"):
  2301. ex = strings.TrimSuffix(ex, "\n")
  2302. }
  2303. if ex != "" {
  2304. c = append(c, ex)
  2305. }
  2306. comment = open[1]
  2307. close := strings.SplitN(comment, "--)", 2)
  2308. if len(close) > 1 {
  2309. comment = close[1]
  2310. } else {
  2311. break
  2312. }
  2313. }
  2314. return strings.Join(c, "")
  2315. }
  2316. // updateOpenAPIDataFromComments updates a OpenAPI object based on a comment
  2317. // from the proto file.
  2318. //
  2319. // First paragraph of a comment is used for summary. Remaining paragraphs of
  2320. // a comment are used for description. If 'Summary' field is not present on
  2321. // the passed swaggerObject, the summary and description are joined by \n\n.
  2322. //
  2323. // If there is a field named 'Info', its 'Summary' and 'Description' fields
  2324. // will be updated instead.
  2325. //
  2326. // If there is no 'Summary', the same behavior will be attempted on 'Title',
  2327. // but only if the last character is not a period.
  2328. func updateOpenAPIDataFromComments(reg *descriptor.Registry, swaggerObject interface{}, data interface{}, comment string, isPackageObject bool) error {
  2329. if len(comment) == 0 {
  2330. return nil
  2331. }
  2332. // Checks whether the "ignore_comments" flag is set to true
  2333. if reg.GetIgnoreComments() {
  2334. return nil
  2335. }
  2336. // Checks whether the "remove_internal_comments" flag is set to true
  2337. if reg.GetRemoveInternalComments() {
  2338. comment = removeInternalComments(comment)
  2339. }
  2340. // Checks whether the "use_go_templates" flag is set to true
  2341. if reg.GetUseGoTemplate() {
  2342. comment = goTemplateComments(comment, data, reg)
  2343. }
  2344. // Figure out what to apply changes to.
  2345. swaggerObjectValue := reflect.ValueOf(swaggerObject)
  2346. infoObjectValue := swaggerObjectValue.Elem().FieldByName("Info")
  2347. if !infoObjectValue.CanSet() {
  2348. // No such field? Apply summary and description directly to
  2349. // passed object.
  2350. infoObjectValue = swaggerObjectValue.Elem()
  2351. }
  2352. // Figure out which properties to update.
  2353. summaryValue := infoObjectValue.FieldByName("Summary")
  2354. descriptionValue := infoObjectValue.FieldByName("Description")
  2355. readOnlyValue := infoObjectValue.FieldByName("ReadOnly")
  2356. if readOnlyValue.Kind() == reflect.Bool && readOnlyValue.CanSet() && strings.Contains(comment, "Output only.") {
  2357. readOnlyValue.Set(reflect.ValueOf(true))
  2358. }
  2359. usingTitle := false
  2360. if !summaryValue.CanSet() {
  2361. summaryValue = infoObjectValue.FieldByName("Title")
  2362. usingTitle = true
  2363. }
  2364. paragraphs := strings.Split(comment, paragraphDeliminator)
  2365. // If there is a summary (or summary-equivalent) and it's empty, use the first
  2366. // paragraph as summary, and the rest as description.
  2367. if summaryValue.CanSet() {
  2368. summary := strings.TrimSpace(paragraphs[0])
  2369. description := strings.TrimSpace(strings.Join(paragraphs[1:], paragraphDeliminator))
  2370. if !usingTitle || (len(summary) > 0 && summary[len(summary)-1] != '.') {
  2371. // overrides the schema value only if it's empty
  2372. // keep the comment precedence when updating the package definition
  2373. if summaryValue.Len() == 0 || isPackageObject {
  2374. summaryValue.Set(reflect.ValueOf(summary))
  2375. }
  2376. if len(description) > 0 {
  2377. if !descriptionValue.CanSet() {
  2378. return errors.New("encountered object type with a summary, but no description")
  2379. }
  2380. // overrides the schema value only if it's empty
  2381. // keep the comment precedence when updating the package definition
  2382. if descriptionValue.Len() == 0 || isPackageObject {
  2383. descriptionValue.Set(reflect.ValueOf(description))
  2384. }
  2385. }
  2386. return nil
  2387. }
  2388. }
  2389. // There was no summary field on the swaggerObject. Try to apply the
  2390. // whole comment into description if the OpenAPI object description is empty.
  2391. if descriptionValue.CanSet() {
  2392. if descriptionValue.Len() == 0 || isPackageObject {
  2393. descriptionValue.Set(reflect.ValueOf(strings.Join(paragraphs, paragraphDeliminator)))
  2394. }
  2395. return nil
  2396. }
  2397. return errors.New("no description nor summary property")
  2398. }
  2399. func fieldProtoComments(reg *descriptor.Registry, msg *descriptor.Message, field *descriptor.Field) string {
  2400. protoPath := protoPathIndex(reflect.TypeOf((*descriptorpb.DescriptorProto)(nil)), "Field")
  2401. for i, f := range msg.Fields {
  2402. if f == field {
  2403. return protoComments(reg, msg.File, msg.Outers, "MessageType", int32(msg.Index), protoPath, int32(i))
  2404. }
  2405. }
  2406. return ""
  2407. }
  2408. func enumValueProtoComments(reg *descriptor.Registry, enum *descriptor.Enum) string {
  2409. protoPath := protoPathIndex(reflect.TypeOf((*descriptorpb.EnumDescriptorProto)(nil)), "Value")
  2410. var comments []string
  2411. for idx, value := range enum.GetValue() {
  2412. if reg.GetOmitEnumDefaultValue() && value.GetNumber() == 0 {
  2413. continue
  2414. }
  2415. if !isVisible(getEnumValueVisibilityOption(value), reg) {
  2416. continue
  2417. }
  2418. name := value.GetName()
  2419. if reg.GetEnumsAsInts() {
  2420. name = strconv.Itoa(int(value.GetNumber()))
  2421. }
  2422. if str := protoComments(reg, enum.File, enum.Outers, "EnumType", int32(enum.Index), protoPath, int32(idx)); str != "" {
  2423. comments = append(comments, name+": "+str)
  2424. }
  2425. }
  2426. if len(comments) > 0 {
  2427. return "- " + strings.Join(comments, "\n - ")
  2428. }
  2429. return ""
  2430. }
  2431. func protoComments(reg *descriptor.Registry, file *descriptor.File, outers []string, typeName string, typeIndex int32, fieldPaths ...int32) string {
  2432. if file.SourceCodeInfo == nil {
  2433. fmt.Fprintln(os.Stderr, file.GetName(), "descriptor.File should not contain nil SourceCodeInfo")
  2434. return ""
  2435. }
  2436. outerPaths := make([]int32, len(outers))
  2437. for i := range outers {
  2438. location := ""
  2439. if file.Package != nil {
  2440. location = file.GetPackage()
  2441. }
  2442. msg, err := reg.LookupMsg(location, strings.Join(outers[:i+1], "."))
  2443. if err != nil {
  2444. panic(err)
  2445. }
  2446. outerPaths[i] = int32(msg.Index)
  2447. }
  2448. for _, loc := range file.SourceCodeInfo.Location {
  2449. if !isProtoPathMatches(loc.Path, outerPaths, typeName, typeIndex, fieldPaths) {
  2450. continue
  2451. }
  2452. comments := ""
  2453. if loc.LeadingComments != nil {
  2454. comments = strings.TrimRight(*loc.LeadingComments, "\n")
  2455. comments = strings.TrimSpace(comments)
  2456. // TODO(ivucica): this is a hack to fix "// " being interpreted as "//".
  2457. // perhaps we should:
  2458. // - split by \n
  2459. // - determine if every (but first and last) line begins with " "
  2460. // - trim every line only if that is the case
  2461. // - join by \n
  2462. comments = strings.ReplaceAll(comments, "\n ", "\n")
  2463. comments = removeInternalComments(comments)
  2464. }
  2465. if loc.TrailingComments != nil {
  2466. trailing := strings.TrimSpace(*loc.TrailingComments)
  2467. if comments == "" {
  2468. comments = trailing
  2469. } else {
  2470. comments += "\n\n" + trailing
  2471. }
  2472. }
  2473. return comments
  2474. }
  2475. return ""
  2476. }
  2477. func goTemplateComments(comment string, data interface{}, reg *descriptor.Registry) string {
  2478. var temp bytes.Buffer
  2479. tpl, err := template.New("").Funcs(template.FuncMap{
  2480. // Allows importing documentation from a file
  2481. "import": func(name string) string {
  2482. file, err := os.ReadFile(name)
  2483. if err != nil {
  2484. return err.Error()
  2485. }
  2486. // Runs template over imported file
  2487. return goTemplateComments(string(file), data, reg)
  2488. },
  2489. // Grabs title and description from a field
  2490. "fieldcomments": func(msg *descriptor.Message, field *descriptor.Field) string {
  2491. return strings.ReplaceAll(fieldProtoComments(reg, msg, field), "\n", "<br>")
  2492. },
  2493. "arg": func(name string) string {
  2494. if v, f := reg.GetGoTemplateArgs()[name]; f {
  2495. return v
  2496. }
  2497. return fmt.Sprintf("goTemplateArg %s not found", name)
  2498. },
  2499. }).Parse(comment)
  2500. if err != nil {
  2501. // If there is an error parsing the templating insert the error as string in the comment
  2502. // to make it easier to debug the template error
  2503. return err.Error()
  2504. }
  2505. if err := tpl.Execute(&temp, data); err != nil {
  2506. // If there is an error executing the templating insert the error as string in the comment
  2507. // to make it easier to debug the error
  2508. return err.Error()
  2509. }
  2510. return temp.String()
  2511. }
  2512. var (
  2513. messageProtoPath = protoPathIndex(reflect.TypeOf((*descriptorpb.FileDescriptorProto)(nil)), "MessageType")
  2514. nestedProtoPath = protoPathIndex(reflect.TypeOf((*descriptorpb.DescriptorProto)(nil)), "NestedType")
  2515. packageProtoPath = protoPathIndex(reflect.TypeOf((*descriptorpb.FileDescriptorProto)(nil)), "Package")
  2516. serviceProtoPath = protoPathIndex(reflect.TypeOf((*descriptorpb.FileDescriptorProto)(nil)), "Service")
  2517. methodProtoPath = protoPathIndex(reflect.TypeOf((*descriptorpb.ServiceDescriptorProto)(nil)), "Method")
  2518. )
  2519. func isProtoPathMatches(paths []int32, outerPaths []int32, typeName string, typeIndex int32, fieldPaths []int32) bool {
  2520. if typeName == "Package" && typeIndex == packageProtoPath {
  2521. // path for package comments is just [2], and all the other processing
  2522. // is too complex for it.
  2523. if len(paths) == 0 || typeIndex != paths[0] {
  2524. return false
  2525. }
  2526. return true
  2527. }
  2528. if len(paths) != len(outerPaths)*2+2+len(fieldPaths) {
  2529. return false
  2530. }
  2531. if typeName == "Method" {
  2532. if paths[0] != serviceProtoPath || paths[2] != methodProtoPath {
  2533. return false
  2534. }
  2535. paths = paths[2:]
  2536. } else {
  2537. typeNameDescriptor := reflect.TypeOf((*descriptorpb.FileDescriptorProto)(nil))
  2538. if len(outerPaths) > 0 {
  2539. if paths[0] != messageProtoPath || paths[1] != outerPaths[0] {
  2540. return false
  2541. }
  2542. paths = paths[2:]
  2543. outerPaths = outerPaths[1:]
  2544. for i, v := range outerPaths {
  2545. if paths[i*2] != nestedProtoPath || paths[i*2+1] != v {
  2546. return false
  2547. }
  2548. }
  2549. paths = paths[len(outerPaths)*2:]
  2550. if typeName == "MessageType" {
  2551. typeName = "NestedType"
  2552. }
  2553. typeNameDescriptor = reflect.TypeOf((*descriptorpb.DescriptorProto)(nil))
  2554. }
  2555. if paths[0] != protoPathIndex(typeNameDescriptor, typeName) || paths[1] != typeIndex {
  2556. return false
  2557. }
  2558. paths = paths[2:]
  2559. }
  2560. for i, v := range fieldPaths {
  2561. if paths[i] != v {
  2562. return false
  2563. }
  2564. }
  2565. return true
  2566. }
  2567. // protoPathIndex returns a path component for google.protobuf.descriptor.SourceCode_Location.
  2568. //
  2569. // Specifically, it returns an id as generated from descriptor proto which
  2570. // can be used to determine what type the id following it in the path is.
  2571. // For example, if we are trying to locate comments related to a field named
  2572. // `Address` in a message named `Person`, the path will be:
  2573. //
  2574. // [4, a, 2, b]
  2575. //
  2576. // While `a` gets determined by the order in which the messages appear in
  2577. // the proto file, and `b` is the field index specified in the proto
  2578. // file itself, the path actually needs to specify that `a` refers to a
  2579. // message and not, say, a service; and that `b` refers to a field and not
  2580. // an option.
  2581. //
  2582. // protoPathIndex figures out the values 4 and 2 in the above example. Because
  2583. // messages are top level objects, the value of 4 comes from field id for
  2584. // `MessageType` inside `google.protobuf.descriptor.FileDescriptor` message.
  2585. // This field has a message type `google.protobuf.descriptor.DescriptorProto`.
  2586. // And inside message `DescriptorProto`, there is a field named `Field` with id
  2587. // 2.
  2588. //
  2589. // Some code generators seem to be hardcoding these values; this method instead
  2590. // interprets them from `descriptor.proto`-derived Go source as necessary.
  2591. func protoPathIndex(descriptorType reflect.Type, what string) int32 {
  2592. field, ok := descriptorType.Elem().FieldByName(what)
  2593. if !ok {
  2594. panic(fmt.Errorf("could not find protobuf descriptor type id for %s", what))
  2595. }
  2596. pbtag := field.Tag.Get("protobuf")
  2597. if pbtag == "" {
  2598. panic(fmt.Errorf("no Go tag 'protobuf' on protobuf descriptor for %s", what))
  2599. }
  2600. path, err := strconv.ParseInt(strings.Split(pbtag, ",")[1], 10, 32)
  2601. if err != nil {
  2602. panic(fmt.Errorf("protobuf descriptor id for %s cannot be converted to a number: %s", what, err.Error()))
  2603. }
  2604. return int32(path)
  2605. }
  2606. // extractOperationOptionFromMethodDescriptor extracts the message of type
  2607. // openapi_options.Operation from a given proto method's descriptor.
  2608. func extractOperationOptionFromMethodDescriptor(meth *descriptorpb.MethodDescriptorProto) (*openapi_options.Operation, error) {
  2609. if meth.Options == nil {
  2610. return nil, nil
  2611. }
  2612. if !proto.HasExtension(meth.Options, openapi_options.E_Openapiv2Operation) {
  2613. return nil, nil
  2614. }
  2615. ext := proto.GetExtension(meth.Options, openapi_options.E_Openapiv2Operation)
  2616. opts, ok := ext.(*openapi_options.Operation)
  2617. if !ok {
  2618. return nil, fmt.Errorf("extension is %T; want an Operation", ext)
  2619. }
  2620. return opts, nil
  2621. }
  2622. // extractSchemaOptionFromMessageDescriptor extracts the message of type
  2623. // openapi_options.Schema from a given proto message's descriptor.
  2624. func extractSchemaOptionFromMessageDescriptor(msg *descriptorpb.DescriptorProto) (*openapi_options.Schema, error) {
  2625. if msg.Options == nil {
  2626. return nil, nil
  2627. }
  2628. if !proto.HasExtension(msg.Options, openapi_options.E_Openapiv2Schema) {
  2629. return nil, nil
  2630. }
  2631. ext := proto.GetExtension(msg.Options, openapi_options.E_Openapiv2Schema)
  2632. opts, ok := ext.(*openapi_options.Schema)
  2633. if !ok {
  2634. return nil, fmt.Errorf("extension is %T; want a Schema", ext)
  2635. }
  2636. return opts, nil
  2637. }
  2638. // extractEnumSchemaOptionFromEnumDescriptor extracts the message of type
  2639. // openapi_options.EnumSchema from a given proto enum's descriptor.
  2640. func extractEnumSchemaOptionFromEnumDescriptor(enum *descriptorpb.EnumDescriptorProto) (*openapi_options.EnumSchema, error) {
  2641. if enum.Options == nil {
  2642. return nil, nil
  2643. }
  2644. if !proto.HasExtension(enum.Options, openapi_options.E_Openapiv2Enum) {
  2645. return nil, nil
  2646. }
  2647. ext := proto.GetExtension(enum.Options, openapi_options.E_Openapiv2Enum)
  2648. opts, ok := ext.(*openapi_options.EnumSchema)
  2649. if !ok {
  2650. return nil, fmt.Errorf("extension is %T; want a EnumSchema", ext)
  2651. }
  2652. return opts, nil
  2653. }
  2654. // extractTagOptionFromServiceDescriptor extracts the tag of type
  2655. // openapi_options.Tag from a given proto service's descriptor.
  2656. func extractTagOptionFromServiceDescriptor(svc *descriptorpb.ServiceDescriptorProto) (*openapi_options.Tag, error) {
  2657. if svc.Options == nil {
  2658. return nil, nil
  2659. }
  2660. if !proto.HasExtension(svc.Options, openapi_options.E_Openapiv2Tag) {
  2661. return nil, nil
  2662. }
  2663. ext := proto.GetExtension(svc.Options, openapi_options.E_Openapiv2Tag)
  2664. opts, ok := ext.(*openapi_options.Tag)
  2665. if !ok {
  2666. return nil, fmt.Errorf("extension is %T; want a Tag", ext)
  2667. }
  2668. return opts, nil
  2669. }
  2670. // extractOpenAPIOptionFromFileDescriptor extracts the message of type
  2671. // openapi_options.OpenAPI from a given proto method's descriptor.
  2672. func extractOpenAPIOptionFromFileDescriptor(file *descriptorpb.FileDescriptorProto) (*openapi_options.Swagger, error) {
  2673. if file.Options == nil {
  2674. return nil, nil
  2675. }
  2676. if !proto.HasExtension(file.Options, openapi_options.E_Openapiv2Swagger) {
  2677. return nil, nil
  2678. }
  2679. ext := proto.GetExtension(file.Options, openapi_options.E_Openapiv2Swagger)
  2680. opts, ok := ext.(*openapi_options.Swagger)
  2681. if !ok {
  2682. return nil, fmt.Errorf("extension is %T; want a OpenAPI object", ext)
  2683. }
  2684. return opts, nil
  2685. }
  2686. func extractJSONSchemaFromFieldDescriptor(fd *descriptorpb.FieldDescriptorProto) (*openapi_options.JSONSchema, error) {
  2687. if fd.Options == nil {
  2688. return nil, nil
  2689. }
  2690. if !proto.HasExtension(fd.Options, openapi_options.E_Openapiv2Field) {
  2691. return nil, nil
  2692. }
  2693. ext := proto.GetExtension(fd.Options, openapi_options.E_Openapiv2Field)
  2694. opts, ok := ext.(*openapi_options.JSONSchema)
  2695. if !ok {
  2696. return nil, fmt.Errorf("extension is %T; want a JSONSchema object", ext)
  2697. }
  2698. return opts, nil
  2699. }
  2700. func extractFieldBehaviorFromFieldDescriptor(fd *descriptorpb.FieldDescriptorProto) ([]annotations.FieldBehavior, error) {
  2701. if fd.Options == nil {
  2702. return nil, nil
  2703. }
  2704. if !proto.HasExtension(fd.Options, annotations.E_FieldBehavior) {
  2705. return nil, nil
  2706. }
  2707. ext := proto.GetExtension(fd.Options, annotations.E_FieldBehavior)
  2708. opts, ok := ext.([]annotations.FieldBehavior)
  2709. if !ok {
  2710. return nil, fmt.Errorf("extension is %T; want a []FieldBehavior object", ext)
  2711. }
  2712. return opts, nil
  2713. }
  2714. func getFieldVisibilityOption(fd *descriptor.Field) *visibility.VisibilityRule {
  2715. if fd.Options == nil {
  2716. return nil
  2717. }
  2718. if !proto.HasExtension(fd.Options, visibility.E_FieldVisibility) {
  2719. return nil
  2720. }
  2721. ext := proto.GetExtension(fd.Options, visibility.E_FieldVisibility)
  2722. opts, ok := ext.(*visibility.VisibilityRule)
  2723. if !ok {
  2724. return nil
  2725. }
  2726. return opts
  2727. }
  2728. func getServiceVisibilityOption(fd *descriptor.Service) *visibility.VisibilityRule {
  2729. if fd.Options == nil {
  2730. return nil
  2731. }
  2732. if !proto.HasExtension(fd.Options, visibility.E_ApiVisibility) {
  2733. return nil
  2734. }
  2735. ext := proto.GetExtension(fd.Options, visibility.E_ApiVisibility)
  2736. opts, ok := ext.(*visibility.VisibilityRule)
  2737. if !ok {
  2738. return nil
  2739. }
  2740. return opts
  2741. }
  2742. func getMethodVisibilityOption(fd *descriptor.Method) *visibility.VisibilityRule {
  2743. if fd.Options == nil {
  2744. return nil
  2745. }
  2746. if !proto.HasExtension(fd.Options, visibility.E_MethodVisibility) {
  2747. return nil
  2748. }
  2749. ext := proto.GetExtension(fd.Options, visibility.E_MethodVisibility)
  2750. opts, ok := ext.(*visibility.VisibilityRule)
  2751. if !ok {
  2752. return nil
  2753. }
  2754. return opts
  2755. }
  2756. func getEnumValueVisibilityOption(fd *descriptorpb.EnumValueDescriptorProto) *visibility.VisibilityRule {
  2757. if fd.Options == nil {
  2758. return nil
  2759. }
  2760. if !proto.HasExtension(fd.Options, visibility.E_ValueVisibility) {
  2761. return nil
  2762. }
  2763. ext := proto.GetExtension(fd.Options, visibility.E_ValueVisibility)
  2764. opts, ok := ext.(*visibility.VisibilityRule)
  2765. if !ok {
  2766. return nil
  2767. }
  2768. return opts
  2769. }
  2770. func getMethodOpenAPIOption(reg *descriptor.Registry, meth *descriptor.Method) (*openapi_options.Operation, error) {
  2771. opts, err := extractOperationOptionFromMethodDescriptor(meth.MethodDescriptorProto)
  2772. if err != nil {
  2773. return nil, err
  2774. }
  2775. if opts != nil {
  2776. return opts, nil
  2777. }
  2778. opts, ok := reg.GetOpenAPIMethodOption(meth.FQMN())
  2779. if !ok {
  2780. return nil, nil
  2781. }
  2782. return opts, nil
  2783. }
  2784. func getMessageOpenAPIOption(reg *descriptor.Registry, msg *descriptor.Message) (*openapi_options.Schema, error) {
  2785. opts, err := extractSchemaOptionFromMessageDescriptor(msg.DescriptorProto)
  2786. if err != nil {
  2787. return nil, err
  2788. }
  2789. if opts != nil {
  2790. return opts, nil
  2791. }
  2792. opts, ok := reg.GetOpenAPIMessageOption(msg.FQMN())
  2793. if !ok {
  2794. return nil, nil
  2795. }
  2796. return opts, nil
  2797. }
  2798. func getEnumOpenAPIOption(reg *descriptor.Registry, enum *descriptor.Enum) (*openapi_options.EnumSchema, error) {
  2799. opts, err := extractEnumSchemaOptionFromEnumDescriptor(enum.EnumDescriptorProto)
  2800. if err != nil {
  2801. return nil, err
  2802. }
  2803. return opts, nil
  2804. }
  2805. func getServiceOpenAPIOption(reg *descriptor.Registry, svc *descriptor.Service) (*openapi_options.Tag, error) {
  2806. if opts, ok := reg.GetOpenAPIServiceOption(svc.FQSN()); ok {
  2807. return opts, nil
  2808. }
  2809. opts, err := extractTagOptionFromServiceDescriptor(svc.ServiceDescriptorProto)
  2810. if err != nil {
  2811. return nil, err
  2812. }
  2813. return opts, nil
  2814. }
  2815. func getFileOpenAPIOption(reg *descriptor.Registry, file *descriptor.File) (*openapi_options.Swagger, error) {
  2816. opts, err := extractOpenAPIOptionFromFileDescriptor(file.FileDescriptorProto)
  2817. if err != nil {
  2818. return nil, err
  2819. }
  2820. if opts != nil {
  2821. return opts, nil
  2822. }
  2823. opts, ok := reg.GetOpenAPIFileOption(*file.Name)
  2824. if !ok {
  2825. return nil, nil
  2826. }
  2827. return opts, nil
  2828. }
  2829. func getFieldOpenAPIOption(reg *descriptor.Registry, fd *descriptor.Field) (*openapi_options.JSONSchema, error) {
  2830. opts, err := extractJSONSchemaFromFieldDescriptor(fd.FieldDescriptorProto)
  2831. if err != nil {
  2832. return nil, err
  2833. }
  2834. if opts != nil {
  2835. return opts, nil
  2836. }
  2837. opts, ok := reg.GetOpenAPIFieldOption(fd.FQFN())
  2838. if !ok {
  2839. return nil, nil
  2840. }
  2841. return opts, nil
  2842. }
  2843. func getFieldBehaviorOption(reg *descriptor.Registry, fd *descriptor.Field) ([]annotations.FieldBehavior, error) {
  2844. opts, err := extractFieldBehaviorFromFieldDescriptor(fd.FieldDescriptorProto)
  2845. if err != nil {
  2846. return nil, err
  2847. }
  2848. if opts != nil {
  2849. return opts, nil
  2850. }
  2851. return opts, nil
  2852. }
  2853. func protoJSONSchemaToOpenAPISchemaCore(j *openapi_options.JSONSchema, reg *descriptor.Registry, refs refMap) schemaCore {
  2854. ret := schemaCore{}
  2855. if j.GetRef() != "" {
  2856. openapiName, ok := fullyQualifiedNameToOpenAPIName(j.GetRef(), reg)
  2857. if ok {
  2858. ret.Ref = "#/definitions/" + openapiName
  2859. if refs != nil {
  2860. refs[j.GetRef()] = struct{}{}
  2861. }
  2862. } else {
  2863. ret.Ref += j.GetRef()
  2864. }
  2865. } else {
  2866. f, t := protoJSONSchemaTypeToFormat(j.GetType())
  2867. ret.Format = f
  2868. ret.Type = t
  2869. }
  2870. return ret
  2871. }
  2872. func updateswaggerObjectFromJSONSchema(s *openapiSchemaObject, j *openapi_options.JSONSchema, reg *descriptor.Registry, data interface{}) {
  2873. s.Title = j.GetTitle()
  2874. s.Description = j.GetDescription()
  2875. if reg.GetUseGoTemplate() {
  2876. s.Title = goTemplateComments(s.Title, data, reg)
  2877. s.Description = goTemplateComments(s.Description, data, reg)
  2878. }
  2879. if s.Type == "array" {
  2880. s.Items.MaxLength = j.GetMaxLength()
  2881. s.Items.MinLength = j.GetMinLength()
  2882. s.Items.Pattern = j.GetPattern()
  2883. s.Items.Default = j.GetDefault()
  2884. s.Items.MaxProperties = j.GetMaxProperties()
  2885. s.Items.MinProperties = j.GetMinProperties()
  2886. s.Items.Required = j.GetRequired()
  2887. s.Items.Minimum = j.GetMinimum()
  2888. s.Items.Maximum = j.GetMaximum()
  2889. s.Items.ReadOnly = j.GetReadOnly()
  2890. s.Items.MultipleOf = j.GetMultipleOf()
  2891. s.Items.ExclusiveMaximum = j.GetExclusiveMaximum()
  2892. s.Items.ExclusiveMinimum = j.GetExclusiveMinimum()
  2893. s.Items.Enum = j.GetEnum()
  2894. if j.GetDefault() == "" {
  2895. s.Items.Default = nil
  2896. }
  2897. if len(j.GetEnum()) == 0 {
  2898. s.Items.Enum = nil
  2899. }
  2900. if j.GetFormat() != "" {
  2901. s.Items.Format = j.GetFormat()
  2902. }
  2903. } else {
  2904. s.MaxLength = j.GetMaxLength()
  2905. s.MinLength = j.GetMinLength()
  2906. s.Pattern = j.GetPattern()
  2907. s.Default = j.GetDefault()
  2908. s.MaxProperties = j.GetMaxProperties()
  2909. s.MinProperties = j.GetMinProperties()
  2910. s.Required = j.GetRequired()
  2911. s.Minimum = j.GetMinimum()
  2912. s.Maximum = j.GetMaximum()
  2913. s.ReadOnly = j.GetReadOnly()
  2914. s.MultipleOf = j.GetMultipleOf()
  2915. s.ExclusiveMaximum = j.GetExclusiveMaximum()
  2916. s.ExclusiveMinimum = j.GetExclusiveMinimum()
  2917. s.Enum = j.GetEnum()
  2918. if j.GetDefault() == "" {
  2919. s.Default = nil
  2920. }
  2921. if len(j.GetEnum()) == 0 {
  2922. s.Enum = nil
  2923. }
  2924. if j.GetFormat() != "" {
  2925. s.Format = j.GetFormat()
  2926. }
  2927. }
  2928. s.UniqueItems = j.GetUniqueItems()
  2929. s.MaxItems = j.GetMaxItems()
  2930. s.MinItems = j.GetMinItems()
  2931. if j.GetExtensions() != nil {
  2932. exts, err := processExtensions(j.GetExtensions())
  2933. if err != nil {
  2934. panic(err)
  2935. }
  2936. s.extensions = exts
  2937. }
  2938. if overrideType := j.GetType(); len(overrideType) > 0 {
  2939. s.Type = strings.ToLower(overrideType[0].String())
  2940. }
  2941. if j.GetExample() != "" {
  2942. s.Example = RawExample(j.GetExample())
  2943. }
  2944. }
  2945. func updateSwaggerObjectFromFieldBehavior(s *openapiSchemaObject, j []annotations.FieldBehavior, reg *descriptor.Registry, field *descriptor.Field) {
  2946. for _, fb := range j {
  2947. switch fb {
  2948. case annotations.FieldBehavior_REQUIRED:
  2949. if reg.GetUseJSONNamesForFields() {
  2950. s.Required = append(s.Required, *field.JsonName)
  2951. } else {
  2952. s.Required = append(s.Required, *field.Name)
  2953. }
  2954. case annotations.FieldBehavior_OUTPUT_ONLY:
  2955. s.ReadOnly = true
  2956. case annotations.FieldBehavior_FIELD_BEHAVIOR_UNSPECIFIED:
  2957. case annotations.FieldBehavior_OPTIONAL:
  2958. case annotations.FieldBehavior_INPUT_ONLY:
  2959. // OpenAPI v3 supports a writeOnly property, but this is not supported in Open API v2
  2960. case annotations.FieldBehavior_IMMUTABLE:
  2961. }
  2962. }
  2963. }
  2964. func openapiSchemaFromProtoEnumSchema(s *openapi_options.EnumSchema, reg *descriptor.Registry, refs refMap, data interface{}) openapiSchemaObject {
  2965. ret := openapiSchemaObject{
  2966. ExternalDocs: protoExternalDocumentationToOpenAPIExternalDocumentation(s.GetExternalDocs(), reg, data),
  2967. }
  2968. jsonSchema := &openapi_options.JSONSchema{
  2969. Ref: s.Ref,
  2970. Title: s.Title,
  2971. Extensions: s.Extensions,
  2972. Description: s.Description,
  2973. Default: s.Default,
  2974. ReadOnly: s.ReadOnly,
  2975. Example: s.Example,
  2976. }
  2977. ret.schemaCore = protoJSONSchemaToOpenAPISchemaCore(jsonSchema, reg, refs)
  2978. updateswaggerObjectFromJSONSchema(&ret, jsonSchema, reg, data)
  2979. return ret
  2980. }
  2981. func openapiSchemaFromProtoSchema(s *openapi_options.Schema, reg *descriptor.Registry, refs refMap, data interface{}) openapiSchemaObject {
  2982. ret := openapiSchemaObject{
  2983. ExternalDocs: protoExternalDocumentationToOpenAPIExternalDocumentation(s.GetExternalDocs(), reg, data),
  2984. }
  2985. ret.schemaCore = protoJSONSchemaToOpenAPISchemaCore(s.GetJsonSchema(), reg, refs)
  2986. updateswaggerObjectFromJSONSchema(&ret, s.GetJsonSchema(), reg, data)
  2987. if s != nil && s.Example != "" {
  2988. ret.Example = RawExample(s.Example)
  2989. }
  2990. return ret
  2991. }
  2992. func openapiExamplesFromProtoExamples(in map[string]string) map[string]interface{} {
  2993. if len(in) == 0 {
  2994. return nil
  2995. }
  2996. out := make(map[string]interface{}, len(in))
  2997. for mimeType, exampleStr := range in {
  2998. switch mimeType {
  2999. case "application/json":
  3000. // JSON example objects are rendered raw.
  3001. out[mimeType] = RawExample(exampleStr)
  3002. default:
  3003. // All other mimetype examples are rendered as strings.
  3004. out[mimeType] = exampleStr
  3005. }
  3006. }
  3007. return out
  3008. }
  3009. func protoJSONSchemaTypeToFormat(in []openapi_options.JSONSchema_JSONSchemaSimpleTypes) (string, string) {
  3010. if len(in) == 0 {
  3011. return "", ""
  3012. }
  3013. // Can't support more than 1 type, just return the first element.
  3014. // This is due to an inconsistency in the design of the openapiv2 proto
  3015. // and that used in schemaCore. schemaCore uses the v3 definition of types,
  3016. // which only allows a single string, while the openapiv2 proto uses the OpenAPI v2
  3017. // definition, which defers to the JSON schema definition, which allows a string or an array.
  3018. // Sources:
  3019. // https://swagger.io/specification/#itemsObject
  3020. // https://tools.ietf.org/html/draft-fge-json-schema-validation-00#section-5.5.2
  3021. switch in[0] {
  3022. case openapi_options.JSONSchema_UNKNOWN, openapi_options.JSONSchema_NULL:
  3023. return "", ""
  3024. case openapi_options.JSONSchema_OBJECT:
  3025. return "object", ""
  3026. case openapi_options.JSONSchema_ARRAY:
  3027. return "array", ""
  3028. case openapi_options.JSONSchema_BOOLEAN:
  3029. // NOTE: in OpenAPI specification, format should be empty on boolean type
  3030. return "boolean", ""
  3031. case openapi_options.JSONSchema_INTEGER:
  3032. return "integer", "int32"
  3033. case openapi_options.JSONSchema_NUMBER:
  3034. return "number", "double"
  3035. case openapi_options.JSONSchema_STRING:
  3036. // NOTE: in OpenAPI specification, format should be empty on string type
  3037. return "string", ""
  3038. default:
  3039. // Maybe panic?
  3040. return "", ""
  3041. }
  3042. }
  3043. func protoExternalDocumentationToOpenAPIExternalDocumentation(in *openapi_options.ExternalDocumentation, reg *descriptor.Registry, data interface{}) *openapiExternalDocumentationObject {
  3044. if in == nil {
  3045. return nil
  3046. }
  3047. if reg.GetUseGoTemplate() {
  3048. in.Description = goTemplateComments(in.Description, data, reg)
  3049. }
  3050. return &openapiExternalDocumentationObject{
  3051. Description: in.Description,
  3052. URL: in.Url,
  3053. }
  3054. }
  3055. func addCustomRefs(d openapiDefinitionsObject, reg *descriptor.Registry, refs refMap) error {
  3056. if len(refs) == 0 {
  3057. return nil
  3058. }
  3059. msgMap := make(messageMap)
  3060. enumMap := make(enumMap)
  3061. for ref := range refs {
  3062. swgName, swgOk := fullyQualifiedNameToOpenAPIName(ref, reg)
  3063. if !swgOk {
  3064. grpclog.Errorf("can't resolve OpenAPI name from CustomRef %q", ref)
  3065. continue
  3066. }
  3067. if _, ok := d[swgName]; ok {
  3068. // Skip already existing definitions
  3069. delete(refs, ref)
  3070. continue
  3071. }
  3072. msg, err := reg.LookupMsg("", ref)
  3073. if err == nil {
  3074. msgMap[swgName] = msg
  3075. continue
  3076. }
  3077. enum, err := reg.LookupEnum("", ref)
  3078. if err == nil {
  3079. enumMap[swgName] = enum
  3080. continue
  3081. }
  3082. // ?? Should be either enum or msg
  3083. }
  3084. if err := renderMessagesAsDefinition(msgMap, d, reg, refs, nil); err != nil {
  3085. return err
  3086. }
  3087. renderEnumerationsAsDefinition(enumMap, d, reg, refs)
  3088. // Run again in case any new refs were added
  3089. return addCustomRefs(d, reg, refs)
  3090. }
  3091. func lowerCamelCase(fieldName string, fields []*descriptor.Field, msgs []*descriptor.Message) string {
  3092. for _, oneField := range fields {
  3093. if oneField.GetName() == fieldName {
  3094. return oneField.GetJsonName()
  3095. }
  3096. }
  3097. messageNameToFieldsToJSONName := make(map[string]map[string]string, len(msgs))
  3098. fieldNameToType := make(map[string]string)
  3099. for _, msg := range msgs {
  3100. fieldNameToJSONName := make(map[string]string)
  3101. for _, oneField := range msg.GetField() {
  3102. fieldNameToJSONName[oneField.GetName()] = oneField.GetJsonName()
  3103. fieldNameToType[oneField.GetName()] = oneField.GetTypeName()
  3104. }
  3105. messageNameToFieldsToJSONName[msg.GetName()] = fieldNameToJSONName
  3106. }
  3107. if strings.Contains(fieldName, ".") {
  3108. fieldNames := strings.Split(fieldName, ".")
  3109. fieldNamesWithCamelCase := make([]string, 0)
  3110. for i := 0; i < len(fieldNames)-1; i++ {
  3111. fieldNamesWithCamelCase = append(fieldNamesWithCamelCase, casing.JSONCamelCase(fieldNames[i]))
  3112. }
  3113. prefix := strings.Join(fieldNamesWithCamelCase, ".")
  3114. reservedJSONName := getReservedJSONName(fieldName, messageNameToFieldsToJSONName, fieldNameToType)
  3115. if reservedJSONName != "" {
  3116. return prefix + "." + reservedJSONName
  3117. }
  3118. }
  3119. return casing.JSONCamelCase(fieldName)
  3120. }
  3121. func getReservedJSONName(fieldName string, messageNameToFieldsToJSONName map[string]map[string]string, fieldNameToType map[string]string) string {
  3122. if len(strings.Split(fieldName, ".")) == 2 {
  3123. fieldNames := strings.Split(fieldName, ".")
  3124. firstVariable := fieldNames[0]
  3125. firstType := fieldNameToType[firstVariable]
  3126. firstTypeShortNames := strings.Split(firstType, ".")
  3127. firstTypeShortName := firstTypeShortNames[len(firstTypeShortNames)-1]
  3128. return messageNameToFieldsToJSONName[firstTypeShortName][fieldNames[1]]
  3129. }
  3130. fieldNames := strings.Split(fieldName, ".")
  3131. return getReservedJSONName(strings.Join(fieldNames[1:], "."), messageNameToFieldsToJSONName, fieldNameToType)
  3132. }
  3133. func find(a []string, x string) int {
  3134. // This is a linear search but we are dealing with a small number of fields
  3135. for i, n := range a {
  3136. if x == n {
  3137. return i
  3138. }
  3139. }
  3140. return -1
  3141. }
  3142. // Make a deep copy of the outer parameters that has paramName as the first component,
  3143. // but remove the first component of the field path.
  3144. func subPathParams(paramName string, outerParams []descriptor.Parameter) []descriptor.Parameter {
  3145. var innerParams []descriptor.Parameter
  3146. for _, p := range outerParams {
  3147. if len(p.FieldPath) > 1 && p.FieldPath[0].Name == paramName {
  3148. subParam := descriptor.Parameter{
  3149. FieldPath: p.FieldPath[1:],
  3150. Target: p.Target,
  3151. Method: p.Method,
  3152. }
  3153. innerParams = append(innerParams, subParam)
  3154. }
  3155. }
  3156. return innerParams
  3157. }
  3158. func getFieldConfiguration(reg *descriptor.Registry, fd *descriptor.Field) *openapi_options.JSONSchema_FieldConfiguration {
  3159. if j, err := getFieldOpenAPIOption(reg, fd); err == nil {
  3160. return j.GetFieldConfiguration()
  3161. }
  3162. return nil
  3163. }