template_test.go 332 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487748874897490749174927493749474957496749774987499750075017502750375047505750675077508750975107511751275137514751575167517751875197520752175227523752475257526752775287529753075317532753375347535753675377538753975407541754275437544754575467547754875497550755175527553755475557556755775587559756075617562756375647565756675677568756975707571757275737574757575767577757875797580758175827583758475857586758775887589759075917592759375947595759675977598759976007601760276037604760576067607760876097610761176127613761476157616761776187619762076217622762376247625762676277628762976307631763276337634763576367637763876397640764176427643764476457646764776487649765076517652765376547655765676577658765976607661766276637664766576667667766876697670767176727673767476757676767776787679768076817682768376847685768676877688768976907691769276937694769576967697769876997700770177027703770477057706770777087709771077117712771377147715771677177718771977207721772277237724772577267727772877297730773177327733773477357736773777387739774077417742774377447745774677477748774977507751775277537754775577567757775877597760776177627763776477657766776777687769777077717772777377747775777677777778777977807781778277837784778577867787778877897790779177927793779477957796779777987799780078017802780378047805780678077808780978107811781278137814781578167817781878197820782178227823782478257826782778287829783078317832783378347835783678377838783978407841784278437844784578467847784878497850785178527853785478557856785778587859786078617862786378647865786678677868786978707871787278737874787578767877787878797880788178827883788478857886788778887889789078917892789378947895789678977898789979007901790279037904790579067907790879097910791179127913791479157916791779187919792079217922792379247925792679277928792979307931793279337934793579367937793879397940794179427943794479457946794779487949795079517952795379547955795679577958795979607961796279637964796579667967796879697970797179727973797479757976797779787979798079817982798379847985798679877988798979907991799279937994799579967997799879998000800180028003800480058006800780088009801080118012801380148015801680178018801980208021802280238024802580268027802880298030803180328033803480358036803780388039804080418042804380448045804680478048804980508051805280538054805580568057805880598060806180628063806480658066806780688069807080718072807380748075807680778078807980808081808280838084808580868087808880898090809180928093809480958096809780988099810081018102810381048105810681078108810981108111811281138114811581168117811881198120812181228123812481258126812781288129813081318132813381348135813681378138813981408141814281438144814581468147814881498150815181528153815481558156815781588159816081618162816381648165816681678168816981708171817281738174817581768177817881798180818181828183818481858186818781888189819081918192819381948195819681978198819982008201820282038204820582068207820882098210821182128213821482158216821782188219822082218222822382248225822682278228822982308231823282338234823582368237823882398240824182428243824482458246824782488249825082518252825382548255825682578258825982608261826282638264826582668267826882698270827182728273827482758276827782788279828082818282828382848285828682878288828982908291829282938294829582968297829882998300830183028303830483058306830783088309831083118312831383148315831683178318831983208321832283238324832583268327832883298330833183328333833483358336833783388339834083418342834383448345834683478348834983508351835283538354835583568357835883598360836183628363836483658366836783688369837083718372837383748375837683778378837983808381838283838384838583868387838883898390839183928393839483958396839783988399840084018402840384048405840684078408840984108411841284138414841584168417841884198420842184228423842484258426842784288429843084318432843384348435843684378438843984408441844284438444844584468447844884498450845184528453845484558456845784588459846084618462846384648465846684678468846984708471847284738474847584768477847884798480848184828483848484858486848784888489849084918492849384948495849684978498849985008501850285038504850585068507850885098510851185128513851485158516851785188519852085218522852385248525852685278528852985308531853285338534853585368537853885398540854185428543854485458546854785488549855085518552855385548555855685578558855985608561856285638564856585668567856885698570857185728573857485758576857785788579858085818582858385848585858685878588858985908591859285938594859585968597859885998600860186028603860486058606860786088609861086118612861386148615861686178618861986208621862286238624862586268627862886298630863186328633863486358636863786388639864086418642864386448645864686478648864986508651865286538654865586568657865886598660866186628663866486658666866786688669867086718672867386748675867686778678867986808681868286838684868586868687868886898690869186928693869486958696869786988699870087018702870387048705870687078708870987108711871287138714871587168717871887198720872187228723872487258726872787288729873087318732873387348735873687378738873987408741874287438744874587468747874887498750875187528753875487558756875787588759876087618762876387648765876687678768876987708771877287738774877587768777877887798780878187828783878487858786878787888789879087918792879387948795879687978798879988008801880288038804880588068807880888098810881188128813881488158816881788188819882088218822882388248825882688278828882988308831883288338834883588368837883888398840884188428843884488458846884788488849885088518852885388548855885688578858885988608861886288638864886588668867886888698870887188728873887488758876887788788879888088818882888388848885888688878888888988908891889288938894889588968897889888998900890189028903890489058906890789088909891089118912891389148915891689178918891989208921892289238924892589268927892889298930893189328933893489358936893789388939894089418942894389448945894689478948894989508951895289538954895589568957895889598960896189628963896489658966896789688969897089718972897389748975897689778978897989808981898289838984898589868987898889898990899189928993899489958996899789988999900090019002900390049005900690079008900990109011901290139014901590169017901890199020902190229023902490259026902790289029903090319032903390349035903690379038903990409041904290439044904590469047904890499050905190529053905490559056905790589059906090619062906390649065906690679068906990709071907290739074907590769077907890799080908190829083908490859086908790889089909090919092909390949095909690979098909991009101910291039104910591069107910891099110911191129113911491159116911791189119912091219122912391249125912691279128912991309131913291339134913591369137913891399140914191429143914491459146914791489149915091519152915391549155915691579158915991609161916291639164916591669167916891699170917191729173917491759176917791789179918091819182918391849185918691879188918991909191919291939194919591969197919891999200920192029203920492059206920792089209921092119212921392149215921692179218921992209221922292239224922592269227922892299230923192329233923492359236923792389239924092419242924392449245924692479248924992509251925292539254925592569257925892599260926192629263926492659266926792689269927092719272927392749275927692779278927992809281928292839284928592869287928892899290929192929293929492959296929792989299930093019302930393049305930693079308930993109311931293139314931593169317931893199320932193229323932493259326932793289329933093319332933393349335933693379338933993409341934293439344934593469347934893499350935193529353935493559356935793589359936093619362936393649365936693679368936993709371937293739374937593769377937893799380938193829383938493859386938793889389939093919392939393949395939693979398939994009401940294039404940594069407940894099410941194129413941494159416941794189419942094219422942394249425942694279428942994309431943294339434943594369437943894399440944194429443944494459446944794489449945094519452945394549455945694579458945994609461946294639464946594669467946894699470947194729473947494759476947794789479948094819482948394849485948694879488948994909491949294939494949594969497949894999500950195029503950495059506950795089509951095119512951395149515951695179518951995209521952295239524952595269527952895299530953195329533953495359536953795389539954095419542954395449545954695479548954995509551955295539554955595569557955895599560956195629563956495659566956795689569957095719572957395749575957695779578957995809581958295839584958595869587958895899590959195929593959495959596959795989599960096019602960396049605960696079608960996109611961296139614961596169617961896199620962196229623962496259626962796289629963096319632963396349635963696379638963996409641964296439644964596469647964896499650965196529653965496559656965796589659966096619662966396649665966696679668966996709671967296739674967596769677967896799680968196829683968496859686968796889689969096919692969396949695969696979698969997009701970297039704970597069707970897099710971197129713971497159716971797189719972097219722972397249725972697279728972997309731973297339734973597369737973897399740974197429743974497459746974797489749975097519752975397549755975697579758975997609761976297639764976597669767976897699770977197729773977497759776977797789779978097819782978397849785978697879788978997909791979297939794979597969797979897999800980198029803980498059806980798089809981098119812981398149815981698179818981998209821982298239824982598269827982898299830983198329833983498359836983798389839984098419842984398449845984698479848984998509851985298539854985598569857985898599860986198629863986498659866986798689869987098719872987398749875987698779878987998809881988298839884988598869887988898899890989198929893989498959896989798989899990099019902990399049905990699079908990999109911991299139914991599169917991899199920992199229923992499259926992799289929993099319932993399349935993699379938993999409941994299439944994599469947994899499950995199529953995499559956995799589959996099619962996399649965996699679968996999709971997299739974997599769977997899799980998199829983998499859986998799889989999099919992999399949995999699979998999910000100011000210003100041000510006100071000810009100101001110012100131001410015100161001710018100191002010021100221002310024100251002610027100281002910030100311003210033100341003510036100371003810039100401004110042100431004410045100461004710048100491005010051100521005310054100551005610057100581005910060100611006210063100641006510066100671006810069100701007110072100731007410075100761007710078100791008010081100821008310084100851008610087100881008910090100911009210093100941009510096100971009810099101001010110102101031010410105101061010710108101091011010111101121011310114101151011610117101181011910120101211012210123101241012510126101271012810129101301013110132101331013410135101361013710138101391014010141101421014310144101451014610147101481014910150101511015210153101541015510156101571015810159101601016110162101631016410165101661016710168101691017010171101721017310174101751017610177101781017910180101811018210183101841018510186101871018810189101901019110192101931019410195101961019710198101991020010201102021020310204102051020610207102081020910210102111021210213102141021510216102171021810219102201022110222102231022410225102261022710228102291023010231102321023310234102351023610237102381023910240102411024210243102441024510246102471024810249102501025110252102531025410255102561025710258102591026010261102621026310264102651026610267102681026910270102711027210273102741027510276102771027810279102801028110282102831028410285102861028710288102891029010291102921029310294102951029610297102981029910300103011030210303103041030510306103071030810309103101031110312103131031410315103161031710318103191032010321103221032310324103251032610327103281032910330103311033210333103341033510336103371033810339103401034110342103431034410345103461034710348103491035010351103521035310354103551035610357103581035910360103611036210363103641036510366103671036810369103701037110372103731037410375103761037710378103791038010381103821038310384103851038610387103881038910390103911039210393103941039510396103971039810399104001040110402104031040410405104061040710408104091041010411104121041310414104151041610417104181041910420104211042210423104241042510426104271042810429104301043110432104331043410435104361043710438104391044010441104421044310444104451044610447104481044910450104511045210453104541045510456104571045810459104601046110462104631046410465104661046710468104691047010471104721047310474104751047610477104781047910480104811048210483104841048510486104871048810489104901049110492104931049410495104961049710498104991050010501105021050310504105051050610507105081050910510105111051210513105141051510516105171051810519105201052110522105231052410525105261052710528105291053010531105321053310534105351053610537105381053910540105411054210543105441054510546105471054810549105501055110552105531055410555105561055710558105591056010561105621056310564105651056610567105681056910570105711057210573105741057510576105771057810579105801058110582105831058410585105861058710588105891059010591105921059310594105951059610597105981059910600106011060210603106041060510606106071060810609106101061110612106131061410615106161061710618106191062010621106221062310624106251062610627106281062910630106311063210633106341063510636106371063810639106401064110642106431064410645106461064710648106491065010651106521065310654106551065610657106581065910660106611066210663106641066510666106671066810669106701067110672106731067410675106761067710678106791068010681106821068310684106851068610687106881068910690106911069210693106941069510696106971069810699107001070110702107031070410705107061070710708107091071010711107121071310714107151071610717107181071910720107211072210723107241072510726107271072810729107301073110732107331073410735107361073710738107391074010741107421074310744107451074610747107481074910750107511075210753107541075510756107571075810759107601076110762107631076410765107661076710768107691077010771107721077310774107751077610777107781077910780107811078210783107841078510786107871078810789107901079110792107931079410795107961079710798107991080010801108021080310804108051080610807108081080910810108111081210813108141081510816108171081810819108201082110822108231082410825108261082710828108291083010831108321083310834108351083610837108381083910840108411084210843108441084510846108471084810849108501085110852108531085410855108561085710858108591086010861108621086310864108651086610867108681086910870108711087210873108741087510876108771087810879108801088110882108831088410885108861088710888108891089010891108921089310894108951089610897108981089910900109011090210903109041090510906109071090810909109101091110912109131091410915109161091710918109191092010921109221092310924109251092610927109281092910930109311093210933109341093510936109371093810939109401094110942109431094410945109461094710948109491095010951109521095310954109551095610957109581095910960109611096210963109641096510966109671096810969109701097110972109731097410975109761097710978109791098010981109821098310984109851098610987109881098910990109911099210993109941099510996109971099810999110001100111002110031100411005110061100711008110091101011011110121101311014110151101611017110181101911020110211102211023110241102511026110271102811029110301103111032110331103411035110361103711038110391104011041110421104311044110451104611047110481104911050110511105211053110541105511056110571105811059110601106111062110631106411065110661106711068110691107011071110721107311074110751107611077110781107911080110811108211083110841108511086110871108811089110901109111092110931109411095110961109711098110991110011101111021110311104111051110611107111081110911110111111111211113111141111511116111171111811119111201112111122111231112411125111261112711128111291113011131111321113311134111351113611137111381113911140111411114211143111441114511146111471114811149111501115111152111531115411155111561115711158111591116011161111621116311164111651116611167111681116911170111711117211173111741117511176111771117811179111801118111182111831118411185111861118711188111891119011191111921119311194111951119611197111981119911200
  1. package genopenapi
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "math"
  8. "os"
  9. "reflect"
  10. "strings"
  11. "testing"
  12. "github.com/google/go-cmp/cmp"
  13. "github.com/grpc-ecosystem/grpc-gateway/v2/internal/descriptor"
  14. "github.com/grpc-ecosystem/grpc-gateway/v2/internal/descriptor/openapiconfig"
  15. "github.com/grpc-ecosystem/grpc-gateway/v2/internal/httprule"
  16. openapi_options "github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-openapiv2/options"
  17. "github.com/grpc-ecosystem/grpc-gateway/v2/runtime"
  18. "google.golang.org/genproto/googleapis/api/annotations"
  19. "google.golang.org/genproto/googleapis/api/visibility"
  20. "google.golang.org/protobuf/encoding/protojson"
  21. "google.golang.org/protobuf/proto"
  22. "google.golang.org/protobuf/reflect/protodesc"
  23. "google.golang.org/protobuf/reflect/protoreflect"
  24. "google.golang.org/protobuf/reflect/protoregistry"
  25. "google.golang.org/protobuf/types/descriptorpb"
  26. "google.golang.org/protobuf/types/known/anypb"
  27. "google.golang.org/protobuf/types/known/durationpb"
  28. "google.golang.org/protobuf/types/known/emptypb"
  29. field_mask "google.golang.org/protobuf/types/known/fieldmaskpb"
  30. "google.golang.org/protobuf/types/known/structpb"
  31. "google.golang.org/protobuf/types/known/timestamppb"
  32. "google.golang.org/protobuf/types/known/wrapperspb"
  33. "google.golang.org/protobuf/types/pluginpb"
  34. )
  35. var marshaler = &runtime.JSONPb{}
  36. func TestOpenapiExamplesFromProtoExamples(t *testing.T) {
  37. examples := openapiExamplesFromProtoExamples(map[string]string{
  38. "application/json": `{"Hello": "Worldr!"}`,
  39. "plain/text": "Hello, World!",
  40. })
  41. testCases := map[Format]string{
  42. FormatJSON: `
  43. {
  44. "application/json": {
  45. "Hello": "Worldr!"
  46. },
  47. "plain/text": "Hello, World!"
  48. }
  49. `,
  50. FormatYAML: `
  51. application/json:
  52. Hello: Worldr!
  53. plain/text: Hello, World!
  54. `,
  55. }
  56. spaceRemover := strings.NewReplacer(" ", "", "\t", "", "\n", "")
  57. for format, expected := range testCases {
  58. t.Run(string(format), func(t *testing.T) {
  59. var buf bytes.Buffer
  60. encoder, err := format.NewEncoder(&buf)
  61. if err != nil {
  62. t.Fatalf("creating encoder: %s", err)
  63. }
  64. err = encoder.Encode(examples)
  65. if err != nil {
  66. t.Fatalf("encoding: %s", err)
  67. }
  68. actual := spaceRemover.Replace(buf.String())
  69. expected = spaceRemover.Replace(expected)
  70. if expected != actual {
  71. t.Fatalf("expected:\n%s\nactual:\n%s", expected, actual)
  72. }
  73. })
  74. }
  75. }
  76. func crossLinkFixture(f *descriptor.File) *descriptor.File {
  77. for _, m := range f.Messages {
  78. m.File = f
  79. }
  80. for _, svc := range f.Services {
  81. svc.File = f
  82. for _, m := range svc.Methods {
  83. m.Service = svc
  84. for _, b := range m.Bindings {
  85. b.Method = m
  86. for _, param := range b.PathParams {
  87. param.Method = m
  88. }
  89. }
  90. }
  91. }
  92. return f
  93. }
  94. func reqFromFile(f *descriptor.File) *pluginpb.CodeGeneratorRequest {
  95. return &pluginpb.CodeGeneratorRequest{
  96. ProtoFile: []*descriptorpb.FileDescriptorProto{
  97. f.FileDescriptorProto,
  98. },
  99. FileToGenerate: []string{f.GetName()},
  100. }
  101. }
  102. func TestMessageToQueryParametersWithEnumAsInt(t *testing.T) {
  103. type test struct {
  104. MsgDescs []*descriptorpb.DescriptorProto
  105. Message string
  106. Params []openapiParameterObject
  107. }
  108. tests := []test{
  109. {
  110. MsgDescs: []*descriptorpb.DescriptorProto{
  111. {
  112. Name: proto.String("ExampleMessage"),
  113. Field: []*descriptorpb.FieldDescriptorProto{
  114. {
  115. Name: proto.String("a"),
  116. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  117. Number: proto.Int32(1),
  118. },
  119. {
  120. Name: proto.String("b"),
  121. Type: descriptorpb.FieldDescriptorProto_TYPE_DOUBLE.Enum(),
  122. Number: proto.Int32(2),
  123. },
  124. {
  125. Name: proto.String("c"),
  126. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  127. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  128. Number: proto.Int32(3),
  129. },
  130. },
  131. },
  132. },
  133. Message: "ExampleMessage",
  134. Params: []openapiParameterObject{
  135. {
  136. Name: "a",
  137. In: "query",
  138. Required: false,
  139. Type: "string",
  140. },
  141. {
  142. Name: "b",
  143. In: "query",
  144. Required: false,
  145. Type: "number",
  146. Format: "double",
  147. },
  148. {
  149. Name: "c",
  150. In: "query",
  151. Required: false,
  152. Type: "array",
  153. CollectionFormat: "multi",
  154. },
  155. },
  156. },
  157. {
  158. MsgDescs: []*descriptorpb.DescriptorProto{
  159. {
  160. Name: proto.String("ExampleMessage"),
  161. Field: []*descriptorpb.FieldDescriptorProto{
  162. {
  163. Name: proto.String("nested"),
  164. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  165. TypeName: proto.String(".example.Nested"),
  166. Number: proto.Int32(1),
  167. },
  168. },
  169. },
  170. {
  171. Name: proto.String("Nested"),
  172. Field: []*descriptorpb.FieldDescriptorProto{
  173. {
  174. Name: proto.String("a"),
  175. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  176. Number: proto.Int32(1),
  177. },
  178. {
  179. Name: proto.String("deep"),
  180. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  181. TypeName: proto.String(".example.Nested.DeepNested"),
  182. Number: proto.Int32(2),
  183. },
  184. },
  185. NestedType: []*descriptorpb.DescriptorProto{{
  186. Name: proto.String("DeepNested"),
  187. Field: []*descriptorpb.FieldDescriptorProto{
  188. {
  189. Name: proto.String("b"),
  190. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  191. Number: proto.Int32(1),
  192. },
  193. {
  194. Name: proto.String("c"),
  195. Type: descriptorpb.FieldDescriptorProto_TYPE_ENUM.Enum(),
  196. TypeName: proto.String(".example.Nested.DeepNested.DeepEnum"),
  197. Number: proto.Int32(2),
  198. },
  199. },
  200. EnumType: []*descriptorpb.EnumDescriptorProto{
  201. {
  202. Name: proto.String("DeepEnum"),
  203. Value: []*descriptorpb.EnumValueDescriptorProto{
  204. {Name: proto.String("FALSE"), Number: proto.Int32(0)},
  205. {Name: proto.String("TRUE"), Number: proto.Int32(1)},
  206. },
  207. },
  208. },
  209. }},
  210. },
  211. },
  212. Message: "ExampleMessage",
  213. Params: []openapiParameterObject{
  214. {
  215. Name: "nested.a",
  216. In: "query",
  217. Required: false,
  218. Type: "string",
  219. },
  220. {
  221. Name: "nested.deep.b",
  222. In: "query",
  223. Required: false,
  224. Type: "string",
  225. },
  226. {
  227. Name: "nested.deep.c",
  228. In: "query",
  229. Required: false,
  230. Type: "integer",
  231. Enum: []int{0, 1},
  232. Default: 0,
  233. },
  234. },
  235. },
  236. }
  237. for _, test := range tests {
  238. reg := descriptor.NewRegistry()
  239. reg.SetEnumsAsInts(true)
  240. var msgs []*descriptor.Message
  241. for _, msgdesc := range test.MsgDescs {
  242. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  243. }
  244. file := descriptor.File{
  245. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  246. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  247. Name: proto.String("example.proto"),
  248. Package: proto.String("example"),
  249. Dependency: []string{},
  250. MessageType: test.MsgDescs,
  251. Service: []*descriptorpb.ServiceDescriptorProto{},
  252. Options: &descriptorpb.FileOptions{
  253. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  254. },
  255. },
  256. GoPkg: descriptor.GoPackage{
  257. Path: "example.com/path/to/example/example.pb",
  258. Name: "example_pb",
  259. },
  260. Messages: msgs,
  261. }
  262. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  263. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  264. })
  265. if err != nil {
  266. t.Fatalf("failed to load code generator request: %v", err)
  267. }
  268. message, err := reg.LookupMsg("", ".example."+test.Message)
  269. if err != nil {
  270. t.Fatalf("failed to lookup message: %s", err)
  271. }
  272. params, err := messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  273. if err != nil {
  274. t.Fatalf("failed to convert message to query parameters: %s", err)
  275. }
  276. // avoid checking Items for array types
  277. for i := range params {
  278. params[i].Items = nil
  279. }
  280. if !reflect.DeepEqual(params, test.Params) {
  281. t.Errorf("expected %v, got %v", test.Params, params)
  282. }
  283. }
  284. }
  285. func TestMessageToQueryParametersWithOmitEnumDefaultValue(t *testing.T) {
  286. type test struct {
  287. MsgDescs []*descriptorpb.DescriptorProto
  288. Message string
  289. Params []openapiParameterObject
  290. }
  291. tests := []test{
  292. {
  293. MsgDescs: []*descriptorpb.DescriptorProto{
  294. {
  295. Name: proto.String("ExampleMessage"),
  296. Field: []*descriptorpb.FieldDescriptorProto{
  297. {
  298. Name: proto.String("a"),
  299. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  300. Number: proto.Int32(1),
  301. },
  302. {
  303. Name: proto.String("b"),
  304. Type: descriptorpb.FieldDescriptorProto_TYPE_DOUBLE.Enum(),
  305. Number: proto.Int32(2),
  306. },
  307. {
  308. Name: proto.String("c"),
  309. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  310. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  311. Number: proto.Int32(3),
  312. },
  313. },
  314. },
  315. },
  316. Message: "ExampleMessage",
  317. Params: []openapiParameterObject{
  318. {
  319. Name: "a",
  320. In: "query",
  321. Required: false,
  322. Type: "string",
  323. },
  324. {
  325. Name: "b",
  326. In: "query",
  327. Required: false,
  328. Type: "number",
  329. Format: "double",
  330. },
  331. {
  332. Name: "c",
  333. In: "query",
  334. Required: false,
  335. Type: "array",
  336. CollectionFormat: "multi",
  337. },
  338. },
  339. },
  340. {
  341. MsgDescs: []*descriptorpb.DescriptorProto{
  342. {
  343. Name: proto.String("ExampleMessage"),
  344. Field: []*descriptorpb.FieldDescriptorProto{
  345. {
  346. Name: proto.String("nested"),
  347. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  348. TypeName: proto.String(".example.Nested"),
  349. Number: proto.Int32(1),
  350. },
  351. },
  352. },
  353. {
  354. Name: proto.String("Nested"),
  355. Field: []*descriptorpb.FieldDescriptorProto{
  356. {
  357. Name: proto.String("a"),
  358. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  359. Number: proto.Int32(1),
  360. },
  361. {
  362. Name: proto.String("deep"),
  363. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  364. TypeName: proto.String(".example.Nested.DeepNested"),
  365. Number: proto.Int32(2),
  366. },
  367. },
  368. NestedType: []*descriptorpb.DescriptorProto{{
  369. Name: proto.String("DeepNested"),
  370. Field: []*descriptorpb.FieldDescriptorProto{
  371. {
  372. Name: proto.String("b"),
  373. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  374. Number: proto.Int32(1),
  375. },
  376. {
  377. Name: proto.String("c"),
  378. Type: descriptorpb.FieldDescriptorProto_TYPE_ENUM.Enum(),
  379. TypeName: proto.String(".example.Nested.DeepNested.DeepEnum"),
  380. Number: proto.Int32(2),
  381. },
  382. },
  383. EnumType: []*descriptorpb.EnumDescriptorProto{
  384. {
  385. Name: proto.String("DeepEnum"),
  386. Value: []*descriptorpb.EnumValueDescriptorProto{
  387. {Name: proto.String("FALSE"), Number: proto.Int32(0)},
  388. {Name: proto.String("TRUE"), Number: proto.Int32(1)},
  389. },
  390. },
  391. },
  392. }},
  393. },
  394. },
  395. Message: "ExampleMessage",
  396. Params: []openapiParameterObject{
  397. {
  398. Name: "nested.a",
  399. In: "query",
  400. Required: false,
  401. Type: "string",
  402. },
  403. {
  404. Name: "nested.deep.b",
  405. In: "query",
  406. Required: false,
  407. Type: "string",
  408. },
  409. {
  410. Name: "nested.deep.c",
  411. In: "query",
  412. Required: false,
  413. Type: "string",
  414. Enum: []string{"TRUE"},
  415. },
  416. },
  417. },
  418. }
  419. for _, test := range tests {
  420. reg := descriptor.NewRegistry()
  421. reg.SetOmitEnumDefaultValue(true)
  422. var msgs []*descriptor.Message
  423. for _, msgdesc := range test.MsgDescs {
  424. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  425. }
  426. file := descriptor.File{
  427. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  428. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  429. Name: proto.String("example.proto"),
  430. Package: proto.String("example"),
  431. Dependency: []string{},
  432. MessageType: test.MsgDescs,
  433. Service: []*descriptorpb.ServiceDescriptorProto{},
  434. Options: &descriptorpb.FileOptions{
  435. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  436. },
  437. },
  438. GoPkg: descriptor.GoPackage{
  439. Path: "example.com/path/to/example/example.pb",
  440. Name: "example_pb",
  441. },
  442. Messages: msgs,
  443. }
  444. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  445. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  446. })
  447. if err != nil {
  448. t.Fatalf("failed to load code generator request: %v", err)
  449. }
  450. message, err := reg.LookupMsg("", ".example."+test.Message)
  451. if err != nil {
  452. t.Fatalf("failed to lookup message: %s", err)
  453. }
  454. params, err := messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  455. if err != nil {
  456. t.Fatalf("failed to convert message to query parameters: %s", err)
  457. }
  458. // avoid checking Items for array types
  459. for i := range params {
  460. params[i].Items = nil
  461. }
  462. if !reflect.DeepEqual(params, test.Params) {
  463. t.Errorf("expected %v, got %v", test.Params, params)
  464. }
  465. }
  466. }
  467. func TestMessageToQueryParameters(t *testing.T) {
  468. type test struct {
  469. MsgDescs []*descriptorpb.DescriptorProto
  470. Message string
  471. Params []openapiParameterObject
  472. }
  473. tests := []test{
  474. {
  475. MsgDescs: []*descriptorpb.DescriptorProto{
  476. {
  477. Name: proto.String("ExampleMessage"),
  478. Field: []*descriptorpb.FieldDescriptorProto{
  479. {
  480. Name: proto.String("a"),
  481. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  482. Number: proto.Int32(1),
  483. },
  484. {
  485. Name: proto.String("b"),
  486. Type: descriptorpb.FieldDescriptorProto_TYPE_DOUBLE.Enum(),
  487. Number: proto.Int32(2),
  488. },
  489. {
  490. Name: proto.String("c"),
  491. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  492. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  493. Number: proto.Int32(3),
  494. },
  495. },
  496. },
  497. },
  498. Message: "ExampleMessage",
  499. Params: []openapiParameterObject{
  500. {
  501. Name: "a",
  502. In: "query",
  503. Required: false,
  504. Type: "string",
  505. },
  506. {
  507. Name: "b",
  508. In: "query",
  509. Required: false,
  510. Type: "number",
  511. Format: "double",
  512. },
  513. {
  514. Name: "c",
  515. In: "query",
  516. Required: false,
  517. Type: "array",
  518. CollectionFormat: "multi",
  519. },
  520. },
  521. },
  522. {
  523. MsgDescs: []*descriptorpb.DescriptorProto{
  524. {
  525. Name: proto.String("ExampleMessage"),
  526. Field: []*descriptorpb.FieldDescriptorProto{
  527. {
  528. Name: proto.String("nested"),
  529. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  530. TypeName: proto.String(".example.Nested"),
  531. Number: proto.Int32(1),
  532. },
  533. },
  534. },
  535. {
  536. Name: proto.String("Nested"),
  537. Field: []*descriptorpb.FieldDescriptorProto{
  538. {
  539. Name: proto.String("a"),
  540. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  541. Number: proto.Int32(1),
  542. },
  543. {
  544. Name: proto.String("deep"),
  545. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  546. TypeName: proto.String(".example.Nested.DeepNested"),
  547. Number: proto.Int32(2),
  548. },
  549. },
  550. NestedType: []*descriptorpb.DescriptorProto{{
  551. Name: proto.String("DeepNested"),
  552. Field: []*descriptorpb.FieldDescriptorProto{
  553. {
  554. Name: proto.String("b"),
  555. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  556. Number: proto.Int32(1),
  557. },
  558. {
  559. Name: proto.String("c"),
  560. Type: descriptorpb.FieldDescriptorProto_TYPE_ENUM.Enum(),
  561. TypeName: proto.String(".example.Nested.DeepNested.DeepEnum"),
  562. Number: proto.Int32(2),
  563. },
  564. },
  565. EnumType: []*descriptorpb.EnumDescriptorProto{
  566. {
  567. Name: proto.String("DeepEnum"),
  568. Value: []*descriptorpb.EnumValueDescriptorProto{
  569. {Name: proto.String("FALSE"), Number: proto.Int32(0)},
  570. {Name: proto.String("TRUE"), Number: proto.Int32(1)},
  571. },
  572. },
  573. },
  574. }},
  575. },
  576. },
  577. Message: "ExampleMessage",
  578. Params: []openapiParameterObject{
  579. {
  580. Name: "nested.a",
  581. In: "query",
  582. Required: false,
  583. Type: "string",
  584. },
  585. {
  586. Name: "nested.deep.b",
  587. In: "query",
  588. Required: false,
  589. Type: "string",
  590. },
  591. {
  592. Name: "nested.deep.c",
  593. In: "query",
  594. Required: false,
  595. Type: "string",
  596. Enum: []string{"FALSE", "TRUE"},
  597. Default: "FALSE",
  598. },
  599. },
  600. },
  601. }
  602. for _, test := range tests {
  603. reg := descriptor.NewRegistry()
  604. msgs := []*descriptor.Message{}
  605. for _, msgdesc := range test.MsgDescs {
  606. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  607. }
  608. file := descriptor.File{
  609. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  610. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  611. Name: proto.String("example.proto"),
  612. Package: proto.String("example"),
  613. Dependency: []string{},
  614. MessageType: test.MsgDescs,
  615. Service: []*descriptorpb.ServiceDescriptorProto{},
  616. Options: &descriptorpb.FileOptions{
  617. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  618. },
  619. },
  620. GoPkg: descriptor.GoPackage{
  621. Path: "example.com/path/to/example/example.pb",
  622. Name: "example_pb",
  623. },
  624. Messages: msgs,
  625. }
  626. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  627. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  628. })
  629. if err != nil {
  630. t.Fatalf("failed to load code generator request: %v", err)
  631. }
  632. message, err := reg.LookupMsg("", ".example."+test.Message)
  633. if err != nil {
  634. t.Fatalf("failed to lookup message: %s", err)
  635. }
  636. params, err := messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  637. if err != nil {
  638. t.Fatalf("failed to convert message to query parameters: %s", err)
  639. }
  640. // avoid checking Items for array types
  641. for i := range params {
  642. params[i].Items = nil
  643. }
  644. if !reflect.DeepEqual(params, test.Params) {
  645. t.Errorf("expected %v, got %v", test.Params, params)
  646. }
  647. }
  648. }
  649. // TestMessageToQueryParametersNoRecursive, is a check that cyclical references between messages
  650. // are not falsely detected given previous known edge-cases.
  651. func TestMessageToQueryParametersNoRecursive(t *testing.T) {
  652. type test struct {
  653. MsgDescs []*descriptorpb.DescriptorProto
  654. Message string
  655. }
  656. tests := []test{
  657. // First test:
  658. // Here is a message that has two of another message adjacent to one another in a nested message.
  659. // There is no loop but this was previously falsely flagged as a cycle.
  660. // Example proto:
  661. // message NonRecursiveMessage {
  662. // string field = 1;
  663. // }
  664. // message BaseMessage {
  665. // NonRecursiveMessage first = 1;
  666. // NonRecursiveMessage second = 2;
  667. // }
  668. // message QueryMessage {
  669. // BaseMessage first = 1;
  670. // string second = 2;
  671. // }
  672. {
  673. MsgDescs: []*descriptorpb.DescriptorProto{
  674. {
  675. Name: proto.String("QueryMessage"),
  676. Field: []*descriptorpb.FieldDescriptorProto{
  677. {
  678. Name: proto.String("first"),
  679. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  680. TypeName: proto.String(".example.BaseMessage"),
  681. Number: proto.Int32(1),
  682. },
  683. {
  684. Name: proto.String("second"),
  685. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  686. Number: proto.Int32(2),
  687. },
  688. },
  689. },
  690. {
  691. Name: proto.String("BaseMessage"),
  692. Field: []*descriptorpb.FieldDescriptorProto{
  693. {
  694. Name: proto.String("first"),
  695. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  696. TypeName: proto.String(".example.NonRecursiveMessage"),
  697. Number: proto.Int32(1),
  698. },
  699. {
  700. Name: proto.String("second"),
  701. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  702. TypeName: proto.String(".example.NonRecursiveMessage"),
  703. Number: proto.Int32(2),
  704. },
  705. },
  706. },
  707. // Note there is no recursive nature to this message
  708. {
  709. Name: proto.String("NonRecursiveMessage"),
  710. Field: []*descriptorpb.FieldDescriptorProto{
  711. {
  712. Name: proto.String("field"),
  713. // Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  714. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  715. Number: proto.Int32(1),
  716. },
  717. },
  718. },
  719. },
  720. Message: "QueryMessage",
  721. },
  722. }
  723. for _, test := range tests {
  724. reg := descriptor.NewRegistry()
  725. msgs := []*descriptor.Message{}
  726. for _, msgdesc := range test.MsgDescs {
  727. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  728. }
  729. file := descriptor.File{
  730. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  731. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  732. Name: proto.String("example.proto"),
  733. Package: proto.String("example"),
  734. Dependency: []string{},
  735. MessageType: test.MsgDescs,
  736. Service: []*descriptorpb.ServiceDescriptorProto{},
  737. Options: &descriptorpb.FileOptions{
  738. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  739. },
  740. },
  741. GoPkg: descriptor.GoPackage{
  742. Path: "example.com/path/to/example/example.pb",
  743. Name: "example_pb",
  744. },
  745. Messages: msgs,
  746. }
  747. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  748. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  749. })
  750. if err != nil {
  751. t.Fatalf("failed to load code generator request: %v", err)
  752. }
  753. message, err := reg.LookupMsg("", ".example."+test.Message)
  754. if err != nil {
  755. t.Fatalf("failed to lookup message: %s", err)
  756. }
  757. _, err = messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  758. if err != nil {
  759. t.Fatalf("No recursion error should be thrown: %s", err)
  760. }
  761. }
  762. }
  763. // TestMessageToQueryParametersRecursive, is a check that cyclical references between messages
  764. // are handled gracefully. The goal is to ensure that attempts to add messages with cyclical
  765. // references to query-parameters returns an error message.
  766. func TestMessageToQueryParametersRecursive(t *testing.T) {
  767. type test struct {
  768. MsgDescs []*descriptorpb.DescriptorProto
  769. Message string
  770. }
  771. tests := []test{
  772. // First test:
  773. // Here we test that a message that references itself through a field will return an error.
  774. // Example proto:
  775. // message DirectRecursiveMessage {
  776. // DirectRecursiveMessage nested = 1;
  777. // }
  778. {
  779. MsgDescs: []*descriptorpb.DescriptorProto{
  780. {
  781. Name: proto.String("DirectRecursiveMessage"),
  782. Field: []*descriptorpb.FieldDescriptorProto{
  783. {
  784. Name: proto.String("nested"),
  785. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  786. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  787. TypeName: proto.String(".example.DirectRecursiveMessage"),
  788. Number: proto.Int32(1),
  789. },
  790. },
  791. },
  792. },
  793. Message: "DirectRecursiveMessage",
  794. },
  795. // Second test:
  796. // Here we test that a cycle through multiple messages is detected and that an error is returned.
  797. // Sample:
  798. // message Root { NodeMessage nested = 1; }
  799. // message NodeMessage { CycleMessage nested = 1; }
  800. // message CycleMessage { Root nested = 1; }
  801. {
  802. MsgDescs: []*descriptorpb.DescriptorProto{
  803. {
  804. Name: proto.String("RootMessage"),
  805. Field: []*descriptorpb.FieldDescriptorProto{
  806. {
  807. Name: proto.String("nested"),
  808. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  809. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  810. TypeName: proto.String(".example.NodeMessage"),
  811. Number: proto.Int32(1),
  812. },
  813. },
  814. },
  815. {
  816. Name: proto.String("NodeMessage"),
  817. Field: []*descriptorpb.FieldDescriptorProto{
  818. {
  819. Name: proto.String("nested"),
  820. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  821. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  822. TypeName: proto.String(".example.CycleMessage"),
  823. Number: proto.Int32(1),
  824. },
  825. },
  826. },
  827. {
  828. Name: proto.String("CycleMessage"),
  829. Field: []*descriptorpb.FieldDescriptorProto{
  830. {
  831. Name: proto.String("nested"),
  832. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  833. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  834. TypeName: proto.String(".example.RootMessage"),
  835. Number: proto.Int32(1),
  836. },
  837. },
  838. },
  839. },
  840. Message: "RootMessage",
  841. },
  842. }
  843. for _, test := range tests {
  844. reg := descriptor.NewRegistry()
  845. msgs := []*descriptor.Message{}
  846. for _, msgdesc := range test.MsgDescs {
  847. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  848. }
  849. file := descriptor.File{
  850. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  851. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  852. Name: proto.String("example.proto"),
  853. Package: proto.String("example"),
  854. Dependency: []string{},
  855. MessageType: test.MsgDescs,
  856. Service: []*descriptorpb.ServiceDescriptorProto{},
  857. Options: &descriptorpb.FileOptions{
  858. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  859. },
  860. },
  861. GoPkg: descriptor.GoPackage{
  862. Path: "example.com/path/to/example/example.pb",
  863. Name: "example_pb",
  864. },
  865. Messages: msgs,
  866. }
  867. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  868. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  869. })
  870. if err != nil {
  871. t.Fatalf("failed to load code generator request: %v", err)
  872. }
  873. message, err := reg.LookupMsg("", ".example."+test.Message)
  874. if err != nil {
  875. t.Fatalf("failed to lookup message: %s", err)
  876. }
  877. _, err = messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  878. if err == nil {
  879. t.Fatalf("It should not be allowed to have recursive query parameters")
  880. }
  881. }
  882. }
  883. func TestMessageToQueryParametersWithJsonName(t *testing.T) {
  884. type test struct {
  885. MsgDescs []*descriptorpb.DescriptorProto
  886. Message string
  887. Params []openapiParameterObject
  888. }
  889. requiredField := []annotations.FieldBehavior{annotations.FieldBehavior_REQUIRED}
  890. requiredFieldOptions := new(descriptorpb.FieldOptions)
  891. proto.SetExtension(requiredFieldOptions, annotations.E_FieldBehavior, requiredField)
  892. messageSchema := &openapi_options.Schema{
  893. JsonSchema: &openapi_options.JSONSchema{
  894. Required: []string{"test_field_b"},
  895. },
  896. }
  897. messageOption := &descriptorpb.MessageOptions{}
  898. proto.SetExtension(messageOption, openapi_options.E_Openapiv2Schema, messageSchema)
  899. tests := []test{
  900. {
  901. MsgDescs: []*descriptorpb.DescriptorProto{
  902. {
  903. Name: proto.String("ExampleMessage"),
  904. Field: []*descriptorpb.FieldDescriptorProto{
  905. {
  906. Name: proto.String("test_field_a"),
  907. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  908. Number: proto.Int32(1),
  909. JsonName: proto.String("testFieldA"),
  910. },
  911. },
  912. },
  913. },
  914. Message: "ExampleMessage",
  915. Params: []openapiParameterObject{
  916. {
  917. Name: "testFieldA",
  918. In: "query",
  919. Required: false,
  920. Type: "string",
  921. },
  922. },
  923. },
  924. {
  925. MsgDescs: []*descriptorpb.DescriptorProto{
  926. {
  927. Name: proto.String("SubMessage"),
  928. Field: []*descriptorpb.FieldDescriptorProto{
  929. {
  930. Name: proto.String("test_field_a"),
  931. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  932. Number: proto.Int32(1),
  933. JsonName: proto.String("testFieldA"),
  934. },
  935. },
  936. },
  937. {
  938. Name: proto.String("ExampleMessage"),
  939. Field: []*descriptorpb.FieldDescriptorProto{
  940. {
  941. Name: proto.String("sub_message"),
  942. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  943. TypeName: proto.String(".example.SubMessage"),
  944. Number: proto.Int32(1),
  945. JsonName: proto.String("subMessage"),
  946. },
  947. },
  948. },
  949. },
  950. Message: "ExampleMessage",
  951. Params: []openapiParameterObject{
  952. {
  953. Name: "subMessage.testFieldA",
  954. In: "query",
  955. Required: false,
  956. Type: "string",
  957. },
  958. },
  959. },
  960. {
  961. MsgDescs: []*descriptorpb.DescriptorProto{
  962. {
  963. Name: proto.String("ExampleMessage"),
  964. Field: []*descriptorpb.FieldDescriptorProto{
  965. {
  966. Name: proto.String("test_field_a"),
  967. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  968. Number: proto.Int32(1),
  969. JsonName: proto.String("testFieldACustom"),
  970. Options: requiredFieldOptions,
  971. },
  972. {
  973. Name: proto.String("test_field_b"),
  974. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  975. Number: proto.Int32(2),
  976. JsonName: proto.String("testFieldBCustom"),
  977. },
  978. },
  979. Options: messageOption,
  980. },
  981. },
  982. Message: "ExampleMessage",
  983. Params: []openapiParameterObject{
  984. {
  985. Name: "testFieldACustom",
  986. In: "query",
  987. Required: true,
  988. Type: "string",
  989. },
  990. {
  991. Name: "testFieldBCustom",
  992. In: "query",
  993. Required: true,
  994. Type: "string",
  995. },
  996. },
  997. },
  998. }
  999. for _, test := range tests {
  1000. reg := descriptor.NewRegistry()
  1001. reg.SetUseJSONNamesForFields(true)
  1002. msgs := []*descriptor.Message{}
  1003. for _, msgdesc := range test.MsgDescs {
  1004. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  1005. }
  1006. file := descriptor.File{
  1007. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  1008. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1009. Name: proto.String("example.proto"),
  1010. Package: proto.String("example"),
  1011. Dependency: []string{},
  1012. MessageType: test.MsgDescs,
  1013. Service: []*descriptorpb.ServiceDescriptorProto{},
  1014. Options: &descriptorpb.FileOptions{
  1015. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  1016. },
  1017. },
  1018. GoPkg: descriptor.GoPackage{
  1019. Path: "example.com/path/to/example/example.pb",
  1020. Name: "example_pb",
  1021. },
  1022. Messages: msgs,
  1023. }
  1024. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  1025. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  1026. })
  1027. if err != nil {
  1028. t.Fatalf("failed to load code generator request: %v", err)
  1029. }
  1030. message, err := reg.LookupMsg("", ".example."+test.Message)
  1031. if err != nil {
  1032. t.Fatalf("failed to lookup message: %s", err)
  1033. }
  1034. params, err := messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  1035. if err != nil {
  1036. t.Fatalf("failed to convert message to query parameters: %s", err)
  1037. }
  1038. if !reflect.DeepEqual(params, test.Params) {
  1039. t.Errorf("expected %#v, got %#v", test.Params, params)
  1040. }
  1041. }
  1042. }
  1043. func TestMessageToQueryParametersWellKnownTypes(t *testing.T) {
  1044. type test struct {
  1045. MsgDescs []*descriptorpb.DescriptorProto
  1046. WellKnownMsgDescs []*descriptorpb.DescriptorProto
  1047. Message string
  1048. Params []openapiParameterObject
  1049. }
  1050. tests := []test{
  1051. {
  1052. MsgDescs: []*descriptorpb.DescriptorProto{
  1053. {
  1054. Name: proto.String("ExampleMessage"),
  1055. Field: []*descriptorpb.FieldDescriptorProto{
  1056. {
  1057. Name: proto.String("a_field_mask"),
  1058. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  1059. TypeName: proto.String(".google.protobuf.FieldMask"),
  1060. Number: proto.Int32(1),
  1061. },
  1062. {
  1063. Name: proto.String("a_timestamp"),
  1064. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  1065. TypeName: proto.String(".google.protobuf.Timestamp"),
  1066. Number: proto.Int32(2),
  1067. },
  1068. },
  1069. },
  1070. },
  1071. WellKnownMsgDescs: []*descriptorpb.DescriptorProto{
  1072. {
  1073. Name: proto.String("FieldMask"),
  1074. Field: []*descriptorpb.FieldDescriptorProto{
  1075. {
  1076. Name: proto.String("paths"),
  1077. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  1078. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  1079. Number: proto.Int32(1),
  1080. },
  1081. },
  1082. },
  1083. {
  1084. Name: proto.String("Timestamp"),
  1085. Field: []*descriptorpb.FieldDescriptorProto{
  1086. {
  1087. Name: proto.String("seconds"),
  1088. Type: descriptorpb.FieldDescriptorProto_TYPE_INT64.Enum(),
  1089. Number: proto.Int32(1),
  1090. },
  1091. {
  1092. Name: proto.String("nanos"),
  1093. Type: descriptorpb.FieldDescriptorProto_TYPE_INT32.Enum(),
  1094. Number: proto.Int32(2),
  1095. },
  1096. },
  1097. },
  1098. },
  1099. Message: "ExampleMessage",
  1100. Params: []openapiParameterObject{
  1101. {
  1102. Name: "a_field_mask",
  1103. In: "query",
  1104. Required: false,
  1105. Type: "string",
  1106. },
  1107. {
  1108. Name: "a_timestamp",
  1109. In: "query",
  1110. Required: false,
  1111. Type: "string",
  1112. Format: "date-time",
  1113. },
  1114. },
  1115. },
  1116. }
  1117. for _, test := range tests {
  1118. reg := descriptor.NewRegistry()
  1119. reg.SetEnumsAsInts(true)
  1120. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  1121. ProtoFile: []*descriptorpb.FileDescriptorProto{
  1122. {
  1123. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1124. Name: proto.String("google/well_known.proto"),
  1125. Package: proto.String("google.protobuf"),
  1126. Dependency: []string{},
  1127. MessageType: test.WellKnownMsgDescs,
  1128. Service: []*descriptorpb.ServiceDescriptorProto{},
  1129. Options: &descriptorpb.FileOptions{
  1130. GoPackage: proto.String("google/well_known"),
  1131. },
  1132. },
  1133. {
  1134. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1135. Name: proto.String("acme/example.proto"),
  1136. Package: proto.String("example"),
  1137. Dependency: []string{"google/well_known.proto"},
  1138. MessageType: test.MsgDescs,
  1139. Service: []*descriptorpb.ServiceDescriptorProto{},
  1140. Options: &descriptorpb.FileOptions{
  1141. GoPackage: proto.String("acme/example"),
  1142. },
  1143. },
  1144. },
  1145. })
  1146. if err != nil {
  1147. t.Fatalf("failed to load CodeGeneratorRequest: %v", err)
  1148. }
  1149. message, err := reg.LookupMsg("", ".example."+test.Message)
  1150. if err != nil {
  1151. t.Fatalf("failed to lookup message: %s", err)
  1152. }
  1153. params, err := messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  1154. if err != nil {
  1155. t.Fatalf("failed to convert message to query parameters: %s", err)
  1156. }
  1157. if !reflect.DeepEqual(params, test.Params) {
  1158. t.Errorf("expected %v, got %v", test.Params, params)
  1159. }
  1160. }
  1161. }
  1162. func TestMessageToQueryParametersWithRequiredField(t *testing.T) {
  1163. type test struct {
  1164. MsgDescs []*descriptorpb.DescriptorProto
  1165. Message string
  1166. Params []openapiParameterObject
  1167. }
  1168. messageSchema := &openapi_options.Schema{
  1169. JsonSchema: &openapi_options.JSONSchema{
  1170. Required: []string{"a"},
  1171. },
  1172. }
  1173. messageOption := &descriptorpb.MessageOptions{}
  1174. proto.SetExtension(messageOption, openapi_options.E_Openapiv2Schema, messageSchema)
  1175. fieldSchema := &openapi_options.JSONSchema{Required: []string{"b"}}
  1176. fieldOption := &descriptorpb.FieldOptions{}
  1177. proto.SetExtension(fieldOption, openapi_options.E_Openapiv2Field, fieldSchema)
  1178. // TODO(makdon): is nested field's test case necessary here?
  1179. tests := []test{
  1180. {
  1181. MsgDescs: []*descriptorpb.DescriptorProto{
  1182. {
  1183. Name: proto.String("ExampleMessage"),
  1184. Field: []*descriptorpb.FieldDescriptorProto{
  1185. {
  1186. Name: proto.String("a"),
  1187. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  1188. Number: proto.Int32(1),
  1189. },
  1190. {
  1191. Name: proto.String("b"),
  1192. Type: descriptorpb.FieldDescriptorProto_TYPE_DOUBLE.Enum(),
  1193. Number: proto.Int32(2),
  1194. Options: fieldOption,
  1195. },
  1196. {
  1197. Name: proto.String("c"),
  1198. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  1199. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  1200. Number: proto.Int32(3),
  1201. },
  1202. },
  1203. Options: messageOption,
  1204. },
  1205. },
  1206. Message: "ExampleMessage",
  1207. Params: []openapiParameterObject{
  1208. {
  1209. Name: "a",
  1210. In: "query",
  1211. Required: true,
  1212. Type: "string",
  1213. },
  1214. {
  1215. Name: "b",
  1216. In: "query",
  1217. Required: true,
  1218. Type: "number",
  1219. Format: "double",
  1220. },
  1221. {
  1222. Name: "c",
  1223. In: "query",
  1224. Required: false,
  1225. Type: "array",
  1226. CollectionFormat: "multi",
  1227. },
  1228. },
  1229. },
  1230. }
  1231. for _, test := range tests {
  1232. reg := descriptor.NewRegistry()
  1233. msgs := []*descriptor.Message{}
  1234. for _, msgdesc := range test.MsgDescs {
  1235. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  1236. }
  1237. file := descriptor.File{
  1238. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  1239. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1240. Name: proto.String("example.proto"),
  1241. Package: proto.String("example"),
  1242. Dependency: []string{},
  1243. MessageType: test.MsgDescs,
  1244. Service: []*descriptorpb.ServiceDescriptorProto{},
  1245. Options: &descriptorpb.FileOptions{
  1246. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  1247. },
  1248. },
  1249. GoPkg: descriptor.GoPackage{
  1250. Path: "example.com/path/to/example/example.pb",
  1251. Name: "example_pb",
  1252. },
  1253. Messages: msgs,
  1254. }
  1255. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  1256. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  1257. })
  1258. if err != nil {
  1259. t.Fatalf("failed to load code generator request: %v", err)
  1260. }
  1261. message, err := reg.LookupMsg("", ".example."+test.Message)
  1262. if err != nil {
  1263. t.Fatalf("failed to lookup message: %s", err)
  1264. }
  1265. params, err := messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  1266. if err != nil {
  1267. t.Fatalf("failed to convert message to query parameters: %s", err)
  1268. }
  1269. // avoid checking Items for array types
  1270. for i := range params {
  1271. params[i].Items = nil
  1272. }
  1273. if !reflect.DeepEqual(params, test.Params) {
  1274. t.Errorf("expected %v, got %v", test.Params, params)
  1275. }
  1276. }
  1277. }
  1278. func TestMessageToQueryParametersWithEnumFieldOption(t *testing.T) {
  1279. type test struct {
  1280. MsgDescs []*descriptorpb.DescriptorProto
  1281. Message string
  1282. Params []openapiParameterObject
  1283. }
  1284. fieldSchema := &openapi_options.JSONSchema{Enum: []string{"enum1", "enum2"}}
  1285. fieldOption := &descriptorpb.FieldOptions{}
  1286. proto.SetExtension(fieldOption, openapi_options.E_Openapiv2Field, fieldSchema)
  1287. tests := []test{
  1288. {
  1289. MsgDescs: []*descriptorpb.DescriptorProto{
  1290. {
  1291. Name: proto.String("ExampleMessage"),
  1292. Field: []*descriptorpb.FieldDescriptorProto{
  1293. {
  1294. Name: proto.String("a"),
  1295. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  1296. Number: proto.Int32(1),
  1297. Options: fieldOption,
  1298. },
  1299. {
  1300. Name: proto.String("b"),
  1301. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  1302. Number: proto.Int32(2),
  1303. },
  1304. {
  1305. Name: proto.String("c"),
  1306. Type: descriptorpb.FieldDescriptorProto_TYPE_ENUM.Enum(),
  1307. TypeName: proto.String(".example.ExampleMessage.EnabledEnum"),
  1308. Number: proto.Int32(3),
  1309. },
  1310. {
  1311. Name: proto.String("d"),
  1312. Type: descriptorpb.FieldDescriptorProto_TYPE_ENUM.Enum(),
  1313. TypeName: proto.String(".example.ExampleMessage.EnabledEnum"),
  1314. Number: proto.Int32(4),
  1315. Options: fieldOption,
  1316. },
  1317. },
  1318. EnumType: []*descriptorpb.EnumDescriptorProto{
  1319. {
  1320. Name: proto.String("EnabledEnum"),
  1321. Value: []*descriptorpb.EnumValueDescriptorProto{
  1322. {Name: proto.String("FALSE"), Number: proto.Int32(0)},
  1323. {Name: proto.String("TRUE"), Number: proto.Int32(1)},
  1324. },
  1325. },
  1326. },
  1327. },
  1328. },
  1329. Message: "ExampleMessage",
  1330. Params: []openapiParameterObject{
  1331. {
  1332. Name: "a",
  1333. In: "query",
  1334. Type: "string",
  1335. Enum: []string{"enum1", "enum2"},
  1336. },
  1337. {
  1338. Name: "b",
  1339. In: "query",
  1340. Type: "string",
  1341. },
  1342. {
  1343. Name: "c",
  1344. In: "query",
  1345. Type: "string",
  1346. Enum: []string{"FALSE", "TRUE"},
  1347. Default: "FALSE",
  1348. },
  1349. {
  1350. Name: "d",
  1351. In: "query",
  1352. Type: "string",
  1353. Enum: []string{"FALSE", "TRUE"},
  1354. Default: "FALSE",
  1355. },
  1356. },
  1357. },
  1358. }
  1359. for _, test := range tests {
  1360. reg := descriptor.NewRegistry()
  1361. msgs := []*descriptor.Message{}
  1362. for _, msgdesc := range test.MsgDescs {
  1363. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  1364. }
  1365. file := descriptor.File{
  1366. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  1367. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1368. Name: proto.String("example.proto"),
  1369. Package: proto.String("example"),
  1370. Dependency: []string{},
  1371. MessageType: test.MsgDescs,
  1372. Service: []*descriptorpb.ServiceDescriptorProto{},
  1373. Options: &descriptorpb.FileOptions{
  1374. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  1375. },
  1376. },
  1377. GoPkg: descriptor.GoPackage{
  1378. Path: "example.com/path/to/example/example.pb",
  1379. Name: "example_pb",
  1380. },
  1381. Messages: msgs,
  1382. }
  1383. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  1384. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  1385. })
  1386. if err != nil {
  1387. t.Fatalf("failed to load code generator request: %v", err)
  1388. }
  1389. message, err := reg.LookupMsg("", ".example."+test.Message)
  1390. if err != nil {
  1391. t.Fatalf("failed to lookup message: %s", err)
  1392. }
  1393. params, err := messageToQueryParameters(message, reg, []descriptor.Parameter{}, nil, "")
  1394. if err != nil {
  1395. t.Fatalf("failed to convert message to query parameters: %s", err)
  1396. }
  1397. // avoid checking Items for array types
  1398. for i := range params {
  1399. params[i].Items = nil
  1400. }
  1401. if !reflect.DeepEqual(params, test.Params) {
  1402. t.Errorf("expected %v, got %v", test.Params, params)
  1403. }
  1404. }
  1405. }
  1406. func TestApplyTemplateSimple(t *testing.T) {
  1407. msgdesc := &descriptorpb.DescriptorProto{
  1408. Name: proto.String("ExampleMessage"),
  1409. }
  1410. meth := &descriptorpb.MethodDescriptorProto{
  1411. Name: proto.String("Example"),
  1412. InputType: proto.String("ExampleMessage"),
  1413. OutputType: proto.String("ExampleMessage"),
  1414. }
  1415. svc := &descriptorpb.ServiceDescriptorProto{
  1416. Name: proto.String("ExampleService"),
  1417. Method: []*descriptorpb.MethodDescriptorProto{meth},
  1418. }
  1419. msg := &descriptor.Message{
  1420. DescriptorProto: msgdesc,
  1421. }
  1422. file := descriptor.File{
  1423. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  1424. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1425. Name: proto.String("example.proto"),
  1426. Package: proto.String("example"),
  1427. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  1428. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  1429. Options: &descriptorpb.FileOptions{
  1430. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  1431. },
  1432. },
  1433. GoPkg: descriptor.GoPackage{
  1434. Path: "example.com/path/to/example/example.pb",
  1435. Name: "example_pb",
  1436. },
  1437. Messages: []*descriptor.Message{msg},
  1438. Services: []*descriptor.Service{
  1439. {
  1440. ServiceDescriptorProto: svc,
  1441. Methods: []*descriptor.Method{
  1442. {
  1443. MethodDescriptorProto: meth,
  1444. RequestType: msg,
  1445. ResponseType: msg,
  1446. Bindings: []*descriptor.Binding{
  1447. {
  1448. HTTPMethod: "GET",
  1449. Body: &descriptor.Body{FieldPath: nil},
  1450. PathTmpl: httprule.Template{
  1451. Version: 1,
  1452. OpCodes: []int{0, 0},
  1453. Template: "/v1/echo", // TODO(achew22): Figure out what this should really be
  1454. },
  1455. },
  1456. },
  1457. },
  1458. },
  1459. },
  1460. },
  1461. }
  1462. reg := descriptor.NewRegistry()
  1463. if err := AddErrorDefs(reg); err != nil {
  1464. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  1465. return
  1466. }
  1467. fileCL := crossLinkFixture(&file)
  1468. err := reg.Load(reqFromFile(fileCL))
  1469. if err != nil {
  1470. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  1471. return
  1472. }
  1473. result, err := applyTemplate(param{File: fileCL, reg: reg})
  1474. if err != nil {
  1475. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  1476. return
  1477. }
  1478. if want, is, name := "2.0", result.Swagger, "Swagger"; !reflect.DeepEqual(is, want) {
  1479. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  1480. }
  1481. if want, is, name := "", result.BasePath, "BasePath"; !reflect.DeepEqual(is, want) {
  1482. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  1483. }
  1484. if want, is, name := ([]string)(nil), result.Schemes, "Schemes"; !reflect.DeepEqual(is, want) {
  1485. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  1486. }
  1487. if want, is, name := []string{"application/json"}, result.Consumes, "Consumes"; !reflect.DeepEqual(is, want) {
  1488. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  1489. }
  1490. if want, is, name := []string{"application/json"}, result.Produces, "Produces"; !reflect.DeepEqual(is, want) {
  1491. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  1492. }
  1493. // If there was a failure, print out the input and the json result for debugging.
  1494. if t.Failed() {
  1495. t.Errorf("had: %s", file)
  1496. t.Errorf("got: %s", fmt.Sprint(result))
  1497. }
  1498. }
  1499. func TestApplyTemplateMultiService(t *testing.T) {
  1500. msgdesc := &descriptorpb.DescriptorProto{
  1501. Name: proto.String("ExampleMessage"),
  1502. }
  1503. meth := &descriptorpb.MethodDescriptorProto{
  1504. Name: proto.String("Example"),
  1505. InputType: proto.String("ExampleMessage"),
  1506. OutputType: proto.String("ExampleMessage"),
  1507. }
  1508. // Create two services that have the same method name. We will test that the
  1509. // operation IDs are different
  1510. svc := &descriptorpb.ServiceDescriptorProto{
  1511. Name: proto.String("ExampleService"),
  1512. Method: []*descriptorpb.MethodDescriptorProto{meth},
  1513. }
  1514. svc2 := &descriptorpb.ServiceDescriptorProto{
  1515. Name: proto.String("OtherService"),
  1516. Method: []*descriptorpb.MethodDescriptorProto{meth},
  1517. }
  1518. msg := &descriptor.Message{
  1519. DescriptorProto: msgdesc,
  1520. }
  1521. file := descriptor.File{
  1522. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  1523. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1524. Name: proto.String("example.proto"),
  1525. Package: proto.String("example"),
  1526. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  1527. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  1528. Options: &descriptorpb.FileOptions{
  1529. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  1530. },
  1531. },
  1532. GoPkg: descriptor.GoPackage{
  1533. Path: "example.com/path/to/example/example.pb",
  1534. Name: "example_pb",
  1535. },
  1536. Messages: []*descriptor.Message{msg},
  1537. Services: []*descriptor.Service{
  1538. {
  1539. ServiceDescriptorProto: svc,
  1540. Methods: []*descriptor.Method{
  1541. {
  1542. MethodDescriptorProto: meth,
  1543. RequestType: msg,
  1544. ResponseType: msg,
  1545. Bindings: []*descriptor.Binding{
  1546. {
  1547. HTTPMethod: "GET",
  1548. Body: &descriptor.Body{FieldPath: nil},
  1549. PathTmpl: httprule.Template{
  1550. Version: 1,
  1551. OpCodes: []int{0, 0},
  1552. Template: "/v1/echo",
  1553. },
  1554. },
  1555. },
  1556. },
  1557. },
  1558. },
  1559. {
  1560. ServiceDescriptorProto: svc2,
  1561. Methods: []*descriptor.Method{
  1562. {
  1563. MethodDescriptorProto: meth,
  1564. RequestType: msg,
  1565. ResponseType: msg,
  1566. Bindings: []*descriptor.Binding{
  1567. {
  1568. HTTPMethod: "GET",
  1569. Body: &descriptor.Body{FieldPath: nil},
  1570. PathTmpl: httprule.Template{
  1571. Version: 1,
  1572. OpCodes: []int{0, 0},
  1573. Template: "/v1/ping",
  1574. },
  1575. },
  1576. },
  1577. },
  1578. },
  1579. },
  1580. },
  1581. }
  1582. reg := descriptor.NewRegistry()
  1583. if err := AddErrorDefs(reg); err != nil {
  1584. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  1585. return
  1586. }
  1587. fileCL := crossLinkFixture(&file)
  1588. err := reg.Load(reqFromFile(fileCL))
  1589. if err != nil {
  1590. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  1591. return
  1592. }
  1593. result, err := applyTemplate(param{File: fileCL, reg: reg})
  1594. if err != nil {
  1595. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  1596. return
  1597. }
  1598. // Check that the two services have unique operation IDs even though they
  1599. // have the same method name.
  1600. if want, is := "ExampleService_Example", result.getPathItemObject("/v1/echo").Get.OperationID; !reflect.DeepEqual(is, want) {
  1601. t.Errorf("applyTemplate(%#v).Paths[0].Get.OperationID = %s want to be %s", file, is, want)
  1602. }
  1603. if want, is := "OtherService_Example", result.getPathItemObject("/v1/ping").Get.OperationID; !reflect.DeepEqual(is, want) {
  1604. t.Errorf("applyTemplate(%#v).Paths[0].Get.OperationID = %s want to be %s", file, is, want)
  1605. }
  1606. // If there was a failure, print out the input and the json result for debugging.
  1607. if t.Failed() {
  1608. t.Errorf("had: %s", file)
  1609. t.Errorf("got: %s", fmt.Sprint(result))
  1610. }
  1611. }
  1612. func TestApplyTemplateOpenAPIConfigFromYAML(t *testing.T) {
  1613. msgdesc := &descriptorpb.DescriptorProto{
  1614. Name: proto.String("ExampleMessage"),
  1615. }
  1616. meth := &descriptorpb.MethodDescriptorProto{
  1617. Name: proto.String("Example"),
  1618. InputType: proto.String("ExampleMessage"),
  1619. OutputType: proto.String("ExampleMessage"),
  1620. }
  1621. svc := &descriptorpb.ServiceDescriptorProto{
  1622. Name: proto.String("ExampleService"),
  1623. Method: []*descriptorpb.MethodDescriptorProto{meth},
  1624. }
  1625. msg := &descriptor.Message{
  1626. DescriptorProto: msgdesc,
  1627. }
  1628. file := descriptor.File{
  1629. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  1630. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1631. Name: proto.String("example.proto"),
  1632. Package: proto.String("example"),
  1633. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  1634. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  1635. Options: &descriptorpb.FileOptions{
  1636. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  1637. },
  1638. },
  1639. GoPkg: descriptor.GoPackage{
  1640. Path: "example.com/path/to/example/example.pb",
  1641. Name: "example_pb",
  1642. },
  1643. Messages: []*descriptor.Message{msg},
  1644. Services: []*descriptor.Service{
  1645. {
  1646. ServiceDescriptorProto: svc,
  1647. Methods: []*descriptor.Method{
  1648. {
  1649. MethodDescriptorProto: meth,
  1650. RequestType: msg,
  1651. ResponseType: msg,
  1652. Bindings: []*descriptor.Binding{
  1653. {
  1654. HTTPMethod: "GET",
  1655. Body: &descriptor.Body{FieldPath: nil},
  1656. PathTmpl: httprule.Template{
  1657. Version: 1,
  1658. OpCodes: []int{0, 0},
  1659. Template: "/v1/echo", // TODO(achew22): Figure out what this should really be
  1660. },
  1661. },
  1662. },
  1663. },
  1664. },
  1665. },
  1666. },
  1667. }
  1668. reg := descriptor.NewRegistry()
  1669. if err := AddErrorDefs(reg); err != nil {
  1670. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  1671. return
  1672. }
  1673. fileCL := crossLinkFixture(&file)
  1674. err := reg.Load(reqFromFile(fileCL))
  1675. if err != nil {
  1676. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  1677. return
  1678. }
  1679. openapiOptions := &openapiconfig.OpenAPIOptions{
  1680. Service: []*openapiconfig.OpenAPIServiceOption{
  1681. {
  1682. Service: "example.ExampleService",
  1683. Option: &openapi_options.Tag{
  1684. Description: "ExampleService description",
  1685. ExternalDocs: &openapi_options.ExternalDocumentation{
  1686. Description: "Find out more about ExampleService",
  1687. },
  1688. },
  1689. },
  1690. },
  1691. }
  1692. if err := reg.RegisterOpenAPIOptions(openapiOptions); err != nil {
  1693. t.Errorf("reg.RegisterOpenAPIOptions for Service %#v failed with %v; want success", openapiOptions.Service, err)
  1694. return
  1695. }
  1696. result, err := applyTemplate(param{File: fileCL, reg: reg})
  1697. if err != nil {
  1698. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  1699. return
  1700. }
  1701. if want, is, name := "ExampleService description", result.Tags[0].Description, "Tags[0].Description"; !reflect.DeepEqual(is, want) {
  1702. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  1703. }
  1704. if want, is, name := "Find out more about ExampleService", result.Tags[0].ExternalDocs.Description, "Tags[0].ExternalDocs.Description"; !reflect.DeepEqual(is, want) {
  1705. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  1706. }
  1707. reg.SetDisableServiceTags(true)
  1708. res, err := applyTemplate(param{File: fileCL, reg: reg})
  1709. if err != nil {
  1710. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  1711. return
  1712. }
  1713. if got, want := len(res.Tags), 0; got != want {
  1714. t.Fatalf("len(applyTemplate(%#v).Tags) = %d want to be %d", file, got, want)
  1715. }
  1716. // If there was a failure, print out the input and the json result for debugging.
  1717. if t.Failed() {
  1718. t.Errorf("had: %s", file)
  1719. t.Errorf("got: %s", fmt.Sprint(result))
  1720. }
  1721. }
  1722. func TestApplyTemplateOverrideWithOperation(t *testing.T) {
  1723. newFile := func() *descriptor.File {
  1724. msgdesc := &descriptorpb.DescriptorProto{
  1725. Name: proto.String("ExampleMessage"),
  1726. }
  1727. meth := &descriptorpb.MethodDescriptorProto{
  1728. Name: proto.String("Example"),
  1729. InputType: proto.String("ExampleMessage"),
  1730. OutputType: proto.String("ExampleMessage"),
  1731. Options: &descriptorpb.MethodOptions{},
  1732. }
  1733. svc := &descriptorpb.ServiceDescriptorProto{
  1734. Name: proto.String("ExampleService"),
  1735. Method: []*descriptorpb.MethodDescriptorProto{meth},
  1736. }
  1737. msg := &descriptor.Message{
  1738. DescriptorProto: msgdesc,
  1739. }
  1740. return &descriptor.File{
  1741. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  1742. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1743. Name: proto.String("example.proto"),
  1744. Package: proto.String("example"),
  1745. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  1746. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  1747. Options: &descriptorpb.FileOptions{
  1748. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  1749. },
  1750. },
  1751. GoPkg: descriptor.GoPackage{
  1752. Path: "example.com/path/to/example/example.pb",
  1753. Name: "example_pb",
  1754. },
  1755. Messages: []*descriptor.Message{msg},
  1756. Services: []*descriptor.Service{
  1757. {
  1758. ServiceDescriptorProto: svc,
  1759. Methods: []*descriptor.Method{
  1760. {
  1761. MethodDescriptorProto: meth,
  1762. RequestType: msg,
  1763. ResponseType: msg,
  1764. Bindings: []*descriptor.Binding{
  1765. {
  1766. HTTPMethod: "GET",
  1767. Body: &descriptor.Body{FieldPath: nil},
  1768. PathTmpl: httprule.Template{
  1769. Version: 1,
  1770. OpCodes: []int{0, 0},
  1771. Template: "/v1/echo", // TODO(achew22): Figure out what this should really be
  1772. },
  1773. },
  1774. },
  1775. },
  1776. },
  1777. },
  1778. },
  1779. }
  1780. }
  1781. verifyTemplateFromReq := func(t *testing.T, reg *descriptor.Registry, file *descriptor.File, opts *openapiconfig.OpenAPIOptions) {
  1782. if err := AddErrorDefs(reg); err != nil {
  1783. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  1784. return
  1785. }
  1786. fileCL := crossLinkFixture(file)
  1787. err := reg.Load(reqFromFile(fileCL))
  1788. if err != nil {
  1789. t.Errorf("reg.Load(%#v) failed with %v; want success", *file, err)
  1790. return
  1791. }
  1792. if opts != nil {
  1793. if err := reg.RegisterOpenAPIOptions(opts); err != nil {
  1794. t.Fatalf("failed to register OpenAPI options: %s", err)
  1795. }
  1796. }
  1797. result, err := applyTemplate(param{File: fileCL, reg: reg})
  1798. if err != nil {
  1799. t.Errorf("applyTemplate(%#v) failed with %v; want success", *file, err)
  1800. return
  1801. }
  1802. if want, is := "MyExample", result.getPathItemObject("/v1/echo").Get.OperationID; !reflect.DeepEqual(is, want) {
  1803. t.Errorf("applyTemplate(%#v).Paths[0].Get.OperationID = %s want to be %s", *file, is, want)
  1804. }
  1805. if want, is := []string{"application/xml"}, result.getPathItemObject("/v1/echo").Get.Consumes; !reflect.DeepEqual(is, want) {
  1806. t.Errorf("applyTemplate(%#v).Paths[0].Get.Consumes = %s want to be %s", *file, is, want)
  1807. }
  1808. if want, is := []string{"application/json", "application/xml"}, result.getPathItemObject("/v1/echo").Get.Produces; !reflect.DeepEqual(is, want) {
  1809. t.Errorf("applyTemplate(%#v).Paths[0].Get.Produces = %s want to be %s", *file, is, want)
  1810. }
  1811. // If there was a failure, print out the input and the json result for debugging.
  1812. if t.Failed() {
  1813. t.Errorf("had: %s", *file)
  1814. t.Errorf("got: %s", fmt.Sprint(result))
  1815. }
  1816. }
  1817. openapiOperation := openapi_options.Operation{
  1818. OperationId: "MyExample",
  1819. Consumes: []string{"application/xml"},
  1820. Produces: []string{"application/json", "application/xml"},
  1821. }
  1822. t.Run("verify override via method option", func(t *testing.T) {
  1823. file := newFile()
  1824. proto.SetExtension(proto.Message(file.Services[0].Methods[0].MethodDescriptorProto.Options),
  1825. openapi_options.E_Openapiv2Operation, &openapiOperation)
  1826. reg := descriptor.NewRegistry()
  1827. verifyTemplateFromReq(t, reg, file, nil)
  1828. })
  1829. t.Run("verify override options annotations", func(t *testing.T) {
  1830. file := newFile()
  1831. reg := descriptor.NewRegistry()
  1832. opts := &openapiconfig.OpenAPIOptions{
  1833. Method: []*openapiconfig.OpenAPIMethodOption{
  1834. {
  1835. Method: "example.ExampleService.Example",
  1836. Option: &openapiOperation,
  1837. },
  1838. },
  1839. }
  1840. verifyTemplateFromReq(t, reg, file, opts)
  1841. })
  1842. }
  1843. func TestApplyTemplateExtensions(t *testing.T) {
  1844. newFile := func() *descriptor.File {
  1845. msgdesc := &descriptorpb.DescriptorProto{
  1846. Name: proto.String("ExampleMessage"),
  1847. }
  1848. meth := &descriptorpb.MethodDescriptorProto{
  1849. Name: proto.String("Example"),
  1850. InputType: proto.String("ExampleMessage"),
  1851. OutputType: proto.String("ExampleMessage"),
  1852. Options: &descriptorpb.MethodOptions{},
  1853. }
  1854. svc := &descriptorpb.ServiceDescriptorProto{
  1855. Name: proto.String("ExampleService"),
  1856. Method: []*descriptorpb.MethodDescriptorProto{meth},
  1857. }
  1858. msg := &descriptor.Message{
  1859. DescriptorProto: msgdesc,
  1860. }
  1861. return &descriptor.File{
  1862. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  1863. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  1864. Name: proto.String("example.proto"),
  1865. Package: proto.String("example"),
  1866. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  1867. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  1868. Options: &descriptorpb.FileOptions{
  1869. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  1870. },
  1871. },
  1872. GoPkg: descriptor.GoPackage{
  1873. Path: "example.com/path/to/example/example.pb",
  1874. Name: "example_pb",
  1875. },
  1876. Messages: []*descriptor.Message{msg},
  1877. Services: []*descriptor.Service{
  1878. {
  1879. ServiceDescriptorProto: svc,
  1880. Methods: []*descriptor.Method{
  1881. {
  1882. MethodDescriptorProto: meth,
  1883. RequestType: msg,
  1884. ResponseType: msg,
  1885. Bindings: []*descriptor.Binding{
  1886. {
  1887. HTTPMethod: "GET",
  1888. Body: &descriptor.Body{FieldPath: nil},
  1889. PathTmpl: httprule.Template{
  1890. Version: 1,
  1891. OpCodes: []int{0, 0},
  1892. Template: "/v1/echo", // TODO(achew22): Figure out what this should really be
  1893. },
  1894. },
  1895. },
  1896. },
  1897. },
  1898. },
  1899. },
  1900. }
  1901. }
  1902. swagger := openapi_options.Swagger{
  1903. Info: &openapi_options.Info{
  1904. Title: "test",
  1905. Extensions: map[string]*structpb.Value{
  1906. "x-info-extension": {Kind: &structpb.Value_StringValue{StringValue: "bar"}},
  1907. },
  1908. },
  1909. Extensions: map[string]*structpb.Value{
  1910. "x-foo": {Kind: &structpb.Value_StringValue{StringValue: "bar"}},
  1911. "x-bar": {Kind: &structpb.Value_ListValue{ListValue: &structpb.ListValue{
  1912. Values: []*structpb.Value{{Kind: &structpb.Value_StringValue{StringValue: "baz"}}},
  1913. }}},
  1914. },
  1915. SecurityDefinitions: &openapi_options.SecurityDefinitions{
  1916. Security: map[string]*openapi_options.SecurityScheme{
  1917. "somescheme": {
  1918. Extensions: map[string]*structpb.Value{
  1919. "x-security-baz": {Kind: &structpb.Value_BoolValue{BoolValue: true}},
  1920. },
  1921. },
  1922. },
  1923. },
  1924. Tags: []*openapi_options.Tag{
  1925. {
  1926. Name: "test tag",
  1927. Description: "test tag description",
  1928. Extensions: map[string]*structpb.Value{
  1929. "x-traitTag": {Kind: &structpb.Value_BoolValue{BoolValue: true}},
  1930. },
  1931. },
  1932. },
  1933. }
  1934. openapiOperation := openapi_options.Operation{
  1935. Responses: map[string]*openapi_options.Response{
  1936. "200": {
  1937. Extensions: map[string]*structpb.Value{
  1938. "x-resp-id": {Kind: &structpb.Value_StringValue{StringValue: "resp1000"}},
  1939. },
  1940. },
  1941. },
  1942. Extensions: map[string]*structpb.Value{
  1943. "x-op-foo": {Kind: &structpb.Value_StringValue{StringValue: "baz"}},
  1944. },
  1945. }
  1946. verifyTemplateExtensions := func(t *testing.T, reg *descriptor.Registry, file *descriptor.File,
  1947. opts *openapiconfig.OpenAPIOptions,
  1948. ) {
  1949. if err := AddErrorDefs(reg); err != nil {
  1950. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  1951. return
  1952. }
  1953. fileCL := crossLinkFixture(file)
  1954. err := reg.Load(reqFromFile(fileCL))
  1955. if err != nil {
  1956. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  1957. return
  1958. }
  1959. if opts != nil {
  1960. if err := reg.RegisterOpenAPIOptions(opts); err != nil {
  1961. t.Fatalf("failed to register OpenAPI annotations: %s", err)
  1962. }
  1963. }
  1964. result, err := applyTemplate(param{File: fileCL, reg: reg})
  1965. if err != nil {
  1966. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  1967. return
  1968. }
  1969. if want, is, name := "2.0", result.Swagger, "Swagger"; !reflect.DeepEqual(is, want) {
  1970. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  1971. }
  1972. if got, want := len(result.extensions), 2; got != want {
  1973. t.Fatalf("len(applyTemplate(%#v).Extensions) = %d want to be %d", file, got, want)
  1974. }
  1975. if got, want := result.extensions[0].key, "x-bar"; got != want {
  1976. t.Errorf("applyTemplate(%#v).Extensions[0].key = %s want to be %s", file, got, want)
  1977. }
  1978. if got, want := result.extensions[1].key, "x-foo"; got != want {
  1979. t.Errorf("applyTemplate(%#v).Extensions[1].key = %s want to be %s", file, got, want)
  1980. }
  1981. {
  1982. var got []string
  1983. err = marshaler.Unmarshal(result.extensions[0].value, &got)
  1984. if err != nil {
  1985. t.Fatalf("marshaler.Unmarshal failed: %v", err)
  1986. }
  1987. want := []string{"baz"}
  1988. if diff := cmp.Diff(got, want); diff != "" {
  1989. t.Error(diff)
  1990. }
  1991. }
  1992. {
  1993. var got string
  1994. err = marshaler.Unmarshal(result.extensions[1].value, &got)
  1995. if err != nil {
  1996. t.Fatalf("marshaler.Unmarshal failed: %v", err)
  1997. }
  1998. want := "bar"
  1999. if diff := cmp.Diff(got, want); diff != "" {
  2000. t.Error(diff)
  2001. }
  2002. }
  2003. var scheme openapiSecuritySchemeObject
  2004. for _, v := range result.SecurityDefinitions {
  2005. scheme = v
  2006. }
  2007. if want, is, name := []extension{
  2008. {key: "x-security-baz", value: json.RawMessage("true")},
  2009. }, scheme.extensions, "SecurityScheme.Extensions"; !reflect.DeepEqual(is, want) {
  2010. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  2011. }
  2012. if want, is, name := []extension{
  2013. {key: "x-info-extension", value: json.RawMessage("\"bar\"")},
  2014. }, result.Info.extensions, "Info.Extensions"; !reflect.DeepEqual(is, want) {
  2015. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  2016. }
  2017. var operation *openapiOperationObject
  2018. var response openapiResponseObject
  2019. for _, v := range result.Paths {
  2020. operation = v.PathItemObject.Get
  2021. response = v.PathItemObject.Get.Responses["200"]
  2022. }
  2023. if want, is, name := []extension{
  2024. {key: "x-op-foo", value: json.RawMessage("\"baz\"")},
  2025. }, operation.extensions, "operation.Extensions"; !reflect.DeepEqual(is, want) {
  2026. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  2027. }
  2028. if want, is, name := []extension{
  2029. {key: "x-resp-id", value: json.RawMessage("\"resp1000\"")},
  2030. }, response.extensions, "response.Extensions"; !reflect.DeepEqual(is, want) {
  2031. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  2032. }
  2033. if len(result.Tags) == 0 {
  2034. t.Errorf("No tags found in result")
  2035. return
  2036. }
  2037. tag := result.Tags[0]
  2038. if want, is, name := []extension{
  2039. {key: "x-traitTag", value: json.RawMessage("true")},
  2040. }, tag.extensions, "Tags[0].Extensions"; !reflect.DeepEqual(is, want) {
  2041. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  2042. }
  2043. }
  2044. t.Run("verify template options set via proto options", func(t *testing.T) {
  2045. file := newFile()
  2046. proto.SetExtension(proto.Message(file.FileDescriptorProto.Options), openapi_options.E_Openapiv2Swagger, &swagger)
  2047. proto.SetExtension(proto.Message(file.Services[0].Methods[0].Options), openapi_options.E_Openapiv2Operation, &openapiOperation)
  2048. reg := descriptor.NewRegistry()
  2049. verifyTemplateExtensions(t, reg, file, nil)
  2050. })
  2051. t.Run("verify template options set via annotations", func(t *testing.T) {
  2052. file := newFile()
  2053. opts := &openapiconfig.OpenAPIOptions{
  2054. File: []*openapiconfig.OpenAPIFileOption{
  2055. {
  2056. File: "example.proto",
  2057. Option: &swagger,
  2058. },
  2059. },
  2060. Method: []*openapiconfig.OpenAPIMethodOption{
  2061. {
  2062. Method: "example.ExampleService.Example",
  2063. Option: &openapiOperation,
  2064. },
  2065. },
  2066. }
  2067. reg := descriptor.NewRegistry()
  2068. verifyTemplateExtensions(t, reg, file, opts)
  2069. })
  2070. }
  2071. func TestApplyTemplateHeaders(t *testing.T) {
  2072. newFile := func() *descriptor.File {
  2073. msgdesc := &descriptorpb.DescriptorProto{
  2074. Name: proto.String("ExampleMessage"),
  2075. }
  2076. meth := &descriptorpb.MethodDescriptorProto{
  2077. Name: proto.String("Example"),
  2078. InputType: proto.String("ExampleMessage"),
  2079. OutputType: proto.String("ExampleMessage"),
  2080. Options: &descriptorpb.MethodOptions{},
  2081. }
  2082. svc := &descriptorpb.ServiceDescriptorProto{
  2083. Name: proto.String("ExampleService"),
  2084. Method: []*descriptorpb.MethodDescriptorProto{meth},
  2085. }
  2086. msg := &descriptor.Message{
  2087. DescriptorProto: msgdesc,
  2088. }
  2089. return &descriptor.File{
  2090. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  2091. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  2092. Name: proto.String("example.proto"),
  2093. Package: proto.String("example"),
  2094. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  2095. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  2096. Options: &descriptorpb.FileOptions{
  2097. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  2098. },
  2099. },
  2100. GoPkg: descriptor.GoPackage{
  2101. Path: "example.com/path/to/example/example.pb",
  2102. Name: "example_pb",
  2103. },
  2104. Messages: []*descriptor.Message{msg},
  2105. Services: []*descriptor.Service{
  2106. {
  2107. ServiceDescriptorProto: svc,
  2108. Methods: []*descriptor.Method{
  2109. {
  2110. MethodDescriptorProto: meth,
  2111. RequestType: msg,
  2112. ResponseType: msg,
  2113. Bindings: []*descriptor.Binding{
  2114. {
  2115. HTTPMethod: "GET",
  2116. Body: &descriptor.Body{FieldPath: nil},
  2117. PathTmpl: httprule.Template{
  2118. Version: 1,
  2119. OpCodes: []int{0, 0},
  2120. Template: "/v1/echo", // TODO(achew22): Figure out what this should really be
  2121. },
  2122. },
  2123. },
  2124. },
  2125. },
  2126. },
  2127. },
  2128. }
  2129. }
  2130. openapiOperation := openapi_options.Operation{
  2131. Responses: map[string]*openapi_options.Response{
  2132. "200": {
  2133. Description: "Testing Headers",
  2134. Headers: map[string]*openapi_options.Header{
  2135. "string": {
  2136. Description: "string header description",
  2137. Type: "string",
  2138. Format: "uuid",
  2139. Pattern: "",
  2140. },
  2141. "boolean": {
  2142. Description: "boolean header description",
  2143. Type: "boolean",
  2144. Default: "true",
  2145. Pattern: "^true|false$",
  2146. },
  2147. "integer": {
  2148. Description: "integer header description",
  2149. Type: "integer",
  2150. Default: "0",
  2151. Pattern: "^[0-9]$",
  2152. },
  2153. "number": {
  2154. Description: "number header description",
  2155. Type: "number",
  2156. Default: "1.2",
  2157. Pattern: "^[-+]?[0-9]*\\.?[0-9]+([eE][-+]?[0-9]+)?$",
  2158. },
  2159. },
  2160. },
  2161. },
  2162. }
  2163. verifyTemplateHeaders := func(t *testing.T, reg *descriptor.Registry, file *descriptor.File,
  2164. opts *openapiconfig.OpenAPIOptions,
  2165. ) {
  2166. if err := AddErrorDefs(reg); err != nil {
  2167. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  2168. return
  2169. }
  2170. fileCL := crossLinkFixture(file)
  2171. err := reg.Load(reqFromFile(fileCL))
  2172. if err != nil {
  2173. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  2174. return
  2175. }
  2176. if opts != nil {
  2177. if err := reg.RegisterOpenAPIOptions(opts); err != nil {
  2178. t.Fatalf("failed to register OpenAPI annotations: %s", err)
  2179. }
  2180. }
  2181. result, err := applyTemplate(param{File: fileCL, reg: reg})
  2182. if err != nil {
  2183. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  2184. return
  2185. }
  2186. if want, is, name := "2.0", result.Swagger, "Swagger"; !reflect.DeepEqual(is, want) {
  2187. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  2188. }
  2189. var response openapiResponseObject
  2190. for _, v := range result.Paths {
  2191. response = v.PathItemObject.Get.Responses["200"]
  2192. }
  2193. if want, is, name := []openapiHeadersObject{
  2194. {
  2195. "String": openapiHeaderObject{
  2196. Description: "string header description",
  2197. Type: "string",
  2198. Format: "uuid",
  2199. Pattern: "",
  2200. },
  2201. "Boolean": openapiHeaderObject{
  2202. Description: "boolean header description",
  2203. Type: "boolean",
  2204. Default: RawExample("true"),
  2205. Pattern: "^true|false$",
  2206. },
  2207. "Integer": openapiHeaderObject{
  2208. Description: "integer header description",
  2209. Type: "integer",
  2210. Default: RawExample("0"),
  2211. Pattern: "^[0-9]$",
  2212. },
  2213. "Number": openapiHeaderObject{
  2214. Description: "number header description",
  2215. Type: "number",
  2216. Default: RawExample("1.2"),
  2217. Pattern: "^[-+]?[0-9]*\\.?[0-9]+([eE][-+]?[0-9]+)?$",
  2218. },
  2219. },
  2220. }[0], response.Headers, "response.Headers"; !reflect.DeepEqual(is, want) {
  2221. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  2222. }
  2223. }
  2224. t.Run("verify template options set via proto options", func(t *testing.T) {
  2225. file := newFile()
  2226. proto.SetExtension(proto.Message(file.Services[0].Methods[0].Options), openapi_options.E_Openapiv2Operation, &openapiOperation)
  2227. reg := descriptor.NewRegistry()
  2228. verifyTemplateHeaders(t, reg, file, nil)
  2229. })
  2230. }
  2231. func TestValidateHeaderType(t *testing.T) {
  2232. type test struct {
  2233. Type string
  2234. Format string
  2235. expectedError error
  2236. }
  2237. tests := []test{
  2238. {
  2239. "string",
  2240. "date-time",
  2241. nil,
  2242. },
  2243. {
  2244. "boolean",
  2245. "",
  2246. nil,
  2247. },
  2248. {
  2249. "integer",
  2250. "uint",
  2251. nil,
  2252. },
  2253. {
  2254. "integer",
  2255. "uint8",
  2256. nil,
  2257. },
  2258. {
  2259. "integer",
  2260. "uint16",
  2261. nil,
  2262. },
  2263. {
  2264. "integer",
  2265. "uint32",
  2266. nil,
  2267. },
  2268. {
  2269. "integer",
  2270. "uint64",
  2271. nil,
  2272. },
  2273. {
  2274. "integer",
  2275. "int",
  2276. nil,
  2277. },
  2278. {
  2279. "integer",
  2280. "int8",
  2281. nil,
  2282. },
  2283. {
  2284. "integer",
  2285. "int16",
  2286. nil,
  2287. },
  2288. {
  2289. "integer",
  2290. "int32",
  2291. nil,
  2292. },
  2293. {
  2294. "integer",
  2295. "int64",
  2296. nil,
  2297. },
  2298. {
  2299. "integer",
  2300. "float64",
  2301. errors.New("the provided format \"float64\" is not a valid extension of the type \"integer\""),
  2302. },
  2303. {
  2304. "integer",
  2305. "uuid",
  2306. errors.New("the provided format \"uuid\" is not a valid extension of the type \"integer\""),
  2307. },
  2308. {
  2309. "number",
  2310. "uint",
  2311. nil,
  2312. },
  2313. {
  2314. "number",
  2315. "uint8",
  2316. nil,
  2317. },
  2318. {
  2319. "number",
  2320. "uint16",
  2321. nil,
  2322. },
  2323. {
  2324. "number",
  2325. "uint32",
  2326. nil,
  2327. },
  2328. {
  2329. "number",
  2330. "uint64",
  2331. nil,
  2332. },
  2333. {
  2334. "number",
  2335. "int",
  2336. nil,
  2337. },
  2338. {
  2339. "number",
  2340. "int8",
  2341. nil,
  2342. },
  2343. {
  2344. "number",
  2345. "int16",
  2346. nil,
  2347. },
  2348. {
  2349. "number",
  2350. "int32",
  2351. nil,
  2352. },
  2353. {
  2354. "number",
  2355. "int64",
  2356. nil,
  2357. },
  2358. {
  2359. "number",
  2360. "float",
  2361. nil,
  2362. },
  2363. {
  2364. "number",
  2365. "float32",
  2366. nil,
  2367. },
  2368. {
  2369. "number",
  2370. "float64",
  2371. nil,
  2372. },
  2373. {
  2374. "number",
  2375. "complex64",
  2376. nil,
  2377. },
  2378. {
  2379. "number",
  2380. "complex128",
  2381. nil,
  2382. },
  2383. {
  2384. "number",
  2385. "double",
  2386. nil,
  2387. },
  2388. {
  2389. "number",
  2390. "byte",
  2391. nil,
  2392. },
  2393. {
  2394. "number",
  2395. "rune",
  2396. nil,
  2397. },
  2398. {
  2399. "number",
  2400. "uintptr",
  2401. nil,
  2402. },
  2403. {
  2404. "number",
  2405. "date",
  2406. errors.New("the provided format \"date\" is not a valid extension of the type \"number\""),
  2407. },
  2408. {
  2409. "array",
  2410. "",
  2411. errors.New("the provided header type \"array\" is not supported"),
  2412. },
  2413. {
  2414. "foo",
  2415. "",
  2416. errors.New("the provided header type \"foo\" is not supported"),
  2417. },
  2418. }
  2419. for _, v := range tests {
  2420. err := validateHeaderTypeAndFormat(v.Type, v.Format)
  2421. if v.expectedError == nil {
  2422. if err != nil {
  2423. t.Errorf("unexpected error %v", err)
  2424. }
  2425. } else {
  2426. if err == nil {
  2427. t.Fatal("expected header error not returned")
  2428. }
  2429. if err.Error() != v.expectedError.Error() {
  2430. t.Errorf("expected error malformed, expected %q, got %q", v.expectedError.Error(), err.Error())
  2431. }
  2432. }
  2433. }
  2434. }
  2435. func TestValidateDefaultValueType(t *testing.T) {
  2436. type test struct {
  2437. Type string
  2438. Value string
  2439. Format string
  2440. expectedError error
  2441. }
  2442. tests := []test{
  2443. {
  2444. "string",
  2445. `"string"`,
  2446. "",
  2447. nil,
  2448. },
  2449. {
  2450. "string",
  2451. "\"2012-11-01T22:08:41+00:00\"",
  2452. "date-time",
  2453. nil,
  2454. },
  2455. {
  2456. "string",
  2457. "\"2012-11-01\"",
  2458. "date",
  2459. nil,
  2460. },
  2461. {
  2462. "string",
  2463. "0",
  2464. "",
  2465. errors.New("the provided default value \"0\" does not match provider type \"string\", or is not properly quoted with escaped quotations"),
  2466. },
  2467. {
  2468. "string",
  2469. "false",
  2470. "",
  2471. errors.New("the provided default value \"false\" does not match provider type \"string\", or is not properly quoted with escaped quotations"),
  2472. },
  2473. {
  2474. "boolean",
  2475. "true",
  2476. "",
  2477. nil,
  2478. },
  2479. {
  2480. "boolean",
  2481. "0",
  2482. "",
  2483. errors.New("the provided default value \"0\" does not match provider type \"boolean\""),
  2484. },
  2485. {
  2486. "boolean",
  2487. `"string"`,
  2488. "",
  2489. errors.New("the provided default value \"\\\"string\\\"\" does not match provider type \"boolean\""),
  2490. },
  2491. {
  2492. "number",
  2493. "1.2",
  2494. "",
  2495. nil,
  2496. },
  2497. {
  2498. "number",
  2499. "123",
  2500. "",
  2501. nil,
  2502. },
  2503. {
  2504. "number",
  2505. "nan",
  2506. "",
  2507. errors.New("the provided number \"nan\" is not a valid JSON number"),
  2508. },
  2509. {
  2510. "number",
  2511. "NaN",
  2512. "",
  2513. errors.New("the provided number \"NaN\" is not a valid JSON number"),
  2514. },
  2515. {
  2516. "number",
  2517. "-459.67",
  2518. "",
  2519. nil,
  2520. },
  2521. {
  2522. "number",
  2523. "inf",
  2524. "",
  2525. errors.New("the provided number \"inf\" is not a valid JSON number"),
  2526. },
  2527. {
  2528. "number",
  2529. "infinity",
  2530. "",
  2531. errors.New("the provided number \"infinity\" is not a valid JSON number"),
  2532. },
  2533. {
  2534. "number",
  2535. "Inf",
  2536. "",
  2537. errors.New("the provided number \"Inf\" is not a valid JSON number"),
  2538. },
  2539. {
  2540. "number",
  2541. "Infinity",
  2542. "",
  2543. errors.New("the provided number \"Infinity\" is not a valid JSON number"),
  2544. },
  2545. {
  2546. "number",
  2547. "false",
  2548. "",
  2549. errors.New("the provided default value \"false\" does not match provider type \"number\""),
  2550. },
  2551. {
  2552. "number",
  2553. `"string"`,
  2554. "",
  2555. errors.New("the provided default value \"\\\"string\\\"\" does not match provider type \"number\""),
  2556. },
  2557. {
  2558. "integer",
  2559. "2",
  2560. "",
  2561. nil,
  2562. },
  2563. {
  2564. "integer",
  2565. fmt.Sprint(math.MaxInt32),
  2566. "int32",
  2567. nil,
  2568. },
  2569. {
  2570. "integer",
  2571. fmt.Sprint(int64(math.MaxInt32) + 1),
  2572. "int32",
  2573. errors.New("the provided default value \"2147483648\" does not match provided format \"int32\""),
  2574. },
  2575. {
  2576. "integer",
  2577. fmt.Sprint(int64(math.MaxInt64)),
  2578. "int64",
  2579. nil,
  2580. },
  2581. {
  2582. "integer",
  2583. "9223372036854775808",
  2584. "int64",
  2585. errors.New("the provided default value \"9223372036854775808\" does not match provided format \"int64\""),
  2586. },
  2587. {
  2588. "integer",
  2589. "18446744073709551615",
  2590. "uint64",
  2591. nil,
  2592. },
  2593. {
  2594. "integer",
  2595. "false",
  2596. "",
  2597. errors.New("the provided default value \"false\" does not match provided type \"integer\""),
  2598. },
  2599. {
  2600. "integer",
  2601. "1.2",
  2602. "",
  2603. errors.New("the provided default value \"1.2\" does not match provided type \"integer\""),
  2604. },
  2605. {
  2606. "integer",
  2607. `"string"`,
  2608. "",
  2609. errors.New("the provided default value \"\\\"string\\\"\" does not match provided type \"integer\""),
  2610. },
  2611. }
  2612. for _, v := range tests {
  2613. err := validateDefaultValueTypeAndFormat(v.Type, v.Value, v.Format)
  2614. if v.expectedError == nil {
  2615. if err != nil {
  2616. t.Errorf("unexpected error '%v'", err)
  2617. }
  2618. } else {
  2619. if err == nil {
  2620. t.Error("expected update error not returned")
  2621. }
  2622. if err.Error() != v.expectedError.Error() {
  2623. t.Errorf("expected error malformed, expected %q, got %q", v.expectedError.Error(), err.Error())
  2624. }
  2625. }
  2626. }
  2627. }
  2628. func TestApplyTemplateRequestWithoutClientStreaming(t *testing.T) {
  2629. msgdesc := &descriptorpb.DescriptorProto{
  2630. Name: proto.String("ExampleMessage"),
  2631. Field: []*descriptorpb.FieldDescriptorProto{
  2632. {
  2633. Name: proto.String("nested"),
  2634. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2635. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  2636. TypeName: proto.String("NestedMessage"),
  2637. Number: proto.Int32(1),
  2638. },
  2639. },
  2640. }
  2641. nesteddesc := &descriptorpb.DescriptorProto{
  2642. Name: proto.String("NestedMessage"),
  2643. Field: []*descriptorpb.FieldDescriptorProto{
  2644. {
  2645. Name: proto.String("int32"),
  2646. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2647. Type: descriptorpb.FieldDescriptorProto_TYPE_INT32.Enum(),
  2648. Number: proto.Int32(1),
  2649. },
  2650. {
  2651. Name: proto.String("bool"),
  2652. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2653. Type: descriptorpb.FieldDescriptorProto_TYPE_BOOL.Enum(),
  2654. Number: proto.Int32(2),
  2655. },
  2656. },
  2657. }
  2658. meth := &descriptorpb.MethodDescriptorProto{
  2659. Name: proto.String("Echo"),
  2660. InputType: proto.String("ExampleMessage"),
  2661. OutputType: proto.String("ExampleMessage"),
  2662. ClientStreaming: proto.Bool(false),
  2663. }
  2664. svc := &descriptorpb.ServiceDescriptorProto{
  2665. Name: proto.String("ExampleService"),
  2666. Method: []*descriptorpb.MethodDescriptorProto{meth},
  2667. }
  2668. meth.ServerStreaming = proto.Bool(false)
  2669. msg := &descriptor.Message{
  2670. DescriptorProto: msgdesc,
  2671. }
  2672. nested := &descriptor.Message{
  2673. DescriptorProto: nesteddesc,
  2674. }
  2675. nestedField := &descriptor.Field{
  2676. Message: msg,
  2677. FieldDescriptorProto: msg.GetField()[0],
  2678. }
  2679. intField := &descriptor.Field{
  2680. Message: nested,
  2681. FieldDescriptorProto: nested.GetField()[0],
  2682. }
  2683. file := descriptor.File{
  2684. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  2685. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  2686. Name: proto.String("example.proto"),
  2687. Package: proto.String("example"),
  2688. MessageType: []*descriptorpb.DescriptorProto{msgdesc, nesteddesc},
  2689. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  2690. Options: &descriptorpb.FileOptions{
  2691. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  2692. },
  2693. },
  2694. GoPkg: descriptor.GoPackage{
  2695. Path: "example.com/path/to/example/example.pb",
  2696. Name: "example_pb",
  2697. },
  2698. Messages: []*descriptor.Message{msg, nested},
  2699. Services: []*descriptor.Service{
  2700. {
  2701. ServiceDescriptorProto: svc,
  2702. Methods: []*descriptor.Method{
  2703. {
  2704. MethodDescriptorProto: meth,
  2705. RequestType: msg,
  2706. ResponseType: msg,
  2707. Bindings: []*descriptor.Binding{
  2708. {
  2709. HTTPMethod: "POST",
  2710. PathTmpl: httprule.Template{
  2711. Version: 1,
  2712. OpCodes: []int{0, 0},
  2713. Template: "/v1/echo", // TODO(achew): Figure out what this should really be
  2714. },
  2715. PathParams: []descriptor.Parameter{
  2716. {
  2717. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  2718. {
  2719. Name: "nested",
  2720. Target: nestedField,
  2721. },
  2722. {
  2723. Name: "int32",
  2724. Target: intField,
  2725. },
  2726. }),
  2727. Target: intField,
  2728. },
  2729. },
  2730. Body: &descriptor.Body{
  2731. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  2732. {
  2733. Name: "nested",
  2734. Target: nestedField,
  2735. },
  2736. }),
  2737. },
  2738. },
  2739. },
  2740. },
  2741. },
  2742. },
  2743. },
  2744. }
  2745. reg := descriptor.NewRegistry()
  2746. if err := AddErrorDefs(reg); err != nil {
  2747. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  2748. return
  2749. }
  2750. fmt.Fprintln(os.Stderr, "fd", file.FileDescriptorProto)
  2751. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  2752. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  2753. })
  2754. if err != nil {
  2755. t.Fatalf("failed to load code generator request: %v", err)
  2756. }
  2757. fmt.Fprintln(os.Stderr, "AllFQMNs", reg.GetAllFQMNs())
  2758. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  2759. if err != nil {
  2760. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  2761. return
  2762. }
  2763. if want, got := "2.0", result.Swagger; !reflect.DeepEqual(got, want) {
  2764. t.Errorf("applyTemplate(%#v).Swagger = %s want to be %s", file, got, want)
  2765. }
  2766. if want, got := "", result.BasePath; !reflect.DeepEqual(got, want) {
  2767. t.Errorf("applyTemplate(%#v).BasePath = %s want to be %s", file, got, want)
  2768. }
  2769. if want, got := ([]string)(nil), result.Schemes; !reflect.DeepEqual(got, want) {
  2770. t.Errorf("applyTemplate(%#v).Schemes = %s want to be %s", file, got, want)
  2771. }
  2772. if want, got := []string{"application/json"}, result.Consumes; !reflect.DeepEqual(got, want) {
  2773. t.Errorf("applyTemplate(%#v).Consumes = %s want to be %s", file, got, want)
  2774. }
  2775. if want, got := []string{"application/json"}, result.Produces; !reflect.DeepEqual(got, want) {
  2776. t.Errorf("applyTemplate(%#v).Produces = %s want to be %s", file, got, want)
  2777. }
  2778. // If there was a failure, print out the input and the json result for debugging.
  2779. if t.Failed() {
  2780. t.Errorf("had: %s", file)
  2781. t.Errorf("got: %s", fmt.Sprint(result))
  2782. }
  2783. }
  2784. func TestApplyTemplateRequestWithClientStreaming(t *testing.T) {
  2785. msgdesc := &descriptorpb.DescriptorProto{
  2786. Name: proto.String("ExampleMessage"),
  2787. Field: []*descriptorpb.FieldDescriptorProto{
  2788. {
  2789. Name: proto.String("nested"),
  2790. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2791. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  2792. TypeName: proto.String("NestedMessage"),
  2793. Number: proto.Int32(1),
  2794. },
  2795. },
  2796. }
  2797. nesteddesc := &descriptorpb.DescriptorProto{
  2798. Name: proto.String("NestedMessage"),
  2799. Field: []*descriptorpb.FieldDescriptorProto{
  2800. {
  2801. Name: proto.String("int32"),
  2802. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2803. Type: descriptorpb.FieldDescriptorProto_TYPE_INT32.Enum(),
  2804. Number: proto.Int32(1),
  2805. },
  2806. {
  2807. Name: proto.String("bool"),
  2808. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2809. Type: descriptorpb.FieldDescriptorProto_TYPE_BOOL.Enum(),
  2810. Number: proto.Int32(2),
  2811. },
  2812. },
  2813. }
  2814. meth := &descriptorpb.MethodDescriptorProto{
  2815. Name: proto.String("Echo"),
  2816. InputType: proto.String("ExampleMessage"),
  2817. OutputType: proto.String("ExampleMessage"),
  2818. ClientStreaming: proto.Bool(true),
  2819. ServerStreaming: proto.Bool(true),
  2820. }
  2821. svc := &descriptorpb.ServiceDescriptorProto{
  2822. Name: proto.String("ExampleService"),
  2823. Method: []*descriptorpb.MethodDescriptorProto{meth},
  2824. }
  2825. msg := &descriptor.Message{
  2826. DescriptorProto: msgdesc,
  2827. }
  2828. nested := &descriptor.Message{
  2829. DescriptorProto: nesteddesc,
  2830. }
  2831. nestedField := &descriptor.Field{
  2832. Message: msg,
  2833. FieldDescriptorProto: msg.GetField()[0],
  2834. }
  2835. intField := &descriptor.Field{
  2836. Message: nested,
  2837. FieldDescriptorProto: nested.GetField()[0],
  2838. }
  2839. file := descriptor.File{
  2840. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  2841. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  2842. Name: proto.String("example.proto"),
  2843. Package: proto.String("example"),
  2844. MessageType: []*descriptorpb.DescriptorProto{msgdesc, nesteddesc},
  2845. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  2846. Options: &descriptorpb.FileOptions{
  2847. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  2848. },
  2849. },
  2850. GoPkg: descriptor.GoPackage{
  2851. Path: "example.com/path/to/example/example.pb",
  2852. Name: "example_pb",
  2853. },
  2854. Messages: []*descriptor.Message{msg, nested},
  2855. Services: []*descriptor.Service{
  2856. {
  2857. ServiceDescriptorProto: svc,
  2858. Methods: []*descriptor.Method{
  2859. {
  2860. MethodDescriptorProto: meth,
  2861. RequestType: msg,
  2862. ResponseType: msg,
  2863. Bindings: []*descriptor.Binding{
  2864. {
  2865. HTTPMethod: "POST",
  2866. PathTmpl: httprule.Template{
  2867. Version: 1,
  2868. OpCodes: []int{0, 0},
  2869. Template: "/v1/echo", // TODO(achew): Figure out what this should really be
  2870. },
  2871. PathParams: []descriptor.Parameter{
  2872. {
  2873. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  2874. {
  2875. Name: "nested",
  2876. Target: nestedField,
  2877. },
  2878. {
  2879. Name: "int32",
  2880. Target: intField,
  2881. },
  2882. }),
  2883. Target: intField,
  2884. },
  2885. },
  2886. Body: &descriptor.Body{
  2887. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  2888. {
  2889. Name: "nested",
  2890. Target: nestedField,
  2891. },
  2892. }),
  2893. },
  2894. },
  2895. },
  2896. },
  2897. },
  2898. },
  2899. },
  2900. }
  2901. reg := descriptor.NewRegistry()
  2902. if err := AddErrorDefs(reg); err != nil {
  2903. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  2904. return
  2905. }
  2906. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  2907. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  2908. })
  2909. if err != nil {
  2910. t.Fatalf("failed to load code generator request: %v", err)
  2911. }
  2912. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  2913. if err != nil {
  2914. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  2915. return
  2916. }
  2917. // Only ExampleMessage must be present, not NestedMessage
  2918. if want, got, name := 3, len(result.Definitions), "len(Definitions)"; !reflect.DeepEqual(got, want) {
  2919. t.Errorf("applyTemplate(%#v).%s = %d want to be %d", file, name, got, want)
  2920. }
  2921. if _, ok := result.getPathItemObject("/v1/echo").Post.Responses["200"]; !ok {
  2922. t.Errorf("applyTemplate(%#v).%s = expected 200 response to be defined", file, `result.getPathItemObject("/v1/echo").Post.Responses["200"]`)
  2923. } else {
  2924. if want, got, name := "A successful response.(streaming responses)", result.getPathItemObject("/v1/echo").Post.Responses["200"].Description, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Description`; !reflect.DeepEqual(got, want) {
  2925. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  2926. }
  2927. streamExampleExampleMessage := result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema
  2928. if want, got, name := "object", streamExampleExampleMessage.Type, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema.Type`; !reflect.DeepEqual(got, want) {
  2929. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  2930. }
  2931. if want, got, name := "Stream result of exampleExampleMessage", streamExampleExampleMessage.Title, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema.Title`; !reflect.DeepEqual(got, want) {
  2932. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  2933. }
  2934. streamExampleExampleMessageProperties := *(streamExampleExampleMessage.Properties)
  2935. if want, got, name := 2, len(streamExampleExampleMessageProperties), `len(StreamDefinitions["exampleExampleMessage"].Properties)`; !reflect.DeepEqual(got, want) {
  2936. t.Errorf("applyTemplate(%#v).%s = %d want to be %d", file, name, got, want)
  2937. } else {
  2938. resultProperty := streamExampleExampleMessageProperties[0]
  2939. if want, got, name := "result", resultProperty.Key, `(*(StreamDefinitions["exampleExampleMessage"].Properties))[0].Key`; !reflect.DeepEqual(got, want) {
  2940. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  2941. }
  2942. result := resultProperty.Value.(openapiSchemaObject)
  2943. if want, got, name := "#/definitions/exampleExampleMessage", result.Ref, `((*(StreamDefinitions["exampleExampleMessage"].Properties))[0].Value.(openapiSchemaObject)).Ref`; !reflect.DeepEqual(got, want) {
  2944. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  2945. }
  2946. errorProperty := streamExampleExampleMessageProperties[1]
  2947. if want, got, name := "error", errorProperty.Key, `(*(StreamDefinitions["exampleExampleMessage"].Properties))[0].Key`; !reflect.DeepEqual(got, want) {
  2948. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  2949. }
  2950. err := errorProperty.Value.(openapiSchemaObject)
  2951. if want, got, name := "#/definitions/rpcStatus", err.Ref, `((*(StreamDefinitions["exampleExampleMessage"].Properties))[0].Value.(openapiSchemaObject)).Ref`; !reflect.DeepEqual(got, want) {
  2952. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  2953. }
  2954. }
  2955. }
  2956. // If there was a failure, print out the input and the json result for debugging.
  2957. if t.Failed() {
  2958. t.Errorf("had: %s", file)
  2959. t.Errorf("got: %s", fmt.Sprint(result))
  2960. }
  2961. }
  2962. func TestApplyTemplateRequestWithServerStreamingAndNoStandardErrors(t *testing.T) {
  2963. msgdesc := &descriptorpb.DescriptorProto{
  2964. Name: proto.String("ExampleMessage"),
  2965. Field: []*descriptorpb.FieldDescriptorProto{
  2966. {
  2967. Name: proto.String("nested"),
  2968. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2969. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  2970. TypeName: proto.String("NestedMessage"),
  2971. Number: proto.Int32(1),
  2972. },
  2973. },
  2974. }
  2975. nesteddesc := &descriptorpb.DescriptorProto{
  2976. Name: proto.String("NestedMessage"),
  2977. Field: []*descriptorpb.FieldDescriptorProto{
  2978. {
  2979. Name: proto.String("int32"),
  2980. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2981. Type: descriptorpb.FieldDescriptorProto_TYPE_INT32.Enum(),
  2982. Number: proto.Int32(1),
  2983. },
  2984. {
  2985. Name: proto.String("bool"),
  2986. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  2987. Type: descriptorpb.FieldDescriptorProto_TYPE_BOOL.Enum(),
  2988. Number: proto.Int32(2),
  2989. },
  2990. },
  2991. }
  2992. meth := &descriptorpb.MethodDescriptorProto{
  2993. Name: proto.String("Echo"),
  2994. InputType: proto.String("ExampleMessage"),
  2995. OutputType: proto.String("ExampleMessage"),
  2996. ClientStreaming: proto.Bool(false),
  2997. ServerStreaming: proto.Bool(true),
  2998. }
  2999. svc := &descriptorpb.ServiceDescriptorProto{
  3000. Name: proto.String("ExampleService"),
  3001. Method: []*descriptorpb.MethodDescriptorProto{meth},
  3002. }
  3003. msg := &descriptor.Message{
  3004. DescriptorProto: msgdesc,
  3005. }
  3006. nested := &descriptor.Message{
  3007. DescriptorProto: nesteddesc,
  3008. }
  3009. nestedField := &descriptor.Field{
  3010. Message: msg,
  3011. FieldDescriptorProto: msg.GetField()[0],
  3012. }
  3013. intField := &descriptor.Field{
  3014. Message: nested,
  3015. FieldDescriptorProto: nested.GetField()[0],
  3016. }
  3017. file := descriptor.File{
  3018. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  3019. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  3020. Name: proto.String("example.proto"),
  3021. Package: proto.String("example"),
  3022. MessageType: []*descriptorpb.DescriptorProto{msgdesc, nesteddesc},
  3023. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  3024. Options: &descriptorpb.FileOptions{
  3025. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  3026. },
  3027. },
  3028. GoPkg: descriptor.GoPackage{
  3029. Path: "example.com/path/to/example/example.pb",
  3030. Name: "example_pb",
  3031. },
  3032. Messages: []*descriptor.Message{msg, nested},
  3033. Services: []*descriptor.Service{
  3034. {
  3035. ServiceDescriptorProto: svc,
  3036. Methods: []*descriptor.Method{
  3037. {
  3038. MethodDescriptorProto: meth,
  3039. RequestType: msg,
  3040. ResponseType: msg,
  3041. Bindings: []*descriptor.Binding{
  3042. {
  3043. HTTPMethod: "POST",
  3044. PathTmpl: httprule.Template{
  3045. Version: 1,
  3046. OpCodes: []int{0, 0},
  3047. Template: "/v1/echo",
  3048. },
  3049. PathParams: []descriptor.Parameter{
  3050. {
  3051. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  3052. {
  3053. Name: "nested",
  3054. Target: nestedField,
  3055. },
  3056. {
  3057. Name: "int32",
  3058. Target: intField,
  3059. },
  3060. }),
  3061. Target: intField,
  3062. },
  3063. },
  3064. Body: &descriptor.Body{
  3065. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  3066. {
  3067. Name: "nested",
  3068. Target: nestedField,
  3069. },
  3070. }),
  3071. },
  3072. },
  3073. },
  3074. },
  3075. },
  3076. },
  3077. },
  3078. }
  3079. reg := descriptor.NewRegistry()
  3080. if err := AddErrorDefs(reg); err != nil {
  3081. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  3082. return
  3083. }
  3084. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  3085. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  3086. })
  3087. if err != nil {
  3088. t.Fatalf("failed to load code generator request: %v", err)
  3089. }
  3090. reg.SetDisableDefaultErrors(true)
  3091. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  3092. if err != nil {
  3093. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  3094. return
  3095. }
  3096. // Should only include the message, no status or any type
  3097. if want, got, name := 1, len(result.Definitions), "len(Definitions)"; !reflect.DeepEqual(got, want) {
  3098. t.Errorf("applyTemplate(%#v).%s = %d want to be %d", file, name, got, want)
  3099. }
  3100. if _, ok := result.getPathItemObject("/v1/echo").Post.Responses["200"]; !ok {
  3101. t.Errorf("applyTemplate(%#v).%s = expected 200 response to be defined", file, `result.getPathItemObject("/v1/echo").Post.Responses["200"]`)
  3102. } else {
  3103. if want, got, name := "A successful response.(streaming responses)", result.getPathItemObject("/v1/echo").Post.Responses["200"].Description, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Description`; !reflect.DeepEqual(got, want) {
  3104. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  3105. }
  3106. streamExampleExampleMessage := result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema
  3107. if want, got, name := "object", streamExampleExampleMessage.Type, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema.Type`; !reflect.DeepEqual(got, want) {
  3108. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  3109. }
  3110. if want, got, name := "Stream result of exampleExampleMessage", streamExampleExampleMessage.Title, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema.Title`; !reflect.DeepEqual(got, want) {
  3111. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  3112. }
  3113. streamExampleExampleMessageProperties := *(streamExampleExampleMessage.Properties)
  3114. if want, got, name := 1, len(streamExampleExampleMessageProperties), `len(StreamDefinitions["exampleExampleMessage"].Properties)`; !reflect.DeepEqual(got, want) {
  3115. t.Errorf("applyTemplate(%#v).%s = %d want to be %d", file, name, got, want)
  3116. } else {
  3117. resultProperty := streamExampleExampleMessageProperties[0]
  3118. if want, got, name := "result", resultProperty.Key, `(*(StreamDefinitions["exampleExampleMessage"].Properties))[0].Key`; !reflect.DeepEqual(got, want) {
  3119. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  3120. }
  3121. result := resultProperty.Value.(openapiSchemaObject)
  3122. if want, got, name := "#/definitions/exampleExampleMessage", result.Ref, `((*(StreamDefinitions["exampleExampleMessage"].Properties))[0].Value.(openapiSchemaObject)).Ref`; !reflect.DeepEqual(got, want) {
  3123. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  3124. }
  3125. }
  3126. }
  3127. // If there was a failure, print out the input and the json result for debugging.
  3128. if t.Failed() {
  3129. t.Errorf("had: %s", file)
  3130. t.Errorf("got: %s", fmt.Sprint(result))
  3131. }
  3132. }
  3133. func TestApplyTemplateRequestWithUnusedReferences(t *testing.T) {
  3134. reqdesc := &descriptorpb.DescriptorProto{
  3135. Name: proto.String("ExampleMessage"),
  3136. Field: []*descriptorpb.FieldDescriptorProto{
  3137. {
  3138. Name: proto.String("string"),
  3139. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  3140. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3141. Number: proto.Int32(1),
  3142. },
  3143. },
  3144. }
  3145. respdesc := &descriptorpb.DescriptorProto{
  3146. Name: proto.String("EmptyMessage"),
  3147. }
  3148. meth := &descriptorpb.MethodDescriptorProto{
  3149. Name: proto.String("Example"),
  3150. InputType: proto.String("ExampleMessage"),
  3151. OutputType: proto.String("EmptyMessage"),
  3152. ClientStreaming: proto.Bool(false),
  3153. ServerStreaming: proto.Bool(false),
  3154. }
  3155. svc := &descriptorpb.ServiceDescriptorProto{
  3156. Name: proto.String("ExampleService"),
  3157. Method: []*descriptorpb.MethodDescriptorProto{meth},
  3158. }
  3159. req := &descriptor.Message{
  3160. DescriptorProto: reqdesc,
  3161. }
  3162. resp := &descriptor.Message{
  3163. DescriptorProto: respdesc,
  3164. }
  3165. stringField := &descriptor.Field{
  3166. Message: req,
  3167. FieldDescriptorProto: req.GetField()[0],
  3168. }
  3169. file := descriptor.File{
  3170. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  3171. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  3172. Name: proto.String("example.proto"),
  3173. Package: proto.String("example"),
  3174. MessageType: []*descriptorpb.DescriptorProto{reqdesc, respdesc},
  3175. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  3176. Options: &descriptorpb.FileOptions{
  3177. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  3178. },
  3179. },
  3180. GoPkg: descriptor.GoPackage{
  3181. Path: "example.com/path/to/example/example.pb",
  3182. Name: "example_pb",
  3183. },
  3184. Messages: []*descriptor.Message{req, resp},
  3185. Services: []*descriptor.Service{
  3186. {
  3187. ServiceDescriptorProto: svc,
  3188. Methods: []*descriptor.Method{
  3189. {
  3190. MethodDescriptorProto: meth,
  3191. RequestType: req,
  3192. ResponseType: resp,
  3193. Bindings: []*descriptor.Binding{
  3194. {
  3195. HTTPMethod: "GET",
  3196. PathTmpl: httprule.Template{
  3197. Version: 1,
  3198. OpCodes: []int{0, 0},
  3199. Template: "/v1/example",
  3200. },
  3201. },
  3202. {
  3203. HTTPMethod: "POST",
  3204. PathTmpl: httprule.Template{
  3205. Version: 1,
  3206. OpCodes: []int{0, 0},
  3207. Template: "/v1/example/{string}",
  3208. },
  3209. PathParams: []descriptor.Parameter{
  3210. {
  3211. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  3212. {
  3213. Name: "string",
  3214. Target: stringField,
  3215. },
  3216. }),
  3217. Target: stringField,
  3218. },
  3219. },
  3220. Body: &descriptor.Body{
  3221. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  3222. {
  3223. Name: "string",
  3224. Target: stringField,
  3225. },
  3226. }),
  3227. },
  3228. },
  3229. },
  3230. },
  3231. },
  3232. },
  3233. },
  3234. }
  3235. reg := descriptor.NewRegistry()
  3236. if err := AddErrorDefs(reg); err != nil {
  3237. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  3238. return
  3239. }
  3240. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  3241. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  3242. })
  3243. if err != nil {
  3244. t.Fatalf("failed to load code generator request: %v", err)
  3245. }
  3246. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  3247. if err != nil {
  3248. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  3249. return
  3250. }
  3251. // Only EmptyMessage must be present, not ExampleMessage (plus error status)
  3252. if want, got, name := 3, len(result.Definitions), "len(Definitions)"; !reflect.DeepEqual(got, want) {
  3253. t.Errorf("applyTemplate(%#v).%s = %d want to be %d", file, name, got, want)
  3254. }
  3255. // If there was a failure, print out the input and the json result for debugging.
  3256. if t.Failed() {
  3257. t.Errorf("had: %s", file)
  3258. t.Errorf("got: %s", fmt.Sprint(result))
  3259. }
  3260. }
  3261. func TestApplyTemplateRequestWithBodyQueryParameters(t *testing.T) {
  3262. bookDesc := &descriptorpb.DescriptorProto{
  3263. Name: proto.String("Book"),
  3264. Field: []*descriptorpb.FieldDescriptorProto{
  3265. {
  3266. Name: proto.String("name"),
  3267. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3268. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3269. Number: proto.Int32(1),
  3270. },
  3271. {
  3272. Name: proto.String("id"),
  3273. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3274. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3275. Number: proto.Int32(2),
  3276. },
  3277. },
  3278. }
  3279. createDesc := &descriptorpb.DescriptorProto{
  3280. Name: proto.String("CreateBookRequest"),
  3281. Field: []*descriptorpb.FieldDescriptorProto{
  3282. {
  3283. Name: proto.String("parent"),
  3284. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3285. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3286. Number: proto.Int32(1),
  3287. },
  3288. {
  3289. Name: proto.String("book"),
  3290. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3291. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3292. Number: proto.Int32(2),
  3293. },
  3294. {
  3295. Name: proto.String("book_id"),
  3296. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3297. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3298. Number: proto.Int32(3),
  3299. },
  3300. },
  3301. }
  3302. meth := &descriptorpb.MethodDescriptorProto{
  3303. Name: proto.String("CreateBook"),
  3304. InputType: proto.String("CreateBookRequest"),
  3305. OutputType: proto.String("Book"),
  3306. }
  3307. svc := &descriptorpb.ServiceDescriptorProto{
  3308. Name: proto.String("BookService"),
  3309. Method: []*descriptorpb.MethodDescriptorProto{meth},
  3310. }
  3311. bookMsg := &descriptor.Message{
  3312. DescriptorProto: bookDesc,
  3313. }
  3314. createMsg := &descriptor.Message{
  3315. DescriptorProto: createDesc,
  3316. }
  3317. parentField := &descriptor.Field{
  3318. Message: createMsg,
  3319. FieldDescriptorProto: createMsg.GetField()[0],
  3320. }
  3321. bookField := &descriptor.Field{
  3322. Message: createMsg,
  3323. FieldMessage: bookMsg,
  3324. FieldDescriptorProto: createMsg.GetField()[1],
  3325. }
  3326. bookIDField := &descriptor.Field{
  3327. Message: createMsg,
  3328. FieldDescriptorProto: createMsg.GetField()[2],
  3329. }
  3330. createMsg.Fields = []*descriptor.Field{parentField, bookField, bookIDField}
  3331. newFile := func() descriptor.File {
  3332. return descriptor.File{
  3333. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  3334. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  3335. Name: proto.String("book.proto"),
  3336. MessageType: []*descriptorpb.DescriptorProto{bookDesc, createDesc},
  3337. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  3338. Options: &descriptorpb.FileOptions{
  3339. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  3340. },
  3341. },
  3342. GoPkg: descriptor.GoPackage{
  3343. Path: "example.com/path/to/book.pb",
  3344. Name: "book_pb",
  3345. },
  3346. Messages: []*descriptor.Message{bookMsg, createMsg},
  3347. Services: []*descriptor.Service{
  3348. {
  3349. ServiceDescriptorProto: svc,
  3350. Methods: []*descriptor.Method{
  3351. {
  3352. MethodDescriptorProto: meth,
  3353. RequestType: createMsg,
  3354. ResponseType: bookMsg,
  3355. Bindings: []*descriptor.Binding{
  3356. {
  3357. HTTPMethod: "POST",
  3358. PathTmpl: httprule.Template{
  3359. Version: 1,
  3360. OpCodes: []int{0, 0},
  3361. Template: "/v1/{parent=publishers/*}/books",
  3362. },
  3363. PathParams: []descriptor.Parameter{
  3364. {
  3365. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  3366. {
  3367. Name: "parent",
  3368. Target: parentField,
  3369. },
  3370. }),
  3371. Target: parentField,
  3372. },
  3373. },
  3374. Body: &descriptor.Body{
  3375. FieldPath: []descriptor.FieldPathComponent{
  3376. {
  3377. Name: "book",
  3378. Target: bookField,
  3379. },
  3380. },
  3381. },
  3382. },
  3383. },
  3384. },
  3385. },
  3386. },
  3387. },
  3388. }
  3389. }
  3390. type args struct {
  3391. file descriptor.File
  3392. }
  3393. type paramOut struct {
  3394. Name string
  3395. In string
  3396. Required bool
  3397. }
  3398. tests := []struct {
  3399. name string
  3400. args args
  3401. want []paramOut
  3402. }{
  3403. {
  3404. name: "book_in_body",
  3405. args: args{file: newFile()},
  3406. want: []paramOut{
  3407. {"parent", "path", true},
  3408. {"book", "body", true},
  3409. {"book_id", "query", false},
  3410. },
  3411. },
  3412. {
  3413. name: "book_in_query",
  3414. args: args{file: func() descriptor.File {
  3415. f := newFile()
  3416. f.Services[0].Methods[0].Bindings[0].Body = nil
  3417. return f
  3418. }()},
  3419. want: []paramOut{
  3420. {"parent", "path", true},
  3421. {"book", "query", false},
  3422. {"book_id", "query", false},
  3423. },
  3424. },
  3425. }
  3426. for _, tt := range tests {
  3427. tt := tt
  3428. t.Run(tt.name, func(t *testing.T) {
  3429. reg := descriptor.NewRegistry()
  3430. if err := AddErrorDefs(reg); err != nil {
  3431. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  3432. return
  3433. }
  3434. err := reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{tt.args.file.FileDescriptorProto}})
  3435. if err != nil {
  3436. t.Errorf("Registry.Load() failed with %v; want success", err)
  3437. return
  3438. }
  3439. result, err := applyTemplate(param{File: crossLinkFixture(&tt.args.file), reg: reg})
  3440. if err != nil {
  3441. t.Errorf("applyTemplate(%#v) failed with %v; want success", tt.args.file, err)
  3442. return
  3443. }
  3444. if _, ok := result.getPathItemObject("/v1/{parent}/books").Post.Responses["200"]; !ok {
  3445. t.Errorf("applyTemplate(%#v).%s = expected 200 response to be defined", tt.args.file, `result.getPathItemObject("/v1/{parent}/books").Post.Responses["200"]`)
  3446. } else {
  3447. if want, got, name := 3, len(result.getPathItemObject("/v1/{parent}/books").Post.Parameters), `len(result.getPathItemObject("/v1/{parent}/books").Post.Parameters)`; !reflect.DeepEqual(got, want) {
  3448. t.Errorf("applyTemplate(%#v).%s = %d want to be %d", tt.args.file, name, got, want)
  3449. }
  3450. for i, want := range tt.want {
  3451. p := result.getPathItemObject("/v1/{parent}/books").Post.Parameters[i]
  3452. if got, name := (paramOut{p.Name, p.In, p.Required}), `result.getPathItemObject("/v1/{parent}/books").Post.Parameters[0]`; !reflect.DeepEqual(got, want) {
  3453. t.Errorf("applyTemplate(%#v).%s = %v want to be %v", tt.args.file, name, got, want)
  3454. }
  3455. }
  3456. }
  3457. // If there was a failure, print out the input and the json result for debugging.
  3458. if t.Failed() {
  3459. t.Errorf("had: %s", tt.args.file)
  3460. t.Errorf("got: %s", fmt.Sprint(result))
  3461. }
  3462. })
  3463. }
  3464. }
  3465. func TestApplyTemplateWithRequestAndBodyParameters(t *testing.T) {
  3466. bookDesc := &descriptorpb.DescriptorProto{
  3467. Name: proto.String("Book"),
  3468. Field: []*descriptorpb.FieldDescriptorProto{
  3469. {
  3470. Name: proto.String("name"),
  3471. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3472. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3473. Number: proto.Int32(1),
  3474. },
  3475. {
  3476. Name: proto.String("id"),
  3477. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3478. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3479. Number: proto.Int32(2),
  3480. },
  3481. },
  3482. }
  3483. createDesc := &descriptorpb.DescriptorProto{
  3484. Name: proto.String("CreateBookRequest"),
  3485. Field: []*descriptorpb.FieldDescriptorProto{
  3486. {
  3487. Name: proto.String("parent"),
  3488. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3489. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3490. Number: proto.Int32(1),
  3491. },
  3492. {
  3493. Name: proto.String("book"),
  3494. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3495. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3496. Number: proto.Int32(2),
  3497. },
  3498. {
  3499. Name: proto.String("book_id"),
  3500. Label: descriptorpb.FieldDescriptorProto_LABEL_REQUIRED.Enum(),
  3501. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3502. Number: proto.Int32(3),
  3503. },
  3504. },
  3505. }
  3506. meth := &descriptorpb.MethodDescriptorProto{
  3507. Name: proto.String("CreateBook"),
  3508. InputType: proto.String("CreateBookRequest"),
  3509. OutputType: proto.String("Book"),
  3510. }
  3511. svc := &descriptorpb.ServiceDescriptorProto{
  3512. Name: proto.String("BookService"),
  3513. Method: []*descriptorpb.MethodDescriptorProto{meth},
  3514. }
  3515. bookMsg := &descriptor.Message{
  3516. DescriptorProto: bookDesc,
  3517. }
  3518. createMsg := &descriptor.Message{
  3519. DescriptorProto: createDesc,
  3520. }
  3521. parentField := &descriptor.Field{
  3522. Message: createMsg,
  3523. FieldDescriptorProto: createMsg.GetField()[0],
  3524. }
  3525. bookField := &descriptor.Field{
  3526. Message: createMsg,
  3527. FieldMessage: bookMsg,
  3528. FieldDescriptorProto: createMsg.GetField()[1],
  3529. }
  3530. bookIDField := &descriptor.Field{
  3531. Message: createMsg,
  3532. FieldDescriptorProto: createMsg.GetField()[2],
  3533. }
  3534. createMsg.Fields = []*descriptor.Field{parentField, bookField, bookIDField}
  3535. file := descriptor.File{
  3536. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  3537. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  3538. Name: proto.String("book.proto"),
  3539. MessageType: []*descriptorpb.DescriptorProto{bookDesc, createDesc},
  3540. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  3541. Options: &descriptorpb.FileOptions{
  3542. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  3543. },
  3544. },
  3545. GoPkg: descriptor.GoPackage{
  3546. Path: "example.com/path/to/book.pb",
  3547. Name: "book_pb",
  3548. },
  3549. Messages: []*descriptor.Message{bookMsg, createMsg},
  3550. Services: []*descriptor.Service{
  3551. {
  3552. ServiceDescriptorProto: svc,
  3553. Methods: []*descriptor.Method{
  3554. {
  3555. MethodDescriptorProto: meth,
  3556. RequestType: createMsg,
  3557. ResponseType: bookMsg,
  3558. Bindings: []*descriptor.Binding{
  3559. {
  3560. HTTPMethod: "POST",
  3561. PathTmpl: httprule.Template{
  3562. Version: 1,
  3563. OpCodes: []int{0, 0},
  3564. Template: "/v1/{parent=publishers/*}/books",
  3565. },
  3566. PathParams: []descriptor.Parameter{
  3567. {
  3568. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  3569. {
  3570. Name: "parent",
  3571. Target: parentField,
  3572. },
  3573. }),
  3574. Target: parentField,
  3575. },
  3576. },
  3577. Body: &descriptor.Body{
  3578. FieldPath: []descriptor.FieldPathComponent{},
  3579. },
  3580. },
  3581. },
  3582. },
  3583. },
  3584. },
  3585. },
  3586. }
  3587. reg := descriptor.NewRegistry()
  3588. if err := AddErrorDefs(reg); err != nil {
  3589. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  3590. return
  3591. }
  3592. fileCL := crossLinkFixture(&file)
  3593. err := reg.Load(reqFromFile(fileCL))
  3594. if err != nil {
  3595. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  3596. return
  3597. }
  3598. result, err := applyTemplate(param{File: fileCL, reg: reg})
  3599. if err != nil {
  3600. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  3601. return
  3602. }
  3603. if want, is, name := "2.0", result.Swagger, "Swagger"; !reflect.DeepEqual(is, want) {
  3604. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  3605. }
  3606. if want, is, name := "", result.BasePath, "BasePath"; !reflect.DeepEqual(is, want) {
  3607. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  3608. }
  3609. if want, is, name := ([]string)(nil), result.Schemes, "Schemes"; !reflect.DeepEqual(is, want) {
  3610. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  3611. }
  3612. if want, is, name := []string{"application/json"}, result.Consumes, "Consumes"; !reflect.DeepEqual(is, want) {
  3613. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  3614. }
  3615. if want, is, name := []string{"application/json"}, result.Produces, "Produces"; !reflect.DeepEqual(is, want) {
  3616. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  3617. }
  3618. if want, is, name := 1, len(result.Paths), "len(result.Paths)"; !reflect.DeepEqual(is, want) {
  3619. t.Errorf("%s = %d want to be %d", name, want, is)
  3620. }
  3621. if want, is, name := 4, len(result.Paths[0].PathItemObject.Post.Parameters), "len(result.Paths[0].PathItemObject.Post.Parameters)"; !reflect.DeepEqual(is, want) {
  3622. t.Errorf("%s = %d want to be %d", name, want, is)
  3623. }
  3624. if want, is, name := "#/definitions/BookServiceCreateBookBody", result.Paths[0].PathItemObject.Post.Parameters[1].Schema.schemaCore.Ref, "result.Paths[0].PathItemObject.Post.Parameters[1].Schema.schemaCore.Ref"; !reflect.DeepEqual(is, want) {
  3625. t.Errorf("%s = %s want to be %s", name, want, is)
  3626. }
  3627. _, found := result.Definitions["BookServiceCreateBookBody"]
  3628. if !found {
  3629. t.Error("expecting definition to contain BookServiceCreateBookBody")
  3630. }
  3631. // If there was a failure, print out the input and the json result for debugging.
  3632. if t.Failed() {
  3633. t.Errorf("had: %s", file)
  3634. t.Errorf("got: %s", fmt.Sprint(result))
  3635. }
  3636. }
  3637. // TestApplyTemplateProtobufAny tests that the protobufAny definition is correctly rendered with the @type field and
  3638. // allowing additional properties.
  3639. func TestApplyTemplateProtobufAny(t *testing.T) {
  3640. // checkProtobufAnyFormat verifies the only property should be @type and additional properties are allowed
  3641. checkProtobufAnyFormat := func(t *testing.T, protobufAny openapiSchemaObject) {
  3642. anyPropsJSON, err := protobufAny.Properties.MarshalJSON()
  3643. if err != nil {
  3644. t.Errorf("protobufAny.Properties.MarshalJSON(), got error = %v", err)
  3645. }
  3646. var anyPropsMap map[string]interface{}
  3647. if err := json.Unmarshal(anyPropsJSON, &anyPropsMap); err != nil {
  3648. t.Errorf("json.Unmarshal(), got error = %v", err)
  3649. }
  3650. // @type should exist
  3651. if _, ok := anyPropsMap["@type"]; !ok {
  3652. t.Errorf("protobufAny.Properties missing key, \"@type\". got = %#v", anyPropsMap)
  3653. }
  3654. // and @type should be the only property
  3655. if len(anyPropsMap) > 1 {
  3656. t.Errorf("len(protobufAny.Properties) = %v, want = %v", len(anyPropsMap), 1)
  3657. }
  3658. // protobufAny should have additionalProperties allowed
  3659. if protobufAny.AdditionalProperties == nil {
  3660. t.Errorf("protobufAny.AdditionalProperties = nil, want not-nil")
  3661. }
  3662. }
  3663. type args struct {
  3664. regConfig func(registry *descriptor.Registry)
  3665. msgContainsAny bool
  3666. }
  3667. tests := []struct {
  3668. name string
  3669. args args
  3670. wantNumDefinitions int
  3671. }{
  3672. {
  3673. // our proto schema doesn't directly use protobufAny, but it is implicitly used by rpcStatus being
  3674. // automatically rendered
  3675. name: "default_protobufAny_from_rpcStatus",
  3676. args: args{
  3677. msgContainsAny: false,
  3678. },
  3679. wantNumDefinitions: 4,
  3680. },
  3681. {
  3682. // we have a protobufAny in a message, it should contain a ref inside the custom message
  3683. name: "protobufAny_referenced_in_message",
  3684. args: args{
  3685. msgContainsAny: true,
  3686. },
  3687. wantNumDefinitions: 4,
  3688. },
  3689. {
  3690. // we have a protobufAny in a message but with automatic rendering of rpcStatus disabled
  3691. name: "protobufAny_referenced_in_message_with_default_errors_disabled",
  3692. args: args{
  3693. msgContainsAny: true,
  3694. regConfig: func(reg *descriptor.Registry) {
  3695. reg.SetDisableDefaultErrors(true)
  3696. },
  3697. },
  3698. wantNumDefinitions: 3,
  3699. },
  3700. {
  3701. // we have a protobufAny in a message but with automatic rendering of responses disabled
  3702. name: "protobufAny_referenced_in_message_with_default_responses_disabled",
  3703. args: args{
  3704. msgContainsAny: true,
  3705. regConfig: func(reg *descriptor.Registry) {
  3706. reg.SetDisableDefaultResponses(true)
  3707. },
  3708. },
  3709. wantNumDefinitions: 4,
  3710. },
  3711. }
  3712. for _, tt := range tests {
  3713. t.Run(tt.name, func(t *testing.T) {
  3714. reqdesc := &descriptorpb.DescriptorProto{
  3715. Name: proto.String("ExampleMessage"),
  3716. Field: []*descriptorpb.FieldDescriptorProto{
  3717. {
  3718. Name: proto.String("name"),
  3719. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  3720. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3721. Number: proto.Int32(1),
  3722. },
  3723. },
  3724. }
  3725. respdesc := &descriptorpb.DescriptorProto{
  3726. Name: proto.String("EmptyMessage"),
  3727. }
  3728. meth := &descriptorpb.MethodDescriptorProto{
  3729. Name: proto.String("Example"),
  3730. InputType: proto.String("ExampleMessage"),
  3731. OutputType: proto.String("EmptyMessage"),
  3732. ClientStreaming: proto.Bool(false),
  3733. ServerStreaming: proto.Bool(false),
  3734. }
  3735. svc := &descriptorpb.ServiceDescriptorProto{
  3736. Name: proto.String("ExampleService"),
  3737. Method: []*descriptorpb.MethodDescriptorProto{meth},
  3738. }
  3739. req := &descriptor.Message{
  3740. DescriptorProto: reqdesc,
  3741. }
  3742. resp := &descriptor.Message{
  3743. DescriptorProto: respdesc,
  3744. }
  3745. file := descriptor.File{
  3746. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  3747. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  3748. Name: proto.String("example.proto"),
  3749. Package: proto.String("example"),
  3750. MessageType: []*descriptorpb.DescriptorProto{reqdesc, respdesc},
  3751. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  3752. Options: &descriptorpb.FileOptions{
  3753. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  3754. },
  3755. },
  3756. GoPkg: descriptor.GoPackage{
  3757. Path: "example.com/path/to/example/example.pb",
  3758. Name: "example_pb",
  3759. },
  3760. Messages: []*descriptor.Message{req, resp},
  3761. Services: []*descriptor.Service{
  3762. {
  3763. ServiceDescriptorProto: svc,
  3764. Methods: []*descriptor.Method{
  3765. {
  3766. MethodDescriptorProto: meth,
  3767. RequestType: req,
  3768. ResponseType: resp,
  3769. },
  3770. },
  3771. },
  3772. },
  3773. }
  3774. reg := descriptor.NewRegistry()
  3775. reg.SetGenerateUnboundMethods(true)
  3776. if tt.args.regConfig != nil {
  3777. tt.args.regConfig(reg)
  3778. }
  3779. if err := AddErrorDefs(reg); err != nil {
  3780. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  3781. return
  3782. }
  3783. protoFiles := []*descriptorpb.FileDescriptorProto{
  3784. file.FileDescriptorProto,
  3785. }
  3786. if tt.args.msgContainsAny {
  3787. // add an Any field to the request message
  3788. reqdesc.Field = append(reqdesc.Field, &descriptorpb.FieldDescriptorProto{
  3789. Name: proto.String("any_value"),
  3790. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  3791. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  3792. TypeName: proto.String(".google.protobuf.Any"),
  3793. Number: proto.Int32(2),
  3794. })
  3795. // update the dependencies to import it
  3796. file.Dependency = append(file.Dependency, "google/protobuf/any.proto")
  3797. anyDescriptorProto := protodesc.ToFileDescriptorProto((&anypb.Any{}).ProtoReflect().Descriptor().ParentFile())
  3798. anyDescriptorProto.SourceCodeInfo = &descriptorpb.SourceCodeInfo{}
  3799. // prepend the anyDescriptorProto to the protoFiles slice so that the dependency can be resolved
  3800. protoFiles = append(append(make([]*descriptorpb.FileDescriptorProto, 0, len(protoFiles)+1), anyDescriptorProto), protoFiles[0:]...)
  3801. }
  3802. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  3803. ProtoFile: protoFiles,
  3804. FileToGenerate: []string{file.GetName()},
  3805. })
  3806. if err != nil {
  3807. t.Fatalf("failed to load code generator request: %v", err)
  3808. }
  3809. target, err := reg.LookupFile(file.GetName())
  3810. if err != nil {
  3811. t.Fatalf("failed to lookup file from reg: %v", err)
  3812. }
  3813. result, err := applyTemplate(param{File: crossLinkFixture(target), reg: reg})
  3814. if err != nil {
  3815. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  3816. return
  3817. }
  3818. if want, got, name := tt.wantNumDefinitions, len(result.Definitions), "len(Definitions)"; !reflect.DeepEqual(got, want) {
  3819. t.Errorf("applyTemplate(%#v).%s = %d want to be %d", file, name, got, want)
  3820. }
  3821. protobufAny, ok := result.Definitions["protobufAny"]
  3822. if !ok {
  3823. t.Error("expecting Definitions to contain protobufAny")
  3824. }
  3825. checkProtobufAnyFormat(t, protobufAny)
  3826. // If there was a failure, print out the input and the json result for debugging.
  3827. if t.Failed() {
  3828. t.Errorf("had: %s", file)
  3829. resultJSON, _ := json.Marshal(result)
  3830. t.Errorf("got: %s", resultJSON)
  3831. }
  3832. })
  3833. }
  3834. }
  3835. func generateFieldsForJSONReservedName() []*descriptor.Field {
  3836. fields := make([]*descriptor.Field, 0)
  3837. fieldName := "json_name"
  3838. fieldJSONName := "jsonNAME"
  3839. fieldDescriptor := descriptorpb.FieldDescriptorProto{Name: &fieldName, JsonName: &fieldJSONName}
  3840. field := &descriptor.Field{FieldDescriptorProto: &fieldDescriptor}
  3841. return append(fields, field)
  3842. }
  3843. func generateMsgsForJSONReservedName() []*descriptor.Message {
  3844. result := make([]*descriptor.Message, 0)
  3845. // The first message, its field is field_abc and its type is NewType
  3846. // NewType field_abc
  3847. fieldName := "field_abc"
  3848. fieldJSONName := "fieldAbc"
  3849. messageName1 := "message1"
  3850. messageType := "pkg.a.NewType"
  3851. pfd := descriptorpb.FieldDescriptorProto{Name: &fieldName, JsonName: &fieldJSONName, TypeName: &messageType}
  3852. result = append(result,
  3853. &descriptor.Message{
  3854. DescriptorProto: &descriptorpb.DescriptorProto{
  3855. Name: &messageName1, Field: []*descriptorpb.FieldDescriptorProto{&pfd},
  3856. },
  3857. })
  3858. // The second message, its name is NewName, its type is string
  3859. // message NewType {
  3860. // string field_newName [json_name = RESERVEDJSONNAME]
  3861. // }
  3862. messageName := "NewType"
  3863. field := "field_newName"
  3864. fieldJSONName2 := "RESERVEDJSONNAME"
  3865. pfd2 := descriptorpb.FieldDescriptorProto{Name: &field, JsonName: &fieldJSONName2}
  3866. result = append(result, &descriptor.Message{
  3867. DescriptorProto: &descriptorpb.DescriptorProto{
  3868. Name: &messageName, Field: []*descriptorpb.FieldDescriptorProto{&pfd2},
  3869. },
  3870. })
  3871. return result
  3872. }
  3873. func TestTemplateWithJsonCamelCase(t *testing.T) {
  3874. tests := []struct {
  3875. input string
  3876. expected string
  3877. }{
  3878. {"/test/{test_id}", "/test/{testId}"},
  3879. {"/test1/{test1_id}/test2/{test2_id}", "/test1/{test1Id}/test2/{test2Id}"},
  3880. {"/test1/{test1_id}/{test2_id}", "/test1/{test1Id}/{test2Id}"},
  3881. {"/test1/test2/{test1_id}/{test2_id}", "/test1/test2/{test1Id}/{test2Id}"},
  3882. {"/test1/{test1_id1_id2}", "/test1/{test1Id1Id2}"},
  3883. {"/test1/{test1_id1_id2}/test2/{test2_id3_id4}", "/test1/{test1Id1Id2}/test2/{test2Id3Id4}"},
  3884. {"/test1/test2/{test1_id1_id2}/{test2_id3_id4}", "/test1/test2/{test1Id1Id2}/{test2Id3Id4}"},
  3885. {"test/{a}", "test/{a}"},
  3886. {"test/{ab}", "test/{ab}"},
  3887. {"test/{a_a}", "test/{aA}"},
  3888. {"test/{ab_c}", "test/{abC}"},
  3889. {"test/{json_name}", "test/{jsonNAME}"},
  3890. {"test/{field_abc.field_newName}", "test/{fieldAbc.RESERVEDJSONNAME}"},
  3891. }
  3892. reg := descriptor.NewRegistry()
  3893. reg.SetUseJSONNamesForFields(true)
  3894. for _, data := range tests {
  3895. actual := templateToOpenAPIPath(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), make(map[string]string))
  3896. if data.expected != actual {
  3897. t.Errorf("Expected templateToOpenAPIPath(%v) = %v, actual: %v", data.input, data.expected, actual)
  3898. }
  3899. }
  3900. }
  3901. func TestTemplateWithoutJsonCamelCase(t *testing.T) {
  3902. tests := []struct {
  3903. input string
  3904. expected string
  3905. }{
  3906. {"/test/{test_id}", "/test/{test_id}"},
  3907. {"/test1/{test1_id}/test2/{test2_id}", "/test1/{test1_id}/test2/{test2_id}"},
  3908. {"/test1/{test1_id}/{test2_id}", "/test1/{test1_id}/{test2_id}"},
  3909. {"/test1/test2/{test1_id}/{test2_id}", "/test1/test2/{test1_id}/{test2_id}"},
  3910. {"/test1/{test1_id1_id2}", "/test1/{test1_id1_id2}"},
  3911. {"/test1/{test1_id1_id2}/test2/{test2_id3_id4}", "/test1/{test1_id1_id2}/test2/{test2_id3_id4}"},
  3912. {"/test1/test2/{test1_id1_id2}/{test2_id3_id4}", "/test1/test2/{test1_id1_id2}/{test2_id3_id4}"},
  3913. {"test/{a}", "test/{a}"},
  3914. {"test/{ab}", "test/{ab}"},
  3915. {"test/{a_a}", "test/{a_a}"},
  3916. {"test/{json_name}", "test/{json_name}"},
  3917. {"test/{field_abc.field_newName}", "test/{field_abc.field_newName}"},
  3918. }
  3919. reg := descriptor.NewRegistry()
  3920. reg.SetUseJSONNamesForFields(false)
  3921. for _, data := range tests {
  3922. actual := templateToOpenAPIPath(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), make(map[string]string))
  3923. if data.expected != actual {
  3924. t.Errorf("Expected templateToOpenAPIPath(%v) = %v, actual: %v", data.input, data.expected, actual)
  3925. }
  3926. }
  3927. }
  3928. func TestTemplateToOpenAPIPath(t *testing.T) {
  3929. tests := []struct {
  3930. input string
  3931. expected string
  3932. }{
  3933. {"/test", "/test"},
  3934. {"/{test}", "/{test}"},
  3935. {"/{test=prefix/*}", "/{test}"},
  3936. {"/{test=prefix/that/has/multiple/parts/to/it/*}", "/{test}"},
  3937. {"/{test1}/{test2}", "/{test1}/{test2}"},
  3938. {"/{test1}/{test2}/", "/{test1}/{test2}/"},
  3939. {"/{name=prefix/*}", "/{name}"},
  3940. {"/{name=prefix1/*/prefix2/*}", "/{name}"},
  3941. {"/{user.name=prefix/*}", "/{user.name}"},
  3942. {"/{user.name=prefix1/*/prefix2/*}", "/{user.name}"},
  3943. {"/{parent=prefix/*}/children", "/{parent}/children"},
  3944. {"/{name=prefix/*}:customMethod", "/{name}:customMethod"},
  3945. {"/{name=prefix1/*/prefix2/*}:customMethod", "/{name}:customMethod"},
  3946. {"/{user.name=prefix/*}:customMethod", "/{user.name}:customMethod"},
  3947. {"/{user.name=prefix1/*/prefix2/*}:customMethod", "/{user.name}:customMethod"},
  3948. {"/{parent=prefix/*}/children:customMethod", "/{parent}/children:customMethod"},
  3949. }
  3950. reg := descriptor.NewRegistry()
  3951. reg.SetUseJSONNamesForFields(false)
  3952. for _, data := range tests {
  3953. actual := templateToOpenAPIPath(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), make(map[string]string))
  3954. if data.expected != actual {
  3955. t.Errorf("Expected templateToOpenAPIPath(%v) = %v, actual: %v", data.input, data.expected, actual)
  3956. }
  3957. }
  3958. reg.SetUseJSONNamesForFields(true)
  3959. for _, data := range tests {
  3960. actual := templateToOpenAPIPath(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), make(map[string]string))
  3961. if data.expected != actual {
  3962. t.Errorf("Expected templateToOpenAPIPath(%v) = %v, actual: %v", data.input, data.expected, actual)
  3963. }
  3964. }
  3965. }
  3966. func getParameters(names []string) []descriptor.Parameter {
  3967. params := make([]descriptor.Parameter, 0)
  3968. for _, name := range names {
  3969. params = append(params, descriptor.Parameter{
  3970. Target: &descriptor.Field{
  3971. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  3972. Name: proto.String(name),
  3973. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  3974. },
  3975. Message: &descriptor.Message{
  3976. File: &descriptor.File{
  3977. FileDescriptorProto: &descriptorpb.FileDescriptorProto{},
  3978. },
  3979. DescriptorProto: &descriptorpb.DescriptorProto{
  3980. Name: proto.String(""),
  3981. },
  3982. },
  3983. FieldMessage: nil,
  3984. ForcePrefixedName: false,
  3985. },
  3986. FieldPath: []descriptor.FieldPathComponent{{
  3987. Name: name,
  3988. Target: nil,
  3989. }},
  3990. Method: nil,
  3991. })
  3992. }
  3993. return params
  3994. }
  3995. func TestTemplateToOpenAPIPathExpandSlashed(t *testing.T) {
  3996. tests := []struct {
  3997. input string
  3998. expected string
  3999. pathParams []descriptor.Parameter
  4000. expectedPathParams []string
  4001. useJSONNames bool
  4002. }{
  4003. {"/v1/{name=projects/*/documents/*}:exportResults", "/v1/projects/{project}/documents/{document}:exportResults", getParameters([]string{"name"}), []string{"project", "document"}, true},
  4004. {"/test/{name=*}", "/test/{name}", getParameters([]string{"name"}), []string{"name"}, true},
  4005. {"/test/{name=*}/", "/test/{name}/", getParameters([]string{"name"}), []string{"name"}, true},
  4006. {"/test/{name=test_cases/*}/", "/test/test_cases/{testCase}/", getParameters([]string{"name"}), []string{"testCase"}, true},
  4007. {"/test/{name=test_cases/*}/", "/test/test_cases/{test_case}/", getParameters([]string{"name"}), []string{"test_case"}, false},
  4008. {"/test/{test_type.name=test_cases/*}/", "/test/test_cases/{testCase}/", getParameters([]string{"test_type.name"}), []string{"testCase"}, true},
  4009. {"/test/{test_type.name=test_cases/*}/", "/test/test_cases/{test_case}/", getParameters([]string{"test_type.name"}), []string{"test_case"}, false},
  4010. }
  4011. reg := descriptor.NewRegistry()
  4012. reg.SetExpandSlashedPathPatterns(true)
  4013. for _, data := range tests {
  4014. reg.SetUseJSONNamesForFields(data.useJSONNames)
  4015. actualParts, actualParams := templateToExpandedPath(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), data.pathParams)
  4016. if data.expected != actualParts {
  4017. t.Errorf("Expected templateToOpenAPIPath(%v) = %v, actual: %v", data.input, data.expected, actualParts)
  4018. }
  4019. pathParamsNames := make([]string, 0)
  4020. for _, param := range actualParams {
  4021. pathParamsNames = append(pathParamsNames, param.FieldPath[0].Name)
  4022. }
  4023. if !reflect.DeepEqual(data.expectedPathParams, pathParamsNames) {
  4024. t.Errorf("Expected mutated path params in templateToOpenAPIPath(%v) = %v, actual: %v", data.input, data.expectedPathParams, data.pathParams)
  4025. }
  4026. }
  4027. }
  4028. func TestExpandedPathParametersStringType(t *testing.T) {
  4029. tests := []struct {
  4030. input string
  4031. }{
  4032. {"/test/{name=test_cases/*}/"}, {"/v1/{name=projects/*/documents/*}:exportResults"},
  4033. }
  4034. reg := descriptor.NewRegistry()
  4035. reg.SetExpandSlashedPathPatterns(true)
  4036. expectedParamType := openapiSchemaObject{
  4037. schemaCore: schemaCore{
  4038. Type: "string",
  4039. },
  4040. }
  4041. for _, data := range tests {
  4042. _, actualParams := templateToExpandedPath(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), getParameters([]string{"name"}))
  4043. for _, param := range actualParams {
  4044. refs := make(refMap)
  4045. actualParamType := schemaOfField(param.Target, reg, refs)
  4046. if !reflect.DeepEqual(actualParamType, expectedParamType) {
  4047. t.Errorf("Expected all path parameters to be type of 'string', actual: %#+v", actualParamType)
  4048. }
  4049. }
  4050. }
  4051. }
  4052. func BenchmarkTemplateToOpenAPIPath(b *testing.B) {
  4053. const input = "/{user.name=prefix1/*/prefix2/*}:customMethod"
  4054. b.Run("with JSON names", func(b *testing.B) {
  4055. reg := descriptor.NewRegistry()
  4056. reg.SetUseJSONNamesForFields(false)
  4057. for i := 0; i < b.N; i++ {
  4058. _ = templateToOpenAPIPath(input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), make(map[string]string))
  4059. }
  4060. })
  4061. b.Run("without JSON names", func(b *testing.B) {
  4062. reg := descriptor.NewRegistry()
  4063. reg.SetUseJSONNamesForFields(true)
  4064. for i := 0; i < b.N; i++ {
  4065. _ = templateToOpenAPIPath(input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), make(map[string]string))
  4066. }
  4067. })
  4068. }
  4069. func TestResolveFullyQualifiedNameToOpenAPIName(t *testing.T) {
  4070. tests := []struct {
  4071. input string
  4072. output string
  4073. listOfFQMNs []string
  4074. namingStrategy string
  4075. }{
  4076. {
  4077. ".a.b.C",
  4078. "C",
  4079. []string{
  4080. ".a.b.C",
  4081. },
  4082. "legacy",
  4083. },
  4084. {
  4085. ".a.b.C",
  4086. "C",
  4087. []string{
  4088. ".a.b.C",
  4089. },
  4090. "simple",
  4091. },
  4092. {
  4093. ".a.b.C",
  4094. "abC",
  4095. []string{
  4096. ".a.C",
  4097. ".a.b.C",
  4098. },
  4099. "legacy",
  4100. },
  4101. {
  4102. ".a.b.C",
  4103. "b.C",
  4104. []string{
  4105. ".a.C",
  4106. ".a.b.C",
  4107. },
  4108. "simple",
  4109. },
  4110. {
  4111. ".a.b.C",
  4112. "abC",
  4113. []string{
  4114. ".C",
  4115. ".a.C",
  4116. ".a.b.C",
  4117. },
  4118. "legacy",
  4119. },
  4120. {
  4121. ".a.b.C",
  4122. "b.C",
  4123. []string{
  4124. ".C",
  4125. ".a.C",
  4126. ".a.b.C",
  4127. },
  4128. "simple",
  4129. },
  4130. {
  4131. ".a.b.C",
  4132. "a.b.C",
  4133. []string{
  4134. ".C",
  4135. ".a.C",
  4136. ".a.b.C",
  4137. },
  4138. "fqn",
  4139. },
  4140. }
  4141. for _, data := range tests {
  4142. names := resolveFullyQualifiedNameToOpenAPINames(data.listOfFQMNs, data.namingStrategy)
  4143. output := names[data.input]
  4144. if output != data.output {
  4145. t.Errorf("Expected fullyQualifiedNameToOpenAPIName(%v, %s) to be %s but got %s",
  4146. data.input, data.namingStrategy, data.output, output)
  4147. }
  4148. }
  4149. }
  4150. func templateToOpenAPIPath(path string, reg *descriptor.Registry, fields []*descriptor.Field, msgs []*descriptor.Message, pathParamNames map[string]string) string {
  4151. return partsToOpenAPIPath(templateToParts(path, reg, fields, msgs), pathParamNames)
  4152. }
  4153. func templateToRegexpMap(path string, reg *descriptor.Registry, fields []*descriptor.Field, msgs []*descriptor.Message) map[string]string {
  4154. return partsToRegexpMap(templateToParts(path, reg, fields, msgs))
  4155. }
  4156. func templateToExpandedPath(path string, reg *descriptor.Registry, fields []*descriptor.Field, msgs []*descriptor.Message, pathParams []descriptor.Parameter) (string, []descriptor.Parameter) {
  4157. pathParts, pathParams := expandPathPatterns(templateToParts(path, reg, fields, msgs), pathParams, reg)
  4158. return partsToOpenAPIPath(pathParts, make(map[string]string)), pathParams
  4159. }
  4160. func TestFQMNToRegexpMap(t *testing.T) {
  4161. tests := []struct {
  4162. input string
  4163. expected map[string]string
  4164. }{
  4165. {"/test", map[string]string{}},
  4166. {"/{test}", map[string]string{}},
  4167. {"/{test" + pathParamUniqueSuffixDeliminator + "1=prefix/*}", map[string]string{"test" + pathParamUniqueSuffixDeliminator + "1": "prefix/[^/]+"}},
  4168. {"/{test=prefix/that/has/multiple/parts/to/it/**}", map[string]string{"test": "prefix/that/has/multiple/parts/to/it/.+"}},
  4169. {"/{test1=organizations/*}/{test2=divisions/*}", map[string]string{
  4170. "test1": "organizations/[^/]+",
  4171. "test2": "divisions/[^/]+",
  4172. }},
  4173. {"/v1/{name=projects/*/topics/*}:delete", map[string]string{"name": "projects/[^/]+/topics/[^/]+"}},
  4174. }
  4175. reg := descriptor.NewRegistry()
  4176. for _, data := range tests {
  4177. actual := templateToRegexpMap(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName())
  4178. if !reflect.DeepEqual(data.expected, actual) {
  4179. t.Errorf("Expected partsToRegexpMap(%v) = %v, actual: %v", data.input, data.expected, actual)
  4180. }
  4181. }
  4182. }
  4183. func TestFQMNtoOpenAPIName(t *testing.T) {
  4184. tests := []struct {
  4185. input string
  4186. expected string
  4187. }{
  4188. {"/test", "/test"},
  4189. {"/{test}", "/{test}"},
  4190. {"/{test=prefix/*}", "/{test}"},
  4191. {"/{test=prefix/that/has/multiple/parts/to/it/*}", "/{test}"},
  4192. {"/{test1}/{test2}", "/{test1}/{test2}"},
  4193. {"/{test1}/{test2}/", "/{test1}/{test2}/"},
  4194. {"/v1/{name=tests/*}/tests", "/v1/{name}/tests"},
  4195. }
  4196. reg := descriptor.NewRegistry()
  4197. reg.SetUseJSONNamesForFields(false)
  4198. for _, data := range tests {
  4199. actual := templateToOpenAPIPath(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), make(map[string]string))
  4200. if data.expected != actual {
  4201. t.Errorf("Expected templateToOpenAPIPath(%v) = %v, actual: %v", data.input, data.expected, actual)
  4202. }
  4203. }
  4204. reg.SetUseJSONNamesForFields(true)
  4205. for _, data := range tests {
  4206. actual := templateToOpenAPIPath(data.input, reg, generateFieldsForJSONReservedName(), generateMsgsForJSONReservedName(), make(map[string]string))
  4207. if data.expected != actual {
  4208. t.Errorf("Expected templateToOpenAPIPath(%v) = %v, actual: %v", data.input, data.expected, actual)
  4209. }
  4210. }
  4211. }
  4212. func TestSchemaOfField(t *testing.T) {
  4213. type test struct {
  4214. field *descriptor.Field
  4215. refs refMap
  4216. expected openapiSchemaObject
  4217. openAPIOptions *openapiconfig.OpenAPIOptions
  4218. useJSONNamesForFields bool
  4219. }
  4220. jsonSchema := &openapi_options.JSONSchema{
  4221. Title: "field title",
  4222. Description: "field description",
  4223. }
  4224. jsonSchemaWithOptions := &openapi_options.JSONSchema{
  4225. Title: "field title",
  4226. Description: "field description",
  4227. MultipleOf: 100,
  4228. Maximum: 101,
  4229. ExclusiveMaximum: true,
  4230. Minimum: 1,
  4231. ExclusiveMinimum: true,
  4232. MaxLength: 10,
  4233. MinLength: 3,
  4234. Pattern: "[a-z]+",
  4235. MaxItems: 20,
  4236. MinItems: 2,
  4237. UniqueItems: true,
  4238. MaxProperties: 33,
  4239. MinProperties: 22,
  4240. Required: []string{"req"},
  4241. ReadOnly: true,
  4242. }
  4243. jsonSchemaRequired := &openapi_options.JSONSchema{
  4244. Required: []string{"required_via_json_schema"},
  4245. }
  4246. jsonSchemaWithFormat := &openapi_options.JSONSchema{
  4247. Format: "uuid",
  4248. }
  4249. fieldOptions := new(descriptorpb.FieldOptions)
  4250. proto.SetExtension(fieldOptions, openapi_options.E_Openapiv2Field, jsonSchema)
  4251. requiredField := []annotations.FieldBehavior{annotations.FieldBehavior_REQUIRED}
  4252. requiredFieldOptions := new(descriptorpb.FieldOptions)
  4253. proto.SetExtension(requiredFieldOptions, annotations.E_FieldBehavior, requiredField)
  4254. outputOnlyField := []annotations.FieldBehavior{annotations.FieldBehavior_OUTPUT_ONLY}
  4255. outputOnlyOptions := new(descriptorpb.FieldOptions)
  4256. proto.SetExtension(outputOnlyOptions, annotations.E_FieldBehavior, outputOnlyField)
  4257. tests := []test{
  4258. {
  4259. field: &descriptor.Field{
  4260. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4261. Name: proto.String("primitive_field"),
  4262. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4263. },
  4264. },
  4265. refs: make(refMap),
  4266. expected: openapiSchemaObject{
  4267. schemaCore: schemaCore{
  4268. Type: "string",
  4269. },
  4270. },
  4271. },
  4272. {
  4273. field: &descriptor.Field{
  4274. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4275. Name: proto.String("repeated_primitive_field"),
  4276. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4277. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4278. },
  4279. },
  4280. refs: make(refMap),
  4281. expected: openapiSchemaObject{
  4282. schemaCore: schemaCore{
  4283. Type: "array",
  4284. Items: &openapiItemsObject{
  4285. schemaCore: schemaCore{
  4286. Type: "string",
  4287. },
  4288. },
  4289. },
  4290. },
  4291. },
  4292. {
  4293. field: &descriptor.Field{
  4294. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4295. Name: proto.String("empty_field"),
  4296. TypeName: proto.String(".google.protobuf.Empty"),
  4297. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4298. },
  4299. },
  4300. refs: make(refMap),
  4301. expected: openapiSchemaObject{
  4302. schemaCore: schemaCore{
  4303. Type: "object",
  4304. },
  4305. Properties: &openapiSchemaObjectProperties{},
  4306. },
  4307. },
  4308. {
  4309. field: &descriptor.Field{
  4310. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4311. Name: proto.String("wrapped_field"),
  4312. TypeName: proto.String(".google.protobuf.FieldMask"),
  4313. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4314. },
  4315. },
  4316. refs: make(refMap),
  4317. expected: openapiSchemaObject{
  4318. schemaCore: schemaCore{
  4319. Type: "string",
  4320. },
  4321. },
  4322. },
  4323. {
  4324. field: &descriptor.Field{
  4325. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4326. Name: proto.String("wrapped_field"),
  4327. TypeName: proto.String(".google.protobuf.Timestamp"),
  4328. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4329. },
  4330. },
  4331. refs: make(refMap),
  4332. expected: openapiSchemaObject{
  4333. schemaCore: schemaCore{
  4334. Type: "string",
  4335. Format: "date-time",
  4336. },
  4337. },
  4338. },
  4339. {
  4340. field: &descriptor.Field{
  4341. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4342. Name: proto.String("wrapped_field"),
  4343. TypeName: proto.String(".google.protobuf.Duration"),
  4344. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4345. },
  4346. },
  4347. refs: make(refMap),
  4348. expected: openapiSchemaObject{
  4349. schemaCore: schemaCore{
  4350. Type: "string",
  4351. },
  4352. },
  4353. },
  4354. {
  4355. field: &descriptor.Field{
  4356. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4357. Name: proto.String("wrapped_field"),
  4358. TypeName: proto.String(".google.protobuf.StringValue"),
  4359. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4360. },
  4361. },
  4362. refs: make(refMap),
  4363. expected: openapiSchemaObject{
  4364. schemaCore: schemaCore{
  4365. Type: "string",
  4366. },
  4367. },
  4368. },
  4369. {
  4370. field: &descriptor.Field{
  4371. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4372. Name: proto.String("repeated_wrapped_field"),
  4373. TypeName: proto.String(".google.protobuf.StringValue"),
  4374. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4375. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4376. },
  4377. },
  4378. refs: make(refMap),
  4379. expected: openapiSchemaObject{
  4380. schemaCore: schemaCore{
  4381. Type: "array",
  4382. Items: &openapiItemsObject{
  4383. schemaCore: schemaCore{
  4384. Type: "string",
  4385. },
  4386. },
  4387. },
  4388. },
  4389. },
  4390. {
  4391. field: &descriptor.Field{
  4392. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4393. Name: proto.String("wrapped_field"),
  4394. TypeName: proto.String(".google.protobuf.BytesValue"),
  4395. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4396. },
  4397. },
  4398. refs: make(refMap),
  4399. expected: openapiSchemaObject{
  4400. schemaCore: schemaCore{
  4401. Type: "string",
  4402. Format: "byte",
  4403. },
  4404. },
  4405. },
  4406. {
  4407. field: &descriptor.Field{
  4408. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4409. Name: proto.String("wrapped_field"),
  4410. TypeName: proto.String(".google.protobuf.Int32Value"),
  4411. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4412. },
  4413. },
  4414. refs: make(refMap),
  4415. expected: openapiSchemaObject{
  4416. schemaCore: schemaCore{
  4417. Type: "integer",
  4418. Format: "int32",
  4419. },
  4420. },
  4421. },
  4422. {
  4423. field: &descriptor.Field{
  4424. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4425. Name: proto.String("wrapped_field"),
  4426. TypeName: proto.String(".google.protobuf.UInt32Value"),
  4427. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4428. },
  4429. },
  4430. refs: make(refMap),
  4431. expected: openapiSchemaObject{
  4432. schemaCore: schemaCore{
  4433. Type: "integer",
  4434. Format: "int64",
  4435. },
  4436. },
  4437. },
  4438. {
  4439. field: &descriptor.Field{
  4440. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4441. Name: proto.String("wrapped_field"),
  4442. TypeName: proto.String(".google.protobuf.Int64Value"),
  4443. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4444. },
  4445. },
  4446. refs: make(refMap),
  4447. expected: openapiSchemaObject{
  4448. schemaCore: schemaCore{
  4449. Type: "string",
  4450. Format: "int64",
  4451. },
  4452. },
  4453. },
  4454. {
  4455. field: &descriptor.Field{
  4456. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4457. Name: proto.String("wrapped_field"),
  4458. TypeName: proto.String(".google.protobuf.UInt64Value"),
  4459. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4460. },
  4461. },
  4462. refs: make(refMap),
  4463. expected: openapiSchemaObject{
  4464. schemaCore: schemaCore{
  4465. Type: "string",
  4466. Format: "uint64",
  4467. },
  4468. },
  4469. },
  4470. {
  4471. field: &descriptor.Field{
  4472. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4473. Name: proto.String("wrapped_field"),
  4474. TypeName: proto.String(".google.protobuf.FloatValue"),
  4475. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4476. },
  4477. },
  4478. refs: make(refMap),
  4479. expected: openapiSchemaObject{
  4480. schemaCore: schemaCore{
  4481. Type: "number",
  4482. Format: "float",
  4483. },
  4484. },
  4485. },
  4486. {
  4487. field: &descriptor.Field{
  4488. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4489. Name: proto.String("wrapped_field"),
  4490. TypeName: proto.String(".google.protobuf.DoubleValue"),
  4491. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4492. },
  4493. },
  4494. refs: make(refMap),
  4495. expected: openapiSchemaObject{
  4496. schemaCore: schemaCore{
  4497. Type: "number",
  4498. Format: "double",
  4499. },
  4500. },
  4501. },
  4502. {
  4503. field: &descriptor.Field{
  4504. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4505. Name: proto.String("wrapped_field"),
  4506. TypeName: proto.String(".google.protobuf.BoolValue"),
  4507. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4508. },
  4509. },
  4510. refs: make(refMap),
  4511. expected: openapiSchemaObject{
  4512. schemaCore: schemaCore{
  4513. Type: "boolean",
  4514. },
  4515. },
  4516. },
  4517. {
  4518. field: &descriptor.Field{
  4519. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4520. Name: proto.String("wrapped_field"),
  4521. TypeName: proto.String(".google.protobuf.Struct"),
  4522. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4523. },
  4524. },
  4525. refs: make(refMap),
  4526. expected: openapiSchemaObject{
  4527. schemaCore: schemaCore{
  4528. Type: "object",
  4529. },
  4530. },
  4531. },
  4532. {
  4533. field: &descriptor.Field{
  4534. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4535. Name: proto.String("wrapped_field"),
  4536. TypeName: proto.String(".google.protobuf.Value"),
  4537. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4538. },
  4539. },
  4540. refs: make(refMap),
  4541. expected: openapiSchemaObject{
  4542. schemaCore: schemaCore{},
  4543. },
  4544. },
  4545. {
  4546. field: &descriptor.Field{
  4547. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4548. Name: proto.String("wrapped_field"),
  4549. TypeName: proto.String(".google.protobuf.ListValue"),
  4550. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4551. },
  4552. },
  4553. refs: make(refMap),
  4554. expected: openapiSchemaObject{
  4555. schemaCore: schemaCore{
  4556. Type: "array",
  4557. Items: &openapiItemsObject{schemaCore: schemaCore{
  4558. Type: "object",
  4559. }},
  4560. },
  4561. },
  4562. },
  4563. {
  4564. field: &descriptor.Field{
  4565. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4566. Name: proto.String("wrapped_field"),
  4567. TypeName: proto.String(".google.protobuf.NullValue"),
  4568. Type: descriptorpb.FieldDescriptorProto_TYPE_ENUM.Enum(),
  4569. },
  4570. },
  4571. refs: make(refMap),
  4572. expected: openapiSchemaObject{
  4573. schemaCore: schemaCore{
  4574. Type: "string",
  4575. },
  4576. },
  4577. },
  4578. {
  4579. field: &descriptor.Field{
  4580. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4581. Name: proto.String("message_field"),
  4582. TypeName: proto.String(".example.Message"),
  4583. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4584. },
  4585. },
  4586. refs: refMap{".example.Message": struct{}{}},
  4587. expected: openapiSchemaObject{
  4588. schemaCore: schemaCore{
  4589. Ref: "#/definitions/exampleMessage",
  4590. },
  4591. },
  4592. },
  4593. {
  4594. field: &descriptor.Field{
  4595. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4596. Name: proto.String("map_field"),
  4597. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4598. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4599. TypeName: proto.String(".example.Message.MapFieldEntry"),
  4600. Options: fieldOptions,
  4601. },
  4602. },
  4603. refs: make(refMap),
  4604. expected: openapiSchemaObject{
  4605. schemaCore: schemaCore{
  4606. Type: "object",
  4607. },
  4608. AdditionalProperties: &openapiSchemaObject{
  4609. schemaCore: schemaCore{Type: "string"},
  4610. },
  4611. Title: "field title",
  4612. Description: "field description",
  4613. },
  4614. },
  4615. {
  4616. field: &descriptor.Field{
  4617. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4618. Name: proto.String("array_field"),
  4619. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4620. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4621. Options: fieldOptions,
  4622. },
  4623. },
  4624. refs: make(refMap),
  4625. expected: openapiSchemaObject{
  4626. schemaCore: schemaCore{
  4627. Type: "array",
  4628. Items: &openapiItemsObject{schemaCore: schemaCore{
  4629. Type: "string",
  4630. }},
  4631. },
  4632. Title: "field title",
  4633. Description: "field description",
  4634. },
  4635. },
  4636. {
  4637. field: &descriptor.Field{
  4638. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4639. Name: proto.String("primitive_field"),
  4640. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  4641. Type: descriptorpb.FieldDescriptorProto_TYPE_INT32.Enum(),
  4642. Options: fieldOptions,
  4643. },
  4644. },
  4645. refs: make(refMap),
  4646. expected: openapiSchemaObject{
  4647. schemaCore: schemaCore{
  4648. Type: "integer",
  4649. Format: "int32",
  4650. },
  4651. Title: "field title",
  4652. Description: "field description",
  4653. },
  4654. },
  4655. {
  4656. field: &descriptor.Field{
  4657. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4658. Name: proto.String("message_field"),
  4659. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  4660. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4661. TypeName: proto.String(".example.Empty"),
  4662. Options: fieldOptions,
  4663. },
  4664. },
  4665. refs: refMap{".example.Empty": struct{}{}},
  4666. expected: openapiSchemaObject{
  4667. schemaCore: schemaCore{
  4668. Ref: "#/definitions/exampleEmpty",
  4669. },
  4670. Title: "field title",
  4671. Description: "field description",
  4672. },
  4673. },
  4674. {
  4675. field: &descriptor.Field{
  4676. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4677. Name: proto.String("map_field"), // should be called map_field_option but it's not valid map field name
  4678. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4679. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4680. TypeName: proto.String(".example.Message.MapFieldEntry"),
  4681. },
  4682. },
  4683. openAPIOptions: &openapiconfig.OpenAPIOptions{
  4684. Field: []*openapiconfig.OpenAPIFieldOption{
  4685. {
  4686. Field: "example.Message.map_field",
  4687. Option: jsonSchema,
  4688. },
  4689. },
  4690. },
  4691. refs: make(refMap),
  4692. expected: openapiSchemaObject{
  4693. schemaCore: schemaCore{
  4694. Type: "object",
  4695. },
  4696. AdditionalProperties: &openapiSchemaObject{
  4697. schemaCore: schemaCore{Type: "string"},
  4698. },
  4699. Title: "field title",
  4700. Description: "field description",
  4701. },
  4702. },
  4703. {
  4704. field: &descriptor.Field{
  4705. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4706. Name: proto.String("array_field_option"),
  4707. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4708. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4709. },
  4710. },
  4711. openAPIOptions: &openapiconfig.OpenAPIOptions{
  4712. Field: []*openapiconfig.OpenAPIFieldOption{
  4713. {
  4714. Field: "example.Message.array_field_option",
  4715. Option: jsonSchema,
  4716. },
  4717. },
  4718. },
  4719. refs: make(refMap),
  4720. expected: openapiSchemaObject{
  4721. schemaCore: schemaCore{
  4722. Type: "array",
  4723. Items: &openapiItemsObject{schemaCore: schemaCore{
  4724. Type: "string",
  4725. }},
  4726. },
  4727. Title: "field title",
  4728. Description: "field description",
  4729. },
  4730. },
  4731. {
  4732. field: &descriptor.Field{
  4733. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4734. Name: proto.String("primitive_field_option"),
  4735. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  4736. Type: descriptorpb.FieldDescriptorProto_TYPE_INT32.Enum(),
  4737. },
  4738. },
  4739. openAPIOptions: &openapiconfig.OpenAPIOptions{
  4740. Field: []*openapiconfig.OpenAPIFieldOption{
  4741. {
  4742. Field: "example.Message.primitive_field_option",
  4743. Option: jsonSchema,
  4744. },
  4745. },
  4746. },
  4747. refs: make(refMap),
  4748. expected: openapiSchemaObject{
  4749. schemaCore: schemaCore{
  4750. Type: "integer",
  4751. Format: "int32",
  4752. },
  4753. Title: "field title",
  4754. Description: "field description",
  4755. },
  4756. },
  4757. {
  4758. field: &descriptor.Field{
  4759. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4760. Name: proto.String("primitive_field_option"),
  4761. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  4762. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum().Enum(),
  4763. },
  4764. },
  4765. openAPIOptions: &openapiconfig.OpenAPIOptions{
  4766. Field: []*openapiconfig.OpenAPIFieldOption{
  4767. {
  4768. Field: "example.Message.primitive_field_option",
  4769. Option: &openapi_options.JSONSchema{
  4770. Title: "field title",
  4771. Description: "field description",
  4772. Format: "uuid",
  4773. },
  4774. },
  4775. },
  4776. },
  4777. refs: make(refMap),
  4778. expected: openapiSchemaObject{
  4779. schemaCore: schemaCore{
  4780. Type: "string",
  4781. Format: "uuid",
  4782. },
  4783. Title: "field title",
  4784. Description: "field description",
  4785. },
  4786. },
  4787. {
  4788. field: &descriptor.Field{
  4789. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4790. Name: proto.String("message_field_option"),
  4791. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  4792. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4793. TypeName: proto.String(".example.Empty"),
  4794. },
  4795. },
  4796. openAPIOptions: &openapiconfig.OpenAPIOptions{
  4797. Field: []*openapiconfig.OpenAPIFieldOption{
  4798. {
  4799. Field: "example.Message.message_field_option",
  4800. Option: jsonSchema,
  4801. },
  4802. },
  4803. },
  4804. refs: refMap{".example.Empty": struct{}{}},
  4805. expected: openapiSchemaObject{
  4806. schemaCore: schemaCore{
  4807. Ref: "#/definitions/exampleEmpty",
  4808. },
  4809. Title: "field title",
  4810. Description: "field description",
  4811. },
  4812. },
  4813. {
  4814. field: &descriptor.Field{
  4815. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4816. Name: proto.String("required_via_field_behavior_field"),
  4817. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4818. Options: requiredFieldOptions,
  4819. },
  4820. },
  4821. refs: make(refMap),
  4822. expected: openapiSchemaObject{
  4823. schemaCore: schemaCore{
  4824. Type: "string",
  4825. },
  4826. Required: []string{"required_via_field_behavior_field"},
  4827. },
  4828. },
  4829. {
  4830. field: &descriptor.Field{
  4831. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4832. Name: proto.String("readonly_via_field_behavior_field"),
  4833. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4834. Options: outputOnlyOptions,
  4835. },
  4836. },
  4837. refs: make(refMap),
  4838. expected: openapiSchemaObject{
  4839. schemaCore: schemaCore{
  4840. Type: "string",
  4841. },
  4842. ReadOnly: true,
  4843. },
  4844. },
  4845. {
  4846. field: &descriptor.Field{
  4847. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4848. Name: proto.String("required_message_field"),
  4849. TypeName: proto.String(".example.Message"),
  4850. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  4851. Options: requiredFieldOptions,
  4852. },
  4853. },
  4854. refs: refMap{".example.Message": struct{}{}},
  4855. expected: openapiSchemaObject{
  4856. schemaCore: schemaCore{
  4857. Ref: "#/definitions/exampleMessage",
  4858. },
  4859. Required: []string{"required_message_field"},
  4860. },
  4861. },
  4862. {
  4863. field: &descriptor.Field{
  4864. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4865. Name: proto.String("array_field_option"),
  4866. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4867. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4868. },
  4869. },
  4870. openAPIOptions: &openapiconfig.OpenAPIOptions{
  4871. Field: []*openapiconfig.OpenAPIFieldOption{
  4872. {
  4873. Field: "example.Message.array_field_option",
  4874. Option: jsonSchemaWithOptions,
  4875. },
  4876. },
  4877. },
  4878. refs: make(refMap),
  4879. expected: openapiSchemaObject{
  4880. schemaCore: schemaCore{
  4881. Type: "array",
  4882. Items: &openapiItemsObject{
  4883. schemaCore: schemaCore{
  4884. Type: "string",
  4885. },
  4886. MultipleOf: 100,
  4887. Maximum: 101,
  4888. ExclusiveMaximum: true,
  4889. Minimum: 1,
  4890. ExclusiveMinimum: true,
  4891. MaxLength: 10,
  4892. MinLength: 3,
  4893. Pattern: "[a-z]+",
  4894. MaxProperties: 33,
  4895. MinProperties: 22,
  4896. Required: []string{"req"},
  4897. ReadOnly: true,
  4898. },
  4899. },
  4900. Title: "field title",
  4901. Description: "field description",
  4902. UniqueItems: true,
  4903. MaxItems: 20,
  4904. MinItems: 2,
  4905. },
  4906. },
  4907. {
  4908. field: &descriptor.Field{
  4909. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4910. Name: proto.String("array_field_option"),
  4911. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4912. Type: descriptorpb.FieldDescriptorProto_TYPE_INT64.Enum(),
  4913. },
  4914. },
  4915. openAPIOptions: &openapiconfig.OpenAPIOptions{
  4916. Field: []*openapiconfig.OpenAPIFieldOption{
  4917. {
  4918. Field: "example.Message.array_field_option",
  4919. Option: jsonSchemaWithOptions,
  4920. },
  4921. },
  4922. },
  4923. refs: make(refMap),
  4924. expected: openapiSchemaObject{
  4925. schemaCore: schemaCore{
  4926. Type: "array",
  4927. Items: &openapiItemsObject{
  4928. schemaCore: schemaCore{
  4929. Type: "string",
  4930. Format: "int64",
  4931. },
  4932. MultipleOf: 100,
  4933. Maximum: 101,
  4934. ExclusiveMaximum: true,
  4935. Minimum: 1,
  4936. ExclusiveMinimum: true,
  4937. MaxLength: 10,
  4938. MinLength: 3,
  4939. Pattern: "[a-z]+",
  4940. MaxProperties: 33,
  4941. MinProperties: 22,
  4942. Required: []string{"req"},
  4943. ReadOnly: true,
  4944. },
  4945. },
  4946. Title: "field title",
  4947. Description: "field description",
  4948. UniqueItems: true,
  4949. MaxItems: 20,
  4950. MinItems: 2,
  4951. },
  4952. },
  4953. {
  4954. field: &descriptor.Field{
  4955. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4956. Name: proto.String("array_field_format"),
  4957. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  4958. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4959. },
  4960. },
  4961. openAPIOptions: &openapiconfig.OpenAPIOptions{
  4962. Field: []*openapiconfig.OpenAPIFieldOption{
  4963. {
  4964. Field: "example.Message.array_field_format",
  4965. Option: jsonSchemaWithFormat,
  4966. },
  4967. },
  4968. },
  4969. refs: make(refMap),
  4970. expected: openapiSchemaObject{
  4971. schemaCore: schemaCore{
  4972. Type: "array",
  4973. Items: &openapiItemsObject{
  4974. schemaCore: schemaCore{
  4975. Type: "string",
  4976. Format: "uuid",
  4977. },
  4978. },
  4979. },
  4980. },
  4981. },
  4982. {
  4983. field: &descriptor.Field{
  4984. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  4985. Name: proto.String("required_via_field_behavior_field_json_name"),
  4986. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  4987. JsonName: proto.String("required_field_custom_name"),
  4988. Options: requiredFieldOptions,
  4989. },
  4990. },
  4991. refs: make(refMap),
  4992. expected: openapiSchemaObject{
  4993. schemaCore: schemaCore{
  4994. Type: "string",
  4995. },
  4996. Required: []string{"required_field_custom_name"},
  4997. },
  4998. useJSONNamesForFields: true,
  4999. },
  5000. {
  5001. field: &descriptor.Field{
  5002. FieldDescriptorProto: &descriptorpb.FieldDescriptorProto{
  5003. Name: proto.String("required_via_json_schema"),
  5004. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5005. JsonName: proto.String("required_via_json_schema_json_name"),
  5006. },
  5007. },
  5008. openAPIOptions: &openapiconfig.OpenAPIOptions{
  5009. Field: []*openapiconfig.OpenAPIFieldOption{
  5010. {
  5011. Field: "example.Message.required_via_json_schema",
  5012. Option: jsonSchemaRequired,
  5013. },
  5014. },
  5015. },
  5016. refs: make(refMap),
  5017. useJSONNamesForFields: true,
  5018. expected: openapiSchemaObject{
  5019. schemaCore: schemaCore{
  5020. Type: "string",
  5021. },
  5022. Required: []string{"required_via_json_schema_json_name"},
  5023. },
  5024. },
  5025. }
  5026. for _, test := range tests {
  5027. reg := descriptor.NewRegistry()
  5028. reg.SetUseJSONNamesForFields(test.useJSONNamesForFields)
  5029. req := &pluginpb.CodeGeneratorRequest{
  5030. ProtoFile: []*descriptorpb.FileDescriptorProto{
  5031. {
  5032. Name: proto.String("third_party/google.proto"),
  5033. Package: proto.String("google.protobuf"),
  5034. Options: &descriptorpb.FileOptions{
  5035. GoPackage: proto.String("third_party/google"),
  5036. },
  5037. MessageType: []*descriptorpb.DescriptorProto{
  5038. protodesc.ToDescriptorProto((&emptypb.Empty{}).ProtoReflect().Descriptor()),
  5039. protodesc.ToDescriptorProto((&structpb.Struct{}).ProtoReflect().Descriptor()),
  5040. protodesc.ToDescriptorProto((&structpb.Value{}).ProtoReflect().Descriptor()),
  5041. protodesc.ToDescriptorProto((&structpb.ListValue{}).ProtoReflect().Descriptor()),
  5042. protodesc.ToDescriptorProto((&field_mask.FieldMask{}).ProtoReflect().Descriptor()),
  5043. protodesc.ToDescriptorProto((&timestamppb.Timestamp{}).ProtoReflect().Descriptor()),
  5044. protodesc.ToDescriptorProto((&durationpb.Duration{}).ProtoReflect().Descriptor()),
  5045. protodesc.ToDescriptorProto((&wrapperspb.StringValue{}).ProtoReflect().Descriptor()),
  5046. protodesc.ToDescriptorProto((&wrapperspb.BytesValue{}).ProtoReflect().Descriptor()),
  5047. protodesc.ToDescriptorProto((&wrapperspb.Int32Value{}).ProtoReflect().Descriptor()),
  5048. protodesc.ToDescriptorProto((&wrapperspb.UInt32Value{}).ProtoReflect().Descriptor()),
  5049. protodesc.ToDescriptorProto((&wrapperspb.Int64Value{}).ProtoReflect().Descriptor()),
  5050. protodesc.ToDescriptorProto((&wrapperspb.UInt64Value{}).ProtoReflect().Descriptor()),
  5051. protodesc.ToDescriptorProto((&wrapperspb.FloatValue{}).ProtoReflect().Descriptor()),
  5052. protodesc.ToDescriptorProto((&wrapperspb.DoubleValue{}).ProtoReflect().Descriptor()),
  5053. protodesc.ToDescriptorProto((&wrapperspb.BoolValue{}).ProtoReflect().Descriptor()),
  5054. },
  5055. EnumType: []*descriptorpb.EnumDescriptorProto{
  5056. protodesc.ToEnumDescriptorProto(structpb.NullValue(0).Descriptor()),
  5057. },
  5058. },
  5059. {
  5060. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  5061. Name: proto.String("example.proto"),
  5062. Package: proto.String("example"),
  5063. Dependency: []string{"third_party/google.proto"},
  5064. Options: &descriptorpb.FileOptions{
  5065. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  5066. },
  5067. MessageType: []*descriptorpb.DescriptorProto{
  5068. {
  5069. Name: proto.String("Message"),
  5070. Field: []*descriptorpb.FieldDescriptorProto{
  5071. {
  5072. Name: proto.String("value"),
  5073. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5074. Number: proto.Int32(1),
  5075. },
  5076. func() *descriptorpb.FieldDescriptorProto {
  5077. fd := test.field.FieldDescriptorProto
  5078. fd.Number = proto.Int32(2)
  5079. return fd
  5080. }(),
  5081. },
  5082. NestedType: []*descriptorpb.DescriptorProto{
  5083. {
  5084. Name: proto.String("MapFieldEntry"),
  5085. Options: &descriptorpb.MessageOptions{MapEntry: proto.Bool(true)},
  5086. Field: []*descriptorpb.FieldDescriptorProto{
  5087. {
  5088. Name: proto.String("key"),
  5089. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  5090. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5091. Number: proto.Int32(1),
  5092. },
  5093. {
  5094. Name: proto.String("value"),
  5095. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  5096. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5097. Number: proto.Int32(2),
  5098. },
  5099. },
  5100. },
  5101. },
  5102. },
  5103. {
  5104. Name: proto.String("Empty"),
  5105. },
  5106. },
  5107. EnumType: []*descriptorpb.EnumDescriptorProto{
  5108. {
  5109. Name: proto.String("MessageType"),
  5110. Value: []*descriptorpb.EnumValueDescriptorProto{
  5111. {
  5112. Name: proto.String("MESSAGE_TYPE_1"),
  5113. Number: proto.Int32(0),
  5114. },
  5115. },
  5116. },
  5117. },
  5118. Service: []*descriptorpb.ServiceDescriptorProto{},
  5119. },
  5120. },
  5121. }
  5122. err := reg.Load(req)
  5123. if err != nil {
  5124. t.Errorf("failed to reg.Load(req): %v", err)
  5125. }
  5126. // set field's parent message pointer to message so field can resolve its FQFN
  5127. test.field.Message = &descriptor.Message{
  5128. DescriptorProto: req.ProtoFile[1].MessageType[0],
  5129. File: &descriptor.File{
  5130. FileDescriptorProto: req.ProtoFile[1],
  5131. },
  5132. }
  5133. if test.openAPIOptions != nil {
  5134. if err := reg.RegisterOpenAPIOptions(test.openAPIOptions); err != nil {
  5135. t.Fatalf("failed to register OpenAPI options: %s", err)
  5136. }
  5137. }
  5138. refs := make(refMap)
  5139. actual := schemaOfField(test.field, reg, refs)
  5140. expectedSchemaObject := test.expected
  5141. if e, a := expectedSchemaObject, actual; !reflect.DeepEqual(a, e) {
  5142. t.Errorf("Expected schemaOfField(%v) = \n%#+v, actual: \n%#+v", test.field, e, a)
  5143. }
  5144. if !reflect.DeepEqual(refs, test.refs) {
  5145. t.Errorf("Expected schemaOfField(%v) to add refs %v, not %v", test.field, test.refs, refs)
  5146. }
  5147. }
  5148. }
  5149. func TestRenderMessagesAsDefinition(t *testing.T) {
  5150. jsonSchema := &openapi_options.JSONSchema{
  5151. Title: "field title",
  5152. Description: "field description",
  5153. Required: []string{"aRequiredField"},
  5154. }
  5155. requiredField := new(descriptorpb.FieldOptions)
  5156. proto.SetExtension(requiredField, openapi_options.E_Openapiv2Field, jsonSchema)
  5157. fieldBehaviorRequired := []annotations.FieldBehavior{annotations.FieldBehavior_REQUIRED}
  5158. requiredFieldOptions := new(descriptorpb.FieldOptions)
  5159. proto.SetExtension(requiredFieldOptions, annotations.E_FieldBehavior, fieldBehaviorRequired)
  5160. fieldBehaviorOutputOnlyField := []annotations.FieldBehavior{annotations.FieldBehavior_OUTPUT_ONLY}
  5161. fieldBehaviorOutputOnlyOptions := new(descriptorpb.FieldOptions)
  5162. proto.SetExtension(fieldBehaviorOutputOnlyOptions, annotations.E_FieldBehavior, fieldBehaviorOutputOnlyField)
  5163. fieldVisibilityFieldInternal := &visibility.VisibilityRule{Restriction: "INTERNAL"}
  5164. fieldVisibilityInternalOption := new(descriptorpb.FieldOptions)
  5165. proto.SetExtension(fieldVisibilityInternalOption, visibility.E_FieldVisibility, fieldVisibilityFieldInternal)
  5166. fieldVisibilityFieldPreview := &visibility.VisibilityRule{Restriction: "INTERNAL,PREVIEW"}
  5167. fieldVisibilityPreviewOption := new(descriptorpb.FieldOptions)
  5168. proto.SetExtension(fieldVisibilityPreviewOption, visibility.E_FieldVisibility, fieldVisibilityFieldPreview)
  5169. tests := []struct {
  5170. descr string
  5171. msgDescs []*descriptorpb.DescriptorProto
  5172. schema map[string]*openapi_options.Schema // per-message schema to add
  5173. defs openapiDefinitionsObject
  5174. openAPIOptions *openapiconfig.OpenAPIOptions
  5175. pathParams []descriptor.Parameter
  5176. UseJSONNamesForFields bool
  5177. UseAllOfForRefs bool
  5178. }{
  5179. {
  5180. descr: "no OpenAPI options",
  5181. msgDescs: []*descriptorpb.DescriptorProto{
  5182. {Name: proto.String("Message")},
  5183. },
  5184. schema: map[string]*openapi_options.Schema{},
  5185. defs: map[string]openapiSchemaObject{
  5186. "Message": {schemaCore: schemaCore{Type: "object"}},
  5187. },
  5188. },
  5189. {
  5190. descr: "example option",
  5191. msgDescs: []*descriptorpb.DescriptorProto{
  5192. {Name: proto.String("Message")},
  5193. },
  5194. schema: map[string]*openapi_options.Schema{
  5195. "Message": {
  5196. Example: `{"foo":"bar"}`,
  5197. },
  5198. },
  5199. defs: map[string]openapiSchemaObject{
  5200. "Message": {schemaCore: schemaCore{
  5201. Type: "object",
  5202. Example: RawExample(`{"foo":"bar"}`),
  5203. }},
  5204. },
  5205. },
  5206. {
  5207. descr: "example option with something non-json",
  5208. msgDescs: []*descriptorpb.DescriptorProto{
  5209. {Name: proto.String("Message")},
  5210. },
  5211. schema: map[string]*openapi_options.Schema{
  5212. "Message": {
  5213. Example: `XXXX anything goes XXXX`,
  5214. },
  5215. },
  5216. defs: map[string]openapiSchemaObject{
  5217. "Message": {schemaCore: schemaCore{
  5218. Type: "object",
  5219. Example: RawExample(`XXXX anything goes XXXX`),
  5220. }},
  5221. },
  5222. },
  5223. {
  5224. descr: "external docs option",
  5225. msgDescs: []*descriptorpb.DescriptorProto{
  5226. {Name: proto.String("Message")},
  5227. },
  5228. schema: map[string]*openapi_options.Schema{
  5229. "Message": {
  5230. ExternalDocs: &openapi_options.ExternalDocumentation{
  5231. Description: "glorious docs",
  5232. Url: "https://nada",
  5233. },
  5234. },
  5235. },
  5236. defs: map[string]openapiSchemaObject{
  5237. "Message": {
  5238. schemaCore: schemaCore{
  5239. Type: "object",
  5240. },
  5241. ExternalDocs: &openapiExternalDocumentationObject{
  5242. Description: "glorious docs",
  5243. URL: "https://nada",
  5244. },
  5245. },
  5246. },
  5247. },
  5248. {
  5249. descr: "JSONSchema options",
  5250. msgDescs: []*descriptorpb.DescriptorProto{
  5251. {Name: proto.String("Message")},
  5252. },
  5253. schema: map[string]*openapi_options.Schema{
  5254. "Message": {
  5255. JsonSchema: &openapi_options.JSONSchema{
  5256. Title: "title",
  5257. Description: "desc",
  5258. MultipleOf: 100,
  5259. Maximum: 101,
  5260. ExclusiveMaximum: true,
  5261. Minimum: 1,
  5262. ExclusiveMinimum: true,
  5263. MaxLength: 10,
  5264. MinLength: 3,
  5265. Pattern: "[a-z]+",
  5266. MaxItems: 20,
  5267. MinItems: 2,
  5268. UniqueItems: true,
  5269. MaxProperties: 33,
  5270. MinProperties: 22,
  5271. Required: []string{"req"},
  5272. ReadOnly: true,
  5273. },
  5274. },
  5275. },
  5276. defs: map[string]openapiSchemaObject{
  5277. "Message": {
  5278. schemaCore: schemaCore{
  5279. Type: "object",
  5280. },
  5281. Title: "title",
  5282. Description: "desc",
  5283. MultipleOf: 100,
  5284. Maximum: 101,
  5285. ExclusiveMaximum: true,
  5286. Minimum: 1,
  5287. ExclusiveMinimum: true,
  5288. MaxLength: 10,
  5289. MinLength: 3,
  5290. Pattern: "[a-z]+",
  5291. MaxItems: 20,
  5292. MinItems: 2,
  5293. UniqueItems: true,
  5294. MaxProperties: 33,
  5295. MinProperties: 22,
  5296. Required: []string{"req"},
  5297. ReadOnly: true,
  5298. },
  5299. },
  5300. },
  5301. {
  5302. descr: "JSONSchema options from registry",
  5303. msgDescs: []*descriptorpb.DescriptorProto{
  5304. {Name: proto.String("Message")},
  5305. },
  5306. openAPIOptions: &openapiconfig.OpenAPIOptions{
  5307. Message: []*openapiconfig.OpenAPIMessageOption{
  5308. {
  5309. Message: "example.Message",
  5310. Option: &openapi_options.Schema{
  5311. JsonSchema: &openapi_options.JSONSchema{
  5312. Title: "title",
  5313. Description: "desc",
  5314. MultipleOf: 100,
  5315. Maximum: 101,
  5316. ExclusiveMaximum: true,
  5317. Minimum: 1,
  5318. ExclusiveMinimum: true,
  5319. MaxLength: 10,
  5320. MinLength: 3,
  5321. Pattern: "[a-z]+",
  5322. MaxItems: 20,
  5323. MinItems: 2,
  5324. UniqueItems: true,
  5325. MaxProperties: 33,
  5326. MinProperties: 22,
  5327. Required: []string{"req"},
  5328. ReadOnly: true,
  5329. },
  5330. },
  5331. },
  5332. },
  5333. },
  5334. defs: map[string]openapiSchemaObject{
  5335. "Message": {
  5336. schemaCore: schemaCore{
  5337. Type: "object",
  5338. },
  5339. Title: "title",
  5340. Description: "desc",
  5341. MultipleOf: 100,
  5342. Maximum: 101,
  5343. ExclusiveMaximum: true,
  5344. Minimum: 1,
  5345. ExclusiveMinimum: true,
  5346. MaxLength: 10,
  5347. MinLength: 3,
  5348. Pattern: "[a-z]+",
  5349. MaxItems: 20,
  5350. MinItems: 2,
  5351. UniqueItems: true,
  5352. MaxProperties: 33,
  5353. MinProperties: 22,
  5354. Required: []string{"req"},
  5355. ReadOnly: true,
  5356. },
  5357. },
  5358. },
  5359. {
  5360. descr: "JSONSchema with required properties",
  5361. msgDescs: []*descriptorpb.DescriptorProto{
  5362. {
  5363. Name: proto.String("Message"),
  5364. Field: []*descriptorpb.FieldDescriptorProto{
  5365. {
  5366. Name: proto.String("FieldOne"),
  5367. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5368. Number: proto.Int32(1),
  5369. },
  5370. {
  5371. Name: proto.String("FieldTwo"),
  5372. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5373. Number: proto.Int32(2),
  5374. Options: requiredFieldOptions,
  5375. },
  5376. {
  5377. Name: proto.String("FieldThree"),
  5378. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5379. Number: proto.Int32(3),
  5380. Options: requiredFieldOptions,
  5381. },
  5382. },
  5383. },
  5384. },
  5385. schema: map[string]*openapi_options.Schema{
  5386. "Message": {
  5387. JsonSchema: &openapi_options.JSONSchema{
  5388. Title: "title",
  5389. Description: "desc",
  5390. Required: []string{"FieldOne", "FieldTwo"},
  5391. },
  5392. },
  5393. },
  5394. defs: map[string]openapiSchemaObject{
  5395. "Message": {
  5396. schemaCore: schemaCore{
  5397. Type: "object",
  5398. },
  5399. Title: "title",
  5400. Description: "desc",
  5401. Required: []string{"FieldOne", "FieldTwo", "FieldThree"},
  5402. Properties: &openapiSchemaObjectProperties{
  5403. {
  5404. Key: "FieldOne",
  5405. Value: openapiSchemaObject{
  5406. schemaCore: schemaCore{
  5407. Type: "string",
  5408. },
  5409. },
  5410. },
  5411. {
  5412. Key: "FieldTwo",
  5413. Value: openapiSchemaObject{
  5414. schemaCore: schemaCore{
  5415. Type: "string",
  5416. },
  5417. },
  5418. },
  5419. {
  5420. Key: "FieldThree",
  5421. Value: openapiSchemaObject{
  5422. schemaCore: schemaCore{
  5423. Type: "string",
  5424. },
  5425. },
  5426. },
  5427. },
  5428. },
  5429. },
  5430. },
  5431. {
  5432. descr: "JSONSchema with required properties",
  5433. msgDescs: []*descriptorpb.DescriptorProto{
  5434. {
  5435. Name: proto.String("Message"),
  5436. Field: []*descriptorpb.FieldDescriptorProto{
  5437. {
  5438. Name: proto.String("FieldOne"),
  5439. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5440. Number: proto.Int32(3),
  5441. Options: requiredFieldOptions,
  5442. },
  5443. },
  5444. },
  5445. },
  5446. schema: map[string]*openapi_options.Schema{
  5447. "Message": {
  5448. JsonSchema: &openapi_options.JSONSchema{
  5449. Title: "title",
  5450. Description: "desc",
  5451. },
  5452. },
  5453. },
  5454. defs: map[string]openapiSchemaObject{
  5455. "Message": {
  5456. schemaCore: schemaCore{
  5457. Type: "object",
  5458. },
  5459. Title: "title",
  5460. Description: "desc",
  5461. Required: []string{"FieldOne"},
  5462. Properties: &openapiSchemaObjectProperties{
  5463. {
  5464. Key: "FieldOne",
  5465. Value: openapiSchemaObject{
  5466. schemaCore: schemaCore{
  5467. Type: "string",
  5468. },
  5469. },
  5470. },
  5471. },
  5472. },
  5473. },
  5474. },
  5475. {
  5476. descr: "JSONSchema with required properties by using annotations",
  5477. msgDescs: []*descriptorpb.DescriptorProto{
  5478. {
  5479. Name: proto.String("Message"),
  5480. Field: []*descriptorpb.FieldDescriptorProto{
  5481. {
  5482. Name: proto.String("FieldOne"),
  5483. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5484. Number: proto.Int32(2),
  5485. Options: requiredFieldOptions,
  5486. },
  5487. },
  5488. },
  5489. },
  5490. schema: map[string]*openapi_options.Schema{
  5491. "Message": {
  5492. JsonSchema: &openapi_options.JSONSchema{
  5493. Title: "title",
  5494. Description: "desc",
  5495. },
  5496. },
  5497. },
  5498. defs: map[string]openapiSchemaObject{
  5499. "Message": {
  5500. schemaCore: schemaCore{
  5501. Type: "object",
  5502. },
  5503. Title: "title",
  5504. Description: "desc",
  5505. Required: []string{"FieldOne"},
  5506. Properties: &openapiSchemaObjectProperties{
  5507. {
  5508. Key: "FieldOne",
  5509. Value: openapiSchemaObject{
  5510. schemaCore: schemaCore{
  5511. Type: "string",
  5512. },
  5513. },
  5514. },
  5515. },
  5516. },
  5517. },
  5518. },
  5519. {
  5520. descr: "JSONSchema with hidden properties",
  5521. msgDescs: []*descriptorpb.DescriptorProto{
  5522. {
  5523. Name: proto.String("Message"),
  5524. Field: []*descriptorpb.FieldDescriptorProto{
  5525. {
  5526. Name: proto.String("aInternalField"),
  5527. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5528. Number: proto.Int32(1),
  5529. Options: fieldVisibilityInternalOption,
  5530. },
  5531. {
  5532. Name: proto.String("aPreviewField"),
  5533. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5534. Number: proto.Int32(2),
  5535. Options: fieldVisibilityPreviewOption,
  5536. },
  5537. {
  5538. Name: proto.String("aVisibleField"),
  5539. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5540. Number: proto.Int32(3),
  5541. },
  5542. },
  5543. },
  5544. },
  5545. schema: map[string]*openapi_options.Schema{
  5546. "Message": {
  5547. JsonSchema: &openapi_options.JSONSchema{
  5548. Title: "title",
  5549. Description: "desc",
  5550. Required: []string{"req"},
  5551. },
  5552. },
  5553. },
  5554. defs: map[string]openapiSchemaObject{
  5555. "Message": {
  5556. schemaCore: schemaCore{
  5557. Type: "object",
  5558. },
  5559. Title: "title",
  5560. Description: "desc",
  5561. Required: []string{"req"},
  5562. Properties: &openapiSchemaObjectProperties{
  5563. {
  5564. Key: "aPreviewField",
  5565. Value: openapiSchemaObject{
  5566. schemaCore: schemaCore{
  5567. Type: "string",
  5568. },
  5569. },
  5570. },
  5571. {
  5572. Key: "aVisibleField",
  5573. Value: openapiSchemaObject{
  5574. schemaCore: schemaCore{
  5575. Type: "string",
  5576. },
  5577. },
  5578. },
  5579. },
  5580. },
  5581. },
  5582. },
  5583. {
  5584. descr: "JSONSchema with path parameters",
  5585. msgDescs: []*descriptorpb.DescriptorProto{
  5586. {
  5587. Name: proto.String("Message"),
  5588. Field: []*descriptorpb.FieldDescriptorProto{
  5589. {
  5590. Name: proto.String("aRequiredField"),
  5591. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5592. Number: proto.Int32(1),
  5593. Options: requiredField,
  5594. },
  5595. {
  5596. Name: proto.String("aPathParameter"),
  5597. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5598. Number: proto.Int32(2),
  5599. },
  5600. },
  5601. },
  5602. },
  5603. schema: map[string]*openapi_options.Schema{
  5604. "Message": {
  5605. JsonSchema: &openapi_options.JSONSchema{
  5606. Title: "title",
  5607. Description: "desc",
  5608. Required: []string{"req"},
  5609. },
  5610. },
  5611. },
  5612. defs: map[string]openapiSchemaObject{
  5613. "Message": {
  5614. schemaCore: schemaCore{
  5615. Type: "object",
  5616. },
  5617. Title: "title",
  5618. Description: "desc",
  5619. Required: []string{"req", "aRequiredField"},
  5620. Properties: &openapiSchemaObjectProperties{
  5621. {
  5622. Key: "aRequiredField",
  5623. Value: openapiSchemaObject{
  5624. schemaCore: schemaCore{
  5625. Type: "string",
  5626. },
  5627. Description: "field description",
  5628. Title: "field title",
  5629. },
  5630. },
  5631. },
  5632. },
  5633. },
  5634. pathParams: []descriptor.Parameter{
  5635. {
  5636. FieldPath: descriptor.FieldPath{
  5637. descriptor.FieldPathComponent{
  5638. Name: ("aPathParameter"),
  5639. },
  5640. },
  5641. },
  5642. },
  5643. },
  5644. {
  5645. descr: "JSONSchema with required properties via field_behavior",
  5646. msgDescs: []*descriptorpb.DescriptorProto{
  5647. {
  5648. Name: proto.String("Message"),
  5649. Field: []*descriptorpb.FieldDescriptorProto{
  5650. {
  5651. Name: proto.String("aRequiredField"),
  5652. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5653. Number: proto.Int32(1),
  5654. Options: requiredFieldOptions,
  5655. },
  5656. {
  5657. Name: proto.String("aOutputOnlyField"),
  5658. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5659. Number: proto.Int32(2),
  5660. Options: fieldBehaviorOutputOnlyOptions,
  5661. },
  5662. },
  5663. },
  5664. },
  5665. schema: map[string]*openapi_options.Schema{
  5666. "Message": {
  5667. JsonSchema: &openapi_options.JSONSchema{
  5668. Title: "title",
  5669. Description: "desc",
  5670. Required: []string{"req"},
  5671. },
  5672. },
  5673. },
  5674. defs: map[string]openapiSchemaObject{
  5675. "Message": {
  5676. schemaCore: schemaCore{
  5677. Type: "object",
  5678. },
  5679. Title: "title",
  5680. Description: "desc",
  5681. Required: []string{"req", "aRequiredField"},
  5682. Properties: &openapiSchemaObjectProperties{
  5683. {
  5684. Key: "aRequiredField",
  5685. Value: openapiSchemaObject{
  5686. schemaCore: schemaCore{
  5687. Type: "string",
  5688. },
  5689. },
  5690. },
  5691. {
  5692. Key: "aOutputOnlyField",
  5693. Value: openapiSchemaObject{
  5694. schemaCore: schemaCore{
  5695. Type: "string",
  5696. },
  5697. ReadOnly: true,
  5698. },
  5699. },
  5700. },
  5701. },
  5702. },
  5703. },
  5704. {
  5705. descr: "JSONSchema with required properties and fields with json_name",
  5706. msgDescs: []*descriptorpb.DescriptorProto{
  5707. {
  5708. Name: proto.String("Message"),
  5709. Field: []*descriptorpb.FieldDescriptorProto{
  5710. {
  5711. Name: proto.String("FieldOne"),
  5712. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5713. Number: proto.Int32(1),
  5714. JsonName: proto.String("custom_json_1"),
  5715. },
  5716. {
  5717. Name: proto.String("FieldTwo"),
  5718. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5719. Number: proto.Int32(2),
  5720. JsonName: proto.String("custom_json_2"),
  5721. Options: requiredFieldOptions,
  5722. },
  5723. {
  5724. Name: proto.String("FieldThree"),
  5725. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  5726. Number: proto.Int32(3),
  5727. JsonName: proto.String("custom_json_3"),
  5728. Options: requiredFieldOptions,
  5729. },
  5730. },
  5731. },
  5732. },
  5733. schema: map[string]*openapi_options.Schema{
  5734. "Message": {
  5735. JsonSchema: &openapi_options.JSONSchema{
  5736. Title: "title",
  5737. Description: "desc",
  5738. Required: []string{"FieldOne", "FieldTwo"},
  5739. },
  5740. },
  5741. },
  5742. defs: map[string]openapiSchemaObject{
  5743. "Message": {
  5744. schemaCore: schemaCore{
  5745. Type: "object",
  5746. },
  5747. Title: "title",
  5748. Description: "desc",
  5749. Required: []string{"custom_json_1", "custom_json_2", "custom_json_3"},
  5750. Properties: &openapiSchemaObjectProperties{
  5751. {
  5752. Key: "custom_json_1",
  5753. Value: openapiSchemaObject{
  5754. schemaCore: schemaCore{
  5755. Type: "string",
  5756. },
  5757. },
  5758. },
  5759. {
  5760. Key: "custom_json_2",
  5761. Value: openapiSchemaObject{
  5762. schemaCore: schemaCore{
  5763. Type: "string",
  5764. },
  5765. },
  5766. },
  5767. {
  5768. Key: "custom_json_3",
  5769. Value: openapiSchemaObject{
  5770. schemaCore: schemaCore{
  5771. Type: "string",
  5772. },
  5773. },
  5774. },
  5775. },
  5776. },
  5777. },
  5778. UseJSONNamesForFields: true,
  5779. },
  5780. {
  5781. descr: "JSONSchema with a read_only nested field",
  5782. msgDescs: []*descriptorpb.DescriptorProto{
  5783. {
  5784. Name: proto.String("Message"),
  5785. Field: []*descriptorpb.FieldDescriptorProto{
  5786. {
  5787. Name: proto.String("nested"),
  5788. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  5789. TypeName: proto.String(".example.Message.Nested"),
  5790. Number: proto.Int32(1),
  5791. Options: fieldBehaviorOutputOnlyOptions,
  5792. },
  5793. },
  5794. NestedType: []*descriptorpb.DescriptorProto{{
  5795. Name: proto.String("Nested"),
  5796. }},
  5797. },
  5798. },
  5799. UseAllOfForRefs: true,
  5800. schema: map[string]*openapi_options.Schema{
  5801. "Message": {
  5802. JsonSchema: &openapi_options.JSONSchema{
  5803. Title: "title",
  5804. Description: "desc",
  5805. Required: []string{},
  5806. },
  5807. },
  5808. },
  5809. openAPIOptions: &openapiconfig.OpenAPIOptions{
  5810. Field: []*openapiconfig.OpenAPIFieldOption{
  5811. {
  5812. Field: "example.Message.nested",
  5813. Option: &openapi_options.JSONSchema{
  5814. Title: "nested field title",
  5815. Description: "nested field desc",
  5816. Example: `"ok":"TRUE"`,
  5817. },
  5818. },
  5819. },
  5820. },
  5821. defs: map[string]openapiSchemaObject{
  5822. "exampleMessage": {
  5823. schemaCore: schemaCore{
  5824. Type: "object",
  5825. },
  5826. Title: "title",
  5827. Description: "desc",
  5828. Required: nil,
  5829. Properties: &openapiSchemaObjectProperties{
  5830. {
  5831. Key: "nested",
  5832. Value: openapiSchemaObject{
  5833. AllOf: []allOfEntry{{Ref: "#/definitions/MessageNested"}},
  5834. ReadOnly: true,
  5835. schemaCore: schemaCore{
  5836. Example: RawExample(`"ok":"TRUE"`),
  5837. },
  5838. Title: "nested field title",
  5839. Description: "nested field desc",
  5840. },
  5841. },
  5842. },
  5843. },
  5844. },
  5845. },
  5846. }
  5847. for _, test := range tests {
  5848. t.Run(test.descr, func(t *testing.T) {
  5849. msgs := []*descriptor.Message{}
  5850. for _, msgdesc := range test.msgDescs {
  5851. msgdesc.Options = &descriptorpb.MessageOptions{}
  5852. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  5853. }
  5854. reg := descriptor.NewRegistry()
  5855. file := descriptor.File{
  5856. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  5857. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  5858. Name: proto.String("example.proto"),
  5859. Package: proto.String("example"),
  5860. Dependency: []string{},
  5861. MessageType: test.msgDescs,
  5862. EnumType: []*descriptorpb.EnumDescriptorProto{},
  5863. Service: []*descriptorpb.ServiceDescriptorProto{},
  5864. Options: &descriptorpb.FileOptions{
  5865. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  5866. },
  5867. },
  5868. Messages: msgs,
  5869. }
  5870. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  5871. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  5872. })
  5873. reg.SetVisibilityRestrictionSelectors([]string{"PREVIEW"})
  5874. if test.UseJSONNamesForFields {
  5875. reg.SetUseJSONNamesForFields(true)
  5876. }
  5877. if test.UseAllOfForRefs {
  5878. reg.SetUseAllOfForRefs(true)
  5879. }
  5880. if err != nil {
  5881. t.Fatalf("failed to load code generator request: %v", err)
  5882. }
  5883. msgMap := map[string]*descriptor.Message{}
  5884. for _, d := range test.msgDescs {
  5885. name := d.GetName()
  5886. msg, err := reg.LookupMsg("example", name)
  5887. if err != nil {
  5888. t.Fatalf("lookup message %v: %v", name, err)
  5889. }
  5890. msgMap[msg.FQMN()] = msg
  5891. if schema, ok := test.schema[name]; ok {
  5892. proto.SetExtension(d.Options, openapi_options.E_Openapiv2Schema, schema)
  5893. }
  5894. }
  5895. if test.openAPIOptions != nil {
  5896. if err := reg.RegisterOpenAPIOptions(test.openAPIOptions); err != nil {
  5897. t.Fatalf("failed to register OpenAPI options: %s", err)
  5898. }
  5899. }
  5900. refs := make(refMap)
  5901. actual := make(openapiDefinitionsObject)
  5902. if err := renderMessagesAsDefinition(msgMap, actual, reg, refs, test.pathParams); err != nil {
  5903. t.Errorf("renderMessagesAsDefinition failed with: %s", err)
  5904. }
  5905. if !reflect.DeepEqual(actual, test.defs) {
  5906. t.Errorf("Expected renderMessagesAsDefinition() to add defs %+v, not %+v", test.defs, actual)
  5907. }
  5908. })
  5909. }
  5910. }
  5911. func TestUpdateOpenAPIDataFromComments(t *testing.T) {
  5912. tests := []struct {
  5913. descr string
  5914. openapiSwaggerObject interface{}
  5915. comments string
  5916. expectedError error
  5917. expectedOpenAPIObject interface{}
  5918. useGoTemplate bool
  5919. goTemplateArgs []string
  5920. }{
  5921. {
  5922. descr: "empty comments",
  5923. openapiSwaggerObject: nil,
  5924. expectedOpenAPIObject: nil,
  5925. comments: "",
  5926. expectedError: nil,
  5927. },
  5928. {
  5929. descr: "set field to read only",
  5930. openapiSwaggerObject: &openapiSchemaObject{},
  5931. expectedOpenAPIObject: &openapiSchemaObject{
  5932. ReadOnly: true,
  5933. Description: "... Output only. ...",
  5934. },
  5935. comments: "... Output only. ...",
  5936. expectedError: nil,
  5937. },
  5938. {
  5939. descr: "set title",
  5940. openapiSwaggerObject: &openapiSchemaObject{},
  5941. expectedOpenAPIObject: &openapiSchemaObject{
  5942. Title: "Comment with no trailing dot",
  5943. },
  5944. comments: "Comment with no trailing dot",
  5945. expectedError: nil,
  5946. },
  5947. {
  5948. descr: "set description",
  5949. openapiSwaggerObject: &openapiSchemaObject{},
  5950. expectedOpenAPIObject: &openapiSchemaObject{
  5951. Description: "Comment with trailing dot.",
  5952. },
  5953. comments: "Comment with trailing dot.",
  5954. expectedError: nil,
  5955. },
  5956. {
  5957. descr: "use info object",
  5958. openapiSwaggerObject: &openapiSwaggerObject{
  5959. Info: openapiInfoObject{},
  5960. },
  5961. expectedOpenAPIObject: &openapiSwaggerObject{
  5962. Info: openapiInfoObject{
  5963. Description: "Comment with trailing dot.",
  5964. },
  5965. },
  5966. comments: "Comment with trailing dot.",
  5967. expectedError: nil,
  5968. },
  5969. {
  5970. descr: "multi line comment with title",
  5971. openapiSwaggerObject: &openapiSchemaObject{},
  5972. expectedOpenAPIObject: &openapiSchemaObject{
  5973. Title: "First line",
  5974. Description: "Second line",
  5975. },
  5976. comments: "First line\n\nSecond line",
  5977. expectedError: nil,
  5978. },
  5979. {
  5980. descr: "multi line comment no title",
  5981. openapiSwaggerObject: &openapiSchemaObject{},
  5982. expectedOpenAPIObject: &openapiSchemaObject{
  5983. Description: "First line.\n\nSecond line",
  5984. },
  5985. comments: "First line.\n\nSecond line",
  5986. expectedError: nil,
  5987. },
  5988. {
  5989. descr: "multi line comment with summary with dot",
  5990. openapiSwaggerObject: &openapiOperationObject{},
  5991. expectedOpenAPIObject: &openapiOperationObject{
  5992. Summary: "First line.",
  5993. Description: "Second line",
  5994. },
  5995. comments: "First line.\n\nSecond line",
  5996. expectedError: nil,
  5997. },
  5998. {
  5999. descr: "multi line comment with summary no dot",
  6000. openapiSwaggerObject: &openapiOperationObject{},
  6001. expectedOpenAPIObject: &openapiOperationObject{
  6002. Summary: "First line",
  6003. Description: "Second line",
  6004. },
  6005. comments: "First line\n\nSecond line",
  6006. expectedError: nil,
  6007. },
  6008. {
  6009. descr: "multi line comment with summary no dot",
  6010. openapiSwaggerObject: &schemaCore{},
  6011. expectedOpenAPIObject: &schemaCore{},
  6012. comments: "Any comment",
  6013. expectedError: errors.New("no description nor summary property"),
  6014. },
  6015. {
  6016. descr: "without use_go_template",
  6017. openapiSwaggerObject: &openapiSchemaObject{},
  6018. expectedOpenAPIObject: &openapiSchemaObject{
  6019. Title: "First line",
  6020. Description: "{{import \"documentation.md\"}}",
  6021. },
  6022. comments: "First line\n\n{{import \"documentation.md\"}}",
  6023. expectedError: nil,
  6024. },
  6025. {
  6026. descr: "error with use_go_template",
  6027. openapiSwaggerObject: &openapiSchemaObject{},
  6028. expectedOpenAPIObject: &openapiSchemaObject{
  6029. Title: "First line",
  6030. Description: "open noneexistingfile.txt: no such file or directory",
  6031. },
  6032. comments: "First line\n\n{{import \"noneexistingfile.txt\"}}",
  6033. expectedError: nil,
  6034. useGoTemplate: true,
  6035. },
  6036. {
  6037. descr: "template with use_go_template",
  6038. openapiSwaggerObject: &openapiSchemaObject{},
  6039. expectedOpenAPIObject: &openapiSchemaObject{
  6040. Title: "Template",
  6041. Description: `Description "which means nothing"`,
  6042. },
  6043. comments: "Template\n\nDescription {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6044. expectedError: nil,
  6045. useGoTemplate: true,
  6046. },
  6047. {
  6048. descr: "template with use_go_template and go_template_args",
  6049. openapiSwaggerObject: &openapiSchemaObject{},
  6050. expectedOpenAPIObject: &openapiSchemaObject{
  6051. Title: "Template",
  6052. Description: `Description "which means nothing" for environment test with value my_value`,
  6053. },
  6054. comments: "Template\n\nDescription {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}} for " +
  6055. "environment {{arg \"environment\"}} with value {{arg \"my_key\"}}",
  6056. expectedError: nil,
  6057. useGoTemplate: true,
  6058. goTemplateArgs: []string{"my_key=my_value", "environment=test"},
  6059. },
  6060. {
  6061. descr: "template with use_go_template and undefined go_template_args",
  6062. openapiSwaggerObject: &openapiSchemaObject{},
  6063. expectedOpenAPIObject: &openapiSchemaObject{
  6064. Title: "Template",
  6065. Description: `Description "which means nothing" for environment test with value ` +
  6066. `goTemplateArg something_undefined not found`,
  6067. },
  6068. comments: "Template\n\nDescription {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}} for " +
  6069. "environment {{arg \"environment\"}} with value {{arg \"something_undefined\"}}",
  6070. expectedError: nil,
  6071. useGoTemplate: true,
  6072. goTemplateArgs: []string{"environment=test"},
  6073. },
  6074. }
  6075. for _, test := range tests {
  6076. t.Run(test.descr, func(t *testing.T) {
  6077. reg := descriptor.NewRegistry()
  6078. if test.useGoTemplate {
  6079. reg.SetUseGoTemplate(true)
  6080. }
  6081. if len(test.goTemplateArgs) > 0 {
  6082. reg.SetGoTemplateArgs(test.goTemplateArgs)
  6083. }
  6084. err := updateOpenAPIDataFromComments(reg, test.openapiSwaggerObject, nil, test.comments, false)
  6085. if test.expectedError == nil {
  6086. if err != nil {
  6087. t.Errorf("unexpected error '%v'", err)
  6088. }
  6089. if !reflect.DeepEqual(test.openapiSwaggerObject, test.expectedOpenAPIObject) {
  6090. t.Errorf("openapiSwaggerObject was not updated correctly, expected '%+v', got '%+v'", test.expectedOpenAPIObject, test.openapiSwaggerObject)
  6091. }
  6092. } else {
  6093. if err == nil {
  6094. t.Error("expected update error not returned")
  6095. }
  6096. if !reflect.DeepEqual(test.openapiSwaggerObject, test.expectedOpenAPIObject) {
  6097. t.Errorf("openapiSwaggerObject was not updated correctly, expected '%+v', got '%+v'", test.expectedOpenAPIObject, test.openapiSwaggerObject)
  6098. }
  6099. if err.Error() != test.expectedError.Error() {
  6100. t.Errorf("expected error malformed, expected %q, got %q", test.expectedError.Error(), err.Error())
  6101. }
  6102. }
  6103. })
  6104. }
  6105. }
  6106. func TestMessageOptionsWithGoTemplate(t *testing.T) {
  6107. tests := []struct {
  6108. descr string
  6109. msgDescs []*descriptorpb.DescriptorProto
  6110. schema map[string]*openapi_options.Schema // per-message schema to add
  6111. defs openapiDefinitionsObject
  6112. openAPIOptions *openapiconfig.OpenAPIOptions
  6113. useGoTemplate bool
  6114. goTemplateArgs []string
  6115. }{
  6116. {
  6117. descr: "external docs option",
  6118. msgDescs: []*descriptorpb.DescriptorProto{
  6119. {Name: proto.String("Message")},
  6120. },
  6121. schema: map[string]*openapi_options.Schema{
  6122. "Message": {
  6123. JsonSchema: &openapi_options.JSONSchema{
  6124. Title: "{{.Name}}",
  6125. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6126. },
  6127. ExternalDocs: &openapi_options.ExternalDocumentation{
  6128. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6129. },
  6130. },
  6131. },
  6132. defs: map[string]openapiSchemaObject{
  6133. "Message": {
  6134. schemaCore: schemaCore{
  6135. Type: "object",
  6136. },
  6137. Title: "Message",
  6138. Description: `Description "which means nothing"`,
  6139. ExternalDocs: &openapiExternalDocumentationObject{
  6140. Description: `Description "which means nothing"`,
  6141. },
  6142. },
  6143. },
  6144. useGoTemplate: true,
  6145. },
  6146. {
  6147. descr: "external docs option",
  6148. msgDescs: []*descriptorpb.DescriptorProto{
  6149. {Name: proto.String("Message")},
  6150. },
  6151. schema: map[string]*openapi_options.Schema{
  6152. "Message": {
  6153. JsonSchema: &openapi_options.JSONSchema{
  6154. Title: "{{.Name}}",
  6155. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6156. },
  6157. ExternalDocs: &openapi_options.ExternalDocumentation{
  6158. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6159. },
  6160. },
  6161. },
  6162. defs: map[string]openapiSchemaObject{
  6163. "Message": {
  6164. schemaCore: schemaCore{
  6165. Type: "object",
  6166. },
  6167. Title: "{{.Name}}",
  6168. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6169. ExternalDocs: &openapiExternalDocumentationObject{
  6170. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6171. },
  6172. },
  6173. },
  6174. useGoTemplate: false,
  6175. },
  6176. {
  6177. descr: "external docs option with go template args",
  6178. msgDescs: []*descriptorpb.DescriptorProto{
  6179. {Name: proto.String("Message")},
  6180. },
  6181. schema: map[string]*openapi_options.Schema{
  6182. "Message": {
  6183. JsonSchema: &openapi_options.JSONSchema{
  6184. Title: "{{.Name}}",
  6185. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}} " +
  6186. "{{arg \"my_key\"}}",
  6187. },
  6188. ExternalDocs: &openapi_options.ExternalDocumentation{
  6189. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}} " +
  6190. "{{arg \"my_key\"}}",
  6191. },
  6192. },
  6193. },
  6194. defs: map[string]openapiSchemaObject{
  6195. "Message": {
  6196. schemaCore: schemaCore{
  6197. Type: "object",
  6198. },
  6199. Title: "Message",
  6200. Description: `Description "which means nothing" too`,
  6201. ExternalDocs: &openapiExternalDocumentationObject{
  6202. Description: `Description "which means nothing" too`,
  6203. },
  6204. },
  6205. },
  6206. useGoTemplate: true,
  6207. goTemplateArgs: []string{"my_key=too"},
  6208. },
  6209. {
  6210. descr: "registered OpenAPIOption",
  6211. msgDescs: []*descriptorpb.DescriptorProto{
  6212. {Name: proto.String("Message")},
  6213. },
  6214. openAPIOptions: &openapiconfig.OpenAPIOptions{
  6215. Message: []*openapiconfig.OpenAPIMessageOption{
  6216. {
  6217. Message: "example.Message",
  6218. Option: &openapi_options.Schema{
  6219. JsonSchema: &openapi_options.JSONSchema{
  6220. Title: "{{.Name}}",
  6221. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6222. },
  6223. ExternalDocs: &openapi_options.ExternalDocumentation{
  6224. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}}",
  6225. },
  6226. },
  6227. },
  6228. },
  6229. },
  6230. defs: map[string]openapiSchemaObject{
  6231. "Message": {
  6232. schemaCore: schemaCore{
  6233. Type: "object",
  6234. },
  6235. Title: "Message",
  6236. Description: `Description "which means nothing"`,
  6237. ExternalDocs: &openapiExternalDocumentationObject{
  6238. Description: `Description "which means nothing"`,
  6239. },
  6240. },
  6241. },
  6242. useGoTemplate: true,
  6243. },
  6244. {
  6245. descr: "registered OpenAPIOption with go template args",
  6246. msgDescs: []*descriptorpb.DescriptorProto{
  6247. {Name: proto.String("Message")},
  6248. },
  6249. openAPIOptions: &openapiconfig.OpenAPIOptions{
  6250. Message: []*openapiconfig.OpenAPIMessageOption{
  6251. {
  6252. Message: "example.Message",
  6253. Option: &openapi_options.Schema{
  6254. JsonSchema: &openapi_options.JSONSchema{
  6255. Title: "{{.Name}}",
  6256. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}} " +
  6257. "{{arg \"my_key\"}}",
  6258. },
  6259. ExternalDocs: &openapi_options.ExternalDocumentation{
  6260. Description: "Description {{with \"which means nothing\"}}{{printf \"%q\" .}}{{end}} " +
  6261. "{{arg \"my_key\"}}",
  6262. },
  6263. },
  6264. },
  6265. },
  6266. },
  6267. defs: map[string]openapiSchemaObject{
  6268. "Message": {
  6269. schemaCore: schemaCore{
  6270. Type: "object",
  6271. },
  6272. Title: "Message",
  6273. Description: `Description "which means nothing" too`,
  6274. ExternalDocs: &openapiExternalDocumentationObject{
  6275. Description: `Description "which means nothing" too`,
  6276. },
  6277. },
  6278. },
  6279. useGoTemplate: true,
  6280. goTemplateArgs: []string{"my_key=too"},
  6281. },
  6282. }
  6283. for _, test := range tests {
  6284. t.Run(test.descr, func(t *testing.T) {
  6285. msgs := []*descriptor.Message{}
  6286. for _, msgdesc := range test.msgDescs {
  6287. msgdesc.Options = &descriptorpb.MessageOptions{}
  6288. msgs = append(msgs, &descriptor.Message{DescriptorProto: msgdesc})
  6289. }
  6290. reg := descriptor.NewRegistry()
  6291. reg.SetUseGoTemplate(test.useGoTemplate)
  6292. reg.SetGoTemplateArgs(test.goTemplateArgs)
  6293. file := descriptor.File{
  6294. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  6295. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  6296. Name: proto.String("example.proto"),
  6297. Package: proto.String("example"),
  6298. Dependency: []string{},
  6299. MessageType: test.msgDescs,
  6300. EnumType: []*descriptorpb.EnumDescriptorProto{},
  6301. Service: []*descriptorpb.ServiceDescriptorProto{},
  6302. Options: &descriptorpb.FileOptions{
  6303. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  6304. },
  6305. },
  6306. Messages: msgs,
  6307. }
  6308. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  6309. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  6310. })
  6311. if err != nil {
  6312. t.Fatalf("failed to load code generator request: %v", err)
  6313. }
  6314. msgMap := map[string]*descriptor.Message{}
  6315. for _, d := range test.msgDescs {
  6316. name := d.GetName()
  6317. msg, err := reg.LookupMsg("example", name)
  6318. if err != nil {
  6319. t.Fatalf("lookup message %v: %v", name, err)
  6320. }
  6321. msgMap[msg.FQMN()] = msg
  6322. if schema, ok := test.schema[name]; ok {
  6323. proto.SetExtension(d.Options, openapi_options.E_Openapiv2Schema, schema)
  6324. }
  6325. }
  6326. if test.openAPIOptions != nil {
  6327. if err := reg.RegisterOpenAPIOptions(test.openAPIOptions); err != nil {
  6328. t.Fatalf("failed to register OpenAPI options: %s", err)
  6329. }
  6330. }
  6331. refs := make(refMap)
  6332. actual := make(openapiDefinitionsObject)
  6333. if err := renderMessagesAsDefinition(msgMap, actual, reg, refs, nil); err != nil {
  6334. t.Errorf("renderMessagesAsDefinition failed with: %s", err)
  6335. }
  6336. if !reflect.DeepEqual(actual, test.defs) {
  6337. t.Errorf("Expected renderMessagesAsDefinition() to add defs %+v, not %+v", test.defs, actual)
  6338. }
  6339. })
  6340. }
  6341. }
  6342. func TestTagsWithGoTemplate(t *testing.T) {
  6343. reg := descriptor.NewRegistry()
  6344. reg.SetUseGoTemplate(true)
  6345. reg.SetGoTemplateArgs([]string{"my_key=my_value"})
  6346. svc := &descriptorpb.ServiceDescriptorProto{
  6347. Name: proto.String("ExampleService"),
  6348. Options: &descriptorpb.ServiceOptions{},
  6349. }
  6350. file := descriptor.File{
  6351. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  6352. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  6353. Name: proto.String("example.proto"),
  6354. Package: proto.String("example"),
  6355. Dependency: []string{},
  6356. MessageType: []*descriptorpb.DescriptorProto{},
  6357. EnumType: []*descriptorpb.EnumDescriptorProto{},
  6358. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  6359. Options: &descriptorpb.FileOptions{
  6360. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  6361. },
  6362. },
  6363. Messages: []*descriptor.Message{},
  6364. Services: []*descriptor.Service{
  6365. {
  6366. ServiceDescriptorProto: svc,
  6367. },
  6368. },
  6369. }
  6370. // Set tag through service extension
  6371. proto.SetExtension(file.GetService()[0].Options, openapi_options.E_Openapiv2Tag, &openapi_options.Tag{
  6372. Name: "service tag",
  6373. Description: "{{ .Name }}!",
  6374. })
  6375. // Set tags through file extension
  6376. swagger := openapi_options.Swagger{
  6377. Tags: []*openapi_options.Tag{
  6378. {
  6379. Name: "not a service tag",
  6380. Description: "{{ import \"file\" }}",
  6381. },
  6382. {
  6383. Name: "ExampleService",
  6384. Description: "ExampleService!",
  6385. },
  6386. {
  6387. Name: "not a service tag 2",
  6388. Description: "{{ import \"file\" }}",
  6389. },
  6390. {
  6391. Name: "Service with my_key",
  6392. Description: "the {{arg \"my_key\"}}",
  6393. },
  6394. },
  6395. }
  6396. proto.SetExtension(proto.Message(file.FileDescriptorProto.Options), openapi_options.E_Openapiv2Swagger, &swagger)
  6397. actual, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  6398. if err != nil {
  6399. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  6400. }
  6401. expectedTags := []openapiTagObject{
  6402. {
  6403. Name: "not a service tag",
  6404. Description: "open file: no such file or directory",
  6405. },
  6406. {
  6407. Name: "ExampleService",
  6408. Description: "ExampleService!",
  6409. },
  6410. {
  6411. Name: "not a service tag 2",
  6412. Description: "open file: no such file or directory",
  6413. },
  6414. {
  6415. Name: "Service with my_key",
  6416. Description: "the my_value",
  6417. },
  6418. {
  6419. Name: "service tag",
  6420. Description: "ExampleService!",
  6421. },
  6422. }
  6423. if !reflect.DeepEqual(actual.Tags, expectedTags) {
  6424. t.Errorf("Expected tags %+v, not %+v", expectedTags, actual.Tags)
  6425. }
  6426. }
  6427. func TestTemplateWithoutErrorDefinition(t *testing.T) {
  6428. msgdesc := &descriptorpb.DescriptorProto{
  6429. Name: proto.String("ExampleMessage"),
  6430. Field: []*descriptorpb.FieldDescriptorProto{},
  6431. }
  6432. meth := &descriptorpb.MethodDescriptorProto{
  6433. Name: proto.String("Echo"),
  6434. InputType: proto.String("ExampleMessage"),
  6435. OutputType: proto.String("ExampleMessage"),
  6436. }
  6437. svc := &descriptorpb.ServiceDescriptorProto{
  6438. Name: proto.String("ExampleService"),
  6439. Method: []*descriptorpb.MethodDescriptorProto{meth},
  6440. }
  6441. msg := &descriptor.Message{
  6442. DescriptorProto: msgdesc,
  6443. }
  6444. file := descriptor.File{
  6445. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  6446. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  6447. Name: proto.String("example.proto"),
  6448. Package: proto.String("example"),
  6449. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  6450. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  6451. Options: &descriptorpb.FileOptions{
  6452. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  6453. },
  6454. },
  6455. GoPkg: descriptor.GoPackage{
  6456. Path: "example.com/path/to/example/example.pb",
  6457. Name: "example_pb",
  6458. },
  6459. Messages: []*descriptor.Message{msg},
  6460. Services: []*descriptor.Service{
  6461. {
  6462. ServiceDescriptorProto: svc,
  6463. Methods: []*descriptor.Method{
  6464. {
  6465. MethodDescriptorProto: meth,
  6466. RequestType: msg,
  6467. ResponseType: msg,
  6468. Bindings: []*descriptor.Binding{
  6469. {
  6470. HTTPMethod: "POST",
  6471. PathTmpl: httprule.Template{
  6472. Version: 1,
  6473. OpCodes: []int{0, 0},
  6474. Template: "/v1/echo",
  6475. },
  6476. Body: &descriptor.Body{
  6477. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  6478. },
  6479. },
  6480. },
  6481. },
  6482. },
  6483. },
  6484. },
  6485. }
  6486. reg := descriptor.NewRegistry()
  6487. err := reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto}})
  6488. if err != nil {
  6489. t.Errorf("failed to reg.Load(): %v", err)
  6490. return
  6491. }
  6492. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  6493. if err != nil {
  6494. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  6495. return
  6496. }
  6497. defRsp, ok := result.getPathItemObject("/v1/echo").Post.Responses["default"]
  6498. if !ok {
  6499. return
  6500. }
  6501. ref := defRsp.Schema.schemaCore.Ref
  6502. refName := strings.TrimPrefix(ref, "#/definitions/")
  6503. if refName == "" {
  6504. t.Fatal("created default Error response with empty reflink")
  6505. }
  6506. if _, ok := result.Definitions[refName]; !ok {
  6507. t.Errorf("default Error response with reflink '%v', but its definition was not found", refName)
  6508. }
  6509. }
  6510. func TestSingleServiceTemplateWithDuplicateHttp1Operations(t *testing.T) {
  6511. fieldType := descriptorpb.FieldDescriptorProto_TYPE_STRING
  6512. field1 := &descriptorpb.FieldDescriptorProto{
  6513. Name: proto.String("name"),
  6514. Number: proto.Int32(1),
  6515. Type: &fieldType,
  6516. }
  6517. getFooMsgDesc := &descriptorpb.DescriptorProto{
  6518. Name: proto.String("GetFooRequest"),
  6519. Field: []*descriptorpb.FieldDescriptorProto{
  6520. field1,
  6521. },
  6522. }
  6523. getFooMsg := &descriptor.Message{
  6524. DescriptorProto: getFooMsgDesc,
  6525. }
  6526. deleteFooMsgDesc := &descriptorpb.DescriptorProto{
  6527. Name: proto.String("DeleteFooRequest"),
  6528. Field: []*descriptorpb.FieldDescriptorProto{
  6529. field1,
  6530. },
  6531. }
  6532. deleteFooMsg := &descriptor.Message{
  6533. DescriptorProto: deleteFooMsgDesc,
  6534. }
  6535. getFoo := &descriptorpb.MethodDescriptorProto{
  6536. Name: proto.String("GetFoo"),
  6537. InputType: proto.String("GetFooRequest"),
  6538. OutputType: proto.String("EmptyMessage"),
  6539. }
  6540. deleteFoo := &descriptorpb.MethodDescriptorProto{
  6541. Name: proto.String("DeleteFoo"),
  6542. InputType: proto.String("DeleteFooRequest"),
  6543. OutputType: proto.String("EmptyMessage"),
  6544. }
  6545. getBarMsgDesc := &descriptorpb.DescriptorProto{
  6546. Name: proto.String("GetBarRequest"),
  6547. Field: []*descriptorpb.FieldDescriptorProto{
  6548. field1,
  6549. },
  6550. }
  6551. getBarMsg := &descriptor.Message{
  6552. DescriptorProto: getBarMsgDesc,
  6553. }
  6554. deleteBarMsgDesc := &descriptorpb.DescriptorProto{
  6555. Name: proto.String("DeleteBarRequest"),
  6556. Field: []*descriptorpb.FieldDescriptorProto{
  6557. field1,
  6558. },
  6559. }
  6560. deleteBarMsg := &descriptor.Message{
  6561. DescriptorProto: deleteBarMsgDesc,
  6562. }
  6563. getBar := &descriptorpb.MethodDescriptorProto{
  6564. Name: proto.String("GetBar"),
  6565. InputType: proto.String("GetBarRequest"),
  6566. OutputType: proto.String("EmptyMessage"),
  6567. }
  6568. deleteBar := &descriptorpb.MethodDescriptorProto{
  6569. Name: proto.String("DeleteBar"),
  6570. InputType: proto.String("DeleteBarRequest"),
  6571. OutputType: proto.String("EmptyMessage"),
  6572. }
  6573. svc1 := &descriptorpb.ServiceDescriptorProto{
  6574. Name: proto.String("Service1"),
  6575. Method: []*descriptorpb.MethodDescriptorProto{getFoo, deleteFoo, getBar, deleteBar},
  6576. }
  6577. emptyMsgDesc := &descriptorpb.DescriptorProto{
  6578. Name: proto.String("EmptyMessage"),
  6579. }
  6580. emptyMsg := &descriptor.Message{
  6581. DescriptorProto: emptyMsgDesc,
  6582. }
  6583. file := descriptor.File{
  6584. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  6585. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  6586. Name: proto.String("service1.proto"),
  6587. Package: proto.String("example"),
  6588. MessageType: []*descriptorpb.DescriptorProto{getBarMsgDesc, deleteBarMsgDesc, getFooMsgDesc, deleteFooMsgDesc, emptyMsgDesc},
  6589. Service: []*descriptorpb.ServiceDescriptorProto{svc1},
  6590. Options: &descriptorpb.FileOptions{
  6591. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  6592. },
  6593. },
  6594. GoPkg: descriptor.GoPackage{
  6595. Path: "example.com/path/to/example/example.pb",
  6596. Name: "example_pb",
  6597. },
  6598. Messages: []*descriptor.Message{getFooMsg, deleteFooMsg, getBarMsg, deleteBarMsg, emptyMsg},
  6599. Services: []*descriptor.Service{
  6600. {
  6601. ServiceDescriptorProto: svc1,
  6602. Methods: []*descriptor.Method{
  6603. {
  6604. MethodDescriptorProto: getFoo,
  6605. RequestType: getFooMsg,
  6606. ResponseType: getFooMsg,
  6607. Bindings: []*descriptor.Binding{
  6608. {
  6609. HTTPMethod: "GET",
  6610. PathTmpl: httprule.Template{
  6611. Version: 1,
  6612. OpCodes: []int{0, 0},
  6613. Template: "/v1/{name=foos/*}",
  6614. },
  6615. PathParams: []descriptor.Parameter{
  6616. {
  6617. Target: &descriptor.Field{
  6618. FieldDescriptorProto: field1,
  6619. Message: getFooMsg,
  6620. },
  6621. FieldPath: descriptor.FieldPath{
  6622. {
  6623. Name: "name",
  6624. },
  6625. },
  6626. },
  6627. },
  6628. Body: &descriptor.Body{
  6629. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  6630. },
  6631. },
  6632. },
  6633. },
  6634. {
  6635. MethodDescriptorProto: deleteFoo,
  6636. RequestType: deleteFooMsg,
  6637. ResponseType: emptyMsg,
  6638. Bindings: []*descriptor.Binding{
  6639. {
  6640. HTTPMethod: "DELETE",
  6641. PathTmpl: httprule.Template{
  6642. Version: 1,
  6643. OpCodes: []int{0, 0},
  6644. Template: "/v1/{name=foos/*}",
  6645. },
  6646. PathParams: []descriptor.Parameter{
  6647. {
  6648. Target: &descriptor.Field{
  6649. FieldDescriptorProto: field1,
  6650. Message: deleteFooMsg,
  6651. },
  6652. FieldPath: descriptor.FieldPath{
  6653. {
  6654. Name: "name",
  6655. },
  6656. },
  6657. },
  6658. },
  6659. Body: &descriptor.Body{
  6660. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  6661. },
  6662. },
  6663. },
  6664. },
  6665. {
  6666. MethodDescriptorProto: getBar,
  6667. RequestType: getBarMsg,
  6668. ResponseType: getBarMsg,
  6669. Bindings: []*descriptor.Binding{
  6670. {
  6671. HTTPMethod: "GET",
  6672. PathTmpl: httprule.Template{
  6673. Version: 1,
  6674. OpCodes: []int{0, 0},
  6675. Template: "/v1/{name=bars/*}",
  6676. },
  6677. PathParams: []descriptor.Parameter{
  6678. {
  6679. Target: &descriptor.Field{
  6680. FieldDescriptorProto: field1,
  6681. Message: getBarMsg,
  6682. },
  6683. FieldPath: descriptor.FieldPath{
  6684. {
  6685. Name: "name",
  6686. },
  6687. },
  6688. },
  6689. },
  6690. Body: &descriptor.Body{
  6691. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  6692. },
  6693. },
  6694. },
  6695. },
  6696. {
  6697. MethodDescriptorProto: deleteBar,
  6698. RequestType: deleteBarMsg,
  6699. ResponseType: emptyMsg,
  6700. Bindings: []*descriptor.Binding{
  6701. {
  6702. HTTPMethod: "DELETE",
  6703. PathTmpl: httprule.Template{
  6704. Version: 1,
  6705. OpCodes: []int{0, 0},
  6706. Template: "/v1/{name=bars/*}",
  6707. },
  6708. PathParams: []descriptor.Parameter{
  6709. {
  6710. Target: &descriptor.Field{
  6711. FieldDescriptorProto: field1,
  6712. Message: deleteBarMsg,
  6713. },
  6714. FieldPath: descriptor.FieldPath{
  6715. {
  6716. Name: "name",
  6717. },
  6718. },
  6719. },
  6720. },
  6721. Body: &descriptor.Body{
  6722. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  6723. },
  6724. },
  6725. },
  6726. },
  6727. },
  6728. },
  6729. },
  6730. }
  6731. reg := descriptor.NewRegistry()
  6732. err := reg.Load(reqFromFile(&file))
  6733. if err != nil {
  6734. t.Fatalf("failed to reg.Load(): %v", err)
  6735. }
  6736. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  6737. if err != nil {
  6738. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  6739. }
  6740. if got, want := len(result.Paths), 2; got != want {
  6741. t.Fatalf("Results path length differed, got %d want %d", got, want)
  6742. }
  6743. firstOpGet := result.getPathItemObject("/v1/{name}").Get
  6744. if got, want := firstOpGet.OperationID, "Service1_GetFoo"; got != want {
  6745. t.Fatalf("First operation GET id differed, got %s want %s", got, want)
  6746. }
  6747. if got, want := len(firstOpGet.Parameters), 2; got != want {
  6748. t.Fatalf("First operation GET params length differed, got %d want %d", got, want)
  6749. }
  6750. if got, want := firstOpGet.Parameters[0].Name, "name"; got != want {
  6751. t.Fatalf("First operation GET first param name differed, got %s want %s", got, want)
  6752. }
  6753. if got, want := firstOpGet.Parameters[0].Pattern, "foos/[^/]+"; got != want {
  6754. t.Fatalf("First operation GET first param pattern differed, got %s want %s", got, want)
  6755. }
  6756. if got, want := firstOpGet.Parameters[1].In, "body"; got != want {
  6757. t.Fatalf("First operation GET second param 'in' differed, got %s want %s", got, want)
  6758. }
  6759. firstOpDelete := result.getPathItemObject("/v1/{name}").Delete
  6760. if got, want := firstOpDelete.OperationID, "Service1_DeleteFoo"; got != want {
  6761. t.Fatalf("First operation id DELETE differed, got %s want %s", got, want)
  6762. }
  6763. if got, want := len(firstOpDelete.Parameters), 2; got != want {
  6764. t.Fatalf("First operation DELETE params length differed, got %d want %d", got, want)
  6765. }
  6766. if got, want := firstOpDelete.Parameters[0].Name, "name"; got != want {
  6767. t.Fatalf("First operation DELETE first param name differed, got %s want %s", got, want)
  6768. }
  6769. if got, want := firstOpDelete.Parameters[0].Pattern, "foos/[^/]+"; got != want {
  6770. t.Fatalf("First operation DELETE first param pattern differed, got %s want %s", got, want)
  6771. }
  6772. if got, want := firstOpDelete.Parameters[1].In, "body"; got != want {
  6773. t.Fatalf("First operation DELETE second param 'in' differed, got %s want %s", got, want)
  6774. }
  6775. secondOpGet := result.getPathItemObject("/v1/{name" + pathParamUniqueSuffixDeliminator + "1}").Get
  6776. if got, want := secondOpGet.OperationID, "Service1_GetBar"; got != want {
  6777. t.Fatalf("Second operation id GET differed, got %s want %s", got, want)
  6778. }
  6779. if got, want := len(secondOpGet.Parameters), 2; got != want {
  6780. t.Fatalf("Second operation GET params length differed, got %d want %d", got, want)
  6781. }
  6782. if got, want := secondOpGet.Parameters[0].Name, "name"+pathParamUniqueSuffixDeliminator+"1"; got != want {
  6783. t.Fatalf("Second operation GET first param name differed, got %s want %s", got, want)
  6784. }
  6785. if got, want := secondOpGet.Parameters[0].Pattern, "bars/[^/]+"; got != want {
  6786. t.Fatalf("Second operation GET first param pattern differed, got %s want %s", got, want)
  6787. }
  6788. if got, want := secondOpGet.Parameters[1].In, "body"; got != want {
  6789. t.Fatalf("Second operation GET second param 'in' differed, got %s want %s", got, want)
  6790. }
  6791. secondOpDelete := result.getPathItemObject("/v1/{name" + pathParamUniqueSuffixDeliminator + "1}").Delete
  6792. if got, want := secondOpDelete.OperationID, "Service1_DeleteBar"; got != want {
  6793. t.Fatalf("Second operation id differed, got %s want %s", got, want)
  6794. }
  6795. if got, want := len(secondOpDelete.Parameters), 2; got != want {
  6796. t.Fatalf("Second operation params length differed, got %d want %d", got, want)
  6797. }
  6798. if got, want := secondOpDelete.Parameters[0].Name, "name"+pathParamUniqueSuffixDeliminator+"1"; got != want {
  6799. t.Fatalf("Second operation first param name differed, got %s want %s", got, want)
  6800. }
  6801. if got, want := secondOpDelete.Parameters[0].Pattern, "bars/[^/]+"; got != want {
  6802. t.Fatalf("Second operation first param pattern differed, got %s want %s", got, want)
  6803. }
  6804. if got, want := secondOpDelete.Parameters[1].In, "body"; got != want {
  6805. t.Fatalf("Second operation third param 'in' differed, got %s want %s", got, want)
  6806. }
  6807. }
  6808. func getOperation(pathItem openapiPathItemObject, httpMethod string) *openapiOperationObject {
  6809. switch httpMethod {
  6810. case "GET":
  6811. return pathItem.Get
  6812. case "POST":
  6813. return pathItem.Post
  6814. case "PUT":
  6815. return pathItem.Put
  6816. case "DELETE":
  6817. return pathItem.Delete
  6818. case "PATCH":
  6819. return pathItem.Patch
  6820. case "HEAD":
  6821. return pathItem.Head
  6822. case "OPTIONS":
  6823. return pathItem.Options
  6824. default:
  6825. return nil
  6826. }
  6827. }
  6828. func TestSingleServiceTemplateWithDuplicateInAllSupportedHttp1Operations(t *testing.T) {
  6829. supportedMethods := []string{"GET", "POST", "PUT", "PATCH", "DELETE", "HEAD", "OPTIONS"}
  6830. for _, method := range supportedMethods {
  6831. fieldType := descriptorpb.FieldDescriptorProto_TYPE_STRING
  6832. field1 := &descriptorpb.FieldDescriptorProto{
  6833. Name: proto.String("name"),
  6834. Number: proto.Int32(1),
  6835. Type: &fieldType,
  6836. }
  6837. methodFooMsgDesc := &descriptorpb.DescriptorProto{
  6838. Name: proto.String(method + "FooRequest"),
  6839. Field: []*descriptorpb.FieldDescriptorProto{
  6840. field1,
  6841. },
  6842. }
  6843. methodFooMsg := &descriptor.Message{
  6844. DescriptorProto: methodFooMsgDesc,
  6845. }
  6846. methodFoo := &descriptorpb.MethodDescriptorProto{
  6847. Name: proto.String(method + "Foo"),
  6848. InputType: proto.String(method + "FooRequest"),
  6849. OutputType: proto.String("EmptyMessage"),
  6850. }
  6851. methodBarMsgDesc := &descriptorpb.DescriptorProto{
  6852. Name: proto.String(method + "BarRequest"),
  6853. Field: []*descriptorpb.FieldDescriptorProto{
  6854. field1,
  6855. },
  6856. }
  6857. methodBarMsg := &descriptor.Message{
  6858. DescriptorProto: methodBarMsgDesc,
  6859. }
  6860. methodBar := &descriptorpb.MethodDescriptorProto{
  6861. Name: proto.String(method + "Bar"),
  6862. InputType: proto.String(method + "BarRequest"),
  6863. OutputType: proto.String("EmptyMessage"),
  6864. }
  6865. svc1 := &descriptorpb.ServiceDescriptorProto{
  6866. Name: proto.String("Service1"),
  6867. Method: []*descriptorpb.MethodDescriptorProto{methodFoo, methodBar},
  6868. }
  6869. emptyMsgDesc := &descriptorpb.DescriptorProto{
  6870. Name: proto.String("EmptyMessage"),
  6871. }
  6872. emptyMsg := &descriptor.Message{
  6873. DescriptorProto: emptyMsgDesc,
  6874. }
  6875. file := descriptor.File{
  6876. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  6877. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  6878. Name: proto.String("service1.proto"),
  6879. Package: proto.String("example"),
  6880. MessageType: []*descriptorpb.DescriptorProto{methodBarMsgDesc, methodFooMsgDesc, emptyMsgDesc},
  6881. Service: []*descriptorpb.ServiceDescriptorProto{svc1},
  6882. Options: &descriptorpb.FileOptions{
  6883. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  6884. },
  6885. },
  6886. GoPkg: descriptor.GoPackage{
  6887. Path: "example.com/path/to/example/example.pb",
  6888. Name: "example_pb",
  6889. },
  6890. Messages: []*descriptor.Message{methodFooMsg, methodBarMsg, emptyMsg},
  6891. Services: []*descriptor.Service{
  6892. {
  6893. ServiceDescriptorProto: svc1,
  6894. Methods: []*descriptor.Method{
  6895. {
  6896. MethodDescriptorProto: methodFoo,
  6897. RequestType: methodFooMsg,
  6898. ResponseType: methodFooMsg,
  6899. Bindings: []*descriptor.Binding{
  6900. {
  6901. HTTPMethod: method,
  6902. PathTmpl: httprule.Template{
  6903. Version: 1,
  6904. OpCodes: []int{0, 0},
  6905. Template: "/v1/{name=foos/*}",
  6906. },
  6907. PathParams: []descriptor.Parameter{
  6908. {
  6909. Target: &descriptor.Field{
  6910. FieldDescriptorProto: field1,
  6911. Message: methodFooMsg,
  6912. },
  6913. FieldPath: descriptor.FieldPath{
  6914. {
  6915. Name: "name",
  6916. },
  6917. },
  6918. },
  6919. },
  6920. Body: &descriptor.Body{
  6921. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  6922. },
  6923. },
  6924. },
  6925. },
  6926. {
  6927. MethodDescriptorProto: methodBar,
  6928. RequestType: methodBarMsg,
  6929. ResponseType: methodBarMsg,
  6930. Bindings: []*descriptor.Binding{
  6931. {
  6932. HTTPMethod: method,
  6933. PathTmpl: httprule.Template{
  6934. Version: 1,
  6935. OpCodes: []int{0, 0},
  6936. Template: "/v1/{name=bars/*}",
  6937. },
  6938. PathParams: []descriptor.Parameter{
  6939. {
  6940. Target: &descriptor.Field{
  6941. FieldDescriptorProto: field1,
  6942. Message: methodBarMsg,
  6943. },
  6944. FieldPath: descriptor.FieldPath{
  6945. {
  6946. Name: "name",
  6947. },
  6948. },
  6949. },
  6950. },
  6951. Body: &descriptor.Body{
  6952. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  6953. },
  6954. },
  6955. },
  6956. },
  6957. },
  6958. },
  6959. },
  6960. }
  6961. reg := descriptor.NewRegistry()
  6962. err := reg.Load(reqFromFile(&file))
  6963. if err != nil {
  6964. t.Fatalf("failed to reg.Load(): %v", err)
  6965. }
  6966. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  6967. if err != nil {
  6968. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  6969. }
  6970. if got, want := len(result.Paths), 2; got != want {
  6971. t.Fatalf("Results path length differed, got %d want %d", got, want)
  6972. }
  6973. firstOpMethod := getOperation(result.getPathItemObject("/v1/{name}"), method)
  6974. if got, want := firstOpMethod.OperationID, "Service1_"+method+"Foo"; got != want {
  6975. t.Fatalf("First operation %s id differed, got %s want %s", method, got, want)
  6976. }
  6977. if got, want := len(firstOpMethod.Parameters), 2; got != want {
  6978. t.Fatalf("First operation %s params length differed, got %d want %d", method, got, want)
  6979. }
  6980. if got, want := firstOpMethod.Parameters[0].Name, "name"; got != want {
  6981. t.Fatalf("First operation %s first param name differed, got %s want %s", method, got, want)
  6982. }
  6983. if got, want := firstOpMethod.Parameters[0].Pattern, "foos/[^/]+"; got != want {
  6984. t.Fatalf("First operation %s first param pattern differed, got %s want %s", method, got, want)
  6985. }
  6986. if got, want := firstOpMethod.Parameters[1].In, "body"; got != want {
  6987. t.Fatalf("First operation %s second param 'in' differed, got %s want %s", method, got, want)
  6988. }
  6989. secondOpMethod := getOperation(result.getPathItemObject("/v1/{name"+pathParamUniqueSuffixDeliminator+"1}"), method)
  6990. if got, want := secondOpMethod.OperationID, "Service1_"+method+"Bar"; got != want {
  6991. t.Fatalf("Second operation id %s differed, got %s want %s", method, got, want)
  6992. }
  6993. if got, want := len(secondOpMethod.Parameters), 2; got != want {
  6994. t.Fatalf("Second operation %s params length differed, got %d want %d", method, got, want)
  6995. }
  6996. if got, want := secondOpMethod.Parameters[0].Name, "name"+pathParamUniqueSuffixDeliminator+"1"; got != want {
  6997. t.Fatalf("Second operation %s first param name differed, got %s want %s", method, got, want)
  6998. }
  6999. if got, want := secondOpMethod.Parameters[0].Pattern, "bars/[^/]+"; got != want {
  7000. t.Fatalf("Second operation %s first param pattern differed, got %s want %s", method, got, want)
  7001. }
  7002. if got, want := secondOpMethod.Parameters[1].In, "body"; got != want {
  7003. t.Fatalf("Second operation %s second param 'in' differed, got %s want %s", method, got, want)
  7004. }
  7005. }
  7006. }
  7007. func TestSingleServiceTemplateWithDuplicateHttp1UnsupportedOperations(t *testing.T) {
  7008. fieldType := descriptorpb.FieldDescriptorProto_TYPE_STRING
  7009. field1 := &descriptorpb.FieldDescriptorProto{
  7010. Name: proto.String("name"),
  7011. Number: proto.Int32(1),
  7012. Type: &fieldType,
  7013. }
  7014. unsupportedFooMsgDesc := &descriptorpb.DescriptorProto{
  7015. Name: proto.String("UnsupportedFooRequest"),
  7016. Field: []*descriptorpb.FieldDescriptorProto{
  7017. field1,
  7018. },
  7019. }
  7020. unsupportedFooMsg := &descriptor.Message{
  7021. DescriptorProto: unsupportedFooMsgDesc,
  7022. }
  7023. unsupportedFoo := &descriptorpb.MethodDescriptorProto{
  7024. Name: proto.String("UnsupportedFoo"),
  7025. InputType: proto.String("UnsupportedFooRequest"),
  7026. OutputType: proto.String("EmptyMessage"),
  7027. }
  7028. unsupportedBarMsgDesc := &descriptorpb.DescriptorProto{
  7029. Name: proto.String("UnsupportedBarRequest"),
  7030. Field: []*descriptorpb.FieldDescriptorProto{
  7031. field1,
  7032. },
  7033. }
  7034. unsupportedBarMsg := &descriptor.Message{
  7035. DescriptorProto: unsupportedBarMsgDesc,
  7036. }
  7037. unsupportedBar := &descriptorpb.MethodDescriptorProto{
  7038. Name: proto.String("UnsupportedBar"),
  7039. InputType: proto.String("UnsupportedBarRequest"),
  7040. OutputType: proto.String("EmptyMessage"),
  7041. }
  7042. svc1 := &descriptorpb.ServiceDescriptorProto{
  7043. Name: proto.String("Service1"),
  7044. Method: []*descriptorpb.MethodDescriptorProto{unsupportedFoo, unsupportedBar},
  7045. }
  7046. emptyMsgDesc := &descriptorpb.DescriptorProto{
  7047. Name: proto.String("EmptyMessage"),
  7048. }
  7049. emptyMsg := &descriptor.Message{
  7050. DescriptorProto: emptyMsgDesc,
  7051. }
  7052. file := descriptor.File{
  7053. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  7054. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  7055. Name: proto.String("service1.proto"),
  7056. Package: proto.String("example"),
  7057. MessageType: []*descriptorpb.DescriptorProto{unsupportedBarMsgDesc, unsupportedFooMsgDesc, emptyMsgDesc},
  7058. Service: []*descriptorpb.ServiceDescriptorProto{svc1},
  7059. Options: &descriptorpb.FileOptions{
  7060. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  7061. },
  7062. },
  7063. GoPkg: descriptor.GoPackage{
  7064. Path: "example.com/path/to/example/example.pb",
  7065. Name: "example_pb",
  7066. },
  7067. Messages: []*descriptor.Message{unsupportedFooMsg, unsupportedBarMsg, emptyMsg},
  7068. Services: []*descriptor.Service{
  7069. {
  7070. ServiceDescriptorProto: svc1,
  7071. Methods: []*descriptor.Method{
  7072. {
  7073. MethodDescriptorProto: unsupportedFoo,
  7074. RequestType: unsupportedFooMsg,
  7075. ResponseType: unsupportedFooMsg,
  7076. Bindings: []*descriptor.Binding{
  7077. {
  7078. HTTPMethod: "UNSUPPORTED",
  7079. PathTmpl: httprule.Template{
  7080. Version: 1,
  7081. OpCodes: []int{0, 0},
  7082. Template: "/v1/{name=foos/*}",
  7083. },
  7084. PathParams: []descriptor.Parameter{
  7085. {
  7086. Target: &descriptor.Field{
  7087. FieldDescriptorProto: field1,
  7088. Message: unsupportedFooMsg,
  7089. },
  7090. FieldPath: descriptor.FieldPath{
  7091. {
  7092. Name: "name",
  7093. },
  7094. },
  7095. },
  7096. },
  7097. Body: &descriptor.Body{
  7098. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  7099. },
  7100. },
  7101. },
  7102. },
  7103. {
  7104. MethodDescriptorProto: unsupportedBar,
  7105. RequestType: unsupportedBarMsg,
  7106. ResponseType: unsupportedBarMsg,
  7107. Bindings: []*descriptor.Binding{
  7108. {
  7109. HTTPMethod: "UNSUPPORTED",
  7110. PathTmpl: httprule.Template{
  7111. Version: 1,
  7112. OpCodes: []int{0, 0},
  7113. Template: "/v1/{name=bars/*}",
  7114. },
  7115. PathParams: []descriptor.Parameter{
  7116. {
  7117. Target: &descriptor.Field{
  7118. FieldDescriptorProto: field1,
  7119. Message: unsupportedBarMsg,
  7120. },
  7121. FieldPath: descriptor.FieldPath{
  7122. {
  7123. Name: "name",
  7124. },
  7125. },
  7126. },
  7127. },
  7128. Body: &descriptor.Body{
  7129. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  7130. },
  7131. },
  7132. },
  7133. },
  7134. },
  7135. },
  7136. },
  7137. }
  7138. reg := descriptor.NewRegistry()
  7139. err := reg.Load(reqFromFile(&file))
  7140. if err != nil {
  7141. t.Fatalf("failed to reg.Load(): %v", err)
  7142. }
  7143. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  7144. if err != nil {
  7145. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  7146. }
  7147. // Just should not crash, no special handling of unsupported HTTP methods
  7148. if got, want := len(result.Paths), 1; got != want {
  7149. t.Fatalf("Results path length differed, got %d want %d", got, want)
  7150. }
  7151. }
  7152. func TestTemplateWithDuplicateHttp1Operations(t *testing.T) {
  7153. fieldType := descriptorpb.FieldDescriptorProto_TYPE_STRING
  7154. field1 := &descriptorpb.FieldDescriptorProto{
  7155. Name: proto.String("name"),
  7156. Number: proto.Int32(1),
  7157. Type: &fieldType,
  7158. }
  7159. field2 := &descriptorpb.FieldDescriptorProto{
  7160. Name: proto.String("role"),
  7161. Number: proto.Int32(2),
  7162. Type: &fieldType,
  7163. }
  7164. msgdesc := &descriptorpb.DescriptorProto{
  7165. Name: proto.String("ExampleMessage"),
  7166. Field: []*descriptorpb.FieldDescriptorProto{
  7167. field1,
  7168. field2,
  7169. },
  7170. }
  7171. meth1 := &descriptorpb.MethodDescriptorProto{
  7172. Name: proto.String("Method1"),
  7173. InputType: proto.String("ExampleMessage"),
  7174. OutputType: proto.String("ExampleMessage"),
  7175. }
  7176. meth2 := &descriptorpb.MethodDescriptorProto{
  7177. Name: proto.String("Method2"),
  7178. InputType: proto.String("ExampleMessage"),
  7179. OutputType: proto.String("ExampleMessage"),
  7180. }
  7181. svc1 := &descriptorpb.ServiceDescriptorProto{
  7182. Name: proto.String("Service1"),
  7183. Method: []*descriptorpb.MethodDescriptorProto{meth1, meth2},
  7184. }
  7185. meth3 := &descriptorpb.MethodDescriptorProto{
  7186. Name: proto.String("Method3"),
  7187. InputType: proto.String("ExampleMessage"),
  7188. OutputType: proto.String("ExampleMessage"),
  7189. }
  7190. meth4 := &descriptorpb.MethodDescriptorProto{
  7191. Name: proto.String("Method4"),
  7192. InputType: proto.String("ExampleMessage"),
  7193. OutputType: proto.String("ExampleMessage"),
  7194. }
  7195. svc2 := &descriptorpb.ServiceDescriptorProto{
  7196. Name: proto.String("Service2"),
  7197. Method: []*descriptorpb.MethodDescriptorProto{meth3, meth4},
  7198. }
  7199. msg := &descriptor.Message{
  7200. DescriptorProto: msgdesc,
  7201. }
  7202. file := descriptor.File{
  7203. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  7204. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  7205. Name: proto.String("service1.proto"),
  7206. Package: proto.String("example"),
  7207. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  7208. Service: []*descriptorpb.ServiceDescriptorProto{svc1, svc2},
  7209. Options: &descriptorpb.FileOptions{
  7210. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  7211. },
  7212. },
  7213. GoPkg: descriptor.GoPackage{
  7214. Path: "example.com/path/to/example/example.pb",
  7215. Name: "example_pb",
  7216. },
  7217. Messages: []*descriptor.Message{msg},
  7218. Services: []*descriptor.Service{
  7219. {
  7220. ServiceDescriptorProto: svc1,
  7221. Methods: []*descriptor.Method{
  7222. {
  7223. MethodDescriptorProto: meth1,
  7224. RequestType: msg,
  7225. ResponseType: msg,
  7226. Bindings: []*descriptor.Binding{
  7227. {
  7228. HTTPMethod: "GET",
  7229. PathTmpl: httprule.Template{
  7230. Version: 1,
  7231. OpCodes: []int{0, 0},
  7232. Template: "/v1/{name=organizations/*}/{role=roles/*}",
  7233. },
  7234. PathParams: []descriptor.Parameter{
  7235. {
  7236. Target: &descriptor.Field{
  7237. FieldDescriptorProto: field1,
  7238. Message: msg,
  7239. },
  7240. FieldPath: descriptor.FieldPath{
  7241. {
  7242. Name: "name",
  7243. },
  7244. },
  7245. },
  7246. {
  7247. Target: &descriptor.Field{
  7248. FieldDescriptorProto: field2,
  7249. Message: msg,
  7250. },
  7251. FieldPath: descriptor.FieldPath{
  7252. {
  7253. Name: "role",
  7254. },
  7255. },
  7256. },
  7257. },
  7258. Body: &descriptor.Body{
  7259. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  7260. },
  7261. },
  7262. },
  7263. },
  7264. {
  7265. MethodDescriptorProto: meth2,
  7266. RequestType: msg,
  7267. ResponseType: msg,
  7268. Bindings: []*descriptor.Binding{
  7269. {
  7270. HTTPMethod: "GET",
  7271. PathTmpl: httprule.Template{
  7272. Version: 1,
  7273. OpCodes: []int{0, 0},
  7274. Template: "/v1/{name=users/*}/{role=roles/*}",
  7275. },
  7276. PathParams: []descriptor.Parameter{
  7277. {
  7278. Target: &descriptor.Field{
  7279. FieldDescriptorProto: field1,
  7280. Message: msg,
  7281. },
  7282. FieldPath: descriptor.FieldPath{
  7283. {
  7284. Name: "name",
  7285. },
  7286. },
  7287. },
  7288. {
  7289. Target: &descriptor.Field{
  7290. FieldDescriptorProto: field2,
  7291. Message: msg,
  7292. },
  7293. FieldPath: descriptor.FieldPath{
  7294. {
  7295. Name: "role",
  7296. },
  7297. },
  7298. },
  7299. },
  7300. Body: &descriptor.Body{
  7301. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  7302. },
  7303. },
  7304. },
  7305. },
  7306. },
  7307. },
  7308. {
  7309. ServiceDescriptorProto: svc2,
  7310. Methods: []*descriptor.Method{
  7311. {
  7312. MethodDescriptorProto: meth3,
  7313. RequestType: msg,
  7314. ResponseType: msg,
  7315. Bindings: []*descriptor.Binding{
  7316. {
  7317. HTTPMethod: "GET",
  7318. PathTmpl: httprule.Template{
  7319. Version: 1,
  7320. OpCodes: []int{0, 0},
  7321. Template: "/v1/{name=users/*}/roles",
  7322. },
  7323. PathParams: []descriptor.Parameter{
  7324. {
  7325. Target: &descriptor.Field{
  7326. FieldDescriptorProto: field1,
  7327. Message: msg,
  7328. },
  7329. FieldPath: descriptor.FieldPath{
  7330. {
  7331. Name: "name",
  7332. },
  7333. },
  7334. },
  7335. },
  7336. Body: &descriptor.Body{
  7337. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  7338. },
  7339. },
  7340. },
  7341. },
  7342. {
  7343. MethodDescriptorProto: meth4,
  7344. RequestType: msg,
  7345. ResponseType: msg,
  7346. Bindings: []*descriptor.Binding{
  7347. {
  7348. HTTPMethod: "GET",
  7349. PathTmpl: httprule.Template{
  7350. Version: 1,
  7351. OpCodes: []int{0, 0},
  7352. Template: "/v1/{name=groups/*}/{role=roles/*}",
  7353. },
  7354. PathParams: []descriptor.Parameter{
  7355. {
  7356. Target: &descriptor.Field{
  7357. FieldDescriptorProto: field1,
  7358. Message: msg,
  7359. },
  7360. FieldPath: descriptor.FieldPath{
  7361. {
  7362. Name: "name",
  7363. },
  7364. },
  7365. },
  7366. {
  7367. Target: &descriptor.Field{
  7368. FieldDescriptorProto: field2,
  7369. Message: msg,
  7370. },
  7371. FieldPath: descriptor.FieldPath{
  7372. {
  7373. Name: "role",
  7374. },
  7375. },
  7376. },
  7377. },
  7378. Body: &descriptor.Body{
  7379. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  7380. },
  7381. },
  7382. },
  7383. },
  7384. },
  7385. },
  7386. },
  7387. }
  7388. reg := descriptor.NewRegistry()
  7389. err := reg.Load(reqFromFile(&file))
  7390. if err != nil {
  7391. t.Fatalf("failed to reg.Load(): %v", err)
  7392. }
  7393. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  7394. if err != nil {
  7395. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  7396. }
  7397. if got, want := len(result.Paths), 4; got != want {
  7398. t.Fatalf("Results path length differed, got %d want %d", got, want)
  7399. }
  7400. firstOp := result.getPathItemObject("/v1/{name}/{role}").Get
  7401. if got, want := firstOp.OperationID, "Service1_Method1"; got != want {
  7402. t.Fatalf("First operation id differed, got %s want %s", got, want)
  7403. }
  7404. if got, want := len(firstOp.Parameters), 3; got != want {
  7405. t.Fatalf("First operation params length differed, got %d want %d", got, want)
  7406. }
  7407. if got, want := firstOp.Parameters[0].Name, "name"; got != want {
  7408. t.Fatalf("First operation first param name differed, got %s want %s", got, want)
  7409. }
  7410. if got, want := firstOp.Parameters[0].Pattern, "organizations/[^/]+"; got != want {
  7411. t.Fatalf("First operation first param pattern differed, got %s want %s", got, want)
  7412. }
  7413. if got, want := firstOp.Parameters[1].Name, "role"; got != want {
  7414. t.Fatalf("First operation second param name differed, got %s want %s", got, want)
  7415. }
  7416. if got, want := firstOp.Parameters[1].Pattern, "roles/[^/]+"; got != want {
  7417. t.Fatalf("First operation second param pattern differed, got %s want %s", got, want)
  7418. }
  7419. if got, want := firstOp.Parameters[2].In, "body"; got != want {
  7420. t.Fatalf("First operation third param 'in' differed, got %s want %s", got, want)
  7421. }
  7422. secondOp := result.getPathItemObject("/v1/{name" + pathParamUniqueSuffixDeliminator + "1}/{role}").Get
  7423. if got, want := secondOp.OperationID, "Service1_Method2"; got != want {
  7424. t.Fatalf("Second operation id differed, got %s want %s", got, want)
  7425. }
  7426. if got, want := len(secondOp.Parameters), 3; got != want {
  7427. t.Fatalf("Second operation params length differed, got %d want %d", got, want)
  7428. }
  7429. if got, want := secondOp.Parameters[0].Name, "name"+pathParamUniqueSuffixDeliminator+"1"; got != want {
  7430. t.Fatalf("Second operation first param name differed, got %s want %s", got, want)
  7431. }
  7432. if got, want := secondOp.Parameters[0].Pattern, "users/[^/]+"; got != want {
  7433. t.Fatalf("Second operation first param pattern differed, got %s want %s", got, want)
  7434. }
  7435. if got, want := secondOp.Parameters[1].Name, "role"; got != want {
  7436. t.Fatalf("Second operation second param name differed, got %s want %s", got, want)
  7437. }
  7438. if got, want := secondOp.Parameters[1].Pattern, "roles/[^/]+"; got != want {
  7439. t.Fatalf("Second operation second param pattern differed, got %s want %s", got, want)
  7440. }
  7441. if got, want := secondOp.Parameters[2].In, "body"; got != want {
  7442. t.Fatalf("Second operation third param 'in' differed, got %s want %s", got, want)
  7443. }
  7444. thirdOp := result.getPathItemObject("/v1/{name}/roles").Get
  7445. if got, want := thirdOp.OperationID, "Service2_Method3"; got != want {
  7446. t.Fatalf("Third operation id differed, got %s want %s", got, want)
  7447. }
  7448. if got, want := len(thirdOp.Parameters), 2; got != want {
  7449. t.Fatalf("Third operation params length differed, got %d want %d", got, want)
  7450. }
  7451. if got, want := thirdOp.Parameters[0].Name, "name"; got != want {
  7452. t.Fatalf("Third operation first param name differed, got %s want %s", got, want)
  7453. }
  7454. if got, want := thirdOp.Parameters[0].Pattern, "users/[^/]+"; got != want {
  7455. t.Fatalf("Third operation first param pattern differed, got %s want %s", got, want)
  7456. }
  7457. if got, want := thirdOp.Parameters[1].In, "body"; got != want {
  7458. t.Fatalf("Third operation second param 'in' differed, got %s want %s", got, want)
  7459. }
  7460. forthOp := result.getPathItemObject("/v1/{name" + pathParamUniqueSuffixDeliminator + "2}/{role}").Get
  7461. if got, want := forthOp.OperationID, "Service2_Method4"; got != want {
  7462. t.Fatalf("Fourth operation id differed, got %s want %s", got, want)
  7463. }
  7464. if got, want := len(forthOp.Parameters), 3; got != want {
  7465. t.Fatalf("Fourth operation params length differed, got %d want %d", got, want)
  7466. }
  7467. if got, want := forthOp.Parameters[0].Name, "name"+pathParamUniqueSuffixDeliminator+"2"; got != want {
  7468. t.Fatalf("Fourth operation first param name differed, got %s want %s", got, want)
  7469. }
  7470. if got, want := forthOp.Parameters[0].Pattern, "groups/[^/]+"; got != want {
  7471. t.Fatalf("Fourth operation first param pattern differed, got %s want %s", got, want)
  7472. }
  7473. if got, want := forthOp.Parameters[1].Name, "role"; got != want {
  7474. t.Fatalf("Fourth operation second param name differed, got %s want %s", got, want)
  7475. }
  7476. if got, want := forthOp.Parameters[1].Pattern, "roles/[^/]+"; got != want {
  7477. t.Fatalf("Fourth operation second param pattern differed, got %s want %s", got, want)
  7478. }
  7479. if got, want := forthOp.Parameters[2].In, "body"; got != want {
  7480. t.Fatalf("Fourth operation second param 'in' differed, got %s want %s", got, want)
  7481. }
  7482. }
  7483. func Test_getReservedJsonName(t *testing.T) {
  7484. type args struct {
  7485. fieldName string
  7486. messageNameToFieldsToJSONName map[string]map[string]string
  7487. fieldNameToType map[string]string
  7488. }
  7489. tests := []struct {
  7490. name string
  7491. args args
  7492. want string
  7493. }{
  7494. {
  7495. "test case 1: single dot use case",
  7496. args{
  7497. fieldName: "abc.a_1",
  7498. messageNameToFieldsToJSONName: map[string]map[string]string{
  7499. "Msg": {
  7500. "a_1": "a1JSONNAME",
  7501. "b_1": "b1JSONNAME",
  7502. },
  7503. },
  7504. fieldNameToType: map[string]string{
  7505. "abc": "pkg1.test.Msg",
  7506. "bcd": "pkg1.test.Msg",
  7507. },
  7508. },
  7509. "a1JSONNAME",
  7510. },
  7511. {
  7512. "test case 2: single dot use case with no existing field",
  7513. args{
  7514. fieldName: "abc.d_1",
  7515. messageNameToFieldsToJSONName: map[string]map[string]string{
  7516. "Msg": {
  7517. "a_1": "a1JSONNAME",
  7518. "b_1": "b1JSONNAME",
  7519. },
  7520. },
  7521. fieldNameToType: map[string]string{
  7522. "abc": "pkg1.test.Msg",
  7523. "bcd": "pkg1.test.Msg",
  7524. },
  7525. },
  7526. "",
  7527. },
  7528. {
  7529. "test case 3: double dot use case",
  7530. args{
  7531. fieldName: "pkg.abc.a_1",
  7532. messageNameToFieldsToJSONName: map[string]map[string]string{
  7533. "Msg": {
  7534. "a_1": "a1JSONNAME",
  7535. "b_1": "b1JSONNAME",
  7536. },
  7537. },
  7538. fieldNameToType: map[string]string{
  7539. "abc": "pkg1.test.Msg",
  7540. "bcd": "pkg1.test.Msg",
  7541. },
  7542. },
  7543. "a1JSONNAME",
  7544. },
  7545. {
  7546. "test case 4: double dot use case with a not existed field",
  7547. args{
  7548. fieldName: "pkg.abc.c_1",
  7549. messageNameToFieldsToJSONName: map[string]map[string]string{
  7550. "Msg": {
  7551. "a_1": "a1JSONNAME",
  7552. "b_1": "b1JSONNAME",
  7553. },
  7554. },
  7555. fieldNameToType: map[string]string{
  7556. "abc": "pkg1.test.Msg",
  7557. "bcd": "pkg1.test.Msg",
  7558. },
  7559. },
  7560. "",
  7561. },
  7562. }
  7563. for _, tt := range tests {
  7564. t.Run(tt.name, func(t *testing.T) {
  7565. if got := getReservedJSONName(tt.args.fieldName, tt.args.messageNameToFieldsToJSONName, tt.args.fieldNameToType); got != tt.want {
  7566. t.Errorf("getReservedJSONName() = %v, want %v", got, tt.want)
  7567. }
  7568. })
  7569. }
  7570. }
  7571. func TestParseIncompleteSecurityRequirement(t *testing.T) {
  7572. swagger := openapi_options.Swagger{
  7573. Security: []*openapi_options.SecurityRequirement{
  7574. {
  7575. SecurityRequirement: map[string]*openapi_options.SecurityRequirement_SecurityRequirementValue{
  7576. "key": nil,
  7577. },
  7578. },
  7579. },
  7580. }
  7581. file := descriptor.File{
  7582. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  7583. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  7584. Name: proto.String("example.proto"),
  7585. Package: proto.String("example"),
  7586. Options: &descriptorpb.FileOptions{
  7587. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  7588. },
  7589. },
  7590. }
  7591. proto.SetExtension(proto.Message(file.FileDescriptorProto.Options), openapi_options.E_Openapiv2Swagger, &swagger)
  7592. reg := descriptor.NewRegistry()
  7593. err := reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto}})
  7594. if err != nil {
  7595. t.Errorf("failed to reg.Load(): %v", err)
  7596. return
  7597. }
  7598. _, err = applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  7599. if err == nil {
  7600. t.Errorf("applyTemplate(%#v) did not error as expected", file)
  7601. return
  7602. }
  7603. }
  7604. func TestSubPathParams(t *testing.T) {
  7605. outerParams := []descriptor.Parameter{
  7606. {
  7607. FieldPath: []descriptor.FieldPathComponent{
  7608. {
  7609. Name: "prefix",
  7610. },
  7611. {
  7612. Name: "first",
  7613. },
  7614. },
  7615. },
  7616. {
  7617. FieldPath: []descriptor.FieldPathComponent{
  7618. {
  7619. Name: "prefix",
  7620. },
  7621. {
  7622. Name: "second",
  7623. },
  7624. {
  7625. Name: "deeper",
  7626. },
  7627. },
  7628. },
  7629. {
  7630. FieldPath: []descriptor.FieldPathComponent{
  7631. {
  7632. Name: "otherprefix",
  7633. },
  7634. {
  7635. Name: "third",
  7636. },
  7637. },
  7638. },
  7639. }
  7640. subParams := subPathParams("prefix", outerParams)
  7641. if got, want := len(subParams), 2; got != want {
  7642. t.Fatalf("Wrong number of path params, got %d want %d", got, want)
  7643. }
  7644. if got, want := len(subParams[0].FieldPath), 1; got != want {
  7645. t.Fatalf("Wrong length of path param 0, got %d want %d", got, want)
  7646. }
  7647. if got, want := subParams[0].FieldPath[0].Name, "first"; got != want {
  7648. t.Fatalf("Wrong path param 0, element 0, got %s want %s", got, want)
  7649. }
  7650. if got, want := len(subParams[1].FieldPath), 2; got != want {
  7651. t.Fatalf("Wrong length of path param 1 got %d want %d", got, want)
  7652. }
  7653. if got, want := subParams[1].FieldPath[0].Name, "second"; got != want {
  7654. t.Fatalf("Wrong path param 1, element 0, got %s want %s", got, want)
  7655. }
  7656. if got, want := subParams[1].FieldPath[1].Name, "deeper"; got != want {
  7657. t.Fatalf("Wrong path param 1, element 1, got %s want %s", got, want)
  7658. }
  7659. }
  7660. func TestRenderServicesParameterDescriptionNoFieldBody(t *testing.T) {
  7661. optionsRaw := `{
  7662. "[grpc.gateway.protoc_gen_openapiv2.options.openapiv2_schema]": {
  7663. "jsonSchema": {
  7664. "title": "aMessage title",
  7665. "description": "aMessage description"
  7666. }
  7667. }
  7668. }`
  7669. options := &descriptorpb.MessageOptions{}
  7670. err := protojson.Unmarshal([]byte(optionsRaw), options)
  7671. if err != nil {
  7672. t.Fatalf("Error while unmarshalling options: %s", err.Error())
  7673. }
  7674. aMessageDesc := &descriptorpb.DescriptorProto{
  7675. Name: proto.String("AMessage"),
  7676. Field: []*descriptorpb.FieldDescriptorProto{
  7677. {
  7678. Name: proto.String("project_id"),
  7679. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  7680. Number: proto.Int32(1),
  7681. },
  7682. {
  7683. Name: proto.String("other_field"),
  7684. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  7685. Number: proto.Int32(2),
  7686. },
  7687. },
  7688. Options: options,
  7689. }
  7690. someResponseDesc := &descriptorpb.DescriptorProto{
  7691. Name: proto.String("SomeResponse"),
  7692. }
  7693. aMeth := &descriptorpb.MethodDescriptorProto{
  7694. Name: proto.String("AMethod"),
  7695. InputType: proto.String("AMessage"),
  7696. OutputType: proto.String("SomeResponse"),
  7697. }
  7698. svc := &descriptorpb.ServiceDescriptorProto{
  7699. Name: proto.String("Test"),
  7700. Method: []*descriptorpb.MethodDescriptorProto{aMeth},
  7701. }
  7702. aMessage := &descriptor.Message{
  7703. DescriptorProto: aMessageDesc,
  7704. }
  7705. someResponseMessage := &descriptor.Message{
  7706. DescriptorProto: someResponseDesc,
  7707. }
  7708. file := descriptor.File{
  7709. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  7710. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  7711. Package: proto.String("api"),
  7712. Name: proto.String("test.proto"),
  7713. MessageType: []*descriptorpb.DescriptorProto{aMessageDesc, someResponseDesc},
  7714. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  7715. Options: &descriptorpb.FileOptions{
  7716. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  7717. },
  7718. },
  7719. GoPkg: descriptor.GoPackage{
  7720. Path: "example.com/path/to/example/example.pb",
  7721. Name: "example_pb",
  7722. },
  7723. Messages: []*descriptor.Message{aMessage, someResponseMessage},
  7724. Services: []*descriptor.Service{
  7725. {
  7726. ServiceDescriptorProto: svc,
  7727. Methods: []*descriptor.Method{
  7728. {
  7729. MethodDescriptorProto: aMeth,
  7730. RequestType: aMessage,
  7731. ResponseType: someResponseMessage,
  7732. Bindings: []*descriptor.Binding{
  7733. {
  7734. HTTPMethod: "POST",
  7735. PathTmpl: httprule.Template{
  7736. Version: 1,
  7737. OpCodes: []int{0, 0},
  7738. Template: "/v1/projects/someotherpath",
  7739. },
  7740. Body: &descriptor.Body{},
  7741. },
  7742. },
  7743. },
  7744. },
  7745. },
  7746. },
  7747. }
  7748. reg := descriptor.NewRegistry()
  7749. reg.SetUseJSONNamesForFields(true)
  7750. err = reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto}})
  7751. if err != nil {
  7752. t.Fatalf("failed to reg.Load(): %v", err)
  7753. }
  7754. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  7755. if err != nil {
  7756. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  7757. }
  7758. got := result.getPathItemObject("/v1/projects/someotherpath").Post.Parameters[0].Description
  7759. want := "aMessage description"
  7760. if got != want {
  7761. t.Fatalf("Wrong description for body parameter, got %s want %s", got, want)
  7762. }
  7763. }
  7764. func TestRenderServicesWithBodyFieldNameInCamelCase(t *testing.T) {
  7765. userDesc := &descriptorpb.DescriptorProto{
  7766. Name: proto.String("User"),
  7767. Field: []*descriptorpb.FieldDescriptorProto{
  7768. {
  7769. Name: proto.String("name"),
  7770. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  7771. Number: proto.Int32(1),
  7772. },
  7773. {
  7774. Name: proto.String("role"),
  7775. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  7776. Number: proto.Int32(2),
  7777. },
  7778. },
  7779. }
  7780. updateDesc := &descriptorpb.DescriptorProto{
  7781. Name: proto.String("UpdateUserRequest"),
  7782. Field: []*descriptorpb.FieldDescriptorProto{
  7783. {
  7784. Name: proto.String("user_object"),
  7785. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  7786. TypeName: proto.String(".example.User"),
  7787. Number: proto.Int32(1),
  7788. },
  7789. },
  7790. }
  7791. meth := &descriptorpb.MethodDescriptorProto{
  7792. Name: proto.String("UpdateUser"),
  7793. InputType: proto.String("UpdateUserRequest"),
  7794. OutputType: proto.String("User"),
  7795. }
  7796. svc := &descriptorpb.ServiceDescriptorProto{
  7797. Name: proto.String("UserService"),
  7798. Method: []*descriptorpb.MethodDescriptorProto{meth},
  7799. }
  7800. userMsg := &descriptor.Message{
  7801. DescriptorProto: userDesc,
  7802. }
  7803. updateMsg := &descriptor.Message{
  7804. DescriptorProto: updateDesc,
  7805. }
  7806. nameField := &descriptor.Field{
  7807. Message: userMsg,
  7808. FieldDescriptorProto: userMsg.GetField()[0],
  7809. }
  7810. nameField.JsonName = proto.String("name")
  7811. roleField := &descriptor.Field{
  7812. Message: userMsg,
  7813. FieldDescriptorProto: userMsg.GetField()[1],
  7814. }
  7815. roleField.JsonName = proto.String("role")
  7816. userMsg.Fields = []*descriptor.Field{nameField, roleField}
  7817. userField := &descriptor.Field{
  7818. Message: updateMsg,
  7819. FieldMessage: userMsg,
  7820. FieldDescriptorProto: updateMsg.GetField()[0],
  7821. }
  7822. userField.JsonName = proto.String("userObject")
  7823. updateMsg.Fields = []*descriptor.Field{userField}
  7824. file := descriptor.File{
  7825. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  7826. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  7827. Package: proto.String("example"),
  7828. Name: proto.String("user_service.proto"),
  7829. MessageType: []*descriptorpb.DescriptorProto{userDesc, updateDesc},
  7830. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  7831. Options: &descriptorpb.FileOptions{
  7832. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  7833. },
  7834. },
  7835. GoPkg: descriptor.GoPackage{
  7836. Path: "example.com/path/to/example/example.pb",
  7837. Name: "example_pb",
  7838. },
  7839. Messages: []*descriptor.Message{userMsg, updateMsg},
  7840. Services: []*descriptor.Service{
  7841. {
  7842. ServiceDescriptorProto: svc,
  7843. Methods: []*descriptor.Method{
  7844. {
  7845. MethodDescriptorProto: meth,
  7846. RequestType: updateMsg,
  7847. ResponseType: userMsg,
  7848. Bindings: []*descriptor.Binding{
  7849. {
  7850. HTTPMethod: "POST",
  7851. PathTmpl: httprule.Template{
  7852. Version: 1,
  7853. OpCodes: []int{0, 0},
  7854. Template: "/v1/users/{user_object.name}",
  7855. },
  7856. PathParams: []descriptor.Parameter{
  7857. {
  7858. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  7859. {
  7860. Name: "user_object",
  7861. },
  7862. {
  7863. Name: "name",
  7864. },
  7865. }),
  7866. Target: nameField,
  7867. },
  7868. },
  7869. Body: &descriptor.Body{
  7870. FieldPath: []descriptor.FieldPathComponent{
  7871. {
  7872. Name: "user_object",
  7873. Target: userField,
  7874. },
  7875. },
  7876. },
  7877. },
  7878. },
  7879. },
  7880. },
  7881. },
  7882. },
  7883. }
  7884. reg := descriptor.NewRegistry()
  7885. reg.SetUseJSONNamesForFields(true)
  7886. err := reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto}})
  7887. if err != nil {
  7888. t.Fatalf("failed to reg.Load(): %v", err)
  7889. }
  7890. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  7891. if err != nil {
  7892. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  7893. }
  7894. paths := GetPaths(result)
  7895. if got, want := len(paths), 1; got != want {
  7896. t.Fatalf("Results path length differed, got %d want %d", got, want)
  7897. }
  7898. if got, want := paths[0], "/v1/users/{userObject.name}"; got != want {
  7899. t.Fatalf("Wrong results path, got %s want %s", got, want)
  7900. }
  7901. operation := *result.getPathItemObject("/v1/users/{userObject.name}").Post
  7902. if got, want := len(operation.Parameters), 2; got != want {
  7903. t.Fatalf("Parameters length differed, got %d want %d", got, want)
  7904. }
  7905. if got, want := operation.Parameters[0].Name, "userObject.name"; got != want {
  7906. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  7907. }
  7908. if got, want := operation.Parameters[0].In, "path"; got != want {
  7909. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  7910. }
  7911. if got, want := operation.Parameters[1].Name, "userObject"; got != want {
  7912. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  7913. }
  7914. if got, want := operation.Parameters[1].In, "body"; got != want {
  7915. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  7916. }
  7917. // The body parameter should be inlined and not contain 'name', as this is a path parameter.
  7918. schema := operation.Parameters[1].Schema
  7919. if got, want := schema.Ref, ""; got != want {
  7920. t.Fatalf("Wrong reference, got %s want %s", got, want)
  7921. }
  7922. props := schema.Properties
  7923. if props == nil {
  7924. t.Fatal("No properties on body parameter")
  7925. }
  7926. if got, want := len(*props), 1; got != want {
  7927. t.Fatalf("Properties length differed, got %d want %d", got, want)
  7928. }
  7929. for _, v := range *props {
  7930. if got, want := v.Key, "role"; got != want {
  7931. t.Fatalf("Wrong key for property, got %s want %s", got, want)
  7932. }
  7933. }
  7934. }
  7935. func TestRenderServicesWithBodyFieldHasFieldMask(t *testing.T) {
  7936. userDesc := &descriptorpb.DescriptorProto{
  7937. Name: proto.String("User"),
  7938. Field: []*descriptorpb.FieldDescriptorProto{
  7939. {
  7940. Name: proto.String("name"),
  7941. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  7942. Number: proto.Int32(1),
  7943. },
  7944. {
  7945. Name: proto.String("role"),
  7946. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  7947. Number: proto.Int32(2),
  7948. },
  7949. },
  7950. }
  7951. updateDesc := &descriptorpb.DescriptorProto{
  7952. Name: proto.String("UpdateUserRequest"),
  7953. Field: []*descriptorpb.FieldDescriptorProto{
  7954. {
  7955. Name: proto.String("user_object"),
  7956. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  7957. TypeName: proto.String(".example.User"),
  7958. Number: proto.Int32(1),
  7959. },
  7960. {
  7961. Name: proto.String("update_mask"),
  7962. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  7963. TypeName: proto.String(".google.protobuf.FieldMask"),
  7964. Number: proto.Int32(2),
  7965. },
  7966. },
  7967. }
  7968. meth := &descriptorpb.MethodDescriptorProto{
  7969. Name: proto.String("UpdateUser"),
  7970. InputType: proto.String("UpdateUserRequest"),
  7971. OutputType: proto.String("User"),
  7972. }
  7973. svc := &descriptorpb.ServiceDescriptorProto{
  7974. Name: proto.String("UserService"),
  7975. Method: []*descriptorpb.MethodDescriptorProto{meth},
  7976. }
  7977. userMsg := &descriptor.Message{
  7978. DescriptorProto: userDesc,
  7979. }
  7980. updateMsg := &descriptor.Message{
  7981. DescriptorProto: updateDesc,
  7982. }
  7983. nameField := &descriptor.Field{
  7984. Message: userMsg,
  7985. FieldDescriptorProto: userMsg.GetField()[0],
  7986. }
  7987. nameField.JsonName = proto.String("name")
  7988. roleField := &descriptor.Field{
  7989. Message: userMsg,
  7990. FieldDescriptorProto: userMsg.GetField()[1],
  7991. }
  7992. roleField.JsonName = proto.String("role")
  7993. userMsg.Fields = []*descriptor.Field{nameField, roleField}
  7994. userField := &descriptor.Field{
  7995. Message: updateMsg,
  7996. FieldMessage: userMsg,
  7997. FieldDescriptorProto: updateMsg.GetField()[0],
  7998. }
  7999. userField.JsonName = proto.String("userObject")
  8000. updateMaskField := &descriptor.Field{
  8001. Message: updateMsg,
  8002. FieldDescriptorProto: updateMsg.GetField()[1],
  8003. }
  8004. updateMaskField.JsonName = proto.String("updateMask")
  8005. updateMsg.Fields = []*descriptor.Field{userField, updateMaskField}
  8006. file := descriptor.File{
  8007. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  8008. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  8009. Package: proto.String("example"),
  8010. Name: proto.String("user_service.proto"),
  8011. Dependency: []string{"google/well_known.proto"},
  8012. MessageType: []*descriptorpb.DescriptorProto{userDesc, updateDesc},
  8013. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  8014. Options: &descriptorpb.FileOptions{
  8015. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  8016. },
  8017. },
  8018. GoPkg: descriptor.GoPackage{
  8019. Path: "example.com/path/to/example/example.pb",
  8020. Name: "example_pb",
  8021. },
  8022. Messages: []*descriptor.Message{userMsg, updateMsg},
  8023. Services: []*descriptor.Service{
  8024. {
  8025. ServiceDescriptorProto: svc,
  8026. Methods: []*descriptor.Method{
  8027. {
  8028. MethodDescriptorProto: meth,
  8029. RequestType: updateMsg,
  8030. ResponseType: userMsg,
  8031. Bindings: []*descriptor.Binding{
  8032. {
  8033. HTTPMethod: "PATCH",
  8034. PathTmpl: httprule.Template{
  8035. Version: 1,
  8036. OpCodes: []int{0, 0},
  8037. Template: "/v1/users/{user_object.name}",
  8038. },
  8039. PathParams: []descriptor.Parameter{
  8040. {
  8041. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  8042. {
  8043. Name: "user_object",
  8044. },
  8045. {
  8046. Name: "name",
  8047. },
  8048. }),
  8049. Target: nameField,
  8050. },
  8051. },
  8052. Body: &descriptor.Body{
  8053. FieldPath: []descriptor.FieldPathComponent{
  8054. {
  8055. Name: "user_object",
  8056. Target: userField,
  8057. },
  8058. },
  8059. },
  8060. },
  8061. },
  8062. },
  8063. },
  8064. },
  8065. },
  8066. }
  8067. reg := descriptor.NewRegistry()
  8068. reg.SetUseJSONNamesForFields(true)
  8069. reg.SetAllowPatchFeature(true)
  8070. err := reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{
  8071. {
  8072. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  8073. Name: proto.String("google/well_known.proto"),
  8074. Package: proto.String("google.protobuf"),
  8075. Dependency: []string{},
  8076. MessageType: []*descriptorpb.DescriptorProto{
  8077. protodesc.ToDescriptorProto((&field_mask.FieldMask{}).ProtoReflect().Descriptor()),
  8078. },
  8079. Service: []*descriptorpb.ServiceDescriptorProto{},
  8080. Options: &descriptorpb.FileOptions{
  8081. GoPackage: proto.String("google/well_known"),
  8082. },
  8083. },
  8084. file.FileDescriptorProto,
  8085. }})
  8086. if err != nil {
  8087. t.Fatalf("failed to reg.Load(): %v", err)
  8088. }
  8089. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  8090. if err != nil {
  8091. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  8092. }
  8093. paths := GetPaths(result)
  8094. if got, want := len(paths), 1; got != want {
  8095. t.Fatalf("Results path length differed, got %d want %d", got, want)
  8096. }
  8097. if got, want := paths[0], "/v1/users/{userObject.name}"; got != want {
  8098. t.Fatalf("Wrong results path, got %s want %s", got, want)
  8099. }
  8100. operation := *result.getPathItemObject("/v1/users/{userObject.name}").Patch
  8101. if got, want := len(operation.Parameters), 2; got != want {
  8102. t.Fatalf("Parameters length differed, got %d want %d", got, want)
  8103. }
  8104. if got, want := operation.Parameters[0].Name, "userObject.name"; got != want {
  8105. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8106. }
  8107. if got, want := operation.Parameters[0].In, "path"; got != want {
  8108. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8109. }
  8110. if got, want := operation.Parameters[1].Name, "userObject"; got != want {
  8111. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8112. }
  8113. if got, want := operation.Parameters[1].In, "body"; got != want {
  8114. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8115. }
  8116. }
  8117. func TestRenderServicesWithColonInPath(t *testing.T) {
  8118. jsonSchema := &openapi_options.JSONSchema{
  8119. FieldConfiguration: &openapi_options.JSONSchema_FieldConfiguration{
  8120. PathParamName: "overrideField",
  8121. },
  8122. }
  8123. fieldOptions := new(descriptorpb.FieldOptions)
  8124. proto.SetExtension(fieldOptions, openapi_options.E_Openapiv2Field, jsonSchema)
  8125. reqDesc := &descriptorpb.DescriptorProto{
  8126. Name: proto.String("MyRequest"),
  8127. Field: []*descriptorpb.FieldDescriptorProto{
  8128. {
  8129. Name: proto.String("field"),
  8130. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8131. Number: proto.Int32(1),
  8132. Options: fieldOptions,
  8133. },
  8134. },
  8135. }
  8136. resDesc := &descriptorpb.DescriptorProto{
  8137. Name: proto.String("MyResponse"),
  8138. Field: []*descriptorpb.FieldDescriptorProto{
  8139. {
  8140. Name: proto.String("field"),
  8141. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8142. Number: proto.Int32(1),
  8143. },
  8144. },
  8145. }
  8146. meth := &descriptorpb.MethodDescriptorProto{
  8147. Name: proto.String("MyMethod"),
  8148. InputType: proto.String("MyRequest"),
  8149. OutputType: proto.String("MyResponse"),
  8150. }
  8151. svc := &descriptorpb.ServiceDescriptorProto{
  8152. Name: proto.String("MyService"),
  8153. Method: []*descriptorpb.MethodDescriptorProto{meth},
  8154. }
  8155. reqMsg := &descriptor.Message{
  8156. DescriptorProto: reqDesc,
  8157. }
  8158. resMsg := &descriptor.Message{
  8159. DescriptorProto: resDesc,
  8160. }
  8161. reqField := &descriptor.Field{
  8162. Message: reqMsg,
  8163. FieldDescriptorProto: reqMsg.GetField()[0],
  8164. }
  8165. resField := &descriptor.Field{
  8166. Message: resMsg,
  8167. FieldDescriptorProto: resMsg.GetField()[0],
  8168. }
  8169. reqField.JsonName = proto.String("field")
  8170. resField.JsonName = proto.String("field")
  8171. reqMsg.Fields = []*descriptor.Field{reqField}
  8172. resMsg.Fields = []*descriptor.Field{resField}
  8173. file := descriptor.File{
  8174. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  8175. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  8176. Package: proto.String("example"),
  8177. Name: proto.String(",my_service.proto"),
  8178. MessageType: []*descriptorpb.DescriptorProto{reqDesc, resDesc},
  8179. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  8180. Options: &descriptorpb.FileOptions{
  8181. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  8182. },
  8183. },
  8184. GoPkg: descriptor.GoPackage{
  8185. Path: "example.com/path/to/example/example.pb",
  8186. Name: "example_pb",
  8187. },
  8188. Messages: []*descriptor.Message{reqMsg, resMsg},
  8189. Services: []*descriptor.Service{
  8190. {
  8191. ServiceDescriptorProto: svc,
  8192. Methods: []*descriptor.Method{
  8193. {
  8194. MethodDescriptorProto: meth,
  8195. RequestType: reqMsg,
  8196. ResponseType: resMsg,
  8197. Bindings: []*descriptor.Binding{
  8198. {
  8199. HTTPMethod: "POST",
  8200. PathTmpl: httprule.Template{
  8201. Version: 1,
  8202. OpCodes: []int{0, 0},
  8203. Template: "/my/{field}:foo",
  8204. },
  8205. PathParams: []descriptor.Parameter{
  8206. {
  8207. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  8208. {
  8209. Name: "field",
  8210. },
  8211. }),
  8212. Target: reqField,
  8213. },
  8214. },
  8215. Body: &descriptor.Body{
  8216. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  8217. },
  8218. },
  8219. },
  8220. },
  8221. },
  8222. },
  8223. },
  8224. }
  8225. reg := descriptor.NewRegistry()
  8226. reg.SetUseJSONNamesForFields(true)
  8227. err := reg.Load(reqFromFile(&file))
  8228. if err != nil {
  8229. t.Fatalf("failed to reg.Load(): %v", err)
  8230. }
  8231. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  8232. if err != nil {
  8233. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  8234. }
  8235. paths := GetPaths(result)
  8236. if got, want := len(paths), 1; got != want {
  8237. t.Fatalf("Results path length differed, got %d want %d", got, want)
  8238. }
  8239. if got, want := paths[0], "/my/{overrideField}:foo"; got != want {
  8240. t.Fatalf("Wrong results path, got %s want %s", got, want)
  8241. }
  8242. operation := *result.getPathItemObject("/my/{overrideField}:foo").Post
  8243. if got, want := len(operation.Parameters), 2; got != want {
  8244. t.Fatalf("Parameters length differed, got %d want %d", got, want)
  8245. }
  8246. if got, want := operation.Parameters[0].Name, "overrideField"; got != want {
  8247. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8248. }
  8249. if got, want := operation.Parameters[0].In, "path"; got != want {
  8250. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8251. }
  8252. if got, want := operation.Parameters[0].Type, "string"; got != want {
  8253. t.Fatalf("Wrong parameter type, got %s want %s", got, want)
  8254. }
  8255. if got, want := operation.Parameters[1].Name, "body"; got != want {
  8256. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8257. }
  8258. if got, want := operation.Parameters[1].In, "body"; got != want {
  8259. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8260. }
  8261. }
  8262. func TestRenderServicesWithDoubleColonInPath(t *testing.T) {
  8263. reqDesc := &descriptorpb.DescriptorProto{
  8264. Name: proto.String("MyRequest"),
  8265. Field: []*descriptorpb.FieldDescriptorProto{
  8266. {
  8267. Name: proto.String("field"),
  8268. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8269. Number: proto.Int32(1),
  8270. },
  8271. },
  8272. }
  8273. resDesc := &descriptorpb.DescriptorProto{
  8274. Name: proto.String("MyResponse"),
  8275. Field: []*descriptorpb.FieldDescriptorProto{
  8276. {
  8277. Name: proto.String("field"),
  8278. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8279. Number: proto.Int32(1),
  8280. },
  8281. },
  8282. }
  8283. meth := &descriptorpb.MethodDescriptorProto{
  8284. Name: proto.String("MyMethod"),
  8285. InputType: proto.String("MyRequest"),
  8286. OutputType: proto.String("MyResponse"),
  8287. }
  8288. svc := &descriptorpb.ServiceDescriptorProto{
  8289. Name: proto.String("MyService"),
  8290. Method: []*descriptorpb.MethodDescriptorProto{meth},
  8291. }
  8292. reqMsg := &descriptor.Message{
  8293. DescriptorProto: reqDesc,
  8294. }
  8295. resMsg := &descriptor.Message{
  8296. DescriptorProto: resDesc,
  8297. }
  8298. reqField := &descriptor.Field{
  8299. Message: reqMsg,
  8300. FieldDescriptorProto: reqMsg.GetField()[0],
  8301. }
  8302. resField := &descriptor.Field{
  8303. Message: resMsg,
  8304. FieldDescriptorProto: resMsg.GetField()[0],
  8305. }
  8306. reqField.JsonName = proto.String("field")
  8307. resField.JsonName = proto.String("field")
  8308. reqMsg.Fields = []*descriptor.Field{reqField}
  8309. resMsg.Fields = []*descriptor.Field{resField}
  8310. file := descriptor.File{
  8311. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  8312. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  8313. Package: proto.String("example"),
  8314. Name: proto.String(",my_service.proto"),
  8315. MessageType: []*descriptorpb.DescriptorProto{reqDesc, resDesc},
  8316. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  8317. Options: &descriptorpb.FileOptions{
  8318. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  8319. },
  8320. },
  8321. GoPkg: descriptor.GoPackage{
  8322. Path: "example.com/path/to/example/example.pb",
  8323. Name: "example_pb",
  8324. },
  8325. Messages: []*descriptor.Message{reqMsg, resMsg},
  8326. Services: []*descriptor.Service{
  8327. {
  8328. ServiceDescriptorProto: svc,
  8329. Methods: []*descriptor.Method{
  8330. {
  8331. MethodDescriptorProto: meth,
  8332. RequestType: reqMsg,
  8333. ResponseType: resMsg,
  8334. Bindings: []*descriptor.Binding{
  8335. {
  8336. HTTPMethod: "POST",
  8337. PathTmpl: httprule.Template{
  8338. Version: 1,
  8339. OpCodes: []int{0, 0},
  8340. Template: "/my/{field}:foo:bar",
  8341. },
  8342. PathParams: []descriptor.Parameter{
  8343. {
  8344. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  8345. {
  8346. Name: "field",
  8347. },
  8348. }),
  8349. Target: reqField,
  8350. },
  8351. },
  8352. Body: &descriptor.Body{
  8353. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  8354. },
  8355. },
  8356. },
  8357. },
  8358. },
  8359. },
  8360. },
  8361. }
  8362. reg := descriptor.NewRegistry()
  8363. reg.SetUseJSONNamesForFields(true)
  8364. err := reg.Load(reqFromFile(&file))
  8365. if err != nil {
  8366. t.Fatalf("failed to reg.Load(): %v", err)
  8367. }
  8368. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  8369. if err != nil {
  8370. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  8371. }
  8372. paths := GetPaths(result)
  8373. if got, want := len(paths), 1; got != want {
  8374. t.Fatalf("Results path length differed, got %d want %d", got, want)
  8375. }
  8376. if got, want := paths[0], "/my/{field}:foo:bar"; got != want {
  8377. t.Fatalf("Wrong results path, got %s want %s", got, want)
  8378. }
  8379. operation := *result.getPathItemObject("/my/{field}:foo:bar").Post
  8380. if got, want := len(operation.Parameters), 2; got != want {
  8381. t.Fatalf("Parameters length differed, got %d want %d", got, want)
  8382. }
  8383. if got, want := operation.Parameters[0].Name, "field"; got != want {
  8384. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8385. }
  8386. if got, want := operation.Parameters[0].In, "path"; got != want {
  8387. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8388. }
  8389. if got, want := operation.Parameters[0].Type, "string"; got != want {
  8390. t.Fatalf("Wrong parameter type, got %s want %s", got, want)
  8391. }
  8392. if got, want := operation.Parameters[1].Name, "body"; got != want {
  8393. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8394. }
  8395. if got, want := operation.Parameters[1].In, "body"; got != want {
  8396. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8397. }
  8398. }
  8399. func TestRenderServicesWithColonLastInPath(t *testing.T) {
  8400. reqDesc := &descriptorpb.DescriptorProto{
  8401. Name: proto.String("MyRequest"),
  8402. Field: []*descriptorpb.FieldDescriptorProto{
  8403. {
  8404. Name: proto.String("field"),
  8405. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8406. Number: proto.Int32(1),
  8407. },
  8408. },
  8409. }
  8410. resDesc := &descriptorpb.DescriptorProto{
  8411. Name: proto.String("MyResponse"),
  8412. Field: []*descriptorpb.FieldDescriptorProto{
  8413. {
  8414. Name: proto.String("field"),
  8415. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8416. Number: proto.Int32(1),
  8417. },
  8418. },
  8419. }
  8420. meth := &descriptorpb.MethodDescriptorProto{
  8421. Name: proto.String("MyMethod"),
  8422. InputType: proto.String("MyRequest"),
  8423. OutputType: proto.String("MyResponse"),
  8424. }
  8425. svc := &descriptorpb.ServiceDescriptorProto{
  8426. Name: proto.String("MyService"),
  8427. Method: []*descriptorpb.MethodDescriptorProto{meth},
  8428. }
  8429. reqMsg := &descriptor.Message{
  8430. DescriptorProto: reqDesc,
  8431. }
  8432. resMsg := &descriptor.Message{
  8433. DescriptorProto: resDesc,
  8434. }
  8435. reqField := &descriptor.Field{
  8436. Message: reqMsg,
  8437. FieldDescriptorProto: reqMsg.GetField()[0],
  8438. }
  8439. resField := &descriptor.Field{
  8440. Message: resMsg,
  8441. FieldDescriptorProto: resMsg.GetField()[0],
  8442. }
  8443. reqField.JsonName = proto.String("field")
  8444. resField.JsonName = proto.String("field")
  8445. reqMsg.Fields = []*descriptor.Field{reqField}
  8446. resMsg.Fields = []*descriptor.Field{resField}
  8447. file := descriptor.File{
  8448. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  8449. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  8450. Package: proto.String("example"),
  8451. Name: proto.String(",my_service.proto"),
  8452. MessageType: []*descriptorpb.DescriptorProto{reqDesc, resDesc},
  8453. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  8454. Options: &descriptorpb.FileOptions{
  8455. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  8456. },
  8457. },
  8458. GoPkg: descriptor.GoPackage{
  8459. Path: "example.com/path/to/example/example.pb",
  8460. Name: "example_pb",
  8461. },
  8462. Messages: []*descriptor.Message{reqMsg, resMsg},
  8463. Services: []*descriptor.Service{
  8464. {
  8465. ServiceDescriptorProto: svc,
  8466. Methods: []*descriptor.Method{
  8467. {
  8468. MethodDescriptorProto: meth,
  8469. RequestType: reqMsg,
  8470. ResponseType: resMsg,
  8471. Bindings: []*descriptor.Binding{
  8472. {
  8473. HTTPMethod: "POST",
  8474. PathTmpl: httprule.Template{
  8475. Version: 1,
  8476. OpCodes: []int{0, 0},
  8477. Template: "/my/{field}:",
  8478. },
  8479. PathParams: []descriptor.Parameter{
  8480. {
  8481. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  8482. {
  8483. Name: "field",
  8484. },
  8485. }),
  8486. Target: reqField,
  8487. },
  8488. },
  8489. Body: &descriptor.Body{
  8490. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  8491. },
  8492. },
  8493. },
  8494. },
  8495. },
  8496. },
  8497. },
  8498. }
  8499. reg := descriptor.NewRegistry()
  8500. reg.SetUseJSONNamesForFields(true)
  8501. err := reg.Load(reqFromFile(&file))
  8502. if err != nil {
  8503. t.Fatalf("failed to reg.Load(): %v", err)
  8504. }
  8505. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  8506. if err != nil {
  8507. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  8508. }
  8509. paths := GetPaths(result)
  8510. if got, want := len(paths), 1; got != want {
  8511. t.Fatalf("Results path length differed, got %d want %d", got, want)
  8512. }
  8513. if got, want := paths[0], "/my/{field}:"; got != want {
  8514. t.Fatalf("Wrong results path, got %s want %s", got, want)
  8515. }
  8516. operation := *result.getPathItemObject("/my/{field}:").Post
  8517. if got, want := len(operation.Parameters), 2; got != want {
  8518. t.Fatalf("Parameters length differed, got %d want %d", got, want)
  8519. }
  8520. if got, want := operation.Parameters[0].Name, "field"; got != want {
  8521. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8522. }
  8523. if got, want := operation.Parameters[0].In, "path"; got != want {
  8524. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8525. }
  8526. if got, want := operation.Parameters[0].Type, "string"; got != want {
  8527. t.Fatalf("Wrong parameter type, got %s want %s", got, want)
  8528. }
  8529. if got, want := operation.Parameters[1].Name, "body"; got != want {
  8530. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8531. }
  8532. if got, want := operation.Parameters[1].In, "body"; got != want {
  8533. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8534. }
  8535. }
  8536. func TestRenderServicesWithColonInSegment(t *testing.T) {
  8537. reqDesc := &descriptorpb.DescriptorProto{
  8538. Name: proto.String("MyRequest"),
  8539. Field: []*descriptorpb.FieldDescriptorProto{
  8540. {
  8541. Name: proto.String("field"),
  8542. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8543. Number: proto.Int32(1),
  8544. },
  8545. },
  8546. }
  8547. resDesc := &descriptorpb.DescriptorProto{
  8548. Name: proto.String("MyResponse"),
  8549. Field: []*descriptorpb.FieldDescriptorProto{
  8550. {
  8551. Name: proto.String("field"),
  8552. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8553. Number: proto.Int32(1),
  8554. },
  8555. },
  8556. }
  8557. meth := &descriptorpb.MethodDescriptorProto{
  8558. Name: proto.String("MyMethod"),
  8559. InputType: proto.String("MyRequest"),
  8560. OutputType: proto.String("MyResponse"),
  8561. }
  8562. svc := &descriptorpb.ServiceDescriptorProto{
  8563. Name: proto.String("MyService"),
  8564. Method: []*descriptorpb.MethodDescriptorProto{meth},
  8565. }
  8566. reqMsg := &descriptor.Message{
  8567. DescriptorProto: reqDesc,
  8568. }
  8569. resMsg := &descriptor.Message{
  8570. DescriptorProto: resDesc,
  8571. }
  8572. reqField := &descriptor.Field{
  8573. Message: reqMsg,
  8574. FieldDescriptorProto: reqMsg.GetField()[0],
  8575. }
  8576. resField := &descriptor.Field{
  8577. Message: resMsg,
  8578. FieldDescriptorProto: resMsg.GetField()[0],
  8579. }
  8580. reqField.JsonName = proto.String("field")
  8581. resField.JsonName = proto.String("field")
  8582. reqMsg.Fields = []*descriptor.Field{reqField}
  8583. resMsg.Fields = []*descriptor.Field{resField}
  8584. file := descriptor.File{
  8585. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  8586. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  8587. Package: proto.String("example"),
  8588. Name: proto.String(",my_service.proto"),
  8589. MessageType: []*descriptorpb.DescriptorProto{reqDesc, resDesc},
  8590. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  8591. Options: &descriptorpb.FileOptions{
  8592. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  8593. },
  8594. },
  8595. GoPkg: descriptor.GoPackage{
  8596. Path: "example.com/path/to/example/example.pb",
  8597. Name: "example_pb",
  8598. },
  8599. Messages: []*descriptor.Message{reqMsg, resMsg},
  8600. Services: []*descriptor.Service{
  8601. {
  8602. ServiceDescriptorProto: svc,
  8603. Methods: []*descriptor.Method{
  8604. {
  8605. MethodDescriptorProto: meth,
  8606. RequestType: reqMsg,
  8607. ResponseType: resMsg,
  8608. Bindings: []*descriptor.Binding{
  8609. {
  8610. HTTPMethod: "POST",
  8611. PathTmpl: httprule.Template{
  8612. Version: 1,
  8613. OpCodes: []int{0, 0},
  8614. Template: "/my/{field=segment/wi:th}",
  8615. },
  8616. PathParams: []descriptor.Parameter{
  8617. {
  8618. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  8619. {
  8620. Name: "field",
  8621. },
  8622. }),
  8623. Target: reqField,
  8624. },
  8625. },
  8626. Body: &descriptor.Body{
  8627. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  8628. },
  8629. },
  8630. },
  8631. },
  8632. },
  8633. },
  8634. },
  8635. }
  8636. reg := descriptor.NewRegistry()
  8637. reg.SetUseJSONNamesForFields(true)
  8638. err := reg.Load(reqFromFile(&file))
  8639. if err != nil {
  8640. t.Fatalf("failed to reg.Load(): %v", err)
  8641. }
  8642. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  8643. if err != nil {
  8644. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  8645. }
  8646. paths := GetPaths(result)
  8647. if got, want := len(paths), 1; got != want {
  8648. t.Fatalf("Results path length differed, got %d want %d", got, want)
  8649. }
  8650. if got, want := paths[0], "/my/{field}"; got != want {
  8651. t.Fatalf("Wrong results path, got %s want %s", got, want)
  8652. }
  8653. operation := *result.getPathItemObject("/my/{field}").Post
  8654. if got, want := len(operation.Parameters), 2; got != want {
  8655. t.Fatalf("Parameters length differed, got %d want %d", got, want)
  8656. }
  8657. if got, want := operation.Parameters[0].Name, "field"; got != want {
  8658. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8659. }
  8660. if got, want := operation.Parameters[0].In, "path"; got != want {
  8661. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8662. }
  8663. if got, want := operation.Parameters[0].Type, "string"; got != want {
  8664. t.Fatalf("Wrong parameter type, got %s want %s", got, want)
  8665. }
  8666. if got, want := operation.Parameters[1].Name, "body"; got != want {
  8667. t.Fatalf("Wrong parameter name, got %s want %s", got, want)
  8668. }
  8669. if got, want := operation.Parameters[1].In, "body"; got != want {
  8670. t.Fatalf("Wrong parameter location, got %s want %s", got, want)
  8671. }
  8672. }
  8673. func TestRenderServiceWithHeaderParameters(t *testing.T) {
  8674. file := func() descriptor.File {
  8675. msgdesc := &descriptorpb.DescriptorProto{
  8676. Name: proto.String("ExampleMessage"),
  8677. }
  8678. meth := &descriptorpb.MethodDescriptorProto{
  8679. Name: proto.String("Example"),
  8680. InputType: proto.String("ExampleMessage"),
  8681. OutputType: proto.String("ExampleMessage"),
  8682. Options: &descriptorpb.MethodOptions{},
  8683. }
  8684. svc := &descriptorpb.ServiceDescriptorProto{
  8685. Name: proto.String("ExampleService"),
  8686. Method: []*descriptorpb.MethodDescriptorProto{meth},
  8687. }
  8688. msg := &descriptor.Message{
  8689. DescriptorProto: msgdesc,
  8690. }
  8691. return descriptor.File{
  8692. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  8693. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  8694. Name: proto.String("example.proto"),
  8695. Package: proto.String("example"),
  8696. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  8697. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  8698. Options: &descriptorpb.FileOptions{
  8699. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  8700. },
  8701. },
  8702. GoPkg: descriptor.GoPackage{
  8703. Path: "example.com/path/to/example/example.pb",
  8704. Name: "example_pb",
  8705. },
  8706. Messages: []*descriptor.Message{msg},
  8707. Services: []*descriptor.Service{
  8708. {
  8709. ServiceDescriptorProto: svc,
  8710. Methods: []*descriptor.Method{
  8711. {
  8712. MethodDescriptorProto: meth,
  8713. RequestType: msg,
  8714. ResponseType: msg,
  8715. Bindings: []*descriptor.Binding{
  8716. {
  8717. HTTPMethod: "GET",
  8718. PathTmpl: httprule.Template{
  8719. Version: 1,
  8720. OpCodes: []int{0, 0},
  8721. Template: "/v1/echo",
  8722. },
  8723. },
  8724. },
  8725. },
  8726. },
  8727. },
  8728. },
  8729. }
  8730. }
  8731. type test struct {
  8732. file func() descriptor.File
  8733. openapiOperation *openapi_options.Operation
  8734. parameters openapiParametersObject
  8735. }
  8736. tests := map[string]*test{
  8737. "type string": {
  8738. file: file,
  8739. openapiOperation: &openapi_options.Operation{
  8740. Parameters: &openapi_options.Parameters{
  8741. Headers: []*openapi_options.HeaderParameter{
  8742. {
  8743. Name: "X-Custom-Header",
  8744. Type: openapi_options.HeaderParameter_STRING,
  8745. },
  8746. },
  8747. },
  8748. },
  8749. parameters: openapiParametersObject{
  8750. {
  8751. Name: "X-Custom-Header",
  8752. In: "header",
  8753. Type: "string",
  8754. },
  8755. },
  8756. },
  8757. "type string with format": {
  8758. file: file,
  8759. openapiOperation: &openapi_options.Operation{
  8760. Parameters: &openapi_options.Parameters{
  8761. Headers: []*openapi_options.HeaderParameter{
  8762. {
  8763. Name: "X-Custom-Header",
  8764. Type: openapi_options.HeaderParameter_STRING,
  8765. Format: "uuid",
  8766. },
  8767. },
  8768. },
  8769. },
  8770. parameters: openapiParametersObject{
  8771. {
  8772. Name: "X-Custom-Header",
  8773. In: "header",
  8774. Type: "string",
  8775. Format: "uuid",
  8776. },
  8777. },
  8778. },
  8779. "type integer": {
  8780. file: file,
  8781. openapiOperation: &openapi_options.Operation{
  8782. Parameters: &openapi_options.Parameters{
  8783. Headers: []*openapi_options.HeaderParameter{
  8784. {
  8785. Name: "X-Custom-Header",
  8786. Type: openapi_options.HeaderParameter_INTEGER,
  8787. },
  8788. },
  8789. },
  8790. },
  8791. parameters: openapiParametersObject{
  8792. {
  8793. Name: "X-Custom-Header",
  8794. In: "header",
  8795. Type: "integer",
  8796. },
  8797. },
  8798. },
  8799. "type number": {
  8800. file: file,
  8801. openapiOperation: &openapi_options.Operation{
  8802. Parameters: &openapi_options.Parameters{
  8803. Headers: []*openapi_options.HeaderParameter{
  8804. {
  8805. Name: "X-Custom-Header",
  8806. Type: openapi_options.HeaderParameter_NUMBER,
  8807. },
  8808. },
  8809. },
  8810. },
  8811. parameters: openapiParametersObject{
  8812. {
  8813. Name: "X-Custom-Header",
  8814. In: "header",
  8815. Type: "number",
  8816. },
  8817. },
  8818. },
  8819. "type boolean": {
  8820. file: file,
  8821. openapiOperation: &openapi_options.Operation{
  8822. Parameters: &openapi_options.Parameters{
  8823. Headers: []*openapi_options.HeaderParameter{
  8824. {
  8825. Name: "X-Custom-Header",
  8826. Type: openapi_options.HeaderParameter_BOOLEAN,
  8827. },
  8828. },
  8829. },
  8830. },
  8831. parameters: openapiParametersObject{
  8832. {
  8833. Name: "X-Custom-Header",
  8834. In: "header",
  8835. Type: "boolean",
  8836. },
  8837. },
  8838. },
  8839. "header required": {
  8840. file: file,
  8841. openapiOperation: &openapi_options.Operation{
  8842. Parameters: &openapi_options.Parameters{
  8843. Headers: []*openapi_options.HeaderParameter{
  8844. {
  8845. Name: "X-Custom-Header",
  8846. Required: true,
  8847. Type: openapi_options.HeaderParameter_STRING,
  8848. },
  8849. },
  8850. },
  8851. },
  8852. parameters: openapiParametersObject{
  8853. {
  8854. Name: "X-Custom-Header",
  8855. In: "header",
  8856. Required: true,
  8857. Type: "string",
  8858. },
  8859. },
  8860. },
  8861. }
  8862. for name, test := range tests {
  8863. test := test
  8864. t.Run(name, func(t *testing.T) {
  8865. file := test.file()
  8866. proto.SetExtension(
  8867. proto.Message(file.Services[0].Methods[0].Options),
  8868. openapi_options.E_Openapiv2Operation,
  8869. test.openapiOperation)
  8870. reg := descriptor.NewRegistry()
  8871. fileCL := crossLinkFixture(&file)
  8872. err := reg.Load(reqFromFile(fileCL))
  8873. if err != nil {
  8874. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  8875. }
  8876. result, err := applyTemplate(param{File: fileCL, reg: reg})
  8877. if err != nil {
  8878. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  8879. }
  8880. params := result.getPathItemObject("/v1/echo").Get.Parameters
  8881. if !reflect.DeepEqual(params, test.parameters) {
  8882. t.Errorf("expected %+v, got %+v", test.parameters, params)
  8883. }
  8884. })
  8885. }
  8886. }
  8887. func GetPaths(req *openapiSwaggerObject) []string {
  8888. paths := make([]string, len(req.Paths))
  8889. i := 0
  8890. for _, k := range req.Paths {
  8891. paths[i] = k.Path
  8892. i++
  8893. }
  8894. return paths
  8895. }
  8896. func TestRenderServicesOpenapiPathsOrderPreserved(t *testing.T) {
  8897. reqDesc := &descriptorpb.DescriptorProto{
  8898. Name: proto.String("MyRequest"),
  8899. Field: []*descriptorpb.FieldDescriptorProto{
  8900. {
  8901. Name: proto.String("field"),
  8902. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8903. Number: proto.Int32(1),
  8904. },
  8905. },
  8906. }
  8907. resDesc := &descriptorpb.DescriptorProto{
  8908. Name: proto.String("MyResponse"),
  8909. Field: []*descriptorpb.FieldDescriptorProto{
  8910. {
  8911. Name: proto.String("field"),
  8912. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  8913. Number: proto.Int32(1),
  8914. },
  8915. },
  8916. }
  8917. meth1 := &descriptorpb.MethodDescriptorProto{
  8918. Name: proto.String("MyMethod1"),
  8919. InputType: proto.String("MyRequest"),
  8920. OutputType: proto.String("MyResponse"),
  8921. }
  8922. meth2 := &descriptorpb.MethodDescriptorProto{
  8923. Name: proto.String("MyMethod2"),
  8924. InputType: proto.String("MyRequest"),
  8925. OutputType: proto.String("MyResponse"),
  8926. }
  8927. svc := &descriptorpb.ServiceDescriptorProto{
  8928. Name: proto.String("MyService"),
  8929. Method: []*descriptorpb.MethodDescriptorProto{meth1, meth2},
  8930. }
  8931. reqMsg := &descriptor.Message{
  8932. DescriptorProto: reqDesc,
  8933. }
  8934. resMsg := &descriptor.Message{
  8935. DescriptorProto: resDesc,
  8936. }
  8937. reqField := &descriptor.Field{
  8938. Message: reqMsg,
  8939. FieldDescriptorProto: reqMsg.GetField()[0],
  8940. }
  8941. resField := &descriptor.Field{
  8942. Message: resMsg,
  8943. FieldDescriptorProto: resMsg.GetField()[0],
  8944. }
  8945. reqField.JsonName = proto.String("field")
  8946. resField.JsonName = proto.String("field")
  8947. reqMsg.Fields = []*descriptor.Field{reqField}
  8948. resMsg.Fields = []*descriptor.Field{resField}
  8949. file := descriptor.File{
  8950. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  8951. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  8952. Package: proto.String("example"),
  8953. Name: proto.String(",my_service.proto"),
  8954. MessageType: []*descriptorpb.DescriptorProto{reqDesc, resDesc},
  8955. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  8956. Options: &descriptorpb.FileOptions{
  8957. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  8958. },
  8959. },
  8960. GoPkg: descriptor.GoPackage{
  8961. Path: "example.com/path/to/example/example.pb",
  8962. Name: "example_pb",
  8963. },
  8964. Messages: []*descriptor.Message{reqMsg, resMsg},
  8965. Services: []*descriptor.Service{
  8966. {
  8967. ServiceDescriptorProto: svc,
  8968. Methods: []*descriptor.Method{
  8969. {
  8970. MethodDescriptorProto: meth1,
  8971. RequestType: reqMsg,
  8972. ResponseType: resMsg,
  8973. Bindings: []*descriptor.Binding{
  8974. {
  8975. HTTPMethod: "POST",
  8976. PathTmpl: httprule.Template{
  8977. Version: 1,
  8978. OpCodes: []int{0, 0},
  8979. Template: "/c/cpath",
  8980. },
  8981. },
  8982. },
  8983. }, {
  8984. MethodDescriptorProto: meth2,
  8985. RequestType: reqMsg,
  8986. ResponseType: resMsg,
  8987. Bindings: []*descriptor.Binding{
  8988. {
  8989. HTTPMethod: "POST",
  8990. PathTmpl: httprule.Template{
  8991. Version: 1,
  8992. OpCodes: []int{0, 0},
  8993. Template: "/b/bpath",
  8994. },
  8995. },
  8996. },
  8997. },
  8998. },
  8999. },
  9000. },
  9001. }
  9002. reg := descriptor.NewRegistry()
  9003. reg.SetPreserveRPCOrder(true)
  9004. err := reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto}})
  9005. if err != nil {
  9006. t.Fatalf("failed to reg.Load(): %v", err)
  9007. }
  9008. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  9009. if err != nil {
  9010. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  9011. }
  9012. paths := result.Paths
  9013. firstRPCPath := file.Services[0].Methods[0].Bindings[0].PathTmpl.Template
  9014. secondRPCPath := file.Services[0].Methods[1].Bindings[0].PathTmpl.Template
  9015. for i, pathData := range paths {
  9016. switch i {
  9017. case 0:
  9018. if got, want := pathData.Path, firstRPCPath; got != want {
  9019. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9020. }
  9021. case 1:
  9022. if got, want := pathData.Path, secondRPCPath; got != want {
  9023. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9024. }
  9025. }
  9026. }
  9027. }
  9028. func TestRenderServicesOpenapiPathsOrderPreservedMultipleServices(t *testing.T) {
  9029. reqDesc := &descriptorpb.DescriptorProto{
  9030. Name: proto.String("MyRequest"),
  9031. Field: []*descriptorpb.FieldDescriptorProto{
  9032. {
  9033. Name: proto.String("field"),
  9034. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  9035. Number: proto.Int32(1),
  9036. },
  9037. },
  9038. }
  9039. resDesc := &descriptorpb.DescriptorProto{
  9040. Name: proto.String("MyResponse"),
  9041. Field: []*descriptorpb.FieldDescriptorProto{
  9042. {
  9043. Name: proto.String("field"),
  9044. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  9045. Number: proto.Int32(1),
  9046. },
  9047. },
  9048. }
  9049. meth1 := &descriptorpb.MethodDescriptorProto{
  9050. Name: proto.String("MyMethod1"),
  9051. InputType: proto.String("MyRequest"),
  9052. OutputType: proto.String("MyResponse"),
  9053. }
  9054. meth2 := &descriptorpb.MethodDescriptorProto{
  9055. Name: proto.String("MyMethod2"),
  9056. InputType: proto.String("MyRequest"),
  9057. OutputType: proto.String("MyResponse"),
  9058. }
  9059. meth3 := &descriptorpb.MethodDescriptorProto{
  9060. Name: proto.String("MyMethod3"),
  9061. InputType: proto.String("MyRequest"),
  9062. OutputType: proto.String("MyResponse"),
  9063. }
  9064. meth4 := &descriptorpb.MethodDescriptorProto{
  9065. Name: proto.String("MyMethod4"),
  9066. InputType: proto.String("MyRequest"),
  9067. OutputType: proto.String("MyResponse"),
  9068. }
  9069. svc1 := &descriptorpb.ServiceDescriptorProto{
  9070. Name: proto.String("MyServiceOne"),
  9071. Method: []*descriptorpb.MethodDescriptorProto{meth1, meth2},
  9072. }
  9073. svc2 := &descriptorpb.ServiceDescriptorProto{
  9074. Name: proto.String("MyServiceTwo"),
  9075. Method: []*descriptorpb.MethodDescriptorProto{meth3, meth4},
  9076. }
  9077. reqMsg := &descriptor.Message{
  9078. DescriptorProto: reqDesc,
  9079. }
  9080. resMsg := &descriptor.Message{
  9081. DescriptorProto: resDesc,
  9082. }
  9083. reqField := &descriptor.Field{
  9084. Message: reqMsg,
  9085. FieldDescriptorProto: reqMsg.GetField()[0],
  9086. }
  9087. resField := &descriptor.Field{
  9088. Message: resMsg,
  9089. FieldDescriptorProto: resMsg.GetField()[0],
  9090. }
  9091. reqField.JsonName = proto.String("field")
  9092. resField.JsonName = proto.String("field")
  9093. reqMsg.Fields = []*descriptor.Field{reqField}
  9094. resMsg.Fields = []*descriptor.Field{resField}
  9095. file := descriptor.File{
  9096. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  9097. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  9098. Package: proto.String("example"),
  9099. Name: proto.String(",my_service.proto"),
  9100. MessageType: []*descriptorpb.DescriptorProto{reqDesc, resDesc},
  9101. Service: []*descriptorpb.ServiceDescriptorProto{svc1, svc2},
  9102. Options: &descriptorpb.FileOptions{
  9103. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  9104. },
  9105. },
  9106. GoPkg: descriptor.GoPackage{
  9107. Path: "example.com/path/to/example/example.pb",
  9108. Name: "example_pb",
  9109. },
  9110. Messages: []*descriptor.Message{reqMsg, resMsg},
  9111. Services: []*descriptor.Service{
  9112. {
  9113. ServiceDescriptorProto: svc1,
  9114. Methods: []*descriptor.Method{
  9115. {
  9116. MethodDescriptorProto: meth1,
  9117. RequestType: reqMsg,
  9118. ResponseType: resMsg,
  9119. Bindings: []*descriptor.Binding{
  9120. {
  9121. HTTPMethod: "POST",
  9122. PathTmpl: httprule.Template{
  9123. Version: 1,
  9124. OpCodes: []int{0, 0},
  9125. Template: "/g/gpath",
  9126. },
  9127. },
  9128. },
  9129. }, {
  9130. MethodDescriptorProto: meth2,
  9131. RequestType: reqMsg,
  9132. ResponseType: resMsg,
  9133. Bindings: []*descriptor.Binding{
  9134. {
  9135. HTTPMethod: "POST",
  9136. PathTmpl: httprule.Template{
  9137. Version: 1,
  9138. OpCodes: []int{0, 0},
  9139. Template: "/f/fpath",
  9140. },
  9141. },
  9142. },
  9143. },
  9144. },
  9145. }, {
  9146. ServiceDescriptorProto: svc1,
  9147. Methods: []*descriptor.Method{
  9148. {
  9149. MethodDescriptorProto: meth3,
  9150. RequestType: reqMsg,
  9151. ResponseType: resMsg,
  9152. Bindings: []*descriptor.Binding{
  9153. {
  9154. HTTPMethod: "POST",
  9155. PathTmpl: httprule.Template{
  9156. Version: 1,
  9157. OpCodes: []int{0, 0},
  9158. Template: "/c/cpath",
  9159. },
  9160. },
  9161. },
  9162. }, {
  9163. MethodDescriptorProto: meth4,
  9164. RequestType: reqMsg,
  9165. ResponseType: resMsg,
  9166. Bindings: []*descriptor.Binding{
  9167. {
  9168. HTTPMethod: "POST",
  9169. PathTmpl: httprule.Template{
  9170. Version: 1,
  9171. OpCodes: []int{0, 0},
  9172. Template: "/b/bpath",
  9173. },
  9174. },
  9175. },
  9176. },
  9177. },
  9178. },
  9179. },
  9180. }
  9181. reg := descriptor.NewRegistry()
  9182. reg.SetPreserveRPCOrder(true)
  9183. reg.SetUseJSONNamesForFields(true)
  9184. err := reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto}})
  9185. if err != nil {
  9186. t.Fatalf("failed to reg.Load(): %v", err)
  9187. }
  9188. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  9189. if err != nil {
  9190. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  9191. }
  9192. paths := result.Paths
  9193. firstRPCPath := file.Services[0].Methods[0].Bindings[0].PathTmpl.Template
  9194. secondRPCPath := file.Services[0].Methods[1].Bindings[0].PathTmpl.Template
  9195. thirdRPCPath := file.Services[1].Methods[0].Bindings[0].PathTmpl.Template
  9196. fourthRPCPath := file.Services[1].Methods[1].Bindings[0].PathTmpl.Template
  9197. for i, pathData := range paths {
  9198. switch i {
  9199. case 0:
  9200. if got, want := pathData.Path, firstRPCPath; got != want {
  9201. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9202. }
  9203. case 1:
  9204. if got, want := pathData.Path, secondRPCPath; got != want {
  9205. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9206. }
  9207. case 2:
  9208. if got, want := pathData.Path, thirdRPCPath; got != want {
  9209. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9210. }
  9211. case 3:
  9212. if got, want := pathData.Path, fourthRPCPath; got != want {
  9213. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9214. }
  9215. }
  9216. }
  9217. }
  9218. func TestRenderServicesOpenapiPathsOrderPreservedAdditionalBindings(t *testing.T) {
  9219. reqDesc := &descriptorpb.DescriptorProto{
  9220. Name: proto.String("MyRequest"),
  9221. Field: []*descriptorpb.FieldDescriptorProto{
  9222. {
  9223. Name: proto.String("field"),
  9224. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  9225. Number: proto.Int32(1),
  9226. },
  9227. },
  9228. }
  9229. resDesc := &descriptorpb.DescriptorProto{
  9230. Name: proto.String("MyResponse"),
  9231. Field: []*descriptorpb.FieldDescriptorProto{
  9232. {
  9233. Name: proto.String("field"),
  9234. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  9235. Number: proto.Int32(1),
  9236. },
  9237. },
  9238. }
  9239. meth1 := &descriptorpb.MethodDescriptorProto{
  9240. Name: proto.String("MyMethod1"),
  9241. InputType: proto.String("MyRequest"),
  9242. OutputType: proto.String("MyResponse"),
  9243. }
  9244. meth2 := &descriptorpb.MethodDescriptorProto{
  9245. Name: proto.String("MyMethod2"),
  9246. InputType: proto.String("MyRequest"),
  9247. OutputType: proto.String("MyResponse"),
  9248. }
  9249. svc := &descriptorpb.ServiceDescriptorProto{
  9250. Name: proto.String("MyService"),
  9251. Method: []*descriptorpb.MethodDescriptorProto{meth1, meth2},
  9252. }
  9253. reqMsg := &descriptor.Message{
  9254. DescriptorProto: reqDesc,
  9255. }
  9256. resMsg := &descriptor.Message{
  9257. DescriptorProto: resDesc,
  9258. }
  9259. reqField := &descriptor.Field{
  9260. Message: reqMsg,
  9261. FieldDescriptorProto: reqMsg.GetField()[0],
  9262. }
  9263. resField := &descriptor.Field{
  9264. Message: resMsg,
  9265. FieldDescriptorProto: resMsg.GetField()[0],
  9266. }
  9267. reqField.JsonName = proto.String("field")
  9268. resField.JsonName = proto.String("field")
  9269. reqMsg.Fields = []*descriptor.Field{reqField}
  9270. resMsg.Fields = []*descriptor.Field{resField}
  9271. file := descriptor.File{
  9272. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  9273. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  9274. Package: proto.String("example"),
  9275. Name: proto.String(",my_service.proto"),
  9276. MessageType: []*descriptorpb.DescriptorProto{reqDesc, resDesc},
  9277. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  9278. Options: &descriptorpb.FileOptions{
  9279. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  9280. },
  9281. },
  9282. GoPkg: descriptor.GoPackage{
  9283. Path: "example.com/path/to/example/example.pb",
  9284. Name: "example_pb",
  9285. },
  9286. Messages: []*descriptor.Message{reqMsg, resMsg},
  9287. Services: []*descriptor.Service{
  9288. {
  9289. ServiceDescriptorProto: svc,
  9290. Methods: []*descriptor.Method{
  9291. {
  9292. MethodDescriptorProto: meth1,
  9293. RequestType: reqMsg,
  9294. ResponseType: resMsg,
  9295. Bindings: []*descriptor.Binding{
  9296. {
  9297. HTTPMethod: "POST",
  9298. PathTmpl: httprule.Template{
  9299. Version: 1,
  9300. OpCodes: []int{0, 0},
  9301. Template: "/c/cpath",
  9302. },
  9303. }, {
  9304. HTTPMethod: "GET",
  9305. PathTmpl: httprule.Template{
  9306. Version: 1,
  9307. OpCodes: []int{0, 0},
  9308. Template: "/additionalbinding",
  9309. },
  9310. },
  9311. },
  9312. }, {
  9313. MethodDescriptorProto: meth2,
  9314. RequestType: reqMsg,
  9315. ResponseType: resMsg,
  9316. Bindings: []*descriptor.Binding{
  9317. {
  9318. HTTPMethod: "POST",
  9319. PathTmpl: httprule.Template{
  9320. Version: 1,
  9321. OpCodes: []int{0, 0},
  9322. Template: "/b/bpath",
  9323. },
  9324. },
  9325. },
  9326. },
  9327. },
  9328. },
  9329. },
  9330. }
  9331. reg := descriptor.NewRegistry()
  9332. reg.SetPreserveRPCOrder(true)
  9333. reg.SetUseJSONNamesForFields(true)
  9334. err := reg.Load(&pluginpb.CodeGeneratorRequest{ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto}})
  9335. if err != nil {
  9336. t.Fatalf("failed to reg.Load(): %v", err)
  9337. }
  9338. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  9339. if err != nil {
  9340. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  9341. }
  9342. paths := result.Paths
  9343. if err != nil {
  9344. t.Fatalf("failed to obtain extension paths: %v", err)
  9345. }
  9346. firstRPCPath := file.Services[0].Methods[0].Bindings[0].PathTmpl.Template
  9347. firstRPCPathAdditionalBinding := file.Services[0].Methods[0].Bindings[1].PathTmpl.Template
  9348. secondRPCPath := file.Services[0].Methods[1].Bindings[0].PathTmpl.Template
  9349. for i, pathData := range paths {
  9350. switch i {
  9351. case 0:
  9352. if got, want := pathData.Path, firstRPCPath; got != want {
  9353. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9354. }
  9355. case 1:
  9356. if got, want := pathData.Path, firstRPCPathAdditionalBinding; got != want {
  9357. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9358. }
  9359. case 2:
  9360. if got, want := pathData.Path, secondRPCPath; got != want {
  9361. t.Fatalf("RPC path order not preserved, got %s want %s", got, want)
  9362. }
  9363. }
  9364. }
  9365. }
  9366. func TestRenderServicesOpenapiRequiredBodyFieldContainingPathParam(t *testing.T) {
  9367. fieldBehaviorRequired := []annotations.FieldBehavior{annotations.FieldBehavior_REQUIRED}
  9368. requiredFieldOptions := new(descriptorpb.FieldOptions)
  9369. proto.SetExtension(requiredFieldOptions, annotations.E_FieldBehavior, fieldBehaviorRequired)
  9370. bookDesc := &descriptorpb.DescriptorProto{
  9371. Name: proto.String("Book"),
  9372. Field: []*descriptorpb.FieldDescriptorProto{
  9373. {
  9374. Name: proto.String("name"),
  9375. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  9376. Number: proto.Int32(1),
  9377. },
  9378. {
  9379. Name: proto.String("type"),
  9380. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  9381. Number: proto.Int32(2),
  9382. },
  9383. },
  9384. }
  9385. addBookReqDesc := &descriptorpb.DescriptorProto{
  9386. Name: proto.String("AddBookReq"),
  9387. Field: []*descriptorpb.FieldDescriptorProto{
  9388. {
  9389. Name: proto.String("book"),
  9390. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  9391. TypeName: proto.String(".Book"),
  9392. Number: proto.Int32(1),
  9393. Options: requiredFieldOptions,
  9394. },
  9395. {
  9396. Name: proto.String("libraryId"),
  9397. Type: descriptorpb.FieldDescriptorProto_TYPE_UINT32.Enum(),
  9398. Number: proto.Int32(2),
  9399. Options: requiredFieldOptions,
  9400. },
  9401. {
  9402. Name: proto.String("isLatestEdition"),
  9403. Type: descriptorpb.FieldDescriptorProto_TYPE_BOOL.Enum(),
  9404. Number: proto.Int32(3),
  9405. },
  9406. },
  9407. }
  9408. meth := &descriptorpb.MethodDescriptorProto{
  9409. Name: proto.String("AddBook"),
  9410. InputType: proto.String("AddBookReq"),
  9411. OutputType: proto.String("Book"),
  9412. }
  9413. svc := &descriptorpb.ServiceDescriptorProto{
  9414. Name: proto.String("BookService"),
  9415. Method: []*descriptorpb.MethodDescriptorProto{meth},
  9416. }
  9417. bookMsg := &descriptor.Message{
  9418. DescriptorProto: bookDesc,
  9419. }
  9420. addBookReqMsg := &descriptor.Message{
  9421. DescriptorProto: addBookReqDesc,
  9422. }
  9423. nameField := &descriptor.Field{
  9424. Message: bookMsg,
  9425. FieldDescriptorProto: bookMsg.GetField()[0],
  9426. }
  9427. typeField := &descriptor.Field{
  9428. Message: bookMsg,
  9429. FieldDescriptorProto: bookMsg.GetField()[1],
  9430. }
  9431. bookMsg.Fields = []*descriptor.Field{nameField, typeField}
  9432. bookField := &descriptor.Field{
  9433. Message: addBookReqMsg,
  9434. FieldMessage: bookMsg,
  9435. FieldDescriptorProto: addBookReqMsg.GetField()[0],
  9436. }
  9437. libraryIdField := &descriptor.Field{
  9438. Message: addBookReqMsg,
  9439. FieldDescriptorProto: addBookReqMsg.GetField()[1],
  9440. }
  9441. isLatestEditionField := &descriptor.Field{
  9442. Message: addBookReqMsg,
  9443. FieldDescriptorProto: addBookReqMsg.GetField()[2],
  9444. }
  9445. addBookReqMsg.Fields = []*descriptor.Field{bookField, libraryIdField, isLatestEditionField}
  9446. file := descriptor.File{
  9447. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  9448. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  9449. Name: proto.String("book.proto"),
  9450. MessageType: []*descriptorpb.DescriptorProto{bookDesc, addBookReqDesc},
  9451. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  9452. Options: &descriptorpb.FileOptions{
  9453. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  9454. },
  9455. },
  9456. GoPkg: descriptor.GoPackage{
  9457. Path: "example.com/path/to/example/example.pb",
  9458. Name: "example_pb",
  9459. },
  9460. Messages: []*descriptor.Message{bookMsg, addBookReqMsg},
  9461. Services: []*descriptor.Service{
  9462. {
  9463. ServiceDescriptorProto: svc,
  9464. Methods: []*descriptor.Method{
  9465. {
  9466. MethodDescriptorProto: meth,
  9467. RequestType: addBookReqMsg,
  9468. ResponseType: bookMsg,
  9469. Bindings: []*descriptor.Binding{
  9470. {
  9471. HTTPMethod: "POST",
  9472. PathTmpl: httprule.Template{
  9473. Version: 1,
  9474. OpCodes: []int{0, 0},
  9475. Template: "/v1/books/{book.type}",
  9476. },
  9477. PathParams: []descriptor.Parameter{
  9478. {
  9479. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{
  9480. {
  9481. Name: "book",
  9482. },
  9483. {
  9484. Name: "type",
  9485. },
  9486. }),
  9487. Target: typeField,
  9488. },
  9489. },
  9490. Body: &descriptor.Body{
  9491. FieldPath: []descriptor.FieldPathComponent{},
  9492. },
  9493. },
  9494. },
  9495. },
  9496. },
  9497. },
  9498. },
  9499. }
  9500. reg := descriptor.NewRegistry()
  9501. fileCL := crossLinkFixture(&file)
  9502. err := reg.Load(reqFromFile(fileCL))
  9503. if err != nil {
  9504. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  9505. return
  9506. }
  9507. result, err := applyTemplate(param{File: fileCL, reg: reg})
  9508. if err != nil {
  9509. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  9510. }
  9511. paths := GetPaths(result)
  9512. if got, want := len(paths), 1; got != want {
  9513. t.Fatalf("Results path length differed, got %d want %d", got, want)
  9514. }
  9515. if got, want := paths[0], "/v1/books/{book.type}"; got != want {
  9516. t.Fatalf("Wrong results path, got %s want %s", got, want)
  9517. }
  9518. operation := *result.getPathItemObject("/v1/books/{book.type}").Post
  9519. if got, want := operation.Parameters[0].Name, "book.type"; got != want {
  9520. t.Fatalf("Wrong parameter name 0, got %s want %s", got, want)
  9521. }
  9522. if got, want := operation.Parameters[0].In, "path"; got != want {
  9523. t.Fatalf("Wrong parameter location 0, got %s want %s", got, want)
  9524. }
  9525. if got, want := operation.Parameters[1].Name, "body"; got != want {
  9526. t.Fatalf("Wrong parameter name 1, got %s want %s", got, want)
  9527. }
  9528. if got, want := operation.Parameters[1].In, "body"; got != want {
  9529. t.Fatalf("Wrong parameter location 1, got %s want %s", got, want)
  9530. }
  9531. if want, is, name := "#/definitions/BookServiceAddBookBody", operation.Parameters[1].Schema.schemaCore.Ref, "operation.Parameters[1].Schema.schemaCore.Ref"; !reflect.DeepEqual(is, want) {
  9532. t.Fatalf("%s = %s want to be %s", name, want, is)
  9533. }
  9534. definition, found := result.Definitions["BookServiceAddBookBody"]
  9535. if !found {
  9536. t.Fatalf("expecting definition to contain BookServiceAddBookBody")
  9537. }
  9538. if want, is, name := 3, len(*definition.Properties), "len(*definition.Properties)"; !reflect.DeepEqual(is, want) {
  9539. t.Fatalf("%s = %d want to be %d", name, want, is)
  9540. }
  9541. for index, keyValue := range []string{"book", "libraryId", "isLatestEdition"} {
  9542. if got, want := (*definition.Properties)[index].Key, keyValue; got != want {
  9543. t.Fatalf("Wrong definition property %d, got %s want %s", index, got, want)
  9544. }
  9545. }
  9546. correctRequiredFields := []string{"book", "libraryId"}
  9547. if got, want := definition.Required, correctRequiredFields; !reflect.DeepEqual(got, want) {
  9548. t.Fatalf("Wrong required fields in body definition, got = %s, want = %s", got, want)
  9549. }
  9550. }
  9551. func TestArrayMessageItemsType(t *testing.T) {
  9552. msgDesc := &descriptorpb.DescriptorProto{
  9553. Name: proto.String("ExampleMessage"),
  9554. Field: []*descriptorpb.FieldDescriptorProto{
  9555. {
  9556. Name: proto.String("children"),
  9557. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  9558. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  9559. TypeName: proto.String(".example.ExampleMessage"),
  9560. Number: proto.Int32(1),
  9561. JsonName: proto.String("children"),
  9562. },
  9563. },
  9564. }
  9565. nestDesc := &descriptorpb.DescriptorProto{
  9566. Name: proto.String("NestDescMessage"),
  9567. Field: []*descriptorpb.FieldDescriptorProto{
  9568. {
  9569. Name: proto.String("children"),
  9570. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  9571. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  9572. TypeName: proto.String(".example.ExampleMessage"),
  9573. Number: proto.Int32(1),
  9574. JsonName: proto.String("children"),
  9575. },
  9576. },
  9577. }
  9578. meth := &descriptorpb.MethodDescriptorProto{
  9579. Name: proto.String("Example"),
  9580. InputType: proto.String("ExampleMessage"),
  9581. OutputType: proto.String("NestDescMessage"),
  9582. }
  9583. svc := &descriptorpb.ServiceDescriptorProto{
  9584. Name: proto.String("ExampleService"),
  9585. Method: []*descriptorpb.MethodDescriptorProto{meth},
  9586. }
  9587. msg := &descriptor.Message{
  9588. DescriptorProto: msgDesc,
  9589. }
  9590. nsg := &descriptor.Message{
  9591. DescriptorProto: nestDesc,
  9592. }
  9593. msg.Fields = []*descriptor.Field{
  9594. {
  9595. Message: msg,
  9596. FieldDescriptorProto: msg.GetField()[0],
  9597. },
  9598. }
  9599. nsg.Fields = []*descriptor.Field{
  9600. {
  9601. Message: nsg,
  9602. FieldDescriptorProto: nsg.GetField()[0],
  9603. },
  9604. }
  9605. file := descriptor.File{
  9606. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  9607. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  9608. Name: proto.String("example.proto"),
  9609. Package: proto.String("example"),
  9610. MessageType: []*descriptorpb.DescriptorProto{msgDesc, nestDesc},
  9611. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  9612. Options: &descriptorpb.FileOptions{
  9613. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  9614. },
  9615. },
  9616. GoPkg: descriptor.GoPackage{
  9617. Path: "example.com/path/to/example/example.pb",
  9618. Name: "example_pb",
  9619. },
  9620. Messages: []*descriptor.Message{msg, nsg},
  9621. Services: []*descriptor.Service{
  9622. {
  9623. ServiceDescriptorProto: svc,
  9624. Methods: []*descriptor.Method{
  9625. {
  9626. MethodDescriptorProto: meth,
  9627. RequestType: msg,
  9628. ResponseType: nsg,
  9629. Bindings: []*descriptor.Binding{
  9630. {
  9631. HTTPMethod: "POST",
  9632. Body: &descriptor.Body{
  9633. FieldPath: descriptor.FieldPath([]descriptor.FieldPathComponent{}),
  9634. },
  9635. PathTmpl: httprule.Template{
  9636. Version: 1,
  9637. OpCodes: []int{0, 0},
  9638. Template: "/v1/echo", // TODO(achew22): Figure out what this should really be
  9639. },
  9640. },
  9641. },
  9642. },
  9643. },
  9644. },
  9645. },
  9646. }
  9647. reg := descriptor.NewRegistry()
  9648. reg.SetUseJSONNamesForFields(true)
  9649. if err := AddErrorDefs(reg); err != nil {
  9650. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  9651. return
  9652. }
  9653. fileCL := crossLinkFixture(&file)
  9654. if err := reg.Load(&pluginpb.CodeGeneratorRequest{
  9655. ProtoFile: []*descriptorpb.FileDescriptorProto{
  9656. {
  9657. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  9658. Name: proto.String("acme/example.proto"),
  9659. Package: proto.String("example"),
  9660. MessageType: []*descriptorpb.DescriptorProto{msgDesc, nestDesc},
  9661. Service: []*descriptorpb.ServiceDescriptorProto{},
  9662. Options: &descriptorpb.FileOptions{
  9663. GoPackage: proto.String("acme/example"),
  9664. },
  9665. },
  9666. },
  9667. }); err != nil {
  9668. t.Errorf("reg.Load(%#v) failed with %v; want success", reg, err)
  9669. return
  9670. }
  9671. expect := openapiDefinitionsObject{
  9672. "rpcStatus": openapiSchemaObject{
  9673. schemaCore: schemaCore{
  9674. Type: "object",
  9675. },
  9676. Properties: &openapiSchemaObjectProperties{
  9677. keyVal{
  9678. Key: "code",
  9679. Value: openapiSchemaObject{
  9680. schemaCore: schemaCore{
  9681. Type: "integer",
  9682. Format: "int32",
  9683. },
  9684. },
  9685. },
  9686. keyVal{
  9687. Key: "message",
  9688. Value: openapiSchemaObject{
  9689. schemaCore: schemaCore{
  9690. Type: "string",
  9691. },
  9692. },
  9693. },
  9694. keyVal{
  9695. Key: "details",
  9696. Value: openapiSchemaObject{
  9697. schemaCore: schemaCore{
  9698. Type: "array",
  9699. Items: &openapiItemsObject{
  9700. schemaCore: schemaCore{
  9701. Type: "object",
  9702. Ref: "#/definitions/protobufAny",
  9703. },
  9704. },
  9705. },
  9706. },
  9707. },
  9708. },
  9709. },
  9710. "exampleExampleMessage": openapiSchemaObject{
  9711. schemaCore: schemaCore{
  9712. Type: "object",
  9713. },
  9714. Properties: &openapiSchemaObjectProperties{
  9715. keyVal{
  9716. Key: "children",
  9717. Value: openapiSchemaObject{
  9718. schemaCore: schemaCore{
  9719. Type: "array",
  9720. Items: &openapiItemsObject{
  9721. schemaCore: schemaCore{
  9722. Type: "object",
  9723. Ref: "#/definitions/exampleExampleMessage",
  9724. },
  9725. },
  9726. },
  9727. },
  9728. },
  9729. },
  9730. },
  9731. "exampleNestDescMessage": openapiSchemaObject{
  9732. schemaCore: schemaCore{
  9733. Type: "object",
  9734. },
  9735. Properties: &openapiSchemaObjectProperties{
  9736. keyVal{
  9737. Key: "children",
  9738. Value: openapiSchemaObject{
  9739. schemaCore: schemaCore{
  9740. Type: "array",
  9741. Items: &openapiItemsObject{
  9742. schemaCore: schemaCore{
  9743. Type: "object",
  9744. Ref: "#/definitions/exampleExampleMessage",
  9745. },
  9746. },
  9747. },
  9748. },
  9749. },
  9750. },
  9751. },
  9752. "protobufAny": openapiSchemaObject{
  9753. schemaCore: schemaCore{
  9754. Type: "object",
  9755. },
  9756. Properties: &openapiSchemaObjectProperties{
  9757. keyVal{
  9758. Key: "@type",
  9759. Value: openapiSchemaObject{
  9760. schemaCore: schemaCore{
  9761. Type: "string",
  9762. },
  9763. },
  9764. },
  9765. },
  9766. AdditionalProperties: &openapiSchemaObject{},
  9767. },
  9768. }
  9769. result, err := applyTemplate(param{File: fileCL, reg: reg})
  9770. if err != nil {
  9771. t.Errorf("applyTemplate(%#v) failed with %v; want success", reg, err)
  9772. return
  9773. }
  9774. if want, is, name := []string{"application/json"}, result.Produces, "Produces"; !reflect.DeepEqual(is, want) {
  9775. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  9776. }
  9777. if want, is, name := expect, result.Definitions, "Produces"; !reflect.DeepEqual(is, want) {
  9778. t.Errorf("applyTemplate(%#v).%s = %v want to be %v", file, name, is, want)
  9779. }
  9780. // If there was a failure, print out the input and the json result for debugging.
  9781. if t.Failed() {
  9782. t.Errorf("had: %s", file)
  9783. t.Errorf("got: %s", fmt.Sprint(result))
  9784. }
  9785. }
  9786. func TestQueryParameterType(t *testing.T) {
  9787. ntDesc := &descriptorpb.DescriptorProto{
  9788. Name: proto.String("AddressEntry"),
  9789. Field: []*descriptorpb.FieldDescriptorProto{
  9790. {
  9791. Name: proto.String("key"),
  9792. Number: proto.Int32(1),
  9793. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  9794. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  9795. JsonName: proto.String("key"),
  9796. },
  9797. {
  9798. Name: proto.String("value"),
  9799. Number: proto.Int32(2),
  9800. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  9801. Type: descriptorpb.FieldDescriptorProto_TYPE_INT32.Enum(),
  9802. JsonName: proto.String("value"),
  9803. },
  9804. },
  9805. Options: &descriptorpb.MessageOptions{
  9806. MapEntry: proto.Bool(true),
  9807. },
  9808. }
  9809. msgDesc := &descriptorpb.DescriptorProto{
  9810. Name: proto.String("Person"),
  9811. Field: []*descriptorpb.FieldDescriptorProto{
  9812. {
  9813. Name: proto.String("Address"),
  9814. Number: proto.Int32(1),
  9815. Label: descriptorpb.FieldDescriptorProto_LABEL_REPEATED.Enum(),
  9816. Type: descriptorpb.FieldDescriptorProto_TYPE_MESSAGE.Enum(),
  9817. TypeName: proto.String(".example.com.Person.AddressEntry"),
  9818. JsonName: proto.String("Address"),
  9819. },
  9820. },
  9821. NestedType: []*descriptorpb.DescriptorProto{
  9822. ntDesc,
  9823. },
  9824. }
  9825. nesteDesc := &descriptorpb.DescriptorProto{
  9826. Name: proto.String("ExampleResponse"),
  9827. Field: []*descriptorpb.FieldDescriptorProto{
  9828. {
  9829. Name: proto.String("Key"),
  9830. Number: proto.Int32(1),
  9831. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  9832. Type: descriptorpb.FieldDescriptorProto_TYPE_STRING.Enum(),
  9833. JsonName: proto.String("Key"),
  9834. },
  9835. {
  9836. Name: proto.String("Value"),
  9837. Number: proto.Int32(2),
  9838. Label: descriptorpb.FieldDescriptorProto_LABEL_OPTIONAL.Enum(),
  9839. Type: descriptorpb.FieldDescriptorProto_TYPE_INT32.Enum(),
  9840. JsonName: proto.String("Value"),
  9841. },
  9842. },
  9843. }
  9844. meth := &descriptorpb.MethodDescriptorProto{
  9845. Name: proto.String("Example"),
  9846. InputType: proto.String("Person"),
  9847. OutputType: proto.String("ExampleResponse"),
  9848. }
  9849. svc := &descriptorpb.ServiceDescriptorProto{
  9850. Name: proto.String("ExampleService"),
  9851. Method: []*descriptorpb.MethodDescriptorProto{meth},
  9852. }
  9853. msg := &descriptor.Message{
  9854. DescriptorProto: msgDesc,
  9855. }
  9856. nt := &descriptor.Message{
  9857. DescriptorProto: ntDesc,
  9858. }
  9859. nest := &descriptor.Message{
  9860. DescriptorProto: nesteDesc,
  9861. }
  9862. msg.Fields = []*descriptor.Field{
  9863. {
  9864. Message: msg,
  9865. FieldDescriptorProto: msg.GetField()[0],
  9866. },
  9867. }
  9868. nt.Fields = []*descriptor.Field{
  9869. {
  9870. Message: nt,
  9871. FieldDescriptorProto: msg.GetField()[0],
  9872. },
  9873. }
  9874. nest.Fields = []*descriptor.Field{
  9875. {
  9876. Message: nest,
  9877. FieldDescriptorProto: nest.GetField()[0],
  9878. },
  9879. {
  9880. Message: nest,
  9881. FieldDescriptorProto: nest.GetField()[1],
  9882. },
  9883. }
  9884. file := descriptor.File{
  9885. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  9886. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  9887. Name: proto.String("person.proto"),
  9888. Package: proto.String("example.com"),
  9889. MessageType: []*descriptorpb.DescriptorProto{msgDesc, nesteDesc},
  9890. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  9891. Options: &descriptorpb.FileOptions{
  9892. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  9893. },
  9894. },
  9895. GoPkg: descriptor.GoPackage{
  9896. Path: "example.com/path/to/example/example.pb",
  9897. Name: "example_pb",
  9898. },
  9899. Messages: []*descriptor.Message{msg, nest},
  9900. Services: []*descriptor.Service{
  9901. {
  9902. ServiceDescriptorProto: svc,
  9903. Methods: []*descriptor.Method{
  9904. {
  9905. MethodDescriptorProto: meth,
  9906. RequestType: msg,
  9907. ResponseType: nest,
  9908. Bindings: []*descriptor.Binding{
  9909. {
  9910. HTTPMethod: "GET",
  9911. PathTmpl: httprule.Template{
  9912. Version: 1,
  9913. OpCodes: []int{0, 0},
  9914. Template: "/v1/echo",
  9915. },
  9916. },
  9917. },
  9918. },
  9919. },
  9920. },
  9921. },
  9922. }
  9923. expect := openapiPathsObject{{
  9924. Path: "/v1/echo",
  9925. PathItemObject: openapiPathItemObject{
  9926. Get: &openapiOperationObject{
  9927. Parameters: openapiParametersObject{
  9928. {
  9929. Name: "Address[string]",
  9930. 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`,
  9931. In: "query",
  9932. Type: "integer",
  9933. },
  9934. },
  9935. },
  9936. },
  9937. }}
  9938. reg := descriptor.NewRegistry()
  9939. reg.SetUseJSONNamesForFields(false)
  9940. if err := AddErrorDefs(reg); err != nil {
  9941. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  9942. return
  9943. }
  9944. fileCL := crossLinkFixture(&file)
  9945. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  9946. ProtoFile: []*descriptorpb.FileDescriptorProto{
  9947. {
  9948. Name: proto.String("person.proto"),
  9949. Package: proto.String("example.com"),
  9950. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  9951. MessageType: []*descriptorpb.DescriptorProto{msgDesc, nesteDesc},
  9952. Service: []*descriptorpb.ServiceDescriptorProto{},
  9953. Options: &descriptorpb.FileOptions{
  9954. GoPackage: proto.String("person.proto"),
  9955. },
  9956. },
  9957. },
  9958. })
  9959. if err != nil {
  9960. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  9961. return
  9962. }
  9963. result, err := applyTemplate(param{File: fileCL, reg: reg})
  9964. if err != nil {
  9965. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  9966. return
  9967. }
  9968. if want, is, name := []string{"application/json"}, result.Produces, "Produces"; !reflect.DeepEqual(is, want) {
  9969. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, is, want)
  9970. }
  9971. if want, is, name := expect[0].PathItemObject.Get.Parameters, result.getPathItemObject("/v1/echo").Get.Parameters, "Produces"; !reflect.DeepEqual(is, want) {
  9972. t.Errorf("applyTemplate(%#v).%s = %v want to be %v", file, name, is, want)
  9973. }
  9974. // If there was a failure, print out the input and the json result for debugging.
  9975. if t.Failed() {
  9976. t.Errorf("had: %s", file)
  9977. t.Errorf("got: %s", fmt.Sprint(result))
  9978. }
  9979. }
  9980. func TestApplyTemplateRequestWithServerStreamingHttpBody(t *testing.T) {
  9981. meth := &descriptorpb.MethodDescriptorProto{
  9982. Name: proto.String("Echo"),
  9983. InputType: proto.String(".google.api.HttpBody"),
  9984. OutputType: proto.String(".google.api.HttpBody"),
  9985. ClientStreaming: proto.Bool(false),
  9986. ServerStreaming: proto.Bool(true),
  9987. }
  9988. svc := &descriptorpb.ServiceDescriptorProto{
  9989. Name: proto.String("ExampleService"),
  9990. Method: []*descriptorpb.MethodDescriptorProto{meth},
  9991. }
  9992. httpBodyFile, err := protoregistry.GlobalFiles.FindFileByPath("google/api/httpbody.proto")
  9993. if err != nil {
  9994. t.Fatal(err)
  9995. }
  9996. httpBodyFile.SourceLocations()
  9997. desc, err := protoregistry.GlobalFiles.FindDescriptorByName("google.api.HttpBody")
  9998. if err != nil {
  9999. t.Fatal(err)
  10000. }
  10001. msg := &descriptor.Message{
  10002. DescriptorProto: protodesc.ToDescriptorProto(desc.(protoreflect.MessageDescriptor)),
  10003. File: &descriptor.File{
  10004. FileDescriptorProto: protodesc.ToFileDescriptorProto(httpBodyFile),
  10005. },
  10006. }
  10007. anyFile, err := protoregistry.GlobalFiles.FindFileByPath("google/protobuf/any.proto")
  10008. if err != nil {
  10009. t.Fatal(err)
  10010. }
  10011. file := descriptor.File{
  10012. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  10013. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  10014. Name: proto.String("example.proto"),
  10015. Package: proto.String("example"),
  10016. Dependency: []string{
  10017. "google/api/httpbody.proto",
  10018. },
  10019. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  10020. Options: &descriptorpb.FileOptions{
  10021. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  10022. },
  10023. },
  10024. GoPkg: descriptor.GoPackage{
  10025. Path: "example.com/path/to/example/example.pb",
  10026. Name: "example_pb",
  10027. },
  10028. Services: []*descriptor.Service{
  10029. {
  10030. ServiceDescriptorProto: svc,
  10031. Methods: []*descriptor.Method{
  10032. {
  10033. MethodDescriptorProto: meth,
  10034. RequestType: msg,
  10035. ResponseType: msg,
  10036. Bindings: []*descriptor.Binding{
  10037. {
  10038. HTTPMethod: "POST",
  10039. PathTmpl: httprule.Template{
  10040. Version: 1,
  10041. OpCodes: []int{0, 0},
  10042. Template: "/v1/echo",
  10043. },
  10044. },
  10045. },
  10046. },
  10047. },
  10048. },
  10049. },
  10050. }
  10051. reg := descriptor.NewRegistry()
  10052. if err := AddErrorDefs(reg); err != nil {
  10053. t.Errorf("AddErrorDefs(%#v) failed with %v; want success", reg, err)
  10054. return
  10055. }
  10056. err = reg.Load(&pluginpb.CodeGeneratorRequest{
  10057. ProtoFile: []*descriptorpb.FileDescriptorProto{
  10058. protodesc.ToFileDescriptorProto(anyFile),
  10059. protodesc.ToFileDescriptorProto(httpBodyFile),
  10060. file.FileDescriptorProto,
  10061. },
  10062. })
  10063. if err != nil {
  10064. t.Fatalf("failed to load code generator request: %v", err)
  10065. }
  10066. result, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  10067. if err != nil {
  10068. t.Errorf("applyTemplate(%#v) failed with %v; want success", file, err)
  10069. return
  10070. }
  10071. if want, got, name := 3, len(result.Definitions), "len(Definitions)"; !reflect.DeepEqual(got, want) {
  10072. t.Errorf("applyTemplate(%#v).%s = %d want to be %d", file, name, got, want)
  10073. }
  10074. if _, ok := result.getPathItemObject("/v1/echo").Post.Responses["200"]; !ok {
  10075. t.Errorf("applyTemplate(%#v).%s = expected 200 response to be defined", file, `result.getPathItemObject("/v1/echo").Post.Responses["200"]`)
  10076. } else {
  10077. if want, got, name := "A successful response.(streaming responses)", result.getPathItemObject("/v1/echo").Post.Responses["200"].Description, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Description`; !reflect.DeepEqual(got, want) {
  10078. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  10079. }
  10080. streamExampleExampleMessage := result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema
  10081. if want, got, name := "string", streamExampleExampleMessage.Type, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema.Type`; !reflect.DeepEqual(got, want) {
  10082. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  10083. }
  10084. if want, got, name := "binary", streamExampleExampleMessage.Format, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema.Format`; !reflect.DeepEqual(got, want) {
  10085. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  10086. }
  10087. if want, got, name := "Free form byte stream", streamExampleExampleMessage.Title, `result.getPathItemObject("/v1/echo").Post.Responses["200"].Schema.Title`; !reflect.DeepEqual(got, want) {
  10088. t.Errorf("applyTemplate(%#v).%s = %s want to be %s", file, name, got, want)
  10089. }
  10090. if len(*streamExampleExampleMessage.Properties) != 0 {
  10091. t.Errorf("applyTemplate(%#v).Properties should be empty", file)
  10092. }
  10093. }
  10094. // If there was a failure, print out the input and the json result for debugging.
  10095. if t.Failed() {
  10096. t.Errorf("had: %s", file)
  10097. t.Errorf("got: %s", fmt.Sprint(result))
  10098. }
  10099. }
  10100. // Returns the openapiPathItemObject associated with a path.
  10101. func (so openapiSwaggerObject) getPathItemObject(path string) openapiPathItemObject {
  10102. for _, pathData := range so.Paths {
  10103. if pathData.Path == path {
  10104. return pathData.PathItemObject
  10105. }
  10106. }
  10107. return openapiPathItemObject{}
  10108. }
  10109. func TestGetPathItemObjectSwaggerObjectMethod(t *testing.T) {
  10110. testCases := [...]struct {
  10111. testName string
  10112. swaggerObject openapiSwaggerObject
  10113. path string
  10114. expectedPathItemObject openapiPathItemObject
  10115. }{
  10116. {
  10117. testName: "Path present in swagger object",
  10118. swaggerObject: openapiSwaggerObject{Paths: openapiPathsObject{{
  10119. Path: "a/path",
  10120. PathItemObject: openapiPathItemObject{
  10121. Get: &openapiOperationObject{
  10122. Description: "A testful description",
  10123. },
  10124. },
  10125. }}},
  10126. path: "a/path",
  10127. expectedPathItemObject: openapiPathItemObject{
  10128. Get: &openapiOperationObject{
  10129. Description: "A testful description",
  10130. },
  10131. },
  10132. }, {
  10133. testName: "Path not present in swaggerObject",
  10134. swaggerObject: openapiSwaggerObject{Paths: openapiPathsObject{{
  10135. Path: "a/path",
  10136. PathItemObject: openapiPathItemObject{
  10137. Get: &openapiOperationObject{
  10138. Description: "A testful description",
  10139. },
  10140. },
  10141. }}},
  10142. path: "b/path",
  10143. expectedPathItemObject: openapiPathItemObject{},
  10144. }, {
  10145. testName: "Path present in swaggerPathsObject with multiple paths",
  10146. swaggerObject: openapiSwaggerObject{Paths: openapiPathsObject{{
  10147. Path: "a/path",
  10148. PathItemObject: openapiPathItemObject{
  10149. Get: &openapiOperationObject{
  10150. Description: "A testful description",
  10151. },
  10152. },
  10153. }, {
  10154. Path: "another/path",
  10155. PathItemObject: openapiPathItemObject{
  10156. Get: &openapiOperationObject{
  10157. Description: "Another testful description",
  10158. },
  10159. },
  10160. }}},
  10161. path: "another/path",
  10162. expectedPathItemObject: openapiPathItemObject{
  10163. Get: &openapiOperationObject{
  10164. Description: "Another testful description",
  10165. },
  10166. },
  10167. }, {
  10168. testName: "Path not present in swaggerObject with no paths",
  10169. swaggerObject: openapiSwaggerObject{},
  10170. path: "b/path",
  10171. expectedPathItemObject: openapiPathItemObject{},
  10172. },
  10173. }
  10174. for _, tc := range testCases {
  10175. tc := tc
  10176. t.Run(tc.testName, func(t *testing.T) {
  10177. actualPathItemObject := tc.swaggerObject.getPathItemObject(tc.path)
  10178. if isEqual := reflect.DeepEqual(actualPathItemObject, tc.expectedPathItemObject); !isEqual {
  10179. t.Fatalf("Got pathItemObject: %#v, want pathItemObject: %#v", actualPathItemObject, tc.expectedPathItemObject)
  10180. }
  10181. })
  10182. }
  10183. }
  10184. func TestGetPathItemObjectFunction(t *testing.T) {
  10185. testCases := [...]struct {
  10186. testName string
  10187. paths openapiPathsObject
  10188. path string
  10189. expectedPathItemObject openapiPathItemObject
  10190. expectedIsPathPresent bool
  10191. }{
  10192. {
  10193. testName: "Path present in openapiPathsObject",
  10194. paths: openapiPathsObject{{
  10195. Path: "a/path",
  10196. PathItemObject: openapiPathItemObject{
  10197. Get: &openapiOperationObject{
  10198. Description: "A testful description",
  10199. },
  10200. },
  10201. }},
  10202. path: "a/path",
  10203. expectedPathItemObject: openapiPathItemObject{
  10204. Get: &openapiOperationObject{
  10205. Description: "A testful description",
  10206. },
  10207. },
  10208. expectedIsPathPresent: true,
  10209. }, {
  10210. testName: "Path not present in openapiPathsObject",
  10211. paths: openapiPathsObject{{
  10212. Path: "a/path",
  10213. PathItemObject: openapiPathItemObject{
  10214. Get: &openapiOperationObject{
  10215. Description: "A testful description",
  10216. },
  10217. },
  10218. }},
  10219. path: "b/path",
  10220. expectedPathItemObject: openapiPathItemObject{},
  10221. expectedIsPathPresent: false,
  10222. }, {
  10223. testName: "Path present in openapiPathsObject with multiple paths",
  10224. paths: openapiPathsObject{{
  10225. Path: "a/path",
  10226. PathItemObject: openapiPathItemObject{
  10227. Get: &openapiOperationObject{
  10228. Description: "A testful description",
  10229. },
  10230. },
  10231. }, {
  10232. Path: "another/path",
  10233. PathItemObject: openapiPathItemObject{
  10234. Get: &openapiOperationObject{
  10235. Description: "Another testful description",
  10236. },
  10237. },
  10238. }},
  10239. path: "another/path",
  10240. expectedPathItemObject: openapiPathItemObject{
  10241. Get: &openapiOperationObject{
  10242. Description: "Another testful description",
  10243. },
  10244. },
  10245. expectedIsPathPresent: true,
  10246. }, {
  10247. testName: "Path not present in empty openapiPathsObject",
  10248. paths: openapiPathsObject{},
  10249. path: "b/path",
  10250. expectedPathItemObject: openapiPathItemObject{},
  10251. expectedIsPathPresent: false,
  10252. },
  10253. }
  10254. for _, tc := range testCases {
  10255. tc := tc
  10256. t.Run(tc.testName, func(t *testing.T) {
  10257. actualPathItemObject, actualIsPathPresent := getPathItemObject(tc.paths, tc.path)
  10258. if isEqual := reflect.DeepEqual(actualPathItemObject, tc.expectedPathItemObject); !isEqual {
  10259. t.Fatalf("Got pathItemObject: %#v, want pathItemObject: %#v", actualPathItemObject, tc.expectedPathItemObject)
  10260. }
  10261. if actualIsPathPresent != tc.expectedIsPathPresent {
  10262. t.Fatalf("Got isPathPresent bool: %t, want isPathPresent bool: %t", actualIsPathPresent, tc.expectedIsPathPresent)
  10263. }
  10264. })
  10265. }
  10266. }
  10267. func TestUpdatePaths(t *testing.T) {
  10268. testCases := [...]struct {
  10269. testName string
  10270. paths openapiPathsObject
  10271. pathToUpdate string
  10272. newPathItemObject openapiPathItemObject
  10273. expectedUpdatedPaths openapiPathsObject
  10274. }{
  10275. {
  10276. testName: "Path present in openapiPathsObject, pathItemObject updated.",
  10277. paths: openapiPathsObject{{
  10278. Path: "a/path",
  10279. PathItemObject: openapiPathItemObject{
  10280. Get: &openapiOperationObject{
  10281. Description: "A testful description",
  10282. },
  10283. },
  10284. }},
  10285. pathToUpdate: "a/path",
  10286. newPathItemObject: openapiPathItemObject{
  10287. Get: &openapiOperationObject{
  10288. Description: "A newly updated testful description",
  10289. },
  10290. },
  10291. expectedUpdatedPaths: openapiPathsObject{{
  10292. Path: "a/path",
  10293. PathItemObject: openapiPathItemObject{
  10294. Get: &openapiOperationObject{
  10295. Description: "A newly updated testful description",
  10296. },
  10297. },
  10298. }},
  10299. }, {
  10300. testName: "Path not present in openapiPathsObject, new path data appended.",
  10301. paths: openapiPathsObject{{
  10302. Path: "c/path",
  10303. PathItemObject: openapiPathItemObject{
  10304. Get: &openapiOperationObject{
  10305. Description: "A testful description",
  10306. },
  10307. },
  10308. }},
  10309. pathToUpdate: "b/path",
  10310. newPathItemObject: openapiPathItemObject{
  10311. Get: &openapiOperationObject{
  10312. Description: "A new testful description to add",
  10313. },
  10314. },
  10315. expectedUpdatedPaths: openapiPathsObject{{
  10316. Path: "c/path",
  10317. PathItemObject: openapiPathItemObject{
  10318. Get: &openapiOperationObject{
  10319. Description: "A testful description",
  10320. },
  10321. },
  10322. }, {
  10323. Path: "b/path",
  10324. PathItemObject: openapiPathItemObject{
  10325. Get: &openapiOperationObject{
  10326. Description: "A new testful description to add",
  10327. },
  10328. },
  10329. }},
  10330. }, {
  10331. testName: "No paths present in openapiPathsObject, new path data appended.",
  10332. paths: openapiPathsObject{},
  10333. pathToUpdate: "b/path",
  10334. newPathItemObject: openapiPathItemObject{
  10335. Get: &openapiOperationObject{
  10336. Description: "A new testful description to add",
  10337. },
  10338. },
  10339. expectedUpdatedPaths: openapiPathsObject{{
  10340. Path: "b/path",
  10341. PathItemObject: openapiPathItemObject{
  10342. Get: &openapiOperationObject{
  10343. Description: "A new testful description to add",
  10344. },
  10345. },
  10346. }},
  10347. },
  10348. }
  10349. for _, tc := range testCases {
  10350. tc := tc
  10351. t.Run(tc.testName, func(t *testing.T) {
  10352. updatePaths(&tc.paths, tc.pathToUpdate, tc.newPathItemObject)
  10353. if pathsCorrectlyUpdated := reflect.DeepEqual(tc.paths, tc.expectedUpdatedPaths); !pathsCorrectlyUpdated {
  10354. t.Fatalf("Paths not correctly updated. Want %#v, got %#v", tc.expectedUpdatedPaths, tc.paths)
  10355. }
  10356. })
  10357. }
  10358. }
  10359. // Test that enum values have internal comments removed
  10360. func TestEnumValueProtoComments(t *testing.T) {
  10361. reg := descriptor.NewRegistry()
  10362. name := "kind"
  10363. comments := "(-- this is a comment --)"
  10364. enum := &descriptor.Enum{
  10365. EnumDescriptorProto: &descriptorpb.EnumDescriptorProto{
  10366. Name: &name,
  10367. },
  10368. File: &descriptor.File{
  10369. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  10370. Name: new(string),
  10371. Package: new(string),
  10372. SourceCodeInfo: &descriptorpb.SourceCodeInfo{
  10373. Location: []*descriptorpb.SourceCodeInfo_Location{
  10374. {
  10375. LeadingComments: &comments,
  10376. },
  10377. },
  10378. },
  10379. },
  10380. },
  10381. }
  10382. comments = enumValueProtoComments(reg, enum)
  10383. if comments != "" {
  10384. t.Errorf("expected '', got '%v'", comments)
  10385. }
  10386. }
  10387. func MustMarshal(v interface{}) []byte {
  10388. b, err := json.Marshal(v)
  10389. if err != nil {
  10390. panic(err)
  10391. }
  10392. return b
  10393. }
  10394. func TestMergeTags(t *testing.T) {
  10395. testCases := [...]struct {
  10396. testName string
  10397. existingTags []openapiTagObject
  10398. newTags []openapiTagObject
  10399. expectedMergedTags []openapiTagObject
  10400. }{
  10401. {
  10402. testName: "Simple merge.",
  10403. existingTags: []openapiTagObject{{
  10404. Name: "tag1",
  10405. Description: "tag1 description",
  10406. }},
  10407. newTags: []openapiTagObject{{
  10408. Name: "tag2",
  10409. Description: "tag2 description",
  10410. }},
  10411. expectedMergedTags: []openapiTagObject{{
  10412. Name: "tag1",
  10413. Description: "tag1 description",
  10414. }, {
  10415. Name: "tag2",
  10416. Description: "tag2 description",
  10417. }},
  10418. },
  10419. {
  10420. testName: "Merge description",
  10421. existingTags: []openapiTagObject{{
  10422. Name: "tag1",
  10423. Description: "tag1 description",
  10424. }, {
  10425. Name: "tag2",
  10426. }, {
  10427. Name: "tag3",
  10428. Description: "tag3 description",
  10429. }},
  10430. newTags: []openapiTagObject{{
  10431. Name: "tag2",
  10432. Description: "tag2 description",
  10433. }},
  10434. expectedMergedTags: []openapiTagObject{{
  10435. Name: "tag1",
  10436. Description: "tag1 description",
  10437. }, {
  10438. Name: "tag2",
  10439. Description: "tag2 description",
  10440. }, {
  10441. Name: "tag3",
  10442. Description: "tag3 description",
  10443. }},
  10444. },
  10445. {
  10446. testName: "Merge external docs",
  10447. existingTags: []openapiTagObject{{
  10448. Name: "tag1",
  10449. ExternalDocs: &openapiExternalDocumentationObject{},
  10450. }, {
  10451. Name: "tag2",
  10452. }, {
  10453. Name: "tag3",
  10454. ExternalDocs: &openapiExternalDocumentationObject{
  10455. Description: "tag3 description",
  10456. },
  10457. }, {
  10458. Name: "tag4",
  10459. ExternalDocs: &openapiExternalDocumentationObject{
  10460. URL: "tag4 url",
  10461. },
  10462. }},
  10463. newTags: []openapiTagObject{{
  10464. Name: "tag1",
  10465. ExternalDocs: &openapiExternalDocumentationObject{
  10466. Description: "tag1 description",
  10467. },
  10468. }, {
  10469. Name: "tag2",
  10470. ExternalDocs: &openapiExternalDocumentationObject{
  10471. Description: "tag2 description",
  10472. URL: "tag2 url",
  10473. },
  10474. }, {
  10475. Name: "tag3",
  10476. ExternalDocs: &openapiExternalDocumentationObject{
  10477. Description: "ignored tag3 description",
  10478. URL: "tag3 url",
  10479. },
  10480. }, {
  10481. Name: "tag4",
  10482. ExternalDocs: &openapiExternalDocumentationObject{
  10483. Description: "tag4 description",
  10484. },
  10485. }},
  10486. expectedMergedTags: []openapiTagObject{{
  10487. Name: "tag1",
  10488. ExternalDocs: &openapiExternalDocumentationObject{
  10489. Description: "tag1 description",
  10490. },
  10491. }, {
  10492. Name: "tag2",
  10493. ExternalDocs: &openapiExternalDocumentationObject{
  10494. Description: "tag2 description",
  10495. URL: "tag2 url",
  10496. },
  10497. }, {
  10498. Name: "tag3",
  10499. ExternalDocs: &openapiExternalDocumentationObject{
  10500. Description: "tag3 description",
  10501. URL: "tag3 url",
  10502. },
  10503. }, {
  10504. Name: "tag4",
  10505. ExternalDocs: &openapiExternalDocumentationObject{
  10506. Description: "tag4 description",
  10507. URL: "tag4 url",
  10508. },
  10509. }},
  10510. },
  10511. {
  10512. testName: "Merge extensions",
  10513. existingTags: []openapiTagObject{{
  10514. Name: "tag1",
  10515. extensions: []extension{{key: "x-key1", value: MustMarshal("key1 extension")}},
  10516. }, {
  10517. Name: "tag2",
  10518. extensions: []extension{
  10519. {key: "x-key1", value: MustMarshal("key1 extension")},
  10520. {key: "x-key2", value: MustMarshal("key2 extension")},
  10521. },
  10522. }, {
  10523. Name: "tag3",
  10524. extensions: []extension{
  10525. {key: "x-key1", value: MustMarshal("key1 extension")},
  10526. },
  10527. }, {
  10528. Name: "tag4",
  10529. extensions: nil,
  10530. }},
  10531. newTags: []openapiTagObject{{
  10532. Name: "tag1",
  10533. extensions: []extension{{key: "x-key2", value: MustMarshal("key2 extension")}},
  10534. }, {
  10535. Name: "tag2",
  10536. extensions: []extension{
  10537. {key: "x-key1", value: MustMarshal("key1 extension")},
  10538. {key: "x-key2", value: MustMarshal("ignored key2 extension")},
  10539. {key: "x-key3", value: MustMarshal("key3 extension")},
  10540. },
  10541. }, {
  10542. Name: "tag3",
  10543. extensions: nil,
  10544. }, {
  10545. Name: "tag4",
  10546. extensions: []extension{
  10547. {key: "x-key1", value: MustMarshal("key1 extension")},
  10548. },
  10549. }},
  10550. expectedMergedTags: []openapiTagObject{{
  10551. Name: "tag1",
  10552. extensions: []extension{
  10553. {key: "x-key1", value: MustMarshal("key1 extension")},
  10554. {key: "x-key2", value: MustMarshal("key2 extension")},
  10555. },
  10556. }, {
  10557. Name: "tag2",
  10558. extensions: []extension{
  10559. {key: "x-key1", value: MustMarshal("key1 extension")},
  10560. {key: "x-key2", value: MustMarshal("key2 extension")},
  10561. {key: "x-key3", value: MustMarshal("key3 extension")},
  10562. },
  10563. }, {
  10564. Name: "tag3",
  10565. extensions: []extension{
  10566. {key: "x-key1", value: MustMarshal("key1 extension")},
  10567. },
  10568. }, {
  10569. Name: "tag4",
  10570. extensions: []extension{
  10571. {key: "x-key1", value: MustMarshal("key1 extension")},
  10572. },
  10573. }},
  10574. },
  10575. }
  10576. for _, tc := range testCases {
  10577. tc := tc
  10578. t.Run(tc.testName, func(t *testing.T) {
  10579. mergedTags := mergeTags(tc.existingTags, tc.newTags)
  10580. if !reflect.DeepEqual(tc.expectedMergedTags, mergedTags) {
  10581. t.Fatalf("%s: Tags not correctly merged. Want %#v, got %#v", tc.testName, tc.expectedMergedTags, mergedTags)
  10582. }
  10583. })
  10584. }
  10585. }
  10586. func TestApiVisibilityOption(t *testing.T) {
  10587. reg := descriptor.NewRegistry()
  10588. msgdesc := &descriptorpb.DescriptorProto{
  10589. Name: proto.String("ExampleMessage"),
  10590. }
  10591. msg := &descriptor.Message{
  10592. DescriptorProto: msgdesc,
  10593. }
  10594. methodExample := &descriptorpb.MethodDescriptorProto{
  10595. Name: proto.String("Example"),
  10596. InputType: proto.String("ExampleMessage"),
  10597. OutputType: proto.String("ExampleMessage"),
  10598. }
  10599. serviceOptions := &descriptorpb.ServiceOptions{}
  10600. proto.SetExtension(serviceOptions, visibility.E_ApiVisibility, &visibility.VisibilityRule{
  10601. Restriction: "INTERNAL",
  10602. })
  10603. svc := &descriptorpb.ServiceDescriptorProto{
  10604. Name: proto.String("ExampleService"),
  10605. Options: serviceOptions,
  10606. Method: []*descriptorpb.MethodDescriptorProto{methodExample},
  10607. }
  10608. file := descriptor.File{
  10609. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  10610. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  10611. Name: proto.String("example.proto"),
  10612. Package: proto.String("example"),
  10613. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  10614. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  10615. Options: &descriptorpb.FileOptions{
  10616. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  10617. },
  10618. },
  10619. GoPkg: descriptor.GoPackage{
  10620. Path: "example.com/path/to/example/example.pb",
  10621. Name: "example_pb",
  10622. },
  10623. Messages: []*descriptor.Message{msg},
  10624. Services: []*descriptor.Service{
  10625. {
  10626. ServiceDescriptorProto: svc,
  10627. Methods: []*descriptor.Method{
  10628. {
  10629. MethodDescriptorProto: methodExample,
  10630. RequestType: msg,
  10631. ResponseType: msg,
  10632. Bindings: []*descriptor.Binding{
  10633. {
  10634. HTTPMethod: "GET",
  10635. Body: &descriptor.Body{FieldPath: nil},
  10636. PathTmpl: httprule.Template{
  10637. Version: 1,
  10638. OpCodes: []int{0, 0},
  10639. Template: "/v1/example",
  10640. },
  10641. },
  10642. },
  10643. },
  10644. },
  10645. },
  10646. },
  10647. }
  10648. err := reg.Load(&pluginpb.CodeGeneratorRequest{
  10649. ProtoFile: []*descriptorpb.FileDescriptorProto{file.FileDescriptorProto},
  10650. })
  10651. if err != nil {
  10652. t.Errorf("failed to reg.Load(req): %v", err)
  10653. }
  10654. actual, err := applyTemplate(param{File: crossLinkFixture(&file), reg: reg})
  10655. if err != nil {
  10656. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  10657. }
  10658. if len(actual.Definitions) != 0 {
  10659. t.Fatal("Definition should be excluded by api visibility option")
  10660. }
  10661. }
  10662. func TestRenderServicesOptionDeprecated(t *testing.T) {
  10663. testCases := [...]struct {
  10664. testName string
  10665. methodOptions descriptorpb.MethodOptions
  10666. openapiOperation *openapi_options.Operation
  10667. expectedDeprecated bool
  10668. }{
  10669. {
  10670. testName: "method option",
  10671. methodOptions: descriptorpb.MethodOptions{
  10672. Deprecated: proto.Bool(true),
  10673. },
  10674. expectedDeprecated: true,
  10675. },
  10676. {
  10677. testName: "openapi option",
  10678. openapiOperation: &openapi_options.Operation{
  10679. Deprecated: true,
  10680. },
  10681. expectedDeprecated: true,
  10682. },
  10683. {
  10684. testName: "empty openapi doesn't override method option",
  10685. methodOptions: descriptorpb.MethodOptions{
  10686. Deprecated: proto.Bool(true),
  10687. },
  10688. openapiOperation: &openapi_options.Operation{},
  10689. expectedDeprecated: true,
  10690. },
  10691. }
  10692. for _, tc := range testCases {
  10693. tc := tc
  10694. t.Run(tc.testName, func(t *testing.T) {
  10695. msgdesc := &descriptorpb.DescriptorProto{
  10696. Name: proto.String("ExampleMessage"),
  10697. }
  10698. meth := &descriptorpb.MethodDescriptorProto{
  10699. Name: proto.String("Example"),
  10700. InputType: proto.String("ExampleMessage"),
  10701. OutputType: proto.String("ExampleMessage"),
  10702. Options: &tc.methodOptions,
  10703. }
  10704. svc := &descriptorpb.ServiceDescriptorProto{
  10705. Name: proto.String("ExampleService"),
  10706. Method: []*descriptorpb.MethodDescriptorProto{meth},
  10707. }
  10708. msg := &descriptor.Message{
  10709. DescriptorProto: msgdesc,
  10710. }
  10711. file := descriptor.File{
  10712. FileDescriptorProto: &descriptorpb.FileDescriptorProto{
  10713. SourceCodeInfo: &descriptorpb.SourceCodeInfo{},
  10714. Name: proto.String("example.proto"),
  10715. Package: proto.String("example"),
  10716. MessageType: []*descriptorpb.DescriptorProto{msgdesc},
  10717. Service: []*descriptorpb.ServiceDescriptorProto{svc},
  10718. Options: &descriptorpb.FileOptions{
  10719. GoPackage: proto.String("github.com/grpc-ecosystem/grpc-gateway/runtime/internal/examplepb;example"),
  10720. },
  10721. },
  10722. GoPkg: descriptor.GoPackage{
  10723. Path: "example.com/path/to/example/example.pb",
  10724. Name: "example_pb",
  10725. },
  10726. Messages: []*descriptor.Message{msg},
  10727. Services: []*descriptor.Service{
  10728. {
  10729. ServiceDescriptorProto: svc,
  10730. Methods: []*descriptor.Method{
  10731. {
  10732. MethodDescriptorProto: meth,
  10733. RequestType: msg,
  10734. ResponseType: msg,
  10735. Bindings: []*descriptor.Binding{
  10736. {
  10737. HTTPMethod: "GET",
  10738. PathTmpl: httprule.Template{
  10739. Version: 1,
  10740. OpCodes: []int{0, 0},
  10741. Template: "/v1/echo",
  10742. },
  10743. },
  10744. },
  10745. },
  10746. },
  10747. },
  10748. },
  10749. }
  10750. if tc.openapiOperation != nil {
  10751. proto.SetExtension(
  10752. proto.Message(file.Services[0].Methods[0].Options),
  10753. openapi_options.E_Openapiv2Operation,
  10754. tc.openapiOperation,
  10755. )
  10756. }
  10757. reg := descriptor.NewRegistry()
  10758. reg.SetEnableRpcDeprecation(true)
  10759. fileCL := crossLinkFixture(&file)
  10760. if err := reg.Load(reqFromFile(fileCL)); err != nil {
  10761. t.Errorf("reg.Load(%#v) failed with %v; want success", file, err)
  10762. }
  10763. result, err := applyTemplate(param{File: fileCL, reg: reg})
  10764. if err != nil {
  10765. t.Fatalf("applyTemplate(%#v) failed with %v; want success", file, err)
  10766. }
  10767. got := result.getPathItemObject("/v1/echo").Get.Deprecated
  10768. if got != tc.expectedDeprecated {
  10769. t.Fatalf("Wrong deprecated field, got %v want %v", got, tc.expectedDeprecated)
  10770. }
  10771. })
  10772. }
  10773. }