1 В избранное 0 Ответвления 0

OSCHINA-MIRROR/hanchuanchuan-goInception

Присоединиться к Gitlife
Откройте для себя и примите участие в публичных проектах с открытым исходным кодом с участием более 10 миллионов разработчиков. Приватные репозитории также полностью бесплатны :)
Присоединиться бесплатно
Это зеркальный репозиторий, синхронизируется ежедневно с исходного репозитория.
Клонировать/Скачать
parser.y 160 КБ
Копировать Редактировать Исходные данные Просмотреть построчно История
hanchuanchuan Отправлено 5 лет назад eb769b6
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487748874897490749174927493749474957496749774987499750075017502750375047505750675077508750975107511751275137514751575167517751875197520752175227523752475257526752775287529753075317532753375347535753675377538753975407541754275437544754575467547754875497550755175527553755475557556755775587559756075617562756375647565756675677568756975707571757275737574757575767577757875797580758175827583758475857586758775887589759075917592759375947595759675977598759976007601760276037604760576067607760876097610761176127613761476157616761776187619762076217622762376247625762676277628762976307631763276337634763576367637763876397640764176427643764476457646764776487649765076517652765376547655765676577658765976607661766276637664766576667667766876697670767176727673767476757676767776787679768076817682768376847685768676877688768976907691769276937694769576967697769876997700770177027703770477057706770777087709771077117712771377147715771677177718771977207721772277237724772577267727772877297730773177327733773477357736773777387739774077417742774377447745774677477748774977507751775277537754775577567757775877597760776177627763776477657766776777687769777077717772777377747775777677777778777977807781778277837784778577867787778877897790779177927793779477957796779777987799780078017802780378047805780678077808780978107811781278137814781578167817781878197820782178227823782478257826782778287829783078317832783378347835783678377838783978407841784278437844784578467847784878497850785178527853785478557856785778587859786078617862786378647865786678677868
%{
// Copyright 2013 The ql Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSES/QL-LICENSE file.
// Copyright 2015 PingCAP, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// See the License for the specific language governing permissions and
// limitations under the License.
// Initial yacc source generated by ebnf2y[1]
// at 2013-10-04 23:10:47.861401015 +0200 CEST
//
// $ ebnf2y -o ql.y -oe ql.ebnf -start StatementList -pkg ql -p _
//
// [1]: http://github.com/cznic/ebnf2y
package parser
import (
"fmt"
"strings"
"github.com/hanchuanchuan/goInception/mysql"
"github.com/hanchuanchuan/goInception/ast"
"github.com/hanchuanchuan/goInception/model"
"github.com/hanchuanchuan/goInception/parser/opcode"
"github.com/hanchuanchuan/goInception/util/auth"
"github.com/hanchuanchuan/goInception/util/charset"
"github.com/hanchuanchuan/goInception/types"
)
%}
%union {
offset int // offset
item interface{}
ident string
expr ast.ExprNode
statement ast.StmtNode
}
%token <ident>
/*yy:token "%c" */ identifier "identifier"
/*yy:token "_%c" */ underscoreCS "UNDERSCORE_CHARSET"
/*yy:token "\"%c\"" */ stringLit "string literal"
singleAtIdentifier "identifier with single leading at"
doubleAtIdentifier "identifier with double leading at"
invalid "a special token never used by parser, used by lexer to indicate error"
hintBegin "hintBegin is a virtual token for optimizer hint grammar"
hintEnd "hintEnd is a virtual token for optimizer hint grammar"
andand "&&"
pipes "||"
/* The following tokens belong to ODBCDateTimeType. */
odbcDateType "d"
odbcTimeType "t"
odbcTimestampType "ts"
/* The following tokens belong to ReservedKeyword. */
add "ADD"
all "ALL"
alter "ALTER"
analyze "ANALYZE"
and "AND"
as "AS"
asc "ASC"
between "BETWEEN"
bigIntType "BIGINT"
binaryType "BINARY"
blobType "BLOB"
both "BOTH"
by "BY"
cascade "CASCADE"
caseKwd "CASE"
change "CHANGE"
character "CHARACTER"
charType "CHAR"
check "CHECK"
collate "COLLATE"
column "COLUMN"
constraint "CONSTRAINT"
convert "CONVERT"
create "CREATE"
cross "CROSS"
currentDate "CURRENT_DATE"
currentTime "CURRENT_TIME"
currentTs "CURRENT_TIMESTAMP"
currentUser "CURRENT_USER"
database "DATABASE"
databases "DATABASES"
dayHour "DAY_HOUR"
dayMicrosecond "DAY_MICROSECOND"
dayMinute "DAY_MINUTE"
daySecond "DAY_SECOND"
decimalType "DECIMAL"
defaultKwd "DEFAULT"
delayed "DELAYED"
deleteKwd "DELETE"
desc "DESC"
describe "DESCRIBE"
distinct "DISTINCT"
distinctRow "DISTINCTROW"
div "DIV"
doubleType "DOUBLE"
drop "DROP"
dual "DUAL"
elseKwd "ELSE"
enclosed "ENCLOSED"
escaped "ESCAPED"
exists "EXISTS"
explain "EXPLAIN"
falseKwd "FALSE"
floatType "FLOAT"
forKwd "FOR"
force "FORCE"
foreign "FOREIGN"
from "FROM"
fulltext "FULLTEXT"
generated "GENERATED"
geometryType "GEOMETRY"
grant "GRANT"
group "GROUP"
having "HAVING"
highPriority "HIGH_PRIORITY"
hourMicrosecond "HOUR_MICROSECOND"
hourMinute "HOUR_MINUTE"
hourSecond "HOUR_SECOND"
ifKwd "IF"
ignore "IGNORE"
in "IN"
index "INDEX"
infile "INFILE"
inner "INNER"
integerType "INTEGER"
interval "INTERVAL"
into "INTO"
is "IS"
insert "INSERT"
intType "INT"
int1Type "INT1"
int2Type "INT2"
int3Type "INT3"
int4Type "INT4"
int8Type "INT8"
join "JOIN"
key "KEY"
keys "KEYS"
kill "KILL"
leading "LEADING"
left "LEFT"
like "LIKE"
limit "LIMIT"
lines "LINES"
linear "LINEAR"
load "LOAD"
localTime "LOCALTIME"
localTs "LOCALTIMESTAMP"
lock "LOCK"
longblobType "LONGBLOB"
longtextType "LONGTEXT"
lowPriority "LOW_PRIORITY"
maxValue "MAXVALUE"
mediumblobType "MEDIUMBLOB"
mediumIntType "MEDIUMINT"
mediumtextType "MEDIUMTEXT"
minuteMicrosecond "MINUTE_MICROSECOND"
minuteSecond "MINUTE_SECOND"
mod "MOD"
not "NOT"
noWriteToBinLog "NO_WRITE_TO_BINLOG"
null "NULL"
numericType "NUMERIC"
nvarcharType "NVARCHAR"
on "ON"
option "OPTION"
or "OR"
order "ORDER"
outer "OUTER"
packKeys "PACK_KEYS"
partition "PARTITION"
precisionType "PRECISION"
primary "PRIMARY"
procedure "PROCEDURE"
shardRowIDBits "SHARD_ROW_ID_BITS"
rangeKwd "RANGE"
read "READ"
realType "REAL"
references "REFERENCES"
regexpKwd "REGEXP"
rename "RENAME"
repeat "REPEAT"
replace "REPLACE"
restrict "RESTRICT"
revoke "REVOKE"
right "RIGHT"
rlike "RLIKE"
secondMicrosecond "SECOND_MICROSECOND"
selectKwd "SELECT"
set "SET"
show "SHOW"
get "GET"
smallIntType "SMALLINT"
spatial "SPATIAL"
sql "SQL"
sqlCalcFoundRows "SQL_CALC_FOUND_ROWS"
starting "STARTING"
straightJoin "STRAIGHT_JOIN"
tableKwd "TABLE"
stored "STORED"
terminated "TERMINATED"
then "THEN"
tinyblobType "TINYBLOB"
tinyIntType "TINYINT"
tinytextType "TINYTEXT"
to "TO"
trailing "TRAILING"
trigger "TRIGGER"
trueKwd "TRUE"
unique "UNIQUE"
union "UNION"
unlock "UNLOCK"
unsigned "UNSIGNED"
update "UPDATE"
usage "USAGE"
use "USE"
using "USING"
utcDate "UTC_DATE"
utcTimestamp "UTC_TIMESTAMP"
utcTime "UTC_TIME"
values "VALUES"
long "LONG"
varcharType "VARCHAR"
varbinaryType "VARBINARY"
virtual "VIRTUAL"
when "WHEN"
where "WHERE"
write "WRITE"
with "WITH"
xor "XOR"
yearMonth "YEAR_MONTH"
zerofill "ZEROFILL"
natural "NATURAL"
/* The following tokens belong to UnReservedKeyword. */
action "ACTION"
after "AFTER"
always "ALWAYS"
algorithm "ALGORITHM"
any "ANY"
ascii "ASCII"
autoIncrement "AUTO_INCREMENT"
avgRowLength "AVG_ROW_LENGTH"
avg "AVG"
begin "BEGIN"
binlog "BINLOG"
bitType "BIT"
booleanType "BOOLEAN"
boolType "BOOL"
btree "BTREE"
byteType "BYTE"
cascaded "CASCADED"
charsetKwd "CHARSET"
checksum "CHECKSUM"
cleanup "CLEANUP"
client "CLIENT"
coalesce "COALESCE"
collation "COLLATION"
columns "COLUMNS"
comment "COMMENT"
commit "COMMIT"
inception "INCEPTION"
inception_magic_start "INCEPTION_MAGIC_START"
inception_magic_commit "INCEPTION_MAGIC_COMMIT"
osc "OSC"
osc_percent "OSC_PERCENT"
stop "STOP"
pause "PAUSE"
resume "RESUME"
committed "COMMITTED"
compact "COMPACT"
compressed "COMPRESSED"
compression "COMPRESSION"
connection "CONNECTION"
consistent "CONSISTENT"
current "CURRENT"
day "DAY"
data "DATA"
dateType "DATE"
datetimeType "DATETIME"
deallocate "DEALLOCATE"
definer "DEFINER"
delayKeyWrite "DELAY_KEY_WRITE"
directory "DIRECTORY"
disable "DISABLE"
do "DO"
duplicate "DUPLICATE"
dynamic "DYNAMIC"
enable "ENABLE"
end "END"
engine "ENGINE"
engines "ENGINES"
enum "ENUM"
event "EVENT"
events "EVENTS"
escape "ESCAPE"
exclusive "EXCLUSIVE"
execute "EXECUTE"
fields "FIELDS"
first "FIRST"
fixed "FIXED"
flush "FLUSH"
format "FORMAT"
full "FULL"
function "FUNCTION"
grants "GRANTS"
hash "HASH"
history "HISTORY"
hour "HOUR"
identified "IDENTIFIED"
isolation "ISOLATION"
indexes "INDEXES"
invisible "INVISIBLE"
invoker "INVOKER"
jsonType "JSON"
keyBlockSize "KEY_BLOCK_SIZE"
local "LOCAL"
less "LESS"
level "LEVEL"
list "LIST"
master "MASTER"
microsecond "MICROSECOND"
minute "MINUTE"
mode "MODE"
modify "MODIFY"
month "MONTH"
maxRows "MAX_ROWS"
maxConnectionsPerHour "MAX_CONNECTIONS_PER_HOUR"
maxQueriesPerHour "MAX_QUERIES_PER_HOUR"
maxUpdatesPerHour "MAX_UPDATES_PER_HOUR"
maxUserConnections "MAX_USER_CONNECTIONS"
merge "MERGE"
minRows "MIN_ROWS"
names "NAMES"
national "NATIONAL"
no "NO"
nodegroup "NODEGROUP"
none "NONE"
offset "OFFSET"
only "ONLY"
password "PASSWORD"
partitions "PARTITIONS"
pipesAsOr
plugins "PLUGINS"
prepare "PREPARE"
privileges "PRIVILEGES"
process "PROCESS"
processlist "PROCESSLIST"
profiles "PROFILES"
quarter "QUARTER"
query "QUERY"
queries "QUERIES"
quick "QUICK"
recover "RECOVER"
redundant "REDUNDANT"
reload "RELOAD"
repeatable "REPEATABLE"
replication "REPLICATION"
reverse "REVERSE"
rollback "ROLLBACK"
routine "ROUTINE"
row "ROW"
rowCount "ROW_COUNT"
rowFormat "ROW_FORMAT"
rtree "RTREE"
second "SECOND"
security "SECURITY"
separator "SEPARATOR"
serializable "SERIALIZABLE"
session "SESSION"
share "SHARE"
shared "SHARED"
signed "SIGNED"
slave "SLAVE"
slow "SLOW"
snapshot "SNAPSHOT"
sqlCache "SQL_CACHE"
sqlNoCache "SQL_NO_CACHE"
start "START"
statsPersistent "STATS_PERSISTENT"
status "STATUS"
systemTime "SYSTEM_TIME"
subpartition "SUBPARTITION"
subpartitions "SUBPARTITIONS"
super "SUPER"
some "SOME"
global "GLOBAL"
tables "TABLES"
tablespace "TABLESPACE"
temporary "TEMPORARY"
temptable "TEMPTABLE"
textType "TEXT"
than "THAN"
timeType "TIME"
timestampType "TIMESTAMP"
trace "TRACE"
transaction "TRANSACTION"
triggers "TRIGGERS"
truncate "TRUNCATE"
tp "TYPE"
uncommitted "UNCOMMITTED"
unknown "UNKNOWN"
user "USER"
undefined "UNDEFINED"
value "VALUE"
variables "VARIABLES"
levels "LEVELS"
view "VIEW"
visible "VISIBLE"
warnings "WARNINGS"
identSQLErrors "ERRORS"
week "WEEK"
yearType "YEAR"
/* The following tokens belong to NotKeywordToken. */
addDate "ADDDATE"
bitAnd "BIT_AND"
bitOr "BIT_OR"
bitXor "BIT_XOR"
cast "CAST"
copyKwd "COPY"
count "COUNT"
curTime "CURTIME"
dateAdd "DATE_ADD"
dateSub "DATE_SUB"
extract "EXTRACT"
getFormat "GET_FORMAT"
groupConcat "GROUP_CONCAT"
inplace "INPLACE"
instant "INSTANT"
internal "INTERNAL"
min "MIN"
max "MAX"
maxExecutionTime "MAX_EXECUTION_TIME"
now "NOW"
position "POSITION"
recent "RECENT"
subDate "SUBDATE"
sum "SUM"
substring "SUBSTRING"
timestampAdd "TIMESTAMPADD"
timestampDiff "TIMESTAMPDIFF"
top "TOP"
trim "TRIM"
/* The following tokens belong to TiDBKeyword. */
admin "ADMIN"
buckets "BUCKETS"
cancel "CANCEL"
ddl "DDL"
jobs "JOBS"
job "JOB"
stats "STATS"
statsMeta "STATS_META"
statsHistograms "STATS_HISTOGRAMS"
statsBuckets "STATS_BUCKETS"
statsHealthy "STATS_HEALTHY"
tidb "TIDB"
tidbHJ "TIDB_HJ"
tidbSMJ "TIDB_SMJ"
tidbINLJ "TIDB_INLJ"
builtinAddDate
builtinBitAnd
builtinBitOr
builtinBitXor
builtinCast
builtinCount
builtinCurDate
builtinCurTime
builtinDateAdd
builtinDateSub
builtinExtract
builtinGroupConcat
builtinMax
builtinMin
builtinNow
builtinPosition
builtinStddevPop
builtinSubDate
builtinSubstring
builtinSum
builtinSysDate
builtinTrim
builtinUser
builtinVarPop
builtinVarSamp
%token <item>
/*yy:token "1.%d" */ floatLit "floating-point literal"
/*yy:token "1.%d" */ decLit "decimal literal"
/*yy:token "%d" */ intLit "integer literal"
/*yy:token "%x" */ hexLit "hexadecimal literal"
/*yy:token "%b" */ bitLit "bit literal"
andnot "&^"
assignmentEq ":="
eq "="
ge ">="
le "<="
jss "->"
juss "->>"
lsh "<<"
neq "!="
neqSynonym "<>"
nulleq "<=>"
paramMarker "?"
rsh ">>"
%token not2
%type <expr>
Expression "expression"
MaxValueOrExpression "maxvalue or expression"
BoolPri "boolean primary expression"
ExprOrDefault "expression or default"
PredicateExpr "Predicate expression factor"
SetExpr "Set variable statement value's expression"
BitExpr "bit expression"
SimpleExpr "simple expression"
SimpleIdent "Simple Identifier expression"
SumExpr "aggregate functions"
FunctionCallGeneric "Function call with Identifier"
FunctionCallKeyword "Function call with keyword as function name"
FunctionCallNonKeyword "Function call with nonkeyword as function name"
Literal "literal value"
Variable "User or system variable"
SystemVariable "System defined variable name"
UserVariable "User defined variable name"
SubSelect "Sub Select"
StringLiteral "text literal"
ExpressionOpt "Optional expression"
SignedLiteral "Literal or NumLiteral with sign"
DefaultValueExpr "DefaultValueExpr(Now or Signed Literal)"
NowSymOptionFraction "NowSym with optional fraction part"
%type <statement>
AdminStmt "Check table statement or show ddl statement"
AlterTableStmt "Alter table statement"
AlterUserStmt "Alter user statement"
AnalyzeTableStmt "Analyze table statement"
BeginTransactionStmt "BEGIN TRANSACTION statement"
BinlogStmt "Binlog base64 statement"
CommitStmt "COMMIT statement"
InceptionStmt "INCEPTION statement"
InceptionStartStmt "INCEPTION_MAGIC_START statement"
InceptionCommitStmt "INCEPTION_MAGIC_COMMIT statement"
CreateTableStmt "CREATE TABLE statement"
CreateViewStmt "CREATE VIEW stetement"
CreateUserStmt "CREATE User statement"
CreateDatabaseStmt "Create Database Statement"
CreateIndexStmt "CREATE INDEX statement"
DoStmt "Do statement"
DropDatabaseStmt "DROP DATABASE statement"
DropIndexStmt "DROP INDEX statement"
DropStatsStmt "DROP STATS statement"
DropTableStmt "DROP TABLE statement"
DropUserStmt "DROP USER"
DropViewStmt "DROP VIEW statement"
DeallocateStmt "Deallocate prepared statement"
DeleteFromStmt "DELETE FROM statement"
EmptyStmt "empty statement"
ExecuteStmt "Execute statement"
ExplainStmt "EXPLAIN statement"
ExplainableStmt "explainable statement"
FlushStmt "Flush statement"
GrantStmt "Grant statement"
InsertIntoStmt "INSERT INTO statement"
KillStmt "Kill statement"
LoadDataStmt "Load data statement"
LoadStatsStmt "Load statistic statement"
LockTablesStmt "Lock tables statement"
PreparedStmt "PreparedStmt"
SelectStmt "SELECT statement"
RenameTableStmt "rename table statement"
ReplaceIntoStmt "REPLACE INTO statement"
RevokeStmt "Revoke statement"
RollbackStmt "ROLLBACK statement"
SetStmt "Set variable statement"
ShowStmt "Show engines/databases/tables/columns/warnings/status statement"
Statement "statement"
TraceStmt "TRACE statement"
TraceableStmt "traceable statment"
TruncateTableStmt "TRUNCATE TABLE statement"
UnlockTablesStmt "Unlock tables statement"
UpdateStmt "UPDATE statement"
UnionStmt "Union select state ment"
UseStmt "USE statement"
%type <item>
AdminShowSlow "Admin Show Slow statement"
AlgorithmClause "Alter table algorithm"
AlterTableOptionListOpt "alter table option list opt"
AlterTableSpec "Alter table specification"
AlterTableSpecList "Alter table specification list"
AnyOrAll "Any or All for subquery"
Assignment "assignment"
AssignmentList "assignment list"
AssignmentListOpt "assignment list opt"
AuthOption "User auth option"
AuthString "Password string value"
OptionalBraces "optional braces"
CastType "Cast function target type"
CharsetName "Character set name"
CollationName "Collation name"
ColumnDef "table column definition"
ColumnDefList "table column definition list"
ColumnName "column name"
ColumnNameList "column name list"
ColumnList "column list"
ColumnNameListOpt "column name list opt"
ColumnNameListOptWithBrackets "column name list opt with brackets"
ColumnSetValue "insert statement set value by column name"
ColumnSetValueList "insert statement set value by column name list"
CompareOp "Compare opcode"
ColumnOption "column definition option"
ColumnOptionList "column definition option list"
VirtualOrStored "indicate generated column is stored or not"
ColumnOptionListOpt "optional column definition option list"
Constraint "table constraint"
ConstraintElem "table constraint element"
ConstraintKeywordOpt "Constraint Keyword or empty"
CreateTableOptionListOpt "create table option list opt"
CreateTableSelectOpt "Select/Union statement in CREATE TABLE ... SELECT"
DatabaseOption "CREATE Database specification"
DatabaseOptionList "CREATE Database specification list"
DatabaseOptionListOpt "CREATE Database specification list opt"
DBName "Database Name"
DistinctOpt "Explicit distinct option"
DefaultFalseDistinctOpt "Distinct option which defaults to false"
DefaultTrueDistinctOpt "Distinct option which defaults to true"
BuggyDefaultFalseDistinctOpt "Distinct option which accepts DISTINCT ALL and defaults to false"
Enclosed "Enclosed by"
EqOpt "= or empty"
EscapedTableRef "escaped table reference"
Escaped "Escaped by"
ExpressionList "expression list"
MaxValueOrExpressionList "maxvalue or expression list"
ExpressionListOpt "expression list opt"
FuncDatetimePrecListOpt "Function datetime precision list opt"
FuncDatetimePrecList "Function datetime precision list"
Field "field expression"
Fields "Fields clause"
FieldsTerminated "Fields terminated by"
FieldAsName "Field alias name"
FieldAsNameOpt "Field alias name opt"
FieldList "field expression list"
FlushOption "Flush option"
TableRefsClause "Table references clause"
FuncDatetimePrec "Function datetime precision"
GlobalScope "The scope of variable"
GroupByClause "GROUP BY clause"
HashString "Hashed string"
HavingClause "HAVING clause"
HandleRange "handle range"
HandleRangeList "handle range list"
IfExists "If Exists"
IfNotExists "If Not Exists"
IgnoreOptional "IGNORE or empty"
IndexColName "Index column name"
IndexColNameList "List of index column name"
IndexHint "index hint"
IndexHintList "index hint list"
IndexHintListOpt "index hint list opt"
IndexHintScope "index hint scope"
IndexHintType "index hint type"
IndexInvisible "index visible/invisible"
IndexKeyTypeOpt "index key type"
IndexLockAndAlgorithmOpt "index lock and algorithm"
IndexName "index name"
IndexNameAndTypeOpt "index name and index type"
IndexNameList "index name list"
IndexOption "Index Option"
IndexOptionList "Index Option List or empty"
IndexType "index type"
IndexTypeName "index type name"
IndexTypeOpt "optional index type"
InsertValues "Rest part of INSERT/REPLACE INTO statement"
JoinTable "join table"
JoinType "join type"
KillOrKillTiDB "Kill or Kill TiDB"
LikeEscapeOpt "like escape option"
LikeTableWithOrWithoutParen "LIKE table_name or ( LIKE table_name )"
LimitClause "LIMIT clause"
LimitOption "Limit option could be integer or parameter marker."
Lines "Lines clause"
LinesTerminated "Lines terminated by"
LocalOpt "Local opt"
LockClause "Alter table lock clause"
MaxNumBuckets "Max number of buckets"
NumLiteral "Num/Int/Float/Decimal Literal"
NoWriteToBinLogAliasOpt "NO_WRITE_TO_BINLOG alias LOCAL or empty"
ObjectType "Grant statement object type"
OnDuplicateKeyUpdate "ON DUPLICATE KEY UPDATE value list"
DuplicateOpt "[IGNORE|REPLACE] in CREATE TABLE ... SELECT statement"
OptFull "Full or empty"
Order "ORDER BY clause optional collation specification"
OrderBy "ORDER BY clause"
OrReplace "or replace"
ByItem "BY item"
OrderByOptional "Optional ORDER BY clause optional"
ByList "BY list"
QuickOptional "QUICK or empty"
PartitionDefinition "Partition definition"
PartitionDefinitionList "Partition definition list"
PartitionDefinitionListOpt "Partition definition list option"
PartitionKeyAlgorithmOpt "ALGORITHM = n option for KEY partition"
PartitionMethod "Partition method"
PartitionOpt "Partition option"
PartitionNameList "Partition name list"
PartitionNameListOpt "table partition names list optional"
PartitionNumOpt "PARTITION NUM option"
PartDefValuesOpt "VALUES {LESS THAN {(expr | value_list) | MAXVALUE} | IN {value_list}"
PartDefOptionList "PartDefOption list"
PartDefOption "COMMENT [=] xxx | TABLESPACE [=] tablespace_name | ENGINE [=] xxx"
PasswordOpt "Password option"
ColumnPosition "Column position [First|After ColumnName]"
PrepareSQL "Prepare statement sql string"
PriorityOpt "Statement priority option"
PrivElem "Privilege element"
PrivElemList "Privilege element list"
PrivLevel "Privilege scope"
PrivType "Privilege type"
ReferDef "Reference definition"
OnDeleteOpt "optional ON DELETE clause"
OnUpdateOpt "optional ON UPDATE clause"
OptGConcatSeparator "optional GROUP_CONCAT SEPARATOR"
ReferOpt "reference option"
RowFormat "Row format option"
RowValue "Row value"
SelectLockOpt "FOR UPDATE or LOCK IN SHARE MODE,"
SelectStmtCalcFoundRows "SELECT statement optional SQL_CALC_FOUND_ROWS"
SelectStmtSQLCache "SELECT statement optional SQL_CAHCE/SQL_NO_CACHE"
SelectStmtStraightJoin "SELECT statement optional STRAIGHT_JOIN"
SelectStmtFieldList "SELECT statement field list"
SelectStmtLimit "SELECT statement optional LIMIT clause"
SelectStmtOpts "Select statement options"
SelectStmtBasic "SELECT statement from constant value"
SelectStmtFromDual "SELECT statement from dual"
SelectStmtFromTable "SELECT statement from table"
SelectStmtGroup "SELECT statement optional GROUP BY clause"
ShowTargetFilterable "Show target that can be filtered by WHERE or LIKE"
ShowDatabaseNameOpt "Show tables/columns statement database name option"
ShowTableAliasOpt "Show table alias option"
ShowLikeOrWhereOpt "Show like or where clause option"
Starting "Starting by"
StatementList "statement list"
StatsPersistentVal "stats_persistent value"
StringName "string literal or identifier"
StringList "string list"
SubPartDefinition "SubPartition definition"
SubPartDefinitionList "SubPartition definition list"
SubPartDefinitionListOpt "SubPartition definition list optional"
SubPartitionMethod "SubPartition method"
SubPartitionOpt "SubPartition option"
SubPartitionNumOpt "SubPartition NUM option"
Symbol "Constraint Symbol"
TableAsName "table alias name"
TableAsNameOpt "table alias name optional"
TableElement "table definition element"
TableElementList "table definition element list"
TableElementListOpt "table definition element list optional"
TableFactor "table factor"
TableLock "Table name and lock type"
TableLockList "Table lock list"
TableName "Table name"
TableNameList "Table name list"
TableNameListOpt "Table name list opt"
TableOption "create table option"
TableOptionList "create table option list"
TableRef "table reference"
TableRefs "table references"
TableToTable "rename table to table"
TableToTableList "rename table to table by list"
TransactionChar "Transaction characteristic"
TransactionChars "Transaction characteristic list"
TrimDirection "Trim string direction"
UnionOpt "Union Option(empty/ALL/DISTINCT)"
UnionClauseList "Union select clause list"
UnionSelect "Union (select) item"
Username "Username"
UsernameList "UsernameList"
UserSpec "Username and auth option"
UserSpecList "Username and auth option list"
UserVariableList "User defined variable name list"
Values "values"
ValuesList "values list"
ValuesOpt "values optional"
VariableAssignment "set variable value"
VariableAssignmentList "set variable value list"
ViewAlgorithm "view algorithm"
ViewCheckOption "view check option"
ViewDefiner "view definer"
ViewName "view name"
ViewFieldList "create view statement field list"
ViewSQLSecurity "view sql security"
WhereClause "WHERE clause"
WhereClauseOptional "Optional WHERE clause"
WhenClause "When clause"
WhenClauseList "When clause list"
WithReadLockOpt "With Read Lock opt"
WithGrantOptionOpt "With Grant Option opt"
ElseOpt "Optional else clause"
Type "Types"
BetweenOrNotOp "Between predicate"
IsOrNotOp "Is predicate"
InOrNotOp "In predicate"
LikeOrNotOp "Like predicate"
RegexpOrNotOp "Regexp predicate"
NumericType "Numeric types"
IntegerType "Integer Types types"
BooleanType "Boolean Types types"
FixedPointType "Exact value types"
FloatingPointType "Approximate value types"
BitValueType "bit value types"
StringType "String types"
BlobType "Blob types"
TextType "Text types"
DateAndTimeType "Date and Time types"
OptFieldLen "Field length or empty"
FieldLen "Field length"
FieldOpts "Field type definition option list"
FieldOpt "Field type definition option"
FloatOpt "Floating-point type option"
Precision "Floating-point precision option"
OptBinary "Optional BINARY"
OptBinMod "Optional BINARY mode"
OptCharset "Optional Character setting"
OptCollate "Optional Collate setting"
IgnoreLines "Ignore num(int) lines"
NUM "A number"
NumList "Some numbers"
LengthNum "Field length num(uint64)"
HintTableList "Table list in optimizer hint"
TableOptimizerHintOpt "Table level optimizer hint"
TableOptimizerHints "Table level optimizer hints"
TableOptimizerHintList "Table level optimizer hint list"
%type <ident>
AsOpt "AS or EmptyString"
KeyOrIndex "{KEY|INDEX}"
ColumnKeywordOpt "Column keyword or empty"
PrimaryOpt "Optional primary keyword"
NowSym "CURRENT_TIMESTAMP/LOCALTIME/LOCALTIMESTAMP"
NowSymFunc "CURRENT_TIMESTAMP/LOCALTIME/LOCALTIMESTAMP/NOW"
DefaultKwdOpt "optional DEFAULT keyword"
DatabaseSym "DATABASE or SCHEMA"
ExplainSym "EXPLAIN or DESCRIBE or DESC"
RegexpSym "REGEXP or RLIKE"
IntoOpt "INTO or EmptyString"
ValueSym "Value or Values"
Varchar "{NATIONAL VARCHAR|VARCHAR|NVARCHAR}"
TimeUnit "Time unit for 'DATE_ADD', 'DATE_SUB', 'ADDDATE', 'SUBDATE', 'EXTRACT'"
TimestampUnit "Time unit for 'TIMESTAMPADD' and 'TIMESTAMPDIFF'"
DeallocateSym "Deallocate or drop"
OuterOpt "optional OUTER clause"
CrossOpt "Cross join option"
TablesTerminalSym "{TABLE|TABLES}"
IsolationLevel "Isolation level"
ShowIndexKwd "Show index/indexs/key keyword"
DistinctKwd "DISTINCT/DISTINCTROW keyword"
FromOrIn "From or In"
OptTable "Optional table keyword"
OptInteger "Optional Integer keyword"
NationalOpt "National option"
CharsetKw "charset or charater set"
CommaOpt "optional comma"
LockType "Table locks type"
logAnd "logical and operator"
logOr "logical or operator"
LinearOpt "linear or empty"
FieldsOrColumns "Fields or columns"
GetFormatSelector "{DATE|DATETIME|TIME|TIMESTAMP}"
%type <ident>
ODBCDateTimeType "ODBC type keywords for date and time literals"
Identifier "identifier or unreserved keyword"
NotKeywordToken "Tokens not mysql keyword but treated specially"
UnReservedKeyword "MySQL unreserved keywords"
TiDBKeyword "TiDB added keywords"
FunctionNameConflict "Built-in function call names which are conflict with keywords"
FunctionNameOptionalBraces "Function with optional braces, all of them are reserved keywords."
FunctionNameDatetimePrecision "Function with optional datetime precision, all of them are reserved keywords."
FunctionNameDateArith "Date arith function call names (date_add or date_sub)"
FunctionNameDateArithMultiForms "Date arith function call names (adddate or subdate)"
%precedence empty
%precedence sqlCache sqlNoCache
%precedence lowerThanIntervalKeyword
%precedence interval
%precedence lowerThanStringLitToken
%precedence stringLit
%precedence lowerThanSetKeyword
%precedence set
%precedence lowerThanInsertValues
%precedence insertValues
%precedence lowerThanCreateTableSelect
%precedence createTableSelect
%precedence lowerThanKey
%precedence key
%left join straightJoin inner cross left right full natural
/* A dummy token to force the priority of TableRef production in a join. */
%left tableRefPriority
%precedence lowerThanOn
%precedence on using
%right assignmentEq
%left pipes or pipesAsOr
%left xor
%left andand and
%left between
%precedence lowerThanEq
%left eq ge le neq neqSynonym '>' '<' is like in
%left '|'
%left '&'
%left rsh lsh
%left '-' '+'
%left '*' '/' '%' div mod
%left '^'
%left '~' neg
%right not not2
%right collate
%precedence '('
%precedence quick
%precedence escape
%precedence lowerThanComma
%precedence ','
%precedence higherThanComma
%start Start
%%
Start:
StatementList
/**************************************AlterTableStmt***************************************
* See https://dev.mysql.com/doc/refman/5.7/en/alter-table.html
*******************************************************************************************/
AlterTableStmt:
"ALTER" IgnoreOptional "TABLE" TableName AlterTableSpecList
{
$$ = &ast.AlterTableStmt{
Table: $4.(*ast.TableName),
Specs: $5.([]*ast.AlterTableSpec),
}
}
| "ALTER" IgnoreOptional "TABLE" TableName "ANALYZE" "PARTITION" PartitionNameList MaxNumBuckets
{
$$ = &ast.AnalyzeTableStmt{TableNames: []*ast.TableName{$4.(*ast.TableName)}, PartitionNames: $7.([]model.CIStr), MaxNumBuckets: $8.(uint64),}
}
| "ALTER" IgnoreOptional "TABLE" TableName "ANALYZE" "PARTITION" PartitionNameList "INDEX" IndexNameList MaxNumBuckets
{
$$ = &ast.AnalyzeTableStmt{
TableNames: []*ast.TableName{$4.(*ast.TableName)},
PartitionNames: $7.([]model.CIStr),
IndexNames: $9.([]model.CIStr),
IndexFlag: true,
MaxNumBuckets: $10.(uint64),
}
}
AlterTableSpec:
AlterTableOptionListOpt
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableOption,
Options:$1.([]*ast.TableOption),
}
}
| "CONVERT" "TO" CharsetKw CharsetName OptCollate
{
op := &ast.AlterTableSpec{
Tp: ast.AlterTableOption,
Options:[]*ast.TableOption{{Tp: ast.TableOptionCharset, StrValue: $4.(string)}},
}
if $5 != "" {
op.Options = append(op.Options, &ast.TableOption{Tp: ast.TableOptionCollate, StrValue: $5.(string)})
}
$$ = op
}
| "ADD" ColumnKeywordOpt ColumnDef ColumnPosition
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableAddColumns,
NewColumns: []*ast.ColumnDef{$3.(*ast.ColumnDef)},
Position: $4.(*ast.ColumnPosition),
}
}
| "ADD" ColumnKeywordOpt '(' ColumnDefList ')'
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableAddColumns,
NewColumns: $4.([]*ast.ColumnDef),
}
}
| "ADD" Constraint
{
constraint := $2.(*ast.Constraint)
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableAddConstraint,
Constraint: constraint,
}
}
| "ADD" "PARTITION" PartitionDefinitionListOpt
{
var defs []*ast.PartitionDefinition
if $3 != nil {
defs = $3.([]*ast.PartitionDefinition)
}
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableAddPartitions,
PartDefinitions: defs,
}
}
| "DROP" ColumnKeywordOpt ColumnName RestrictOrCascadeOpt
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableDropColumn,
OldColumnName: $3.(*ast.ColumnName),
}
}
| "DROP" "PRIMARY" "KEY"
{
$$ = &ast.AlterTableSpec{Tp: ast.AlterTableDropPrimaryKey}
}
| "DROP" "PARTITION" Identifier
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableDropPartition,
Name: $3,
}
}
| "DROP" KeyOrIndex Identifier
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableDropIndex,
Name: $3,
}
}
| "DROP" "FOREIGN" "KEY" Symbol
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableDropForeignKey,
Name: $4.(string),
}
}
| "DISABLE" "KEYS"
{
$$ = &ast.AlterTableSpec{}
}
| "ENABLE" "KEYS"
{
$$ = &ast.AlterTableSpec{}
}
| "MODIFY" ColumnKeywordOpt ColumnDef ColumnPosition
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableModifyColumn,
NewColumns: []*ast.ColumnDef{$3.(*ast.ColumnDef)},
Position: $4.(*ast.ColumnPosition),
}
}
| "CHANGE" ColumnKeywordOpt ColumnName ColumnDef ColumnPosition
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableChangeColumn,
OldColumnName: $3.(*ast.ColumnName),
NewColumns: []*ast.ColumnDef{$4.(*ast.ColumnDef)},
Position: $5.(*ast.ColumnPosition),
}
}
| "ALTER" ColumnKeywordOpt ColumnName "SET" "DEFAULT" SignedLiteral
{
option := &ast.ColumnOption{Expr: $6}
colDef := &ast.ColumnDef{
Name: $3.(*ast.ColumnName),
Options: []*ast.ColumnOption{option},
}
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableAlterColumn,
NewColumns: []*ast.ColumnDef{colDef},
}
}
| "ALTER" ColumnKeywordOpt ColumnName "DROP" "DEFAULT"
{
colDef := &ast.ColumnDef{
Name: $3.(*ast.ColumnName),
}
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableAlterColumn,
NewColumns: []*ast.ColumnDef{colDef},
}
}
| "RENAME" "TO" TableName
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableRenameTable,
NewTable: $3.(*ast.TableName),
}
}
| "RENAME" TableName
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableRenameTable,
NewTable: $2.(*ast.TableName),
}
}
| "RENAME" "AS" TableName
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableRenameTable,
NewTable: $3.(*ast.TableName),
}
}
| "RENAME" KeyOrIndex Identifier "TO" Identifier
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableRenameIndex,
FromKey: model.NewCIStr($3),
ToKey: model.NewCIStr($5),
}
}
| LockClause
{
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableLock,
LockType: $1.(ast.LockType),
}
}
| "ALGORITHM" EqOpt AlterAlgorithm
{
// Parse it and ignore it. Just for compatibility.
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableAlgorithm,
}
}
| "FORCE"
{
// Parse it and ignore it. Just for compatibility.
$$ = &ast.AlterTableSpec{
Tp: ast.AlterTableForce,
}
}
AlterAlgorithm:
"DEFAULT" | "INPLACE" | "COPY"
AlgorithmClause:
"ALGORITHM" EqOpt "DEFAULT"
{
$$ = ast.AlgorithmTypeDefault
}
| "ALGORITHM" EqOpt "COPY"
{
$$ = ast.AlgorithmTypeCopy
}
| "ALGORITHM" EqOpt "INPLACE"
{
$$ = ast.AlgorithmTypeInplace
}
| "ALGORITHM" EqOpt "INSTANT"
{
$$ = ast.AlgorithmTypeInstant
}
| "ALGORITHM" EqOpt identifier
{
yylex.AppendError(ErrUnknownAlterAlgorithm.GenWithStackByArgs($1))
return 1
}
LockClause:
"LOCK" EqOpt "DEFAULT"
{
$$ = ast.LockTypeDefault
}
| "LOCK" EqOpt Identifier
{
id := strings.ToUpper($3)
if id == "NONE" {
$$ = ast.LockTypeNone
} else if id == "SHARED" {
$$ = ast.LockTypeShared
} else if id == "EXCLUSIVE" {
$$ = ast.LockTypeExclusive
} else {
yylex.AppendError(ErrUnknownAlterLock.GenWithStackByArgs($3))
return 1
}
}
KeyOrIndex: "KEY" | "INDEX"
KeyOrIndexOpt:
{}
| KeyOrIndex
ColumnKeywordOpt:
{}
| "COLUMN"
ColumnPosition:
{
$$ = &ast.ColumnPosition{Tp: ast.ColumnPositionNone}
}
| "FIRST"
{
$$ = &ast.ColumnPosition{Tp: ast.ColumnPositionFirst}
}
| "AFTER" ColumnName
{
$$ = &ast.ColumnPosition{
Tp: ast.ColumnPositionAfter,
RelativeColumn: $2.(*ast.ColumnName),
}
}
AlterTableSpecList:
AlterTableSpec
{
$$ = []*ast.AlterTableSpec{$1.(*ast.AlterTableSpec)}
}
| AlterTableSpecList ',' AlterTableSpec
{
$$ = append($1.([]*ast.AlterTableSpec), $3.(*ast.AlterTableSpec))
}
PartitionNameList:
Identifier
{
$$ = []model.CIStr{model.NewCIStr($1)}
}
| PartitionNameList ',' Identifier
{
$$ = append($1.([]model.CIStr), model.NewCIStr($3))
}
ConstraintKeywordOpt:
{
$$ = nil
}
| "CONSTRAINT"
{
$$ = nil
}
| "CONSTRAINT" Symbol
{
$$ = $2.(string)
}
Symbol:
Identifier
{
$$ = $1
}
/**************************************RenameTableStmt***************************************
* See http://dev.mysql.com/doc/refman/5.7/en/rename-table.html
*
* TODO: refactor this when you are going to add full support for multiple schema changes.
* Currently it is only useful for syncer which depends heavily on tidb parser to do some dirty work.
*******************************************************************************************/
RenameTableStmt:
"RENAME" "TABLE" TableToTableList
{
$$ = &ast.RenameTableStmt{
OldTable: $3.([]*ast.TableToTable)[0].OldTable,
NewTable: $3.([]*ast.TableToTable)[0].NewTable,
TableToTables: $3.([]*ast.TableToTable),
}
}
TableToTableList:
TableToTable
{
$$ = []*ast.TableToTable{$1.(*ast.TableToTable)}
}
| TableToTableList ',' TableToTable
{
$$ = append($1.([]*ast.TableToTable), $3.(*ast.TableToTable))
}
TableToTable:
TableName "TO" TableName
{
$$ = &ast.TableToTable{
OldTable: $1.(*ast.TableName),
NewTable: $3.(*ast.TableName),
}
}
/*******************************************************************************************/
AnalyzeTableStmt:
"ANALYZE" "TABLE" TableNameList MaxNumBuckets
{
$$ = &ast.AnalyzeTableStmt{TableNames: $3.([]*ast.TableName), MaxNumBuckets: $4.(uint64)}
}
| "ANALYZE" "TABLE" TableName "INDEX" IndexNameList MaxNumBuckets
{
$$ = &ast.AnalyzeTableStmt{TableNames: []*ast.TableName{$3.(*ast.TableName)}, IndexNames: $5.([]model.CIStr), IndexFlag: true, MaxNumBuckets: $6.(uint64)}
}
| "ANALYZE" "TABLE" TableName "PARTITION" PartitionNameList MaxNumBuckets
{
$$ = &ast.AnalyzeTableStmt{TableNames: []*ast.TableName{$3.(*ast.TableName)}, PartitionNames: $5.([]model.CIStr), MaxNumBuckets: $6.(uint64),}
}
| "ANALYZE" "TABLE" TableName "PARTITION" PartitionNameList "INDEX" IndexNameList MaxNumBuckets
{
$$ = &ast.AnalyzeTableStmt{
TableNames: []*ast.TableName{$3.(*ast.TableName)},
PartitionNames: $5.([]model.CIStr),
IndexNames: $7.([]model.CIStr),
IndexFlag: true,
MaxNumBuckets: $8.(uint64),
}
}
MaxNumBuckets:
{
$$ = uint64(0)
}
| "WITH" NUM "BUCKETS"
{
$$ = getUint64FromNUM($2)
}
/*******************************************************************************************/
Assignment:
ColumnName eq Expression
{
$$ = &ast.Assignment{Column: $1.(*ast.ColumnName), Expr:$3}
}
AssignmentList:
Assignment
{
$$ = []*ast.Assignment{$1.(*ast.Assignment)}
}
| AssignmentList ',' Assignment
{
$$ = append($1.([]*ast.Assignment), $3.(*ast.Assignment))
}
AssignmentListOpt:
/* EMPTY */
{
$$ = []*ast.Assignment{}
}
| AssignmentList
BeginTransactionStmt:
"BEGIN"
{
$$ = &ast.BeginStmt{}
}
| "START" "TRANSACTION"
{
$$ = &ast.BeginStmt{}
}
| "START" "TRANSACTION" "WITH" "CONSISTENT" "SNAPSHOT"
{
$$ = &ast.BeginStmt{}
}
BinlogStmt:
"BINLOG" stringLit
{
$$ = &ast.BinlogStmt{Str: $2}
}
ColumnDefList:
ColumnDef
{
$$ = []*ast.ColumnDef{$1.(*ast.ColumnDef)}
}
| ColumnDefList ',' ColumnDef
{
$$ = append($1.([]*ast.ColumnDef), $3.(*ast.ColumnDef))
}
ColumnDef:
ColumnName Type ColumnOptionListOpt
{
$$ = &ast.ColumnDef{Name: $1.(*ast.ColumnName), Tp: $2.(*types.FieldType), Options: $3.([]*ast.ColumnOption)}
}
ColumnName:
Identifier
{
$$ = &ast.ColumnName{Name: model.NewCIStr($1)}
}
| Identifier '.' Identifier
{
$$ = &ast.ColumnName{Table: model.NewCIStr($1), Name: model.NewCIStr($3)}
}
| Identifier '.' Identifier '.' Identifier
{
$$ = &ast.ColumnName{Schema: model.NewCIStr($1), Table: model.NewCIStr($3), Name: model.NewCIStr($5)}
}
ColumnNameList:
ColumnName
{
$$ = []*ast.ColumnName{$1.(*ast.ColumnName)}
}
| ColumnNameList ',' ColumnName
{
$$ = append($1.([]*ast.ColumnName), $3.(*ast.ColumnName))
}
ColumnNameListOpt:
/* EMPTY */
{
$$ = []*ast.ColumnName{}
}
| ColumnNameList
{
$$ = $1.([]*ast.ColumnName)
}
ColumnNameListOptWithBrackets:
/* EMPTY */
{
$$ = []*ast.ColumnName{}
}
| '(' ColumnNameListOpt ')'
{
$$ = $2.([]*ast.ColumnName)
}
CommitStmt:
"COMMIT"
{
$$ = &ast.CommitStmt{}
}
InceptionStmt:
"INCEPTION" "GET" "OSC" "PROCESSLIST"
{
$$ = &ast.ShowOscStmt{}
}
| "INCEPTION" "GET" "OSC_PERCENT" AuthString
{
$$ = &ast.ShowOscStmt{
Sqlsha1: $4.(string),
}
}
| "INCEPTION" "KILL" "OSC" AuthString
{
$$ = &ast.ShowOscStmt{
Sqlsha1: $4.(string),
Tp: ast.OscOptionKill,
}
}
| "INCEPTION" "STOP" "ALTER" AuthString
{
$$ = &ast.ShowOscStmt{
Sqlsha1: $4.(string),
Tp: ast.OscOptionKill,
}
}
| "INCEPTION" "PAUSE" "OSC" AuthString
{
$$ = &ast.ShowOscStmt{
Sqlsha1: $4.(string),
Tp: ast.OscOptionPause,
}
}
| "INCEPTION" "PAUSE" "ALTER" AuthString
{
$$ = &ast.ShowOscStmt{
Sqlsha1: $4.(string),
Tp: ast.OscOptionPause,
}
}
| "INCEPTION" "RESUME" "OSC" AuthString
{
$$ = &ast.ShowOscStmt{
Sqlsha1: $4.(string),
Tp: ast.OscOptionResume,
}
}
| "INCEPTION" "RESUME" "ALTER" AuthString
{
$$ = &ast.ShowOscStmt{
Sqlsha1: $4.(string),
Tp: ast.OscOptionResume,
}
}
| "INCEPTION" "SHOW" ShowTargetFilterable ShowLikeOrWhereOpt
{
stmt := $3.(*ast.ShowStmt)
if $4 != nil {
if x, ok := $4.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $4.(ast.ExprNode)
}
}
stmt.IsInception = true
$$ = stmt
}
| "INCEPTION" "GET" ShowTargetFilterable ShowLikeOrWhereOpt
{
stmt := $3.(*ast.ShowStmt)
if $4 != nil {
if x, ok := $4.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $4.(ast.ExprNode)
}
}
stmt.IsInception = true
$$ = stmt
}
| "INCEPTION" "SHOW" "CREATE" "TABLE" TableName
{
$$ = &ast.ShowStmt{
Tp: ast.ShowCreateTable,
Table: $5.(*ast.TableName),
IsInception: true,
}
}
| "INCEPTION" "SHOW" "CREATE" "DATABASE" DBName
{
$$ = &ast.ShowStmt{
Tp: ast.ShowCreateDatabase,
DBName: $5.(string),
IsInception: true,
}
}
| "INCEPTION" "SHOW" "GRANTS"
{
// See https://dev.mysql.com/doc/refman/5.7/en/show-grants.html
$$ = &ast.ShowStmt{
Tp: ast.ShowGrants,
IsInception: true,
}
}
| "INCEPTION" "SHOW" "GRANTS" "FOR" Username
{
// See https://dev.mysql.com/doc/refman/5.7/en/show-grants.html
$$ = &ast.ShowStmt{
Tp: ast.ShowGrants,
User: $5.(*auth.UserIdentity),
IsInception: true,
}
}
| "INCEPTION" "SHOW" OptFull "PROCESSLIST"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowProcessList,
Full: $3.(bool),
IsInception: true,
}
}
| "INCEPTION" "GET" OptFull "PROCESSLIST"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowProcessList,
Full: $3.(bool),
IsInception: true,
}
}
| "INCEPTION" "SHOW" "STATS_META" ShowLikeOrWhereOpt
{
stmt := &ast.ShowStmt{
Tp: ast.ShowStatsMeta,
IsInception: true,
}
if $4 != nil {
if x, ok := $4.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $4.(ast.ExprNode)
}
}
$$ = stmt
}
| "INCEPTION" "SHOW" "STATS_HISTOGRAMS" ShowLikeOrWhereOpt
{
stmt := &ast.ShowStmt{
Tp: ast.ShowStatsHistograms,
IsInception: true,
}
if $4 != nil {
if x, ok := $4.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $4.(ast.ExprNode)
}
}
$$ = stmt
}
| "INCEPTION" "SHOW" "STATS_BUCKETS" ShowLikeOrWhereOpt
{
stmt := &ast.ShowStmt{
Tp: ast.ShowStatsBuckets,
IsInception: true,
}
if $4 != nil {
if x, ok := $4.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $4.(ast.ExprNode)
}
}
$$ = stmt
}
| "INCEPTION" "SHOW" "STATS_HEALTHY" ShowLikeOrWhereOpt
{
stmt := &ast.ShowStmt{
Tp: ast.ShowStatsHealthy,
IsInception: true,
}
if $4 != nil {
if x, ok := $4.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $4.(ast.ExprNode)
}
}
$$ = stmt
}
| "INCEPTION" "SHOW" "PROFILES"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowProfiles,
IsInception: true,
}
}
| "INCEPTION" "SHOW" "PRIVILEGES"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowPrivileges,
IsInception: true,
}
}
| "INCEPTION" "SET" VariableAssignmentList
{
$$ = &ast.InceptionSetStmt{Variables: $3.([]*ast.VariableAssignment)}
}
| "INCEPTION" "SET" "LEVEL" VariableAssignmentList
{
vars := $4.([]*ast.VariableAssignment)
for _, v := range vars {
v.IsLevel = true
}
$$ = &ast.InceptionSetStmt{Variables: vars}
}
InceptionStartStmt:
"INCEPTION_MAGIC_START"
{
$$ = &ast.InceptionStartStmt{}
}
InceptionCommitStmt:
"INCEPTION_MAGIC_COMMIT"
{
$$ = &ast.InceptionCommitStmt{}
}
PrimaryOpt:
{}
| "PRIMARY"
ColumnOption:
"NOT" "NULL"
{
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionNotNull}
}
| "NULL"
{
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionNull}
}
| "AUTO_INCREMENT"
{
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionAutoIncrement}
}
| PrimaryOpt "KEY"
{
// KEY is normally a synonym for INDEX. The key attribute PRIMARY KEY
// can also be specified as just KEY when given in a column definition.
// See http://dev.mysql.com/doc/refman/5.7/en/create-table.html
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionPrimaryKey}
}
| "UNIQUE" %prec lowerThanKey
{
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionUniqKey}
}
| "UNIQUE" "KEY"
{
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionUniqKey}
}
| "DEFAULT" DefaultValueExpr
{
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionDefaultValue, Expr: $2}
}
| "ON" "UPDATE" NowSymOptionFraction
{
nowFunc := &ast.FuncCallExpr{FnName: model.NewCIStr("CURRENT_TIMESTAMP")}
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionOnUpdate, Expr: nowFunc}
}
| "COMMENT" stringLit
{
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionComment, Expr: ast.NewValueExpr($2)}
}
| "CHECK" '(' Expression ')'
{
// See https://dev.mysql.com/doc/refman/5.7/en/create-table.html
// The CHECK clause is parsed but ignored by all storage engines.
$$ = &ast.ColumnOption{}
}
| GeneratedAlways "AS" '(' Expression ')' VirtualOrStored
{
startOffset := parser.startOffset(&yyS[yypt-2])
endOffset := parser.endOffset(&yyS[yypt-1])
expr := $4
expr.SetText(parser.src[startOffset:endOffset])
$$ = &ast.ColumnOption{
Tp: ast.ColumnOptionGenerated,
Expr: expr,
Stored: $6.(bool),
}
}
| ReferDef
{
$$ = &ast.ColumnOption{
Tp: ast.ColumnOptionReference,
Refer: $1.(*ast.ReferenceDef),
}
}
| "COLLATE" CollationName
{
$$ = &ast.ColumnOption{Tp: ast.ColumnOptionCollate, StrValue: $2.(string)}
}
GeneratedAlways: | "GENERATED" "ALWAYS"
VirtualOrStored:
{
$$ = false
}
| "VIRTUAL"
{
$$ = false
}
| "STORED"
{
$$ = true
}
ColumnOptionList:
ColumnOption
{
$$ = []*ast.ColumnOption{$1.(*ast.ColumnOption)}
}
| ColumnOptionList ColumnOption
{
$$ = append($1.([]*ast.ColumnOption), $2.(*ast.ColumnOption))
}
ColumnOptionListOpt:
{
$$ = []*ast.ColumnOption{}
}
| ColumnOptionList
{
$$ = $1.([]*ast.ColumnOption)
}
ConstraintElem:
"PRIMARY" "KEY" IndexNameAndTypeOpt '(' IndexColNameList ')' IndexOptionList
{
c := &ast.Constraint{
Tp: ast.ConstraintPrimaryKey,
Keys: $5.([]*ast.IndexColName),
}
if $7 != nil {
c.Option = $7.(*ast.IndexOption)
}
if indexType := $3.([]interface{})[1]; indexType != nil {
if c.Option == nil {
c.Option = &ast.IndexOption{}
}
c.Option.Tp = indexType.(model.IndexType)
}
$$ = c
}
| "FULLTEXT" KeyOrIndexOpt IndexName '(' IndexColNameList ')' IndexOptionList
{
c := &ast.Constraint{
Tp: ast.ConstraintFulltext,
Keys: $5.([]*ast.IndexColName),
Name: $3.(string),
}
if $7 != nil {
c.Option = $7.(*ast.IndexOption)
}
$$ = c
}
| "SPATIAL" KeyOrIndexOpt IndexName '(' IndexColNameList ')' IndexOptionList
{
c := &ast.Constraint{
Tp: ast. ConstraintSpatial,
Keys: $5.([]*ast.IndexColName),
Name: $3.(string),
}
if $7 != nil {
c.Option = $7.(*ast.IndexOption)
}
$$ = c
}
| KeyOrIndex IfNotExists IndexNameAndTypeOpt '(' IndexColNameList ')' IndexOptionList
{
c := &ast.Constraint{
IfNotExists: $2.(bool),
Tp: ast.ConstraintIndex,
Keys: $5.([]*ast.IndexColName),
}
if $7 != nil {
c.Option = $7.(*ast.IndexOption)
}
c.Name = $3.([]interface{})[0].(string)
if indexType := $3.([]interface{})[1]; indexType != nil {
if c.Option == nil {
c.Option = &ast.IndexOption{}
}
c.Option.Tp = indexType.(model.IndexType)
}
$$ = c
}
| "UNIQUE" KeyOrIndexOpt IndexNameAndTypeOpt '(' IndexColNameList ')' IndexOptionList
{
c := &ast.Constraint{
Tp: ast.ConstraintUniq,
Keys: $5.([]*ast.IndexColName),
}
if $7 != nil {
c.Option = $7.(*ast.IndexOption)
}
c.Name = $3.([]interface{})[0].(string)
if indexType := $3.([]interface{})[1]; indexType != nil {
if c.Option == nil {
c.Option = &ast.IndexOption{}
}
c.Option.Tp = indexType.(model.IndexType)
}
$$ = c
}
| "FOREIGN" "KEY" IfNotExists IndexName '(' IndexColNameList ')' ReferDef
{
$$ = &ast.Constraint{
IfNotExists: $3.(bool),
Tp: ast.ConstraintForeignKey,
Keys: $6.([]*ast.IndexColName),
Name: $4.(string),
Refer: $8.(*ast.ReferenceDef),
}
}
ReferDef:
"REFERENCES" TableName '(' IndexColNameList ')' OnDeleteOpt OnUpdateOpt
{
var onDeleteOpt *ast.OnDeleteOpt
if $6 != nil {
onDeleteOpt = $6.(*ast.OnDeleteOpt)
}
var onUpdateOpt *ast.OnUpdateOpt
if $7 != nil {
onUpdateOpt = $7.(*ast.OnUpdateOpt)
}
$$ = &ast.ReferenceDef{
Table: $2.(*ast.TableName),
IndexColNames: $4.([]*ast.IndexColName),
OnDelete: onDeleteOpt,
OnUpdate: onUpdateOpt,
}
}
OnDeleteOpt:
{
$$ = &ast.OnDeleteOpt{}
} %prec lowerThanOn
| "ON" "DELETE" ReferOpt
{
$$ = &ast.OnDeleteOpt{ReferOpt: $3.(ast.ReferOptionType)}
}
OnUpdateOpt:
{
$$ = &ast.OnUpdateOpt{}
} %prec lowerThanOn
| "ON" "UPDATE" ReferOpt
{
$$ = &ast.OnUpdateOpt{ReferOpt: $3.(ast.ReferOptionType)}
}
ReferOpt:
"RESTRICT"
{
$$ = ast.ReferOptionRestrict
}
| "CASCADE"
{
$$ = ast.ReferOptionCascade
}
| "SET" "NULL"
{
$$ = ast.ReferOptionSetNull
}
| "NO" "ACTION"
{
$$ = ast.ReferOptionNoAction
}
/*
* The DEFAULT clause specifies a default value for a column.
* With one exception, the default value must be a constant;
* it cannot be a function or an expression. This means, for example,
* that you cannot set the default for a date column to be the value of
* a function such as NOW() or CURRENT_DATE. The exception is that you
* can specify CURRENT_TIMESTAMP as the default for a TIMESTAMP or DATETIME column.
*
* See http://dev.mysql.com/doc/refman/5.7/en/create-table.html
* https://github.com/mysql/mysql-server/blob/5.7/sql/sql_yacc.yy#L6832
*/
DefaultValueExpr:
NowSymOptionFraction | SignedLiteral
NowSymOptionFraction:
NowSym
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr("CURRENT_TIMESTAMP")}
}
| NowSymFunc '(' ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr("CURRENT_TIMESTAMP")}
}
| NowSymFunc '(' NUM ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr("CURRENT_TIMESTAMP")}
}
/*
* See https://dev.mysql.com/doc/refman/5.7/en/date-and-time-functions.html#function_localtime
* TODO: Process other three keywords
*/
NowSymFunc:
"CURRENT_TIMESTAMP" | "LOCALTIME" | "LOCALTIMESTAMP" | builtinNow
NowSym:
"CURRENT_TIMESTAMP" | "LOCALTIME" | "LOCALTIMESTAMP"
SignedLiteral:
Literal
{
$$ = ast.NewValueExpr($1)
}
| '+' NumLiteral
{
$$ = &ast.UnaryOperationExpr{Op: opcode.Plus, V: ast.NewValueExpr($2)}
}
| '-' NumLiteral
{
$$ = &ast.UnaryOperationExpr{Op: opcode.Minus, V: ast.NewValueExpr($2)}
}
NumLiteral:
intLit
| floatLit
| decLit
/**************************************CreateIndexStmt***************************************
* See https://dev.mysql.com/doc/refman/8.0/en/create-index.html
*
* TYPE type_name is recognized as a synonym for USING type_name. However, USING is the preferred form.
*
* CREATE [UNIQUE | FULLTEXT | SPATIAL] INDEX index_name
* [index_type]
* ON tbl_name (key_part,...)
* [index_option]
* [algorithm_option | lock_option] ...
*
* key_part: {col_name [(length)] | (expr)} [ASC | DESC]
*
* index_option:
* KEY_BLOCK_SIZE [=] value
* | index_type
* | WITH PARSER parser_name
* | COMMENT 'string'
* | {VISIBLE | INVISIBLE}
*
* index_type:
* USING {BTREE | HASH}
*
* algorithm_option:
* ALGORITHM [=] {DEFAULT | INPLACE | COPY}
*
* lock_option:
* LOCK [=] {DEFAULT | NONE | SHARED | EXCLUSIVE}
*******************************************************************************************/
CreateIndexStmt:
"CREATE" IndexKeyTypeOpt "INDEX" IfNotExists Identifier IndexTypeOpt "ON" TableName '(' IndexColNameList ')' IndexOptionList IndexLockAndAlgorithmOpt
{
var indexOption *ast.IndexOption
if $12 != nil {
indexOption = $12.(*ast.IndexOption)
if indexOption.Tp == model.IndexTypeInvalid {
if $6 != nil {
indexOption.Tp = $6.(model.IndexType)
}
}
} else {
indexOption = &ast.IndexOption{}
if $6 != nil {
indexOption.Tp = $6.(model.IndexType)
}
}
var indexLockAndAlgorithm *ast.IndexLockAndAlgorithm
if $13 != nil {
indexLockAndAlgorithm = $13.(*ast.IndexLockAndAlgorithm)
if indexLockAndAlgorithm.LockTp == ast.LockTypeDefault && indexLockAndAlgorithm.AlgorithmTp == ast.AlgorithmTypeDefault {
indexLockAndAlgorithm = nil
}
}
$$ = &ast.CreateIndexStmt{
IfNotExists: $4.(bool),
IndexName: $5,
Table: $8.(*ast.TableName),
IndexColNames: $10.([]*ast.IndexColName),
IndexOption: indexOption,
KeyType: $2.(ast.IndexKeyType),
Unique: $2.(ast.IndexKeyType) == ast.IndexKeyTypeUnique,
LockAlg: indexLockAndAlgorithm,
}
}
IndexColName:
ColumnName OptFieldLen Order
{
//Order is parsed but just ignored as MySQL did
$$ = &ast.IndexColName{Column: $1.(*ast.ColumnName), Length: $2.(int)}
}
IndexColNameList:
IndexColName
{
$$ = []*ast.IndexColName{$1.(*ast.IndexColName)}
}
| IndexColNameList ',' IndexColName
{
$$ = append($1.([]*ast.IndexColName), $3.(*ast.IndexColName))
}
IndexLockAndAlgorithmOpt:
{
$$ = nil
}
| LockClause
{
$$ = &ast.IndexLockAndAlgorithm{
LockTp: $1.(ast.LockType),
AlgorithmTp: ast.AlgorithmTypeDefault,
}
}
| AlgorithmClause
{
$$ = &ast.IndexLockAndAlgorithm{
LockTp: ast.LockTypeDefault,
AlgorithmTp: $1.(ast.AlgorithmType),
}
}
| LockClause AlgorithmClause
{
$$ = &ast.IndexLockAndAlgorithm{
LockTp: $1.(ast.LockType),
AlgorithmTp: $2.(ast.AlgorithmType),
}
}
| AlgorithmClause LockClause
{
$$ = &ast.IndexLockAndAlgorithm{
LockTp: $2.(ast.LockType),
AlgorithmTp: $1.(ast.AlgorithmType),
}
}
IndexKeyTypeOpt:
{
$$ = ast.IndexKeyTypeNone
}
| "UNIQUE"
{
$$ = ast.IndexKeyTypeUnique
}
| "SPATIAL"
{
$$ = ast.IndexKeyTypeSpatial
}
| "FULLTEXT"
{
$$ = ast.IndexKeyTypeFullText
}
/*******************************************************************
*
* Create Database Statement
* CREATE {DATABASE | SCHEMA} [IF NOT EXISTS] db_name
* [create_specification] ...
*
* create_specification:
* [DEFAULT] CHARACTER SET [=] charset_name
* | [DEFAULT] COLLATE [=] collation_name
*******************************************************************/
CreateDatabaseStmt:
"CREATE" DatabaseSym IfNotExists DBName DatabaseOptionListOpt
{
$$ = &ast.CreateDatabaseStmt{
IfNotExists: $3.(bool),
Name: $4.(string),
Options: $5.([]*ast.DatabaseOption),
}
}
DBName:
Identifier
{
$$ = $1
}
DatabaseOption:
DefaultKwdOpt CharsetKw EqOpt CharsetName
{
$$ = &ast.DatabaseOption{Tp: ast.DatabaseOptionCharset, Value: $4.(string)}
}
| DefaultKwdOpt "COLLATE" EqOpt CollationName
{
$$ = &ast.DatabaseOption{Tp: ast.DatabaseOptionCollate, Value: $4.(string)}
}
DatabaseOptionListOpt:
{
$$ = []*ast.DatabaseOption{}
}
| DatabaseOptionList
DatabaseOptionList:
DatabaseOption
{
$$ = []*ast.DatabaseOption{$1.(*ast.DatabaseOption)}
}
| DatabaseOptionList DatabaseOption
{
$$ = append($1.([]*ast.DatabaseOption), $2.(*ast.DatabaseOption))
}
/*******************************************************************
*
* Create Table Statement
*
* Example:
* CREATE TABLE Persons
* (
* P_Id int NOT NULL,
* LastName varchar(255) NOT NULL,
* FirstName varchar(255),
* Address varchar(255),
* City varchar(255),
* PRIMARY KEY (P_Id)
* )
*******************************************************************/
CreateTableStmt:
"CREATE" "TABLE" IfNotExists TableName TableElementListOpt CreateTableOptionListOpt PartitionOpt DuplicateOpt AsOpt CreateTableSelectOpt
{
stmt := $5.(*ast.CreateTableStmt)
stmt.Table = $4.(*ast.TableName)
stmt.IfNotExists = $3.(bool)
stmt.Options = $6.([]*ast.TableOption)
if $7 != nil {
stmt.Partition = $7.(*ast.PartitionOptions)
}
stmt.OnDuplicate = $8.(ast.OnDuplicateKeyHandlingType)
stmt.Select = $10.(*ast.CreateTableStmt).Select
$$ = stmt
}
| "CREATE" "TABLE" IfNotExists TableName LikeTableWithOrWithoutParen
{
$$ = &ast.CreateTableStmt{
Table: $4.(*ast.TableName),
ReferTable: $5.(*ast.TableName),
IfNotExists: $3.(bool),
}
}
DefaultKwdOpt:
{}
| "DEFAULT"
PartitionOpt:
{
$$ = nil
}
| "PARTITION" "BY" PartitionMethod PartitionNumOpt SubPartitionOpt PartitionDefinitionListOpt
{
method := $3.(*ast.PartitionMethod)
method.Num = $4.(uint64)
sub, _ := $5.(*ast.PartitionMethod)
defs, _ := $6.([]*ast.PartitionDefinition)
opt := &ast.PartitionOptions{
PartitionMethod: *method,
Sub: sub,
Definitions: defs,
}
if err := opt.Validate(); err != nil {
yylex.AppendError(err)
return 1
}
$$ = opt
}
SubPartitionMethod:
LinearOpt "KEY" PartitionKeyAlgorithmOpt '(' ColumnNameListOpt ')'
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeKey,
Linear: len($1) != 0,
ColumnNames: $5.([]*ast.ColumnName),
}
}
| LinearOpt "HASH" '(' Expression ')'
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeHash,
Linear: len($1) != 0,
Expr: $4.(ast.ExprNode),
}
}
PartitionKeyAlgorithmOpt:
/* empty */
{}
| "ALGORITHM" '=' NUM
{}
PartitionMethod:
SubPartitionMethod
{
$$ = $1
}
| "RANGE" '(' Expression ')'
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeRange,
Expr: $3.(ast.ExprNode),
}
}
| "RANGE" "COLUMNS" '(' ColumnNameList ')'
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeRange,
ColumnNames: $4.([]*ast.ColumnName),
}
}
| "LIST" '(' Expression ')'
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeList,
Expr: $3.(ast.ExprNode),
}
}
| "LIST" "COLUMNS" '(' ColumnNameList ')'
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeList,
ColumnNames: $4.([]*ast.ColumnName),
}
}
| "SYSTEM_TIME" "INTERVAL" Expression TimeUnit
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeSystemTime,
Expr: $3.(ast.ExprNode),
Unit: ast.NewValueExpr($4),
}
}
| "SYSTEM_TIME" "LIMIT" LengthNum
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeSystemTime,
Limit: $3.(uint64),
}
}
| "SYSTEM_TIME"
{
$$ = &ast.PartitionMethod{
Tp: model.PartitionTypeSystemTime,
}
}
LinearOpt:
{
$$ = ""
}
| "LINEAR"
{
$$ = $1
}
SubPartitionOpt:
{
$$ = nil
}
| "SUBPARTITION" "BY" SubPartitionMethod SubPartitionNumOpt
{
method := $3.(*ast.PartitionMethod)
method.Num = $4.(uint64)
$$ = method
}
SubPartitionNumOpt:
{
$$ = uint64(0)
}
| "SUBPARTITIONS" LengthNum
{
res := $2.(uint64)
if res == 0 {
yylex.AppendError(ast.ErrNoParts.GenWithStackByArgs("subpartitions"))
return 1
}
$$ = res
}
PartitionNumOpt:
{
$$ = uint64(0)
}
| "PARTITIONS" LengthNum
{
res := $2.(uint64)
if res == 0 {
yylex.AppendError(ast.ErrNoParts.GenWithStackByArgs("partitions"))
return 1
}
$$ = res
}
PartitionDefinitionListOpt:
/* empty */ %prec lowerThanCreateTableSelect
{
$$ = nil
}
| '(' PartitionDefinitionList ')'
{
$$ = $2.([]*ast.PartitionDefinition)
}
PartitionDefinitionList:
PartitionDefinition
{
$$ = []*ast.PartitionDefinition{$1.(*ast.PartitionDefinition)}
}
| PartitionDefinitionList ',' PartitionDefinition
{
$$ = append($1.([]*ast.PartitionDefinition), $3.(*ast.PartitionDefinition))
}
PartitionDefinition:
"PARTITION" Identifier PartDefValuesOpt PartDefOptionList SubPartDefinitionListOpt
{
$$ = &ast.PartitionDefinition{
Name: model.NewCIStr($2),
Clause: $3.(ast.PartitionDefinitionClause),
Options: $4.([]*ast.TableOption),
Sub: $5.([]*ast.SubPartitionDefinition),
}
}
SubPartDefinitionListOpt:
/*empty*/
{
$$ = make([]*ast.SubPartitionDefinition, 0)
}
| '(' SubPartDefinitionList ')'
{
$$ = $2
}
SubPartDefinitionList:
SubPartDefinition
{
$$ = []*ast.SubPartitionDefinition{$1.(*ast.SubPartitionDefinition)}
}
| SubPartDefinitionList ',' SubPartDefinition
{
list := $1.([]*ast.SubPartitionDefinition)
$$ = append(list, $3.(*ast.SubPartitionDefinition))
}
SubPartDefinition:
"SUBPARTITION" Identifier PartDefOptionList
{
$$ = &ast.SubPartitionDefinition{
Name: model.NewCIStr($2),
Options: $3.([]*ast.TableOption),
}
}
PartDefOptionList:
/*empty*/
{
$$ = make([]*ast.TableOption, 0)
}
| PartDefOptionList PartDefOption
{
list := $1.([]*ast.TableOption)
$$ = append(list, $2.(*ast.TableOption))
}
PartDefOption:
"COMMENT" EqOpt stringLit
{
$$ = &ast.TableOption{Tp: ast.TableOptionComment, StrValue: $3}
}
| "ENGINE" EqOpt StringName
{
$$ = &ast.TableOption{Tp: ast.TableOptionEngine, StrValue: $3.(string)}
}
| "DATA" "DIRECTORY" EqOpt stringLit
{
$$ = &ast.TableOption{Tp: ast.TableOptionDataDirectory, StrValue: $4}
}
| "INDEX" "DIRECTORY" EqOpt stringLit
{
$$ = &ast.TableOption{Tp: ast.TableOptionIndexDirectory, StrValue: $4}
}
| "MAX_ROWS" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionMaxRows, UintValue: $3.(uint64)}
}
| "MIN_ROWS" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionMinRows, UintValue: $3.(uint64)}
}
| "TABLESPACE" EqOpt Identifier
{
$$ = &ast.TableOption{Tp: ast.TableOptionTablespace, StrValue: $3}
}
| "NODEGROUP" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionNodegroup, UintValue: $3.(uint64)}
}
PartDefValuesOpt:
{
$$ = &ast.PartitionDefinitionClauseNone{}
}
| "VALUES" "LESS" "THAN" "MAXVALUE"
{
$$ = &ast.PartitionDefinitionClauseLessThan{
Exprs: []ast.ExprNode{&ast.MaxValueExpr{}},
}
}
| "VALUES" "LESS" "THAN" '(' MaxValueOrExpressionList ')'
{
$$ = &ast.PartitionDefinitionClauseLessThan{
Exprs: $5.([]ast.ExprNode),
}
}
| "DEFAULT"
{
$$ = &ast.PartitionDefinitionClauseIn{}
}
| "VALUES" "IN" '(' ExpressionList ')'
{
exprs := $4.([]ast.ExprNode)
values := make([][]ast.ExprNode, 0, len(exprs))
for _, expr := range exprs {
if row, ok := expr.(*ast.RowExpr); ok {
values = append(values, row.Values)
} else {
values = append(values, []ast.ExprNode{expr})
}
}
$$ = &ast.PartitionDefinitionClauseIn{Values: values}
}
| "HISTORY"
{
$$ = &ast.PartitionDefinitionClauseHistory{Current: false}
}
| "CURRENT"
{
$$ = &ast.PartitionDefinitionClauseHistory{Current: true}
}
DuplicateOpt:
{
$$ = ast.OnDuplicateKeyHandlingError
}
| "IGNORE"
{
$$ = ast.OnDuplicateKeyHandlingIgnore
}
| "REPLACE"
{
$$ = ast.OnDuplicateKeyHandlingReplace
}
AsOpt:
{}
| "AS"
{}
CreateTableSelectOpt:
/* empty */
{
$$ = &ast.CreateTableStmt{}
}
|
SelectStmt
{
$$ = &ast.CreateTableStmt{Select: $1}
}
|
UnionStmt
{
$$ = &ast.CreateTableStmt{Select: $1}
}
|
SubSelect %prec createTableSelect
// TODO: We may need better solution as issue #320.
{
$$ = &ast.CreateTableStmt{Select: $1}
}
LikeTableWithOrWithoutParen:
"LIKE" TableName
{
$$ = $2
}
|
'(' "LIKE" TableName ')'
{
$$ = $3
}
/*******************************************************************
*
* Create View Statement
*
* Example:
* CREATE VIEW OR REPLACE ALGORITHM = MERGE DEFINER="root@localhost" SQL SECURITY = definer view_name (col1,col2)
* as select Col1,Col2 from table WITH LOCAL CHECK OPTION
*******************************************************************/
CreateViewStmt:
"CREATE" OrReplace ViewAlgorithm ViewDefiner ViewSQLSecurity "VIEW" ViewName ViewFieldList "AS" SelectStmt ViewCheckOption
{
startOffset := parser.startOffset(&yyS[yypt-1])
selStmt := $10.(*ast.SelectStmt)
selStmt.SetText(string(parser.src[startOffset:]))
x := &ast.CreateViewStmt {
OrReplace: $2.(bool),
ViewName: $7.(*ast.TableName),
Select: selStmt,
}
if $8 != nil{
x.Cols = $8.([]model.CIStr)
}
$$ = x
}
OrReplace:
{
$$ = false
}
| "OR" "REPLACE"
{
$$ = true
}
ViewAlgorithm:
/* EMPTY */
{
$$ = "UNDEFINED"
}
| "ALGORITHM" "=" "UNDEFINED"
{
$$ = strings.ToUpper($3)
}
| "ALGORITHM" "=" "MERGE"
{
$$ = strings.ToUpper($3)
}
| "ALGORITHM" "=" "TEMPTABLE"
{
$$ = strings.ToUpper($3)
}
ViewDefiner:
/* EMPTY */
{
$$ = nil
}
| "DEFINER" "=" Username
{
$$ = $3
}
ViewSQLSecurity:
/* EMPTY */
{
$$ = "DEFINER"
}
| "SQL" "SECURITY" "DEFINER"
{
$$ = $3
}
| "SQL" "SECURITY" "INVOKER"
{
$$ = $3
}
ViewName:
TableName
{
$$ = $1.(*ast.TableName)
}
ViewFieldList:
/* Empty */
{
$$ = nil
}
| '(' ColumnList ')'
{
$$ = $2.([]model.CIStr)
}
ColumnList:
Identifier
{
$$ = []model.CIStr{model.NewCIStr($1)}
}
| ColumnList ',' Identifier
{
$$ = append($1.([]model.CIStr), model.NewCIStr($3))
}
ViewCheckOption:
/* EMPTY */
{
$$ = nil
}
| "WITH" "CASCADED" "CHECK" "OPTION"
{
$$ = $2
}
| "WITH" "LOCAL" "CHECK" "OPTION"
{
$$ = $2
}
/******************************************************************
* Do statement
* See https://dev.mysql.com/doc/refman/5.7/en/do.html
******************************************************************/
DoStmt:
"DO" ExpressionList
{
$$ = &ast.DoStmt {
Exprs: $2.([]ast.ExprNode),
}
}
/*******************************************************************
*
* Delete Statement
*
*******************************************************************/
DeleteFromStmt:
"DELETE" TableOptimizerHints PriorityOpt QuickOptional IgnoreOptional "FROM" TableName IndexHintListOpt WhereClauseOptional OrderByOptional LimitClause
{
// Single Table
tn := $7.(*ast.TableName)
tn.IndexHints = $8.([]*ast.IndexHint)
join := &ast.Join{Left: &ast.TableSource{Source: tn}, Right: nil}
x := &ast.DeleteStmt{
TableRefs: &ast.TableRefsClause{TableRefs: join},
Priority: $3.(mysql.PriorityEnum),
Quick: $4.(bool),
IgnoreErr: $5.(bool),
}
if $9 != nil {
x.Where = $9.(ast.ExprNode)
}
if $10 != nil {
x.Order = $10.(*ast.OrderByClause)
}
if $11 != nil {
x.Limit = $11.(*ast.Limit)
}
$$ = x
}
| "DELETE" TableOptimizerHints PriorityOpt QuickOptional IgnoreOptional TableNameList "FROM" TableRefs WhereClauseOptional
{
// Multiple Table
x := &ast.DeleteStmt{
Priority: $3.(mysql.PriorityEnum),
Quick: $4.(bool),
IgnoreErr: $5.(bool),
IsMultiTable: true,
BeforeFrom: true,
Tables: &ast.DeleteTableList{Tables: $6.([]*ast.TableName)},
TableRefs: &ast.TableRefsClause{TableRefs: $8.(*ast.Join)},
}
if $2 != nil {
x.TableHints = $2.([]*ast.TableOptimizerHint)
}
if $9 != nil {
x.Where = $9.(ast.ExprNode)
}
$$ = x
}
| "DELETE" TableOptimizerHints PriorityOpt QuickOptional IgnoreOptional "FROM" TableNameList "USING" TableRefs WhereClauseOptional
{
// Multiple Table
x := &ast.DeleteStmt{
Priority: $3.(mysql.PriorityEnum),
Quick: $4.(bool),
IgnoreErr: $5.(bool),
IsMultiTable: true,
Tables: &ast.DeleteTableList{Tables: $7.([]*ast.TableName)},
TableRefs: &ast.TableRefsClause{TableRefs: $9.(*ast.Join)},
}
if $2 != nil {
x.TableHints = $2.([]*ast.TableOptimizerHint)
}
if $10 != nil {
x.Where = $10.(ast.ExprNode)
}
$$ = x
}
DatabaseSym:
"DATABASE"
DropDatabaseStmt:
"DROP" DatabaseSym IfExists DBName
{
$$ = &ast.DropDatabaseStmt{IfExists: $3.(bool), Name: $4.(string)}
}
/******************************************************************
* Drop Index Statement
* See https://dev.mysql.com/doc/refman/8.0/en/drop-index.html
*
* DROP INDEX index_name ON tbl_name
* [algorithm_option | lock_option] ...
*
* algorithm_option:
* ALGORITHM [=] {DEFAULT|INPLACE|COPY}
*
* lock_option:
* LOCK [=] {DEFAULT|NONE|SHARED|EXCLUSIVE}
******************************************************************/
DropIndexStmt:
"DROP" "INDEX" IfExists Identifier "ON" TableName IndexLockAndAlgorithmOpt
{
var indexLockAndAlgorithm *ast.IndexLockAndAlgorithm
if $7 != nil {
indexLockAndAlgorithm = $7.(*ast.IndexLockAndAlgorithm)
if indexLockAndAlgorithm.LockTp == ast.LockTypeDefault && indexLockAndAlgorithm.AlgorithmTp == ast.AlgorithmTypeDefault {
indexLockAndAlgorithm = nil
}
}
$$ = &ast.DropIndexStmt{IfExists: $3.(bool), IndexName: $4, Table: $6.(*ast.TableName), LockAlg: indexLockAndAlgorithm}
}
DropTableStmt:
"DROP" TableOrTables TableNameList RestrictOrCascadeOpt
{
$$ = &ast.DropTableStmt{Tables: $3.([]*ast.TableName)}
}
| "DROP" TableOrTables "IF" "EXISTS" TableNameList RestrictOrCascadeOpt
{
$$ = &ast.DropTableStmt{IfExists: true, Tables: $5.([]*ast.TableName)}
}
DropViewStmt:
"DROP" "VIEW" "IF" "EXISTS" TableNameList
{
$$ = &ast.DoStmt{}
}
DropUserStmt:
"DROP" "USER" UsernameList
{
$$ = &ast.DropUserStmt{IfExists: false, UserList: $3.([]*auth.UserIdentity)}
}
| "DROP" "USER" "IF" "EXISTS" UsernameList
{
$$ = &ast.DropUserStmt{IfExists: true, UserList: $5.([]*auth.UserIdentity)}
}
DropStatsStmt:
"DROP" "STATS" TableName
{
$$ = &ast.DropStatsStmt{Table: $3.(*ast.TableName)}
}
RestrictOrCascadeOpt:
{}
| "RESTRICT"
| "CASCADE"
TableOrTables:
"TABLE"
| "TABLES"
EqOpt:
{}
| eq
EmptyStmt:
/* EMPTY */
{
$$ = nil
}
TraceStmt:
"TRACE" TraceableStmt
{
$$ = &ast.TraceStmt{
Stmt: $2,
Format: "row",
}
}
ExplainSym:
"EXPLAIN" | "DESCRIBE" | "DESC"
ExplainStmt:
ExplainSym TableName
{
$$ = &ast.ExplainStmt{
Stmt: &ast.ShowStmt{
Tp: ast.ShowColumns,
Table: $2.(*ast.TableName),
},
}
}
| ExplainSym TableName ColumnName
{
$$ = &ast.ExplainStmt{
Stmt: &ast.ShowStmt{
Tp: ast.ShowColumns,
Table: $2.(*ast.TableName),
Column: $3.(*ast.ColumnName),
},
}
}
| ExplainSym ExplainableStmt
{
$$ = &ast.ExplainStmt{
Stmt: $2,
Format: "row",
}
}
| ExplainSym "FORMAT" "=" stringLit ExplainableStmt
{
$$ = &ast.ExplainStmt{
Stmt: $5,
Format: $4,
}
}
| ExplainSym "ANALYZE" ExplainableStmt
{
$$ = &ast.ExplainStmt {
Stmt: $3,
Format: "row",
Analyze: true,
}
}
LengthNum:
NUM
{
$$ = getUint64FromNUM($1)
}
NUM:
intLit
Expression:
singleAtIdentifier assignmentEq Expression %prec assignmentEq
{
v := $1
v = strings.TrimPrefix(v, "@")
$$ = &ast.VariableExpr{
Name: v,
IsGlobal: false,
IsSystem: false,
Value: $3,
}
}
| Expression logOr Expression %prec pipes
{
$$ = &ast.BinaryOperationExpr{Op: opcode.LogicOr, L: $1, R: $3}
}
| Expression "XOR" Expression %prec xor
{
$$ = &ast.BinaryOperationExpr{Op: opcode.LogicXor, L: $1, R: $3}
}
| Expression logAnd Expression %prec andand
{
$$ = &ast.BinaryOperationExpr{Op: opcode.LogicAnd, L: $1, R: $3}
}
| "NOT" Expression %prec not
{
expr, ok := $2.(*ast.ExistsSubqueryExpr)
if ok {
expr.Not = true
$$ = $2
} else {
$$ = &ast.UnaryOperationExpr{Op: opcode.Not, V: $2}
}
}
| BoolPri IsOrNotOp trueKwd %prec is
{
$$ = &ast.IsTruthExpr{Expr:$1, Not: !$2.(bool), True: int64(1)}
}
| BoolPri IsOrNotOp falseKwd %prec is
{
$$ = &ast.IsTruthExpr{Expr:$1, Not: !$2.(bool), True: int64(0)}
}
| BoolPri IsOrNotOp "UNKNOWN" %prec is
{
/* https://dev.mysql.com/doc/refman/5.7/en/comparison-operators.html#operator_is */
$$ = &ast.IsNullExpr{Expr: $1, Not: !$2.(bool)}
}
| BoolPri
MaxValueOrExpression:
"MAXVALUE"
{
$$ = &ast.MaxValueExpr{}
}
| Expression
{
$$ = $1
}
logOr:
pipesAsOr
| "OR"
logAnd:
"&&" | "AND"
ExpressionList:
Expression
{
$$ = []ast.ExprNode{$1}
}
| ExpressionList ',' Expression
{
$$ = append($1.([]ast.ExprNode), $3)
}
MaxValueOrExpressionList:
MaxValueOrExpression
{
$$ = []ast.ExprNode{$1}
}
| MaxValueOrExpressionList ',' MaxValueOrExpression
{
$$ = append($1.([]ast.ExprNode), $3)
}
ExpressionListOpt:
{
$$ = []ast.ExprNode{}
}
| ExpressionList
FuncDatetimePrecListOpt:
{
$$ = []ast.ExprNode{}
}
| FuncDatetimePrecList
{
$$ = $1
}
FuncDatetimePrecList:
intLit
{
expr := ast.NewValueExpr($1)
$$ = []ast.ExprNode{expr}
}
BoolPri:
BoolPri IsOrNotOp "NULL" %prec is
{
$$ = &ast.IsNullExpr{Expr: $1, Not: !$2.(bool)}
}
| BoolPri CompareOp PredicateExpr %prec eq
{
$$ = &ast.BinaryOperationExpr{Op: $2.(opcode.Op), L: $1, R: $3}
}
| BoolPri CompareOp AnyOrAll SubSelect %prec eq
{
sq := $4.(*ast.SubqueryExpr)
sq.MultiRows = true
$$ = &ast.CompareSubqueryExpr{Op: $2.(opcode.Op), L: $1, R: sq, All: $3.(bool)}
}
| BoolPri CompareOp singleAtIdentifier assignmentEq PredicateExpr %prec assignmentEq
{
v := $3
v = strings.TrimPrefix(v, "@")
variable := &ast.VariableExpr{
Name: v,
IsGlobal: false,
IsSystem: false,
Value: $5,
}
$$ = &ast.BinaryOperationExpr{Op: $2.(opcode.Op), L: $1, R: variable}
}
| PredicateExpr
CompareOp:
">="
{
$$ = opcode.GE
}
| '>'
{
$$ = opcode.GT
}
| "<="
{
$$ = opcode.LE
}
| '<'
{
$$ = opcode.LT
}
| "!="
{
$$ = opcode.NE
}
| "<>"
{
$$ = opcode.NE
}
| "="
{
$$ = opcode.EQ
}
| "<=>"
{
$$ = opcode.NullEQ
}
BetweenOrNotOp:
"BETWEEN"
{
$$ = true
}
| "NOT" "BETWEEN"
{
$$ = false
}
IsOrNotOp:
"IS"
{
$$ = true
}
| "IS" "NOT"
{
$$ = false
}
InOrNotOp:
"IN"
{
$$ = true
}
| "NOT" "IN"
{
$$ = false
}
LikeOrNotOp:
"LIKE"
{
$$ = true
}
| "NOT" "LIKE"
{
$$ = false
}
RegexpOrNotOp:
RegexpSym
{
$$ = true
}
| "NOT" RegexpSym
{
$$ = false
}
AnyOrAll:
"ANY"
{
$$ = false
}
| "SOME"
{
$$ = false
}
| "ALL"
{
$$ = true
}
PredicateExpr:
BitExpr InOrNotOp '(' ExpressionList ')'
{
$$ = &ast.PatternInExpr{Expr: $1, Not: !$2.(bool), List: $4.([]ast.ExprNode)}
}
| BitExpr InOrNotOp SubSelect
{
sq := $3.(*ast.SubqueryExpr)
sq.MultiRows = true
$$ = &ast.PatternInExpr{Expr: $1, Not: !$2.(bool), Sel: sq}
}
| BitExpr BetweenOrNotOp BitExpr "AND" PredicateExpr
{
$$ = &ast.BetweenExpr{
Expr: $1,
Left: $3,
Right: $5,
Not: !$2.(bool),
}
}
| BitExpr LikeOrNotOp SimpleExpr LikeEscapeOpt
{
escape := $4.(string)
if len(escape) > 1 {
yylex.AppendError(ErrWrongArguments.GenWithStackByArgs("ESCAPE"))
return 1
} else if len(escape) == 0 {
escape = "\\"
}
$$ = &ast.PatternLikeExpr{
Expr: $1,
Pattern: $3,
Not: !$2.(bool),
Escape: escape[0],
}
}
| BitExpr RegexpOrNotOp SimpleExpr
{
$$ = &ast.PatternRegexpExpr{Expr: $1, Pattern: $3, Not: !$2.(bool)}
}
| BitExpr
RegexpSym:
"REGEXP" | "RLIKE"
LikeEscapeOpt:
%prec empty
{
$$ = "\\"
}
| "ESCAPE" stringLit
{
$$ = $2
}
Field:
'*'
{
$$ = &ast.SelectField{WildCard: &ast.WildCardField{}}
}
| Identifier '.' '*'
{
wildCard := &ast.WildCardField{Table: model.NewCIStr($1)}
$$ = &ast.SelectField{WildCard: wildCard}
}
| Identifier '.' Identifier '.' '*'
{
wildCard := &ast.WildCardField{Schema: model.NewCIStr($1), Table: model.NewCIStr($3)}
$$ = &ast.SelectField{WildCard: wildCard}
}
| Expression FieldAsNameOpt
{
expr := $1
asName := $2.(string)
$$ = &ast.SelectField{Expr: expr, AsName: model.NewCIStr(asName)}
}
| '{' Identifier Expression '}' FieldAsNameOpt
{
/*
* ODBC escape syntax.
* See https://dev.mysql.com/doc/refman/5.7/en/expressions.html
*/
expr := $3
asName := $5.(string)
$$ = &ast.SelectField{Expr: expr, AsName: model.NewCIStr(asName)}
}
FieldAsNameOpt:
/* EMPTY */
{
$$ = ""
}
| FieldAsName
{
$$ = $1
}
FieldAsName:
Identifier
{
$$ = $1
}
| "AS" Identifier
{
$$ = $2
}
| stringLit
{
$$ = $1
}
| "AS" stringLit
{
$$ = $2
}
FieldList:
Field
{
field := $1.(*ast.SelectField)
field.Offset = parser.startOffset(&yyS[yypt])
$$ = []*ast.SelectField{field}
}
| FieldList ',' Field
{
fl := $1.([]*ast.SelectField)
last := fl[len(fl)-1]
if last.Expr != nil && last.AsName.O == "" {
lastEnd := parser.endOffset(&yyS[yypt-1])
last.SetText(parser.src[last.Offset:lastEnd])
}
newField := $3.(*ast.SelectField)
newField.Offset = parser.startOffset(&yyS[yypt])
$$ = append(fl, newField)
}
GroupByClause:
"GROUP" "BY" ByList
{
$$ = &ast.GroupByClause{Items: $3.([]*ast.ByItem)}
}
HavingClause:
{
$$ = nil
}
| "HAVING" Expression
{
$$ = &ast.HavingClause{Expr: $2}
}
IfExists:
{
$$ = false
}
| "IF" "EXISTS"
{
$$ = true
}
IfNotExists:
{
$$ = false
}
| "IF" "NOT" "EXISTS"
{
$$ = true
}
IgnoreOptional:
{
$$ = false
}
| "IGNORE"
{
$$ = true
}
IndexName:
{
$$ = ""
}
| Identifier
{
//"index name"
$$ = $1
}
IndexOptionList:
{
$$ = nil
}
| IndexOptionList IndexOption
{
// Merge the options
if $1 == nil {
$$ = $2
} else {
opt1 := $1.(*ast.IndexOption)
opt2 := $2.(*ast.IndexOption)
if len(opt2.Comment) > 0 {
opt1.Comment = opt2.Comment
} else if opt2.Tp != 0 {
opt1.Tp = opt2.Tp
}
$$ = opt1
}
}
IndexOption:
"KEY_BLOCK_SIZE" EqOpt LengthNum
{
$$ = &ast.IndexOption{
KeyBlockSize: $3.(uint64),
}
}
| IndexType
{
$$ = &ast.IndexOption {
Tp: $1.(model.IndexType),
}
}
| "COMMENT" stringLit
{
$$ = &ast.IndexOption {
Comment: $2,
}
}
| IndexInvisible
{
$$ = &ast.IndexOption {
Visibility: $1.(ast.IndexVisibility),
}
}
/*
See: https://github.com/mysql/mysql-server/blob/8.0/sql/sql_yacc.yy#L7179
The syntax for defining an index is:
... INDEX [index_name] [USING|TYPE] <index_type> ...
The problem is that whereas USING is a reserved word, TYPE is not. We can
still handle it if an index name is supplied, i.e.:
... INDEX type TYPE <index_type> ...
here the index's name is unmbiguously 'type', but for this:
... INDEX TYPE <index_type> ...
it's impossible to know what this actually mean - is 'type' the name or the
type? For this reason we accept the TYPE syntax only if a name is supplied.
*/
IndexNameAndTypeOpt:
IndexName
{
$$ = []interface{}{$1, nil}
}
| IndexName "USING" IndexTypeName
{
$$ = []interface{}{$1, $3}
}
| Identifier "TYPE" IndexTypeName
{
$$ = []interface{}{$1, $3}
}
IndexTypeOpt:
{
$$ = nil
}
| IndexType
{
$$ = $1
}
IndexType:
"USING" IndexTypeName
{
$$ = $2
}
| "TYPE" IndexTypeName
{
$$ = $2
}
IndexTypeName:
"BTREE"
{
$$ = model.IndexTypeBtree
}
| "HASH"
{
$$ = model.IndexTypeHash
}
| "RTREE"
{
$$ = model.IndexTypeRtree
}
IndexInvisible:
"VISIBLE"
{
$$ = ast.IndexVisibilityVisible
}
| "INVISIBLE"
{
$$ = ast.IndexVisibilityInvisible
}
/**********************************Identifier********************************************/
Identifier:
identifier | UnReservedKeyword | NotKeywordToken | TiDBKeyword
UnReservedKeyword:
"ACTION" | "ASCII" | "AUTO_INCREMENT" | "AFTER" | "ALWAYS" | "AVG" | "BEGIN" | "BIT" | "BOOL" | "BOOLEAN" | "BTREE" | "BYTE" | "CLEANUP" | "CHARSET"
| "COLUMNS" | "COMMIT" | "CURRENT" |"INCEPTION" | "INCEPTION_MAGIC_START" | "INCEPTION_MAGIC_COMMIT" | "OSC" | "OSC_PERCENT" | "STOP" | "PAUSE" | "RESUME" | "COMPACT" | "COMPRESSED" | "CONSISTENT" | "DATA" | "DATE" %prec lowerThanStringLitToken| "DATETIME" | "DAY" | "DEALLOCATE" | "DO" | "DUPLICATE"
| "DYNAMIC"| "END" | "ENGINE" | "ENGINES" | "ENUM" | "ERRORS" | "ESCAPE" | "EXECUTE" | "FIELDS" | "FIRST" | "FIXED" | "FLUSH" | "FORMAT" | "FULL" |"GLOBAL"
| "HASH" | "HOUR" | "LESS" | "LIST" | "LOCAL" | "NAMES" | "OFFSET" | "PASSWORD" %prec lowerThanEq | "PREPARE" | "QUICK" | "REDUNDANT"
| "ROLLBACK" | "SESSION" | "SIGNED" | "SNAPSHOT" | "START" | "STATUS" | "SUBPARTITIONS" | "SUBPARTITION" | "SYSTEM_TIME" | "TABLES" | "TABLESPACE" | "TEXT" | "THAN" | "TIME" %prec lowerThanStringLitToken
| "TIMESTAMP" %prec lowerThanStringLitToken | "TRACE" | "TRANSACTION" | "TRUNCATE" | "UNKNOWN" | "VALUE" | "WARNINGS" | "YEAR" | "MODE" | "WEEK" | "ANY" | "SOME" | "USER" | "IDENTIFIED"
| "COLLATION" | "COMMENT" | "AVG_ROW_LENGTH" | "CONNECTION" | "CHECKSUM" | "COMPRESSION" | "KEY_BLOCK_SIZE" | "MASTER" | "MAX_ROWS"
| "MIN_ROWS" | "NATIONAL" | "ROW" | "ROW_FORMAT" | "QUARTER" | "GRANTS" | "TRIGGERS" | "DELAY_KEY_WRITE" | "ISOLATION" | "JSON"
| "REPEATABLE" | "COMMITTED" | "UNCOMMITTED" | "ONLY" | "SERIALIZABLE" | "LEVEL" | "VARIABLES" | "LEVELS" | "SQL_CACHE" | "INDEXES" | "PROCESSLIST"
| "SQL_NO_CACHE" | "DISABLE" | "ENABLE" | "REVERSE" | "PRIVILEGES" | "NO" | "BINLOG" | "FUNCTION" | "VIEW" | "MODIFY" | "EVENTS" | "PARTITIONS"
| "NONE" | "SUPER" | "EXCLUSIVE" | "STATS_PERSISTENT" | "ROW_COUNT" | "COALESCE" | "MONTH" | "PROCESS" | "PROFILES"
| "MICROSECOND" | "MINUTE" | "PLUGINS" | "QUERY" | "QUERIES" | "SECOND" | "SEPARATOR" | "SHARE" | "SHARED" | "SLOW" | "MAX_CONNECTIONS_PER_HOUR" | "MAX_QUERIES_PER_HOUR" | "MAX_UPDATES_PER_HOUR"
| "MAX_USER_CONNECTIONS" | "REPLICATION" | "CLIENT" | "SLAVE" | "RELOAD" | "TEMPORARY" | "ROUTINE" | "EVENT" | "ALGORITHM" | "DEFINER" | "INVOKER" | "MERGE" | "TEMPTABLE" | "UNDEFINED" | "SECURITY" | "CASCADED" | "RECOVER" | "HISTORY" | "DIRECTORY" | "NODEGROUP"
| "RTREE" | "INVISIBLE" | "VISIBLE" | "TYPE"
TiDBKeyword:
"ADMIN" | "BUCKETS" | "CANCEL" | "DDL" | "JOBS" | "JOB" | "STATS" | "STATS_META" | "STATS_HISTOGRAMS" | "STATS_BUCKETS" | "STATS_HEALTHY" | "TIDB" | "TIDB_HJ" | "TIDB_SMJ" | "TIDB_INLJ"
NotKeywordToken:
"ADDDATE" | "BIT_AND" | "BIT_OR" | "BIT_XOR" | "CAST" | "COPY" | "COUNT" | "CURTIME" | "DATE_ADD" | "DATE_SUB" | "EXTRACT" | "GET_FORMAT" | "GROUP_CONCAT" | "INPLACE" | "INTERNAL"
| "INSTANT" |"MIN" | "MAX" | "MAX_EXECUTION_TIME" | "NOW" | "RECENT" | "POSITION" | "SUBDATE" | "SUBSTRING" | "SUM" | "TIMESTAMPADD" | "TIMESTAMPDIFF" | "TOP" | "TRIM"
/************************************************************************************
*
* Insert Statements
*
* TODO: support PARTITION
**********************************************************************************/
InsertIntoStmt:
"INSERT" PriorityOpt IgnoreOptional IntoOpt TableName InsertValues OnDuplicateKeyUpdate
{
x := $6.(*ast.InsertStmt)
x.Priority = $2.(mysql.PriorityEnum)
x.IgnoreErr = $3.(bool)
// Wraps many layers here so that it can be processed the same way as select statement.
ts := &ast.TableSource{Source: $5.(*ast.TableName)}
x.Table = &ast.TableRefsClause{TableRefs: &ast.Join{Left: ts}}
if $7 != nil {
x.OnDuplicate = $7.([]*ast.Assignment)
}
$$ = x
}
IntoOpt:
{}
| "INTO"
InsertValues:
'(' ColumnNameListOpt ')' ValueSym ValuesList
{
$$ = &ast.InsertStmt{
Columns: $2.([]*ast.ColumnName),
Lists: $5.([][]ast.ExprNode),
}
}
| '(' ColumnNameListOpt ')' SelectStmt
{
$$ = &ast.InsertStmt{Columns: $2.([]*ast.ColumnName), Select: $4.(*ast.SelectStmt)}
}
| '(' ColumnNameListOpt ')' '(' SelectStmt ')'
{
$$ = &ast.InsertStmt{Columns: $2.([]*ast.ColumnName), Select: $5.(*ast.SelectStmt)}
}
| '(' ColumnNameListOpt ')' UnionStmt
{
$$ = &ast.InsertStmt{Columns: $2.([]*ast.ColumnName), Select: $4.(*ast.UnionStmt)}
}
| ValueSym ValuesList %prec insertValues
{
$$ = &ast.InsertStmt{Lists: $2.([][]ast.ExprNode)}
}
| '(' SelectStmt ')'
{
$$ = &ast.InsertStmt{Select: $2.(*ast.SelectStmt)}
}
| SelectStmt
{
$$ = &ast.InsertStmt{Select: $1.(*ast.SelectStmt)}
}
| UnionStmt
{
$$ = &ast.InsertStmt{Select: $1.(*ast.UnionStmt)}
}
| "SET" ColumnSetValueList
{
$$ = &ast.InsertStmt{Setlist: $2.([]*ast.Assignment)}
}
ValueSym:
"VALUE" | "VALUES"
ValuesList:
RowValue
{
$$ = [][]ast.ExprNode{$1.([]ast.ExprNode)}
}
| ValuesList ',' RowValue
{
$$ = append($1.([][]ast.ExprNode), $3.([]ast.ExprNode))
}
RowValue:
'(' ValuesOpt ')'
{
$$ = $2
}
ValuesOpt:
{
$$ = []ast.ExprNode{}
}
| Values
Values:
Values ',' ExprOrDefault
{
$$ = append($1.([]ast.ExprNode), $3)
}
| ExprOrDefault
{
$$ = []ast.ExprNode{$1}
}
ExprOrDefault:
Expression
| "DEFAULT"
{
$$ = &ast.DefaultExpr{}
}
ColumnSetValue:
ColumnName eq Expression
{
$$ = &ast.Assignment{
Column: $1.(*ast.ColumnName),
Expr: $3,
}
}
ColumnSetValueList:
{
$$ = []*ast.Assignment{}
}
| ColumnSetValue
{
$$ = []*ast.Assignment{$1.(*ast.Assignment)}
}
| ColumnSetValueList ',' ColumnSetValue
{
$$ = append($1.([]*ast.Assignment), $3.(*ast.Assignment))
}
/*
* ON DUPLICATE KEY UPDATE col_name=expr [, col_name=expr] ...
* See https://dev.mysql.com/doc/refman/5.7/en/insert-on-duplicate.html
*/
OnDuplicateKeyUpdate:
{
$$ = nil
}
| "ON" "DUPLICATE" "KEY" "UPDATE" AssignmentList
{
$$ = $5
}
/***********************************Insert Statements END************************************/
/************************************************************************************
* Replace Statements
* See https://dev.mysql.com/doc/refman/5.7/en/replace.html
*
* TODO: support PARTITION
**********************************************************************************/
ReplaceIntoStmt:
"REPLACE" PriorityOpt IntoOpt TableName InsertValues
{
x := $5.(*ast.InsertStmt)
x.IsReplace = true
x.Priority = $2.(mysql.PriorityEnum)
ts := &ast.TableSource{Source: $4.(*ast.TableName)}
x.Table = &ast.TableRefsClause{TableRefs: &ast.Join{Left: ts}}
$$ = x
}
/***********************************Replace Statements END************************************/
ODBCDateTimeType:
"d"
{
$$ = ast.DateLiteral
}
| "t"
{
$$ = ast.TimeLiteral
}
| "ts"
{
$$ = ast.TimestampLiteral
}
Literal:
"FALSE"
{
$$ = ast.NewValueExpr(false)
}
| "NULL"
{
$$ = ast.NewValueExpr(nil)
}
| "TRUE"
{
$$ = ast.NewValueExpr(true)
}
| floatLit
{
$$ = ast.NewValueExpr($1)
}
| decLit
{
$$ = ast.NewValueExpr($1)
}
| intLit
{
$$ = ast.NewValueExpr($1)
}
| StringLiteral %prec lowerThanStringLitToken
{
$$ = $1
}
| "UNDERSCORE_CHARSET" stringLit
{
// See https://dev.mysql.com/doc/refman/5.7/en/charset-literal.html
co, err := charset.GetDefaultCollation($1)
if err != nil {
yylex.AppendError(yylex.Errorf("Get collation error for charset: %s", $1))
return 1
}
expr := ast.NewValueExpr($2)
tp := expr.GetType()
tp.Charset = $1
tp.Collate = co
if tp.Collate == charset.CollationBin {
tp.Flag |= mysql.BinaryFlag
}
$$ = expr
}
| hexLit
{
$$ = ast.NewValueExpr($1)
}
| bitLit
{
$$ = ast.NewValueExpr($1)
}
StringLiteral:
stringLit
{
expr := ast.NewValueExpr($1)
$$ = expr
}
| StringLiteral stringLit
{
valExpr := $1.(*ast.ValueExpr)
strLit := valExpr.GetString()
expr := ast.NewValueExpr(strLit+$2)
// Fix #4239, use first string literal as projection name.
if valExpr.GetProjectionOffset() >= 0 {
expr.SetProjectionOffset(valExpr.GetProjectionOffset())
} else {
expr.SetProjectionOffset(len(strLit))
}
$$ = expr
}
OrderBy:
"ORDER" "BY" ByList
{
$$ = &ast.OrderByClause{Items: $3.([]*ast.ByItem)}
}
ByList:
ByItem
{
$$ = []*ast.ByItem{$1.(*ast.ByItem)}
}
| ByList ',' ByItem
{
$$ = append($1.([]*ast.ByItem), $3.(*ast.ByItem))
}
ByItem:
Expression Order
{
expr := $1
valueExpr, ok := expr.(*ast.ValueExpr)
if ok {
position, isPosition := valueExpr.GetValue().(int64)
if isPosition {
expr = &ast.PositionExpr{N: int(position)}
}
}
$$ = &ast.ByItem{Expr: expr, Desc: $2.(bool)}
}
Order:
/* EMPTY */
{
$$ = false // ASC by default
}
| "ASC"
{
$$ = false
}
| "DESC"
{
$$ = true
}
OrderByOptional:
{
$$ = nil
}
| OrderBy
{
$$ = $1
}
BitExpr:
BitExpr '|' BitExpr %prec '|'
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Or, L: $1, R: $3}
}
| BitExpr '&' BitExpr %prec '&'
{
$$ = &ast.BinaryOperationExpr{Op: opcode.And, L: $1, R: $3}
}
| BitExpr "<<" BitExpr %prec lsh
{
$$ = &ast.BinaryOperationExpr{Op: opcode.LeftShift, L: $1, R: $3}
}
| BitExpr ">>" BitExpr %prec rsh
{
$$ = &ast.BinaryOperationExpr{Op: opcode.RightShift, L: $1, R: $3}
}
| BitExpr '+' BitExpr %prec '+'
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Plus, L: $1, R: $3}
}
| BitExpr '-' BitExpr %prec '-'
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Minus, L: $1, R: $3}
}
| BitExpr '+' "INTERVAL" Expression TimeUnit %prec '+'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr("DATE_ADD"),
Args: []ast.ExprNode{
$1,
$4,
ast.NewValueExpr($5),
},
}
}
| BitExpr '-' "INTERVAL" Expression TimeUnit %prec '+'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr("DATE_SUB"),
Args: []ast.ExprNode{
$1,
$4,
ast.NewValueExpr($5),
},
}
}
| BitExpr '*' BitExpr %prec '*'
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Mul, L: $1, R: $3}
}
| BitExpr '/' BitExpr %prec '/'
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Div, L: $1, R: $3}
}
| BitExpr '%' BitExpr %prec '%'
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Mod, L: $1, R: $3}
}
| BitExpr "DIV" BitExpr %prec div
{
$$ = &ast.BinaryOperationExpr{Op: opcode.IntDiv, L: $1, R: $3}
}
| BitExpr "MOD" BitExpr %prec mod
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Mod, L: $1, R: $3}
}
| BitExpr '^' BitExpr
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Xor, L: $1, R: $3}
}
| SimpleExpr
SimpleIdent:
Identifier
{
$$ = &ast.ColumnNameExpr{Name: &ast.ColumnName{
Name: model.NewCIStr($1),
}}
}
| Identifier '.' Identifier
{
$$ = &ast.ColumnNameExpr{Name: &ast.ColumnName{
Table: model.NewCIStr($1),
Name: model.NewCIStr($3),
}}
}
| '.' Identifier '.' Identifier
{
$$ = &ast.ColumnNameExpr{Name: &ast.ColumnName{
Table: model.NewCIStr($2),
Name: model.NewCIStr($4),
}}
}
| Identifier '.' Identifier '.' Identifier
{
$$ = &ast.ColumnNameExpr{Name: &ast.ColumnName{
Schema: model.NewCIStr($1),
Table: model.NewCIStr($3),
Name: model.NewCIStr($5),
}}
}
SimpleExpr:
SimpleIdent
| FunctionCallKeyword
| FunctionCallNonKeyword
| FunctionCallGeneric
| SimpleExpr "COLLATE" StringName %prec neg
{
// TODO: Create a builtin function hold expr and collation. When do evaluation, convert expr result using the collation.
$$ = $1
}
| Literal
| paramMarker
{
$$ = &ast.ParamMarkerExpr{
Offset: yyS[yypt].offset,
}
}
| Variable
| SumExpr
| '!' SimpleExpr %prec neg
{
$$ = &ast.UnaryOperationExpr{Op: opcode.Not, V: $2}
}
| '~' SimpleExpr %prec neg
{
$$ = &ast.UnaryOperationExpr{Op: opcode.BitNeg, V: $2}
}
| '-' SimpleExpr %prec neg
{
$$ = &ast.UnaryOperationExpr{Op: opcode.Minus, V: $2}
}
| '+' SimpleExpr %prec neg
{
$$ = &ast.UnaryOperationExpr{Op: opcode.Plus, V: $2}
}
| SimpleExpr pipes SimpleExpr
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr(ast.Concat), Args: []ast.ExprNode{$1, $3}}
}
| not2 SimpleExpr %prec neg
{
$$ = &ast.UnaryOperationExpr{Op: opcode.Not, V: $2}
}
| SubSelect
| '(' Expression ')' {
startOffset := parser.startOffset(&yyS[yypt-1])
endOffset := parser.endOffset(&yyS[yypt])
expr := $2
expr.SetText(parser.src[startOffset:endOffset])
$$ = &ast.ParenthesesExpr{Expr: expr}
}
| '(' ExpressionList ',' Expression ')'
{
values := append($2.([]ast.ExprNode), $4)
$$ = &ast.RowExpr{Values: values}
}
| "ROW" '(' ExpressionList ',' Expression ')'
{
values := append($3.([]ast.ExprNode), $5)
$$ = &ast.RowExpr{Values: values}
}
| "EXISTS" SubSelect
{
sq := $2.(*ast.SubqueryExpr)
sq.Exists = true
$$ = &ast.ExistsSubqueryExpr{Sel: sq}
}
| "BINARY" SimpleExpr %prec neg
{
// See https://dev.mysql.com/doc/refman/5.7/en/cast-functions.html#operator_binary
x := types.NewFieldType(mysql.TypeString)
x.Charset = charset.CharsetBin
x.Collate = charset.CharsetBin
$$ = &ast.FuncCastExpr{
Expr: $2,
Tp: x,
FunctionType: ast.CastBinaryOperator,
}
}
| builtinCast '(' Expression "AS" CastType ')'
{
/* See https://dev.mysql.com/doc/refman/5.7/en/cast-functions.html#function_cast */
tp := $5.(*types.FieldType)
defaultFlen, defaultDecimal := mysql.GetDefaultFieldLengthAndDecimalForCast(tp.Tp)
if tp.Flen == types.UnspecifiedLength {
tp.Flen = defaultFlen
}
if tp.Decimal == types.UnspecifiedLength {
tp.Decimal = defaultDecimal
}
$$ = &ast.FuncCastExpr{
Expr: $3,
Tp: tp,
FunctionType: ast.CastFunction,
}
}
| "CASE" ExpressionOpt WhenClauseList ElseOpt "END"
{
x := &ast.CaseExpr{WhenClauses: $3.([]*ast.WhenClause)}
if $2 != nil {
x.Value = $2
}
if $4 != nil {
x.ElseClause = $4.(ast.ExprNode)
}
$$ = x
}
| "CONVERT" '(' Expression ',' CastType ')'
{
// See https://dev.mysql.com/doc/refman/5.7/en/cast-functions.html#function_convert
tp := $5.(*types.FieldType)
defaultFlen, defaultDecimal := mysql.GetDefaultFieldLengthAndDecimalForCast(tp.Tp)
if tp.Flen == types.UnspecifiedLength {
tp.Flen = defaultFlen
}
if tp.Decimal == types.UnspecifiedLength {
tp.Decimal = defaultDecimal
}
$$ = &ast.FuncCastExpr{
Expr: $3,
Tp: tp,
FunctionType: ast.CastConvertFunction,
}
}
| "CONVERT" '(' Expression "USING" StringName ')'
{
// See https://dev.mysql.com/doc/refman/5.7/en/cast-functions.html#function_convert
charset1 := ast.NewValueExpr($5)
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$3, charset1},
}
}
| "DEFAULT" '(' SimpleIdent ')'
{
$$ = &ast.DefaultExpr{Name: $3.(*ast.ColumnNameExpr).Name}
}
| "VALUES" '(' SimpleIdent ')' %prec lowerThanInsertValues
{
$$ = &ast.ValuesExpr{Column: $3.(*ast.ColumnNameExpr)}
}
| SimpleIdent jss stringLit
{
expr := ast.NewValueExpr($3)
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr(ast.JSONExtract), Args: []ast.ExprNode{$1, expr}}
}
| SimpleIdent juss stringLit
{
expr := ast.NewValueExpr($3)
extract := &ast.FuncCallExpr{FnName: model.NewCIStr(ast.JSONExtract), Args: []ast.ExprNode{$1, expr}}
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr(ast.JSONUnquote), Args: []ast.ExprNode{extract}}
}
DistinctKwd:
"DISTINCT"
| "DISTINCTROW"
DistinctOpt:
"ALL"
{
$$ = false
}
| DistinctKwd
{
$$ = true
}
DefaultFalseDistinctOpt:
{
$$ = false
}
| DistinctOpt
DefaultTrueDistinctOpt:
{
$$ = true
}
| DistinctOpt
BuggyDefaultFalseDistinctOpt:
DefaultFalseDistinctOpt
| DistinctKwd "ALL"
{
$$ = true
}
FunctionNameConflict:
"ASCII"
| "CHARSET"
| "COALESCE"
| "COLLATION"
| "DATE"
| "DATABASE"
| "DAY"
| "HOUR"
| "IF"
| "INTERVAL" %prec lowerThanIntervalKeyword
| "FORMAT"
| "LEFT"
| "MICROSECOND"
| "MINUTE"
| "MONTH"
| builtinNow
| "QUARTER"
| "REPEAT"
| "REPLACE"
| "REVERSE"
| "RIGHT"
| "ROW_COUNT"
| "SECOND"
| "TIME"
| "TIMESTAMP"
| "TRUNCATE"
| "USER"
| "WEEK"
| "YEAR"
OptionalBraces:
{} | '(' ')' {}
FunctionNameOptionalBraces:
"CURRENT_USER"
| "CURRENT_DATE"
| "UTC_DATE"
FunctionNameDatetimePrecision:
"CURRENT_TIME"
| "CURRENT_TIMESTAMP"
| "LOCALTIME"
| "LOCALTIMESTAMP"
| "UTC_TIME"
| "UTC_TIMESTAMP"
FunctionCallKeyword:
FunctionNameConflict '(' ExpressionListOpt ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1), Args: $3.([]ast.ExprNode)}
}
| builtinUser '(' ExpressionListOpt ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1), Args: $3.([]ast.ExprNode)}
}
| FunctionNameOptionalBraces OptionalBraces
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1)}
}
| builtinCurDate '(' ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1)}
}
| FunctionNameDatetimePrecision FuncDatetimePrec
{
args := []ast.ExprNode{}
if $2 != nil {
args = append(args, $2.(ast.ExprNode))
}
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1), Args: args}
}
| "CHAR" '(' ExpressionList ')'
{
nilVal := ast.NewValueExpr(nil)
args := $3.([]ast.ExprNode)
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr(ast.CharFunc),
Args: append(args, nilVal),
}
}
| "CHAR" '(' ExpressionList "USING" StringName ')'
{
charset1 := ast.NewValueExpr($5)
args := $3.([]ast.ExprNode)
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr(ast.CharFunc),
Args: append(args, charset1),
}
}
| "DATE" stringLit
{
expr := ast.NewValueExpr($2)
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr(ast.DateLiteral), Args: []ast.ExprNode{expr}}
}
| "TIME" stringLit
{
expr := ast.NewValueExpr($2)
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr(ast.TimeLiteral), Args: []ast.ExprNode{expr}}
}
| "TIMESTAMP" stringLit
{
expr := ast.NewValueExpr($2)
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr(ast.TimestampLiteral), Args: []ast.ExprNode{expr}}
}
| "INSERT" '(' ExpressionListOpt ')'
{
$$ = &ast.FuncCallExpr{FnName:model.NewCIStr(ast.InsertFunc), Args: $3.([]ast.ExprNode)}
}
| "MOD" '(' BitExpr ',' BitExpr ')'
{
$$ = &ast.BinaryOperationExpr{Op: opcode.Mod, L: $3, R: $5}
}
| "PASSWORD" '(' ExpressionListOpt ')'
{
$$ = &ast.FuncCallExpr{FnName:model.NewCIStr(ast.PasswordFunc), Args: $3.([]ast.ExprNode)}
}
| '{' ODBCDateTimeType stringLit '}'
{
// This is ODBC syntax for date and time literals.
// See: https://dev.mysql.com/doc/refman/5.7/en/date-and-time-literals.html
expr := ast.NewValueExpr($3)
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($2), Args: []ast.ExprNode{expr}}
}
FunctionCallNonKeyword:
builtinCurTime '(' FuncDatetimePrecListOpt ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1), Args: $3.([]ast.ExprNode)}
}
| builtinSysDate '(' FuncDatetimePrecListOpt ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1), Args: $3.([]ast.ExprNode)}
}
| FunctionNameDateArithMultiForms '(' Expression ',' Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{
$3,
$5,
ast.NewValueExpr("DAY"),
},
}
}
| FunctionNameDateArithMultiForms '(' Expression ',' "INTERVAL" Expression TimeUnit ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{
$3,
$6,
ast.NewValueExpr($7),
},
}
}
| FunctionNameDateArith '(' Expression ',' "INTERVAL" Expression TimeUnit ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{
$3,
$6,
ast.NewValueExpr($7),
},
}
}
| builtinExtract '(' TimeUnit "FROM" Expression ')'
{
timeUnit := ast.NewValueExpr($3)
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{timeUnit, $5},
}
}
| "GET_FORMAT" '(' GetFormatSelector ',' Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{ast.NewValueExpr($3), $5},
}
}
| builtinPosition '(' BitExpr "IN" Expression ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1), Args: []ast.ExprNode{$3, $5}}
}
| builtinSubstring '(' Expression ',' Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$3, $5},
}
}
| builtinSubstring '(' Expression "FROM" Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$3, $5},
}
}
| builtinSubstring '(' Expression ',' Expression ',' Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$3, $5, $7},
}
}
| builtinSubstring '(' Expression "FROM" Expression "FOR" Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$3, $5, $7},
}
}
| "TIMESTAMPADD" '(' TimestampUnit ',' Expression ',' Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{ast.NewValueExpr($3), $5, $7},
}
}
| "TIMESTAMPDIFF" '(' TimestampUnit ',' Expression ',' Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{ast.NewValueExpr($3), $5, $7},
}
}
| builtinTrim '(' Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$3},
}
}
| builtinTrim '(' Expression "FROM" Expression ')'
{
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$5, $3},
}
}
| builtinTrim '(' TrimDirection "FROM" Expression ')'
{
nilVal := ast.NewValueExpr(nil)
direction := ast.NewValueExpr(int($3.(ast.TrimDirectionType)))
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$5, nilVal, direction},
}
}
| builtinTrim '(' TrimDirection Expression "FROM" Expression ')'
{
direction := ast.NewValueExpr(int($3.(ast.TrimDirectionType)))
$$ = &ast.FuncCallExpr{
FnName: model.NewCIStr($1),
Args: []ast.ExprNode{$6, $4, direction},
}
}
GetFormatSelector:
"DATE"
{
$$ = strings.ToUpper($1)
}
| "DATETIME"
{
$$ = strings.ToUpper($1)
}
| "TIME"
{
$$ = strings.ToUpper($1)
}
| "TIMESTAMP"
{
$$ = strings.ToUpper($1)
}
FunctionNameDateArith:
builtinDateAdd
| builtinDateSub
FunctionNameDateArithMultiForms:
builtinAddDate
| builtinSubDate
TrimDirection:
"BOTH"
{
$$ = ast.TrimBoth
}
| "LEADING"
{
$$ = ast.TrimLeading
}
| "TRAILING"
{
$$ = ast.TrimTrailing
}
SumExpr:
"AVG" '(' BuggyDefaultFalseDistinctOpt Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$4}, Distinct: $3.(bool)}
}
| builtinBitAnd '(' Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$3}}
}
| builtinBitAnd '(' "ALL" Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$4}}
}
| builtinBitOr '(' Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$3}}
}
| builtinBitOr '(' "ALL" Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$4}}
}
| builtinBitXor '(' Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$3}}
}
| builtinBitXor '(' "ALL" Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$4}}
}
| builtinCount '(' DistinctKwd ExpressionList ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: $4.([]ast.ExprNode), Distinct: true}
}
| builtinCount '(' "ALL" Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$4}}
}
| builtinCount '(' Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$3}}
}
| builtinCount '(' '*' ')'
{
args := []ast.ExprNode{ast.NewValueExpr(1)}
$$ = &ast.AggregateFuncExpr{F: $1, Args: args}
}
| builtinGroupConcat '(' BuggyDefaultFalseDistinctOpt ExpressionList OrderByOptional OptGConcatSeparator ')'
{
args := $4.([]ast.ExprNode)
args = append(args, $6.(ast.ExprNode))
$$ = &ast.AggregateFuncExpr{F: $1, Args: args, Distinct: $3.(bool)}
}
| builtinMax '(' BuggyDefaultFalseDistinctOpt Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$4}, Distinct: $3.(bool)}
}
| builtinMin '(' BuggyDefaultFalseDistinctOpt Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$4}, Distinct: $3.(bool)}
}
| builtinSum '(' BuggyDefaultFalseDistinctOpt Expression ')'
{
$$ = &ast.AggregateFuncExpr{F: $1, Args: []ast.ExprNode{$4}, Distinct: $3.(bool)}
}
OptGConcatSeparator:
{
$$ = ast.NewValueExpr(",")
}
| "SEPARATOR" stringLit
{
$$ = ast.NewValueExpr($2)
}
FunctionCallGeneric:
identifier '(' ExpressionListOpt ')'
{
$$ = &ast.FuncCallExpr{FnName: model.NewCIStr($1), Args: $3.([]ast.ExprNode)}
}
FuncDatetimePrec:
{
$$ = nil
}
| '(' ')'
{
$$ = nil
}
| '(' intLit ')'
{
expr := ast.NewValueExpr($2)
$$ = expr
}
TimeUnit:
"MICROSECOND"
{
$$ = strings.ToUpper($1)
}
| "SECOND"
{
$$ = strings.ToUpper($1)
}
| "MINUTE"
{
$$ = strings.ToUpper($1)
}
| "HOUR"
{
$$ = strings.ToUpper($1)
}
| "DAY"
{
$$ = strings.ToUpper($1)
}
| "WEEK"
{
$$ = strings.ToUpper($1)
}
| "MONTH"
{
$$ = strings.ToUpper($1)
}
| "QUARTER"
{
$$ = strings.ToUpper($1)
}
| "YEAR"
{
$$ = strings.ToUpper($1)
}
| "SECOND_MICROSECOND"
{
$$ = strings.ToUpper($1)
}
| "MINUTE_MICROSECOND"
{
$$ = strings.ToUpper($1)
}
| "MINUTE_SECOND"
{
$$ = strings.ToUpper($1)
}
| "HOUR_MICROSECOND"
{
$$ = strings.ToUpper($1)
}
| "HOUR_SECOND"
{
$$ = strings.ToUpper($1)
}
| "HOUR_MINUTE"
{
$$ = strings.ToUpper($1)
}
| "DAY_MICROSECOND"
{
$$ = strings.ToUpper($1)
}
| "DAY_SECOND"
{
$$ = strings.ToUpper($1)
}
| "DAY_MINUTE"
{
$$ = strings.ToUpper($1)
}
| "DAY_HOUR"
{
$$ = strings.ToUpper($1)
}
| "YEAR_MONTH"
{
$$ = strings.ToUpper($1)
}
TimestampUnit:
"MICROSECOND"
{
$$ = strings.ToUpper($1)
}
| "SECOND"
{
$$ = strings.ToUpper($1)
}
| "MINUTE"
{
$$ = strings.ToUpper($1)
}
| "HOUR"
{
$$ = strings.ToUpper($1)
}
| "DAY"
{
$$ = strings.ToUpper($1)
}
| "WEEK"
{
$$ = strings.ToUpper($1)
}
| "MONTH"
{
$$ = strings.ToUpper($1)
}
| "QUARTER"
{
$$ = strings.ToUpper($1)
}
| "YEAR"
{
$$ = strings.ToUpper($1)
}
ExpressionOpt:
{
$$ = nil
}
| Expression
{
$$ = $1
}
WhenClauseList:
WhenClause
{
$$ = []*ast.WhenClause{$1.(*ast.WhenClause)}
}
| WhenClauseList WhenClause
{
$$ = append($1.([]*ast.WhenClause), $2.(*ast.WhenClause))
}
WhenClause:
"WHEN" Expression "THEN" Expression
{
$$ = &ast.WhenClause{
Expr: $2,
Result: $4,
}
}
ElseOpt:
/* empty */
{
$$ = nil
}
| "ELSE" Expression
{
$$ = $2
}
CastType:
"BINARY" OptFieldLen
{
x := types.NewFieldType(mysql.TypeVarString)
x.Flen = $2.(int) // TODO: Flen should be the flen of expression
if x.Flen != types.UnspecifiedLength {
x.Tp = mysql.TypeString
}
x.Charset = charset.CharsetBin
x.Collate = charset.CollationBin
x.Flag |= mysql.BinaryFlag
$$ = x
}
| "CHAR" OptFieldLen OptBinary
{
x := types.NewFieldType(mysql.TypeVarString)
x.Flen = $2.(int) // TODO: Flen should be the flen of expression
x.Charset = $3.(*ast.OptBinary).Charset
if $3.(*ast.OptBinary).IsBinary{
x.Flag |= mysql.BinaryFlag
}
if x.Charset == "" {
x.Charset = mysql.DefaultCharset
x.Collate = mysql.DefaultCollationName
}
$$ = x
}
| "DATE"
{
x := types.NewFieldType(mysql.TypeDate)
x.Charset = charset.CharsetBin
x.Collate = charset.CollationBin
x.Flag |= mysql.BinaryFlag
$$ = x
}
| "DATETIME" OptFieldLen
{
x := types.NewFieldType(mysql.TypeDatetime)
x.Flen, _ = mysql.GetDefaultFieldLengthAndDecimalForCast(mysql.TypeDatetime)
x.Decimal = $2.(int)
if x.Decimal > 0 {
x.Flen = x.Flen + 1 + x.Decimal
}
x.Charset = charset.CharsetBin
x.Collate = charset.CollationBin
x.Flag |= mysql.BinaryFlag
$$ = x
}
| "DECIMAL" FloatOpt
{
fopt := $2.(*ast.FloatOpt)
x := types.NewFieldType(mysql.TypeNewDecimal)
x.Flen = fopt.Flen
x.Decimal = fopt.Decimal
x.Charset = charset.CharsetBin
x.Collate = charset.CollationBin
x.Flag |= mysql.BinaryFlag
$$ = x
}
| "TIME" OptFieldLen
{
x := types.NewFieldType(mysql.TypeDuration)
x.Flen, _ = mysql.GetDefaultFieldLengthAndDecimalForCast(mysql.TypeDuration)
x.Decimal = $2.(int)
if x.Decimal > 0 {
x.Flen = x.Flen + 1 + x.Decimal
}
x.Charset = charset.CharsetBin
x.Collate = charset.CollationBin
x.Flag |= mysql.BinaryFlag
$$ = x
}
| "SIGNED" OptInteger
{
x := types.NewFieldType(mysql.TypeLonglong)
x.Charset = charset.CharsetBin
x.Collate = charset.CollationBin
x.Flag |= mysql.BinaryFlag
$$ = x
}
| "UNSIGNED" OptInteger
{
x := types.NewFieldType(mysql.TypeLonglong)
x.Flag |= mysql.UnsignedFlag | mysql.BinaryFlag
x.Charset = charset.CharsetBin
x.Collate = charset.CollationBin
$$ = x
}
| "JSON"
{
x := types.NewFieldType(mysql.TypeJSON)
x.Flag |= mysql.BinaryFlag | (mysql.ParseToJSONFlag)
x.Charset = mysql.DefaultCharset
x.Collate = mysql.DefaultCollationName
$$ = x
}
PriorityOpt:
{
$$ = mysql.NoPriority
}
| "LOW_PRIORITY"
{
$$ = mysql.LowPriority
}
| "HIGH_PRIORITY"
{
$$ = mysql.HighPriority
}
| "DELAYED"
{
$$ = mysql.DelayedPriority
}
TableName:
Identifier
{
$$ = &ast.TableName{Name:model.NewCIStr($1)}
}
| Identifier '.' Identifier
{
$$ = &ast.TableName{Schema:model.NewCIStr($1), Name:model.NewCIStr($3)}
}
TableNameList:
TableName
{
tbl := []*ast.TableName{$1.(*ast.TableName)}
$$ = tbl
}
| TableNameList ',' TableName
{
$$ = append($1.([]*ast.TableName), $3.(*ast.TableName))
}
QuickOptional:
%prec empty
{
$$ = false
}
| "QUICK"
{
$$ = true
}
/***************************Prepared Statement Start******************************
* See https://dev.mysql.com/doc/refman/5.7/en/prepare.html
* Example:
* PREPARE stmt_name FROM 'SELECT SQRT(POW(?,2) + POW(?,2)) AS hypotenuse';
* OR
* SET @s = 'SELECT SQRT(POW(?,2) + POW(?,2)) AS hypotenuse';
* PREPARE stmt_name FROM @s;
*/
PreparedStmt:
"PREPARE" Identifier "FROM" PrepareSQL
{
var sqlText string
var sqlVar *ast.VariableExpr
switch $4.(type) {
case string:
sqlText = $4.(string)
case *ast.VariableExpr:
sqlVar = $4.(*ast.VariableExpr)
}
$$ = &ast.PrepareStmt{
Name: $2,
SQLText: sqlText,
SQLVar: sqlVar,
}
}
PrepareSQL:
stringLit
{
$$ = $1
}
| UserVariable
{
$$ = $1.(interface{})
}
/*
* See https://dev.mysql.com/doc/refman/5.7/en/execute.html
* Example:
* EXECUTE stmt1 USING @a, @b;
* OR
* EXECUTE stmt1;
*/
ExecuteStmt:
"EXECUTE" Identifier
{
$$ = &ast.ExecuteStmt{Name: $2}
}
| "EXECUTE" Identifier "USING" UserVariableList
{
$$ = &ast.ExecuteStmt{
Name: $2,
UsingVars: $4.([]ast.ExprNode),
}
}
UserVariableList:
UserVariable
{
$$ = []ast.ExprNode{$1}
}
| UserVariableList ',' UserVariable
{
$$ = append($1.([]ast.ExprNode), $3)
}
/*
* See https://dev.mysql.com/doc/refman/5.0/en/deallocate-prepare.html
*/
DeallocateStmt:
DeallocateSym "PREPARE" Identifier
{
$$ = &ast.DeallocateStmt{Name: $3}
}
DeallocateSym:
"DEALLOCATE" | "DROP"
/****************************Prepared Statement End*******************************/
RollbackStmt:
"ROLLBACK"
{
$$ = &ast.RollbackStmt{}
}
SelectStmtBasic:
"SELECT" SelectStmtOpts SelectStmtFieldList
{
st := &ast.SelectStmt {
SelectStmtOpts: $2.(*ast.SelectStmtOpts),
Distinct: $2.(*ast.SelectStmtOpts).Distinct,
Fields: $3.(*ast.FieldList),
}
$$ = st
}
SelectStmtFromDual:
SelectStmtBasic FromDual WhereClauseOptional OrderByOptional
{
st := $1.(*ast.SelectStmt)
lastField := st.Fields.Fields[len(st.Fields.Fields)-1]
if lastField.Expr != nil && lastField.AsName.O == "" {
lastEnd := yyS[yypt-2].offset-1
lastField.SetText(parser.src[lastField.Offset:lastEnd])
}
if $3 != nil {
st.Where = $3.(ast.ExprNode)
}
if $4 != nil {
st.OrderBy = $4.(*ast.OrderByClause)
}
}
SelectStmtFromTable:
SelectStmtBasic "FROM"
TableRefsClause WhereClauseOptional SelectStmtGroup HavingClause
{
st := $1.(*ast.SelectStmt)
st.From = $3.(*ast.TableRefsClause)
if st.SelectStmtOpts.TableHints != nil {
st.TableHints = st.SelectStmtOpts.TableHints
}
lastField := st.Fields.Fields[len(st.Fields.Fields)-1]
if lastField.Expr != nil && lastField.AsName.O == "" {
lastEnd := parser.endOffset(&yyS[yypt-4])
lastField.SetText(parser.src[lastField.Offset:lastEnd])
}
if $4 != nil {
st.Where = $4.(ast.ExprNode)
}
if $5 != nil {
st.GroupBy = $5.(*ast.GroupByClause)
}
if $6 != nil {
st.Having = $6.(*ast.HavingClause)
}
$$ = st
}
SelectStmt:
SelectStmtBasic OrderByOptional SelectStmtLimit SelectLockOpt
{
st := $1.(*ast.SelectStmt)
st.LockTp = $4.(ast.SelectLockType)
lastField := st.Fields.Fields[len(st.Fields.Fields)-1]
if lastField.Expr != nil && lastField.AsName.O == "" {
src := parser.src
var lastEnd int
if $2 != nil {
lastEnd = yyS[yypt-2].offset-1
} else if $3 != nil {
lastEnd = yyS[yypt-1].offset-1
} else if $4 != ast.SelectLockNone {
lastEnd = yyS[yypt].offset-1
} else {
lastEnd = len(src)
if src[lastEnd-1] == ';' {
lastEnd--
}
}
lastField.SetText(src[lastField.Offset:lastEnd])
}
if $2 != nil {
st.OrderBy = $2.(*ast.OrderByClause)
}
if $3 != nil {
st.Limit = $3.(*ast.Limit)
}
$$ = st
}
| SelectStmtFromDual SelectStmtLimit SelectLockOpt
{
st := $1.(*ast.SelectStmt)
st.LockTp = $3.(ast.SelectLockType)
if $2 != nil {
st.Limit = $2.(*ast.Limit)
}
$$ = st
}
| SelectStmtFromTable OrderByOptional SelectStmtLimit SelectLockOpt
{
st := $1.(*ast.SelectStmt)
st.LockTp = $4.(ast.SelectLockType)
if $2 != nil {
st.OrderBy = $2.(*ast.OrderByClause)
}
if $3 != nil {
st.Limit = $3.(*ast.Limit)
}
$$ = st
}
FromDual:
"FROM" "DUAL"
TableRefsClause:
TableRefs
{
$$ = &ast.TableRefsClause{TableRefs: $1.(*ast.Join)}
}
TableRefs:
EscapedTableRef
{
if j, ok := $1.(*ast.Join); ok {
// if $1 is Join, use it directly
$$ = j
} else {
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: nil}
}
}
| TableRefs ',' EscapedTableRef
{
/* from a, b is default cross join */
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $3.(ast.ResultSetNode), Tp: ast.CrossJoin}
}
EscapedTableRef:
TableRef %prec lowerThanSetKeyword
{
$$ = $1
}
| '{' Identifier TableRef '}'
{
/*
* ODBC escape syntax for outer join is { OJ join_table }
* Use an Identifier for OJ
*/
$$ = $3
}
TableRef:
TableFactor
{
$$ = $1
}
| JoinTable
{
$$ = $1
}
TableFactor:
TableName TableAsNameOpt IndexHintListOpt
{
tn := $1.(*ast.TableName)
tn.IndexHints = $3.([]*ast.IndexHint)
$$ = &ast.TableSource{Source: tn, AsName: $2.(model.CIStr)}
}
| '(' SelectStmt ')' TableAsName
{
st := $2.(*ast.SelectStmt)
endOffset := parser.endOffset(&yyS[yypt-1])
parser.setLastSelectFieldText(st, endOffset)
$$ = &ast.TableSource{Source: $2.(*ast.SelectStmt), AsName: $4.(model.CIStr)}
}
| '(' UnionStmt ')' TableAsName
{
$$ = &ast.TableSource{Source: $2.(*ast.UnionStmt), AsName: $4.(model.CIStr)}
}
| '(' TableRefs ')'
{
$$ = $2
}
PartitionNameListOpt:
/* empty */
{
$$ = []model.CIStr{}
}
| "PARTITION" '(' PartitionNameList ')'
{
$$ = $3
}
TableAsNameOpt:
{
$$ = model.CIStr{}
}
| TableAsName
{
$$ = $1
}
TableAsName:
Identifier
{
$$ = model.NewCIStr($1)
}
| "AS" Identifier
{
$$ = model.NewCIStr($2)
}
IndexHintType:
"USE" KeyOrIndex
{
$$ = ast.HintUse
}
| "IGNORE" KeyOrIndex
{
$$ = ast.HintIgnore
}
| "FORCE" KeyOrIndex
{
$$ = ast.HintForce
}
IndexHintScope:
{
$$ = ast.HintForScan
}
| "FOR" "JOIN"
{
$$ = ast.HintForJoin
}
| "FOR" "ORDER" "BY"
{
$$ = ast.HintForOrderBy
}
| "FOR" "GROUP" "BY"
{
$$ = ast.HintForGroupBy
}
IndexHint:
IndexHintType IndexHintScope '(' IndexNameList ')'
{
$$ = &ast.IndexHint{
IndexNames: $4.([]model.CIStr),
HintType: $1.(ast.IndexHintType),
HintScope: $2.(ast.IndexHintScope),
}
}
IndexNameList:
{
var nameList []model.CIStr
$$ = nameList
}
| Identifier
{
$$ = []model.CIStr{model.NewCIStr($1)}
}
| IndexNameList ',' Identifier
{
$$ = append($1.([]model.CIStr), model.NewCIStr($3))
}
| "PRIMARY"
{
$$ = []model.CIStr{model.NewCIStr($1)}
}
IndexHintList:
IndexHint
{
$$ = []*ast.IndexHint{$1.(*ast.IndexHint)}
}
| IndexHintList IndexHint
{
$$ = append($1.([]*ast.IndexHint), $2.(*ast.IndexHint))
}
IndexHintListOpt:
{
var hintList []*ast.IndexHint
$$ = hintList
}
| IndexHintList
{
$$ = $1
}
JoinTable:
/* Use %prec to evaluate production TableRef before cross join */
TableRef CrossOpt TableRef %prec tableRefPriority
{
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $3.(ast.ResultSetNode), Tp: ast.CrossJoin}
}
| TableRef CrossOpt TableRef "ON" Expression
{
on := &ast.OnCondition{Expr: $5}
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $3.(ast.ResultSetNode), Tp: ast.CrossJoin, On: on}
}
| TableRef CrossOpt TableRef "USING" '(' ColumnNameList ')'
{
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $3.(ast.ResultSetNode), Tp: ast.CrossJoin, Using: $6.([]*ast.ColumnName)}
}
| TableRef JoinType OuterOpt "JOIN" TableRef "ON" Expression
{
on := &ast.OnCondition{Expr: $7}
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $5.(ast.ResultSetNode), Tp: $2.(ast.JoinType), On: on}
}
| TableRef JoinType OuterOpt "JOIN" TableRef "USING" '(' ColumnNameList ')'
{
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $5.(ast.ResultSetNode), Tp: $2.(ast.JoinType), Using: $8.([]*ast.ColumnName)}
}
| TableRef "NATURAL" "JOIN" TableRef
{
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $4.(ast.ResultSetNode), NaturalJoin: true}
}
| TableRef "NATURAL" JoinType OuterOpt "JOIN" TableRef
{
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $6.(ast.ResultSetNode), Tp: $3.(ast.JoinType), NaturalJoin: true}
}
| TableRef "STRAIGHT_JOIN" TableRef
{
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $3.(ast.ResultSetNode), StraightJoin: true}
}
| TableRef "STRAIGHT_JOIN" TableRef "ON" Expression
{
on := &ast.OnCondition{Expr: $5}
$$ = &ast.Join{Left: $1.(ast.ResultSetNode), Right: $3.(ast.ResultSetNode), StraightJoin: true, On: on}
}
JoinType:
"LEFT"
{
$$ = ast.LeftJoin
}
| "RIGHT"
{
$$ = ast.RightJoin
}
OuterOpt:
{}
| "OUTER"
CrossOpt:
"JOIN"
| "CROSS" "JOIN"
| "INNER" "JOIN"
LimitClause:
{
$$ = nil
}
| "LIMIT" LimitOption
{
$$ = &ast.Limit{Count: $2.(ast.ExprNode)}
}
LimitOption:
LengthNum
{
$$ = ast.NewValueExpr($1)
}
| paramMarker
{
$$ = &ast.ParamMarkerExpr{
Offset: yyS[yypt].offset,
}
}
SelectStmtLimit:
{
$$ = nil
}
| "LIMIT" LimitOption
{
$$ = &ast.Limit{Count: $2.(ast.ExprNode)}
}
| "LIMIT" LimitOption ',' LimitOption
{
$$ = &ast.Limit{Offset: $2.(ast.ExprNode), Count: $4.(ast.ExprNode)}
}
| "LIMIT" LimitOption "OFFSET" LimitOption
{
$$ = &ast.Limit{Offset: $4.(ast.ExprNode), Count: $2.(ast.ExprNode)}
}
SelectStmtOpts:
TableOptimizerHints DefaultFalseDistinctOpt PriorityOpt SelectStmtSQLCache SelectStmtCalcFoundRows SelectStmtStraightJoin
{
opt := &ast.SelectStmtOpts{}
if $1 != nil {
opt.TableHints = $1.([]*ast.TableOptimizerHint)
}
if $2 != nil {
opt.Distinct = $2.(bool)
}
if $3 != nil {
opt.Priority = $3.(mysql.PriorityEnum)
}
if $4 != nil {
opt.SQLCache = $4.(bool)
}
if $5 != nil {
opt.CalcFoundRows = $5.(bool)
}
if $6 != nil {
opt.StraightJoin = $6.(bool)
}
$$ = opt
}
TableOptimizerHints:
/* empty */
{
$$ = nil
}
| hintBegin TableOptimizerHintList hintEnd
{
$$ = $2
}
HintTableList:
Identifier
{
$$ = []model.CIStr{model.NewCIStr($1)}
}
| HintTableList ',' Identifier
{
$$ = append($1.([]model.CIStr), model.NewCIStr($3))
}
TableOptimizerHintList:
TableOptimizerHintOpt
{
$$ = []*ast.TableOptimizerHint{$1.(*ast.TableOptimizerHint)}
}
| TableOptimizerHintList TableOptimizerHintOpt
{
$$ = append($1.([]*ast.TableOptimizerHint), $2.(*ast.TableOptimizerHint))
}
TableOptimizerHintOpt:
tidbSMJ '(' HintTableList ')'
{
$$ = &ast.TableOptimizerHint{HintName: model.NewCIStr($1), Tables: $3.([]model.CIStr)}
}
| tidbINLJ '(' HintTableList ')'
{
$$ = &ast.TableOptimizerHint{HintName: model.NewCIStr($1), Tables: $3.([]model.CIStr)}
}
| tidbHJ '(' HintTableList ')'
{
$$ = &ast.TableOptimizerHint{HintName: model.NewCIStr($1), Tables: $3.([]model.CIStr)}
}
| maxExecutionTime '(' NUM ')'
{
$$ = &ast.TableOptimizerHint{HintName: model.NewCIStr($1), MaxExecutionTime: getUint64FromNUM($3)}
}
SelectStmtCalcFoundRows:
{
$$ = false
}
| "SQL_CALC_FOUND_ROWS"
{
$$ = true
}
SelectStmtSQLCache:
%prec empty
{
$$ = true
}
| "SQL_CACHE"
{
$$ = true
}
| "SQL_NO_CACHE"
{
$$ = false
}
SelectStmtStraightJoin:
%prec empty
{
$$ = false
}
| "STRAIGHT_JOIN"
{
$$ = true
}
SelectStmtFieldList:
FieldList
{
$$ = &ast.FieldList{Fields: $1.([]*ast.SelectField)}
}
SelectStmtGroup:
/* EMPTY */
{
$$ = nil
}
| GroupByClause
// See https://dev.mysql.com/doc/refman/5.7/en/subqueries.html
SubSelect:
'(' SelectStmt ')'
{
s := $2.(*ast.SelectStmt)
endOffset := parser.endOffset(&yyS[yypt])
parser.setLastSelectFieldText(s, endOffset)
src := parser.src
// See the implementation of yyParse function
s.SetText(src[yyS[yypt-1].offset:yyS[yypt].offset])
$$ = &ast.SubqueryExpr{Query: s}
}
| '(' UnionStmt ')'
{
s := $2.(*ast.UnionStmt)
src := parser.src
// See the implementation of yyParse function
s.SetText(src[yyS[yypt-1].offset:yyS[yypt].offset])
$$ = &ast.SubqueryExpr{Query: s}
}
// See https://dev.mysql.com/doc/refman/5.7/en/innodb-locking-reads.html
SelectLockOpt:
/* empty */
{
$$ = ast.SelectLockNone
}
| "FOR" "UPDATE"
{
$$ = ast.SelectLockForUpdate
}
| "LOCK" "IN" "SHARE" "MODE"
{
$$ = ast.SelectLockInShareMode
}
// See https://dev.mysql.com/doc/refman/5.7/en/union.html
UnionStmt:
UnionClauseList "UNION" UnionOpt SelectStmtBasic OrderByOptional SelectStmtLimit SelectLockOpt
{
st := $4.(*ast.SelectStmt)
union := $1.(*ast.UnionStmt)
st.IsAfterUnionDistinct = $3.(bool)
lastSelect := union.SelectList.Selects[len(union.SelectList.Selects)-1]
endOffset := parser.endOffset(&yyS[yypt-5])
parser.setLastSelectFieldText(lastSelect, endOffset)
union.SelectList.Selects = append(union.SelectList.Selects, st)
if $5 != nil {
union.OrderBy = $5.(*ast.OrderByClause)
}
if $6 != nil {
union.Limit = $6.(*ast.Limit)
}
if $5 == nil && $6 == nil {
st.LockTp = $7.(ast.SelectLockType)
}
$$ = union
}
| UnionClauseList "UNION" UnionOpt SelectStmtFromDual SelectStmtLimit SelectLockOpt
{
st := $4.(*ast.SelectStmt)
union := $1.(*ast.UnionStmt)
st.IsAfterUnionDistinct = $3.(bool)
lastSelect := union.SelectList.Selects[len(union.SelectList.Selects)-1]
endOffset := parser.endOffset(&yyS[yypt-4])
parser.setLastSelectFieldText(lastSelect, endOffset)
union.SelectList.Selects = append(union.SelectList.Selects, st)
if $5 != nil {
union.Limit = $5.(*ast.Limit)
} else {
st.LockTp = $6.(ast.SelectLockType)
}
$$ = union
}
| UnionClauseList "UNION" UnionOpt SelectStmtFromTable OrderByOptional
SelectStmtLimit SelectLockOpt
{
st := $4.(*ast.SelectStmt)
union := $1.(*ast.UnionStmt)
st.IsAfterUnionDistinct = $3.(bool)
lastSelect := union.SelectList.Selects[len(union.SelectList.Selects)-1]
endOffset := parser.endOffset(&yyS[yypt-5])
parser.setLastSelectFieldText(lastSelect, endOffset)
union.SelectList.Selects = append(union.SelectList.Selects, st)
if $5 != nil {
union.OrderBy = $5.(*ast.OrderByClause)
}
if $6 != nil {
union.Limit = $6.(*ast.Limit)
}
if $5 == nil && $6 == nil {
st.LockTp = $7.(ast.SelectLockType)
}
$$ = union
}
| UnionClauseList "UNION" UnionOpt '(' SelectStmt ')' OrderByOptional SelectStmtLimit
{
union := $1.(*ast.UnionStmt)
lastSelect := union.SelectList.Selects[len(union.SelectList.Selects)-1]
endOffset := parser.endOffset(&yyS[yypt-6])
parser.setLastSelectFieldText(lastSelect, endOffset)
st := $5.(*ast.SelectStmt)
st.IsInBraces = true
st.IsAfterUnionDistinct = $3.(bool)
endOffset = parser.endOffset(&yyS[yypt-2])
parser.setLastSelectFieldText(st, endOffset)
union.SelectList.Selects = append(union.SelectList.Selects, st)
if $7 != nil {
union.OrderBy = $7.(*ast.OrderByClause)
}
if $8 != nil {
union.Limit = $8.(*ast.Limit)
}
$$ = union
}
UnionClauseList:
UnionSelect
{
selectList := &ast.UnionSelectList{Selects: []*ast.SelectStmt{$1.(*ast.SelectStmt)}}
$$ = &ast.UnionStmt{
SelectList: selectList,
}
}
| UnionClauseList "UNION" UnionOpt UnionSelect
{
union := $1.(*ast.UnionStmt)
st := $4.(*ast.SelectStmt)
st.IsAfterUnionDistinct = $3.(bool)
lastSelect := union.SelectList.Selects[len(union.SelectList.Selects)-1]
endOffset := parser.endOffset(&yyS[yypt-2])
parser.setLastSelectFieldText(lastSelect, endOffset)
union.SelectList.Selects = append(union.SelectList.Selects, st)
$$ = union
}
UnionSelect:
SelectStmt
{
$$ = $1.(interface{})
}
| '(' SelectStmt ')'
{
st := $2.(*ast.SelectStmt)
st.IsInBraces = true
endOffset := parser.endOffset(&yyS[yypt])
parser.setLastSelectFieldText(st, endOffset)
$$ = $2
}
UnionOpt:
DefaultTrueDistinctOpt
/********************Set Statement*******************************/
SetStmt:
"SET" VariableAssignmentList
{
$$ = &ast.SetStmt{Variables: $2.([]*ast.VariableAssignment)}
}
| "SET" "PASSWORD" eq PasswordOpt
{
$$ = &ast.SetPwdStmt{Password: $4.(string)}
}
| "SET" "PASSWORD" "FOR" Username eq PasswordOpt
{
$$ = &ast.SetPwdStmt{User: $4.(*auth.UserIdentity), Password: $6.(string)}
}
| "SET" "GLOBAL" "TRANSACTION" TransactionChars
{
vars := $4.([]*ast.VariableAssignment)
for _, v := range vars {
v.IsGlobal = true
}
$$ = &ast.SetStmt{Variables: vars}
}
| "SET" "SESSION" "TRANSACTION" TransactionChars
{
$$ = &ast.SetStmt{Variables: $4.([]*ast.VariableAssignment)}
}
| "SET" "TRANSACTION" TransactionChars
{
assigns := $3.([]*ast.VariableAssignment)
for i:=0; i<len(assigns); i++ {
if assigns[i].Name == "tx_isolation" {
// A special session variable that make setting tx_isolation take effect one time.
assigns[i].Name = "tx_isolation_one_shot"
}
}
$$ = &ast.SetStmt{Variables: assigns}
}
TransactionChars:
TransactionChar
{
if $1 != nil {
$$ = $1
} else {
$$ = []*ast.VariableAssignment{}
}
}
| TransactionChars ',' TransactionChar
{
if $3 != nil {
varAssigns := $3.([]*ast.VariableAssignment)
$$ = append($1.([]*ast.VariableAssignment), varAssigns...)
} else {
$$ = $1
}
}
TransactionChar:
"ISOLATION" "LEVEL" IsolationLevel
{
varAssigns := []*ast.VariableAssignment{}
expr := ast.NewValueExpr($3)
varAssigns = append(varAssigns, &ast.VariableAssignment{Name: "tx_isolation", Value: expr, IsSystem: true})
$$ = varAssigns
}
| "READ" "WRITE"
{
varAssigns := []*ast.VariableAssignment{}
expr := ast.NewValueExpr("0")
varAssigns = append(varAssigns, &ast.VariableAssignment{Name: "tx_read_only", Value: expr, IsSystem: true})
$$ = varAssigns
}
| "READ" "ONLY"
{
varAssigns := []*ast.VariableAssignment{}
expr := ast.NewValueExpr("1")
varAssigns = append(varAssigns, &ast.VariableAssignment{Name: "tx_read_only", Value: expr, IsSystem: true})
$$ = varAssigns
}
IsolationLevel:
"REPEATABLE" "READ"
{
$$ = ast.RepeatableRead
}
| "READ" "COMMITTED"
{
$$ = ast.ReadCommitted
}
| "READ" "UNCOMMITTED"
{
$$ = ast.ReadUncommitted
}
| "SERIALIZABLE"
{
$$ = ast.Serializable
}
SetExpr:
"ON"
{
$$ = ast.NewValueExpr("ON")
}
| ExprOrDefault
VariableAssignment:
Identifier eq SetExpr
{
$$ = &ast.VariableAssignment{Name: $1, Value: $3, IsSystem: true}
}
| "GLOBAL" Identifier eq SetExpr
{
$$ = &ast.VariableAssignment{Name: $2, Value: $4, IsGlobal: true, IsSystem: true}
}
| "SESSION" Identifier eq SetExpr
{
$$ = &ast.VariableAssignment{Name: $2, Value: $4, IsSystem: true}
}
| "LOCAL" Identifier eq Expression
{
$$ = &ast.VariableAssignment{Name: $2, Value: $4, IsSystem: true}
}
| doubleAtIdentifier eq SetExpr
{
v := strings.ToLower($1)
var isGlobal bool
if strings.HasPrefix(v, "@@global.") {
isGlobal = true
v = strings.TrimPrefix(v, "@@global.")
} else if strings.HasPrefix(v, "@@session.") {
v = strings.TrimPrefix(v, "@@session.")
} else if strings.HasPrefix(v, "@@local.") {
v = strings.TrimPrefix(v, "@@local.")
} else if strings.HasPrefix(v, "@@") {
v = strings.TrimPrefix(v, "@@")
}
$$ = &ast.VariableAssignment{Name: v, Value: $3, IsGlobal: isGlobal, IsSystem: true}
}
| singleAtIdentifier eq Expression
{
v := $1
v = strings.TrimPrefix(v, "@")
$$ = &ast.VariableAssignment{Name: v, Value: $3}
}
| singleAtIdentifier assignmentEq Expression
{
v := $1
v = strings.TrimPrefix(v, "@")
$$ = &ast.VariableAssignment{Name: v, Value: $3}
}
| "NAMES" CharsetName
{
$$ = &ast.VariableAssignment{
Name: ast.SetNames,
Value: ast.NewValueExpr($2.(string)),
}
}
| "NAMES" CharsetName "COLLATE" "DEFAULT"
{
$$ = &ast.VariableAssignment{
Name: ast.SetNames,
Value: ast.NewValueExpr($2.(string)),
}
}
| "NAMES" CharsetName "COLLATE" StringName
{
$$ = &ast.VariableAssignment{
Name: ast.SetNames,
Value: ast.NewValueExpr($2.(string)),
ExtendValue: ast.NewValueExpr($4.(string)),
}
}
| CharsetKw CharsetName
{
$$ = &ast.VariableAssignment{
Name: ast.SetNames,
Value: ast.NewValueExpr($2.(string)),
}
}
CharsetName:
StringName
{
$$ = $1
}
| binaryType
{
$$ = charset.CharsetBin
}
CollationName:
StringName
{
info, err := charset.GetCollationByName($1.(string))
if err != nil {
yylex.AppendError(err)
return 1
}
$$ = info.Name
}
VariableAssignmentList:
{
$$ = []*ast.VariableAssignment{}
}
| VariableAssignment
{
$$ = []*ast.VariableAssignment{$1.(*ast.VariableAssignment)}
}
| VariableAssignmentList ',' VariableAssignment
{
$$ = append($1.([]*ast.VariableAssignment), $3.(*ast.VariableAssignment))
}
Variable:
SystemVariable | UserVariable
SystemVariable:
doubleAtIdentifier
{
v := strings.ToLower($1)
var isGlobal bool
explicitScope := true
if strings.HasPrefix(v, "@@global.") {
isGlobal = true
v = strings.TrimPrefix(v, "@@global.")
} else if strings.HasPrefix(v, "@@session.") {
v = strings.TrimPrefix(v, "@@session.")
} else if strings.HasPrefix(v, "@@local.") {
v = strings.TrimPrefix(v, "@@local.")
} else if strings.HasPrefix(v, "@@") {
v, explicitScope = strings.TrimPrefix(v, "@@"), false
}
$$ = &ast.VariableExpr{Name: v, IsGlobal: isGlobal, IsSystem: true, ExplicitScope: explicitScope}
}
UserVariable:
singleAtIdentifier
{
v := $1
v = strings.TrimPrefix(v, "@")
$$ = &ast.VariableExpr{Name: v, IsGlobal: false, IsSystem: false}
}
Username:
StringName
{
$$ = &auth.UserIdentity{Username: $1.(string), Hostname: "%"}
}
| StringName '@' StringName
{
$$ = &auth.UserIdentity{Username: $1.(string), Hostname: $3.(string)}
}
| StringName singleAtIdentifier
{
$$ = &auth.UserIdentity{Username: $1.(string), Hostname: strings.TrimPrefix($2, "@")}
}
| "CURRENT_USER" OptionalBraces
{
$$ = &auth.UserIdentity{CurrentUser: true}
}
UsernameList:
Username
{
$$ = []*auth.UserIdentity{$1.(*auth.UserIdentity)}
}
| UsernameList ',' Username
{
$$ = append($1.([]*auth.UserIdentity), $3.(*auth.UserIdentity))
}
PasswordOpt:
stringLit
{
$$ = $1
}
| "PASSWORD" '(' AuthString ')'
{
$$ = $3.(string)
}
AuthString:
stringLit
{
$$ = $1
}
/****************************Admin Statement*******************************/
AdminStmt:
"ADMIN" "SHOW" "DDL"
{
$$ = &ast.AdminStmt{Tp: ast.AdminShowDDL}
}
| "ADMIN" "SHOW" "DDL" "JOBS"
{
$$ = &ast.AdminStmt{Tp: ast.AdminShowDDLJobs}
}
| "ADMIN" "SHOW" "DDL" "JOBS" NUM
{
$$ = &ast.AdminStmt{
Tp: ast.AdminShowDDLJobs,
JobNumber: $5.(int64),
}
}
| "ADMIN" "CHECK" "TABLE" TableNameList
{
$$ = &ast.AdminStmt{
Tp: ast.AdminCheckTable,
Tables: $4.([]*ast.TableName),
}
}
| "ADMIN" "CHECK" "INDEX" TableName Identifier
{
$$ = &ast.AdminStmt{
Tp: ast.AdminCheckIndex,
Tables: []*ast.TableName{$4.(*ast.TableName)},
Index: string($5),
}
}
| "ADMIN" "RECOVER" "INDEX" TableName Identifier
{
$$ = &ast.AdminStmt{
Tp: ast.AdminRecoverIndex,
Tables: []*ast.TableName{$4.(*ast.TableName)},
Index: string($5),
}
}
| "ADMIN" "CLEANUP" "INDEX" TableName Identifier
{
$$ = &ast.AdminStmt{
Tp: ast.AdminCleanupIndex,
Tables: []*ast.TableName{$4.(*ast.TableName)},
Index: string($5),
}
}
| "ADMIN" "CHECK" "INDEX" TableName Identifier HandleRangeList
{
$$ = &ast.AdminStmt{
Tp: ast.AdminCheckIndexRange,
Tables: []*ast.TableName{$4.(*ast.TableName)},
Index: string($5),
HandleRanges: $6.([]ast.HandleRange),
}
}
| "ADMIN" "CHECKSUM" "TABLE" TableNameList
{
$$ = &ast.AdminStmt{
Tp: ast.AdminChecksumTable,
Tables: $4.([]*ast.TableName),
}
}
| "ADMIN" "CANCEL" "DDL" "JOBS" NumList
{
$$ = &ast.AdminStmt{
Tp: ast.AdminCancelDDLJobs,
JobIDs: $5.([]int64),
}
}
| "ADMIN" "SHOW" "DDL" "JOB" "QUERIES" NumList
{
$$ = &ast.AdminStmt{
Tp: ast.AdminShowDDLJobQueries,
JobIDs: $6.([]int64),
}
}
| "ADMIN" "SHOW" "SLOW" AdminShowSlow
{
$$ = &ast.AdminStmt{
Tp: ast.AdminShowSlow,
ShowSlow: $4.(*ast.ShowSlow),
}
}
AdminShowSlow:
"RECENT" NUM
{
$$ = &ast.ShowSlow{
Tp: ast.ShowSlowRecent,
Count: getUint64FromNUM($2),
}
}
| "TOP" NUM
{
$$ = &ast.ShowSlow{
Tp: ast.ShowSlowTop,
Kind: ast.ShowSlowKindDefault,
Count: getUint64FromNUM($2),
}
}
| "TOP" "INTERNAL" NUM
{
$$ = &ast.ShowSlow{
Tp: ast.ShowSlowTop,
Kind: ast.ShowSlowKindInternal,
Count: getUint64FromNUM($3),
}
}
| "TOP" "ALL" NUM
{
$$ = &ast.ShowSlow{
Tp: ast.ShowSlowTop,
Kind: ast.ShowSlowKindAll,
Count: getUint64FromNUM($3),
}
}
HandleRangeList:
HandleRange
{
$$ = []ast.HandleRange{$1.(ast.HandleRange)}
}
| HandleRangeList ',' HandleRange
{
$$ = append($1.([]ast.HandleRange), $3.(ast.HandleRange))
}
HandleRange:
'(' NUM ',' NUM ')'
{
$$ = ast.HandleRange{Begin: $2.(int64), End: $4.(int64)}
}
NumList:
NUM
{
$$ = []int64{$1.(int64)}
}
|
NumList ',' NUM
{
$$ = append($1.([]int64), $3.(int64))
}
/****************************Show Statement*******************************/
ShowStmt:
"SHOW" ShowTargetFilterable ShowLikeOrWhereOpt
{
stmt := $2.(*ast.ShowStmt)
if $3 != nil {
if x, ok := $3.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $3.(ast.ExprNode)
}
}
$$ = stmt
}
| "SHOW" "CREATE" "TABLE" TableName
{
$$ = &ast.ShowStmt{
Tp: ast.ShowCreateTable,
Table: $4.(*ast.TableName),
}
}
| "SHOW" "CREATE" "DATABASE" DBName
{
$$ = &ast.ShowStmt{
Tp: ast.ShowCreateDatabase,
DBName: $4.(string),
}
}
| "SHOW" "GRANTS"
{
// See https://dev.mysql.com/doc/refman/5.7/en/show-grants.html
$$ = &ast.ShowStmt{Tp: ast.ShowGrants}
}
| "SHOW" "GRANTS" "FOR" Username
{
// See https://dev.mysql.com/doc/refman/5.7/en/show-grants.html
$$ = &ast.ShowStmt{
Tp: ast.ShowGrants,
User: $4.(*auth.UserIdentity),
}
}
| "SHOW" "MASTER" "STATUS"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowMasterStatus,
}
}
| "SHOW" OptFull "PROCESSLIST"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowProcessList,
Full: $2.(bool),
}
}
| "SHOW" "STATS_META" ShowLikeOrWhereOpt
{
stmt := &ast.ShowStmt{
Tp: ast.ShowStatsMeta,
}
if $3 != nil {
if x, ok := $3.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $3.(ast.ExprNode)
}
}
$$ = stmt
}
| "SHOW" "STATS_HISTOGRAMS" ShowLikeOrWhereOpt
{
stmt := &ast.ShowStmt{
Tp: ast.ShowStatsHistograms,
}
if $3 != nil {
if x, ok := $3.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $3.(ast.ExprNode)
}
}
$$ = stmt
}
| "SHOW" "STATS_BUCKETS" ShowLikeOrWhereOpt
{
stmt := &ast.ShowStmt{
Tp: ast.ShowStatsBuckets,
}
if $3 != nil {
if x, ok := $3.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $3.(ast.ExprNode)
}
}
$$ = stmt
}
| "SHOW" "STATS_HEALTHY" ShowLikeOrWhereOpt
{
stmt := &ast.ShowStmt{
Tp: ast.ShowStatsHealthy,
}
if $3 != nil {
if x, ok := $3.(*ast.PatternLikeExpr); ok {
stmt.Pattern = x
} else {
stmt.Where = $3.(ast.ExprNode)
}
}
$$ = stmt
}
| "SHOW" "PROFILES"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowProfiles,
}
}
| "SHOW" "PRIVILEGES"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowPrivileges,
}
}
ShowIndexKwd:
"INDEX"
| "INDEXES"
| "KEYS"
FromOrIn:
"FROM" | "IN"
ShowTargetFilterable:
"ENGINES"
{
$$ = &ast.ShowStmt{Tp: ast.ShowEngines}
}
| "DATABASES"
{
$$ = &ast.ShowStmt{Tp: ast.ShowDatabases}
}
| CharsetKw
{
$$ = &ast.ShowStmt{Tp: ast.ShowCharset}
}
| OptFull "TABLES" ShowDatabaseNameOpt
{
$$ = &ast.ShowStmt{
Tp: ast.ShowTables,
DBName: $3.(string),
Full: $1.(bool),
}
}
| "TABLE" "STATUS" ShowDatabaseNameOpt
{
$$ = &ast.ShowStmt{
Tp: ast.ShowTableStatus,
DBName: $3.(string),
}
}
| ShowIndexKwd FromOrIn TableName
{
$$ = &ast.ShowStmt{
Tp: ast.ShowIndex,
Table: $3.(*ast.TableName),
}
}
| ShowIndexKwd FromOrIn Identifier FromOrIn Identifier
{
show := &ast.ShowStmt{
Tp: ast.ShowIndex,
Table: &ast.TableName{Name:model.NewCIStr($3), Schema: model.NewCIStr($5)},
}
$$ = show
}
| OptFull "COLUMNS" ShowTableAliasOpt ShowDatabaseNameOpt
{
$$ = &ast.ShowStmt{
Tp: ast.ShowColumns,
Table: $3.(*ast.TableName),
DBName: $4.(string),
Full: $1.(bool),
}
}
| OptFull "FIELDS" ShowTableAliasOpt ShowDatabaseNameOpt
{
// SHOW FIELDS is a synonym for SHOW COLUMNS.
$$ = &ast.ShowStmt{
Tp: ast.ShowColumns,
Table: $3.(*ast.TableName),
DBName: $4.(string),
Full: $1.(bool),
}
}
| "WARNINGS"
{
$$ = &ast.ShowStmt{Tp: ast.ShowWarnings}
}
| "ERRORS"
{
$$ = &ast.ShowStmt{Tp: ast.ShowErrors}
}
| GlobalScope "VARIABLES"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowVariables,
GlobalScope: $1.(bool),
}
}
| GlobalScope "LEVELS"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowLevels,
GlobalScope: $1.(bool),
}
}
| GlobalScope "STATUS"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowStatus,
GlobalScope: $1.(bool),
}
}
| "COLLATION"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowCollation,
}
}
| "TRIGGERS" ShowDatabaseNameOpt
{
$$ = &ast.ShowStmt{
Tp: ast.ShowTriggers,
DBName: $2.(string),
}
}
| "PROCEDURE" "STATUS"
{
$$ = &ast.ShowStmt {
Tp: ast.ShowProcedureStatus,
}
}
| "FUNCTION" "STATUS"
{
// This statement is similar to SHOW PROCEDURE STATUS but for stored functions.
// See http://dev.mysql.com/doc/refman/5.7/en/show-function-status.html
// We do not support neither stored functions nor stored procedures.
// So we reuse show procedure status process logic.
$$ = &ast.ShowStmt {
Tp: ast.ShowProcedureStatus,
}
}
| "EVENTS" ShowDatabaseNameOpt
{
$$ = &ast.ShowStmt{
Tp: ast.ShowEvents,
DBName: $2.(string),
}
}
| "PLUGINS"
{
$$ = &ast.ShowStmt{
Tp: ast.ShowPlugins,
}
}
ShowLikeOrWhereOpt:
{
$$ = nil
}
| "LIKE" SimpleExpr
{
$$ = &ast.PatternLikeExpr{
Pattern: $2,
Escape: '\\',
}
}
| "WHERE" Expression
{
$$ = $2
}
GlobalScope:
{
$$ = false
}
| "GLOBAL"
{
$$ = true
}
| "SESSION"
{
$$ = false
}
OptFull:
{
$$ = false
}
| "FULL"
{
$$ = true
}
ShowDatabaseNameOpt:
{
$$ = ""
}
| FromOrIn DBName
{
$$ = $2.(string)
}
ShowTableAliasOpt:
FromOrIn TableName
{
$$ = $2.(*ast.TableName)
}
FlushStmt:
"FLUSH" NoWriteToBinLogAliasOpt FlushOption
{
tmp := $3.(*ast.FlushStmt)
tmp.NoWriteToBinLog = $2.(bool)
$$ = tmp
}
FlushOption:
"PRIVILEGES"
{
$$ = &ast.FlushStmt{
Tp: ast.FlushPrivileges,
}
}
| "STATUS"
{
$$ = &ast.FlushStmt{
Tp: ast.FlushStatus,
}
}
| TableOrTables TableNameListOpt WithReadLockOpt
{
$$ = &ast.FlushStmt{
Tp: ast.FlushTables,
Tables: $2.([]*ast.TableName),
ReadLock: $3.(bool),
}
}
NoWriteToBinLogAliasOpt:
{
$$ = false
}
| "NO_WRITE_TO_BINLOG"
{
$$ = true
}
| "LOCAL"
{
$$ = true
}
TableNameListOpt:
%prec empty
{
$$ = []*ast.TableName{}
}
| TableNameList
{
$$ = $1
}
WithReadLockOpt:
{
$$ = false
}
| "WITH" "READ" "LOCK"
{
$$ = true
}
Statement:
EmptyStmt
| AdminStmt
| AlterTableStmt
| AlterUserStmt
| AnalyzeTableStmt
| BeginTransactionStmt
| BinlogStmt
| CommitStmt
| InceptionStmt
| InceptionStartStmt
| InceptionCommitStmt
| DeallocateStmt
| DeleteFromStmt
| ExecuteStmt
| ExplainStmt
| CreateDatabaseStmt
| CreateIndexStmt
| CreateTableStmt
| CreateViewStmt
| CreateUserStmt
| DoStmt
| DropDatabaseStmt
| DropIndexStmt
| DropTableStmt
| DropViewStmt
| DropUserStmt
| DropStatsStmt
| FlushStmt
| GrantStmt
| InsertIntoStmt
| KillStmt
| LoadDataStmt
| LoadStatsStmt
| PreparedStmt
| RollbackStmt
| RenameTableStmt
| ReplaceIntoStmt
| RevokeStmt
| SelectStmt
| UnionStmt
| SetStmt
| ShowStmt
| SubSelect
{
// `(select 1)`; is a valid select statement
// TODO: This is used to fix issue #320. There may be a better solution.
$$ = $1.(*ast.SubqueryExpr).Query.(ast.StmtNode)
}
| TraceStmt
| TruncateTableStmt
| UpdateStmt
| UseStmt
| UnlockTablesStmt
| LockTablesStmt
TraceableStmt:
SelectStmt
| DeleteFromStmt
| UpdateStmt
| InsertIntoStmt
| ReplaceIntoStmt
| UnionStmt
ExplainableStmt:
SelectStmt
| DeleteFromStmt
| UpdateStmt
| InsertIntoStmt
| ReplaceIntoStmt
| UnionStmt
StatementList:
Statement
{
if $1 != nil {
s := $1
if lexer, ok := yylex.(stmtTexter); ok {
s.SetText(lexer.stmtText())
}
parser.result = append(parser.result, s)
}
}
| StatementList ';' Statement
{
if $3 != nil {
s := $3
if lexer, ok := yylex.(stmtTexter); ok {
s.SetText(lexer.stmtText())
}
parser.result = append(parser.result, s)
}
}
Constraint:
ConstraintKeywordOpt ConstraintElem
{
cst := $2.(*ast.Constraint)
if $1 != nil {
cst.Name = $1.(string)
}
$$ = cst
}
TableElement:
ColumnDef
{
$$ = $1.(*ast.ColumnDef)
}
| Constraint
{
$$ = $1.(*ast.Constraint)
}
| "CHECK" '(' Expression ')'
{
/* Nothing to do now */
$$ = nil
}
TableElementList:
TableElement
{
if $1 != nil {
$$ = []interface{}{$1.(interface{})}
} else {
$$ = []interface{}{}
}
}
| TableElementList ',' TableElement
{
if $3 != nil {
$$ = append($1.([]interface{}), $3)
} else {
$$ = $1
}
}
TableElementListOpt:
/* empty */ %prec lowerThanCreateTableSelect
{
var columnDefs []*ast.ColumnDef
var constraints []*ast.Constraint
$$ = &ast.CreateTableStmt{
Cols: columnDefs,
Constraints: constraints,
}
}
| '(' TableElementList ')'
{
tes := $2.([]interface {})
var columnDefs []*ast.ColumnDef
var constraints []*ast.Constraint
for _, te := range tes {
switch te := te.(type) {
case *ast.ColumnDef:
columnDefs = append(columnDefs, te)
case *ast.Constraint:
constraints = append(constraints, te)
}
}
$$ = &ast.CreateTableStmt{
Cols: columnDefs,
Constraints: constraints,
}
}
TableOption:
"ENGINE" StringName
{
$$ = &ast.TableOption{Tp: ast.TableOptionEngine, StrValue: $2.(string)}
}
| "ENGINE" eq StringName
{
$$ = &ast.TableOption{Tp: ast.TableOptionEngine, StrValue: $3.(string)}
}
| DefaultKwdOpt CharsetKw EqOpt CharsetName
{
$$ = &ast.TableOption{Tp: ast.TableOptionCharset, StrValue: $4.(string)}
}
| DefaultKwdOpt "COLLATE" EqOpt CollationName
{
$$ = &ast.TableOption{Tp: ast.TableOptionCollate, StrValue: $4.(string)}
}
| "AUTO_INCREMENT" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionAutoIncrement, UintValue: $3.(uint64)}
}
| "COMMENT" EqOpt stringLit
{
$$ = &ast.TableOption{Tp: ast.TableOptionComment, StrValue: $3}
}
| "AVG_ROW_LENGTH" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionAvgRowLength, UintValue: $3.(uint64)}
}
| "CONNECTION" EqOpt stringLit
{
$$ = &ast.TableOption{Tp: ast.TableOptionConnection, StrValue: $3}
}
| "CHECKSUM" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionCheckSum, UintValue: $3.(uint64)}
}
| "PASSWORD" EqOpt stringLit
{
$$ = &ast.TableOption{Tp: ast.TableOptionPassword, StrValue: $3}
}
| "COMPRESSION" EqOpt stringLit
{
$$ = &ast.TableOption{Tp: ast.TableOptionCompression, StrValue: $3}
}
| "KEY_BLOCK_SIZE" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionKeyBlockSize, UintValue: $3.(uint64)}
}
| "MAX_ROWS" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionMaxRows, UintValue: $3.(uint64)}
}
| "MIN_ROWS" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionMinRows, UintValue: $3.(uint64)}
}
| "DELAY_KEY_WRITE" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionDelayKeyWrite, UintValue: $3.(uint64)}
}
| RowFormat
{
$$ = &ast.TableOption{Tp: ast.TableOptionRowFormat, UintValue: $1.(uint64)}
}
| "STATS_PERSISTENT" EqOpt StatsPersistentVal
{
$$ = &ast.TableOption{Tp: ast.TableOptionStatsPersistent}
}
| "SHARD_ROW_ID_BITS" EqOpt LengthNum
{
$$ = &ast.TableOption{Tp: ast.TableOptionShardRowID, UintValue: $3.(uint64)}
}
| "PACK_KEYS" EqOpt StatsPersistentVal
{
// Parse it but will ignore it.
$$ = &ast.TableOption{Tp: ast.TableOptionPackKeys}
}
StatsPersistentVal:
"DEFAULT"
{}
| LengthNum
{}
AlterTableOptionListOpt:
{
$$ = []*ast.TableOption{}
}
| TableOptionList %prec higherThanComma
CreateTableOptionListOpt:
/* empty */ %prec lowerThanCreateTableSelect
{
$$ = []*ast.TableOption{}
}
| TableOptionList %prec lowerThanComma
TableOptionList:
TableOption
{
$$ = []*ast.TableOption{$1.(*ast.TableOption)}
}
| TableOptionList TableOption
{
$$ = append($1.([]*ast.TableOption), $2.(*ast.TableOption))
}
| TableOptionList ',' TableOption
{
$$ = append($1.([]*ast.TableOption), $3.(*ast.TableOption))
}
OptTable:
{}
| "TABLE"
TruncateTableStmt:
"TRUNCATE" OptTable TableName
{
$$ = &ast.TruncateTableStmt{Table: $3.(*ast.TableName)}
}
RowFormat:
"ROW_FORMAT" EqOpt "DEFAULT"
{
$$ = ast.RowFormatDefault
}
| "ROW_FORMAT" EqOpt "DYNAMIC"
{
$$ = ast.RowFormatDynamic
}
| "ROW_FORMAT" EqOpt "FIXED"
{
$$ = ast.RowFormatFixed
}
| "ROW_FORMAT" EqOpt "COMPRESSED"
{
$$ = ast.RowFormatCompressed
}
| "ROW_FORMAT" EqOpt "REDUNDANT"
{
$$ = ast.RowFormatRedundant
}
| "ROW_FORMAT" EqOpt "COMPACT"
{
$$ = ast.RowFormatCompact
}
/*************************************Type Begin***************************************/
Type:
NumericType
{
$$ = $1
}
| StringType
{
$$ = $1
}
| DateAndTimeType
{
$$ = $1
}
NumericType:
IntegerType OptFieldLen FieldOpts
{
// TODO: check flen 0
x := types.NewFieldType($1.(byte))
x.Flen = $2.(int)
for _, o := range $3.([]*ast.TypeOpt) {
if o.IsUnsigned {
x.Flag |= mysql.UnsignedFlag
}
if o.IsZerofill {
x.Flag |= mysql.ZerofillFlag
}
}
$$ = x
}
| BooleanType FieldOpts
{
// TODO: check flen 0
x := types.NewFieldType($1.(byte))
x.Flen = 1
for _, o := range $2.([]*ast.TypeOpt) {
if o.IsUnsigned {
x.Flag |= mysql.UnsignedFlag
}
if o.IsZerofill {
x.Flag |= mysql.ZerofillFlag
}
}
$$ = x
}
| FixedPointType FloatOpt FieldOpts
{
fopt := $2.(*ast.FloatOpt)
x := types.NewFieldType($1.(byte))
x.Flen = fopt.Flen
x.Decimal = fopt.Decimal
for _, o := range $3.([]*ast.TypeOpt) {
if o.IsUnsigned {
x.Flag |= mysql.UnsignedFlag
}
if o.IsZerofill {
x.Flag |= mysql.ZerofillFlag
}
}
$$ = x
}
| FloatingPointType FloatOpt FieldOpts
{
fopt := $2.(*ast.FloatOpt)
x := types.NewFieldType($1.(byte))
x.Flen = fopt.Flen
if x.Tp == mysql.TypeFloat {
if x.Flen > 24 {
x.Tp = mysql.TypeDouble
}
}
x.Decimal = fopt.Decimal
for _, o := range $3.([]*ast.TypeOpt) {
if o.IsUnsigned {
x.Flag |= mysql.UnsignedFlag
}
if o.IsZerofill {
x.Flag |= mysql.ZerofillFlag
}
}
$$ = x
}
| BitValueType OptFieldLen
{
x := types.NewFieldType($1.(byte))
x.Flen = $2.(int)
if x.Flen == types.UnspecifiedLength {
x.Flen = 1
}
$$ = x
}
IntegerType:
"TINYINT"
{
$$ = mysql.TypeTiny
}
| "SMALLINT"
{
$$ = mysql.TypeShort
}
| "MEDIUMINT"
{
$$ = mysql.TypeInt24
}
| "INT"
{
$$ = mysql.TypeLong
}
| "INT1"
{
$$ = mysql.TypeTiny
}
| "INT2"
{
$$ = mysql.TypeShort
}
| "INT3"
{
$$ = mysql.TypeInt24
}
| "INT4"
{
$$ = mysql.TypeLong
}
| "INT8"
{
$$ = mysql.TypeLonglong
}
| "INTEGER"
{
$$ = mysql.TypeLong
}
| "BIGINT"
{
$$ = mysql.TypeLonglong
}
BooleanType:
"BOOL"
{
$$ = mysql.TypeTiny
}
| "BOOLEAN"
{
$$ = mysql.TypeTiny
}
OptInteger:
{}
| "INTEGER"
| "INT"
FixedPointType:
"DECIMAL"
{
$$ = mysql.TypeNewDecimal
}
| "NUMERIC"
{
$$ = mysql.TypeNewDecimal
}
FloatingPointType:
"FLOAT"
{
$$ = mysql.TypeFloat
}
| "REAL"
{
if parser.lexer.GetSQLMode().HasRealAsFloatMode() {
$$ = mysql.TypeFloat
} else {
$$ = mysql.TypeDouble
}
}
| "DOUBLE"
{
$$ = mysql.TypeDouble
}
| "DOUBLE" "PRECISION"
{
$$ = mysql.TypeDouble
}
BitValueType:
"BIT"
{
$$ = mysql.TypeBit
}
StringType:
NationalOpt "CHAR" FieldLen OptBinary
{
x := types.NewFieldType(mysql.TypeString)
x.Flen = $3.(int)
x.Charset = $4.(*ast.OptBinary).Charset
if $4.(*ast.OptBinary).IsBinary {
x.Flag |= mysql.BinaryFlag
}
$$ = x
}
| NationalOpt "CHAR" OptBinary
{
x := types.NewFieldType(mysql.TypeString)
x.Charset = $3.(*ast.OptBinary).Charset
if $3.(*ast.OptBinary).IsBinary {
x.Flag |= mysql.BinaryFlag
}
$$ = x
}
| "NATIONAL" "CHARACTER" FieldLen OptBinary
{
x := types.NewFieldType(mysql.TypeString)
x.Flen = $3.(int)
x.Charset = $4.(*ast.OptBinary).Charset
if $4.(*ast.OptBinary).IsBinary {
x.Flag |= mysql.BinaryFlag
}
$$ = x
}
| Varchar FieldLen OptBinary
{
x := types.NewFieldType(mysql.TypeVarchar)
x.Flen = $2.(int)
x.Charset = $3.(*ast.OptBinary).Charset
if $3.(*ast.OptBinary).IsBinary {
x.Flag |= mysql.BinaryFlag
}
$$ = x
}
| "BINARY" OptFieldLen
{
x := types.NewFieldType(mysql.TypeString)
x.Flen = $2.(int)
x.Charset = charset.CharsetBin
x.Collate = charset.CharsetBin
x.Flag |= mysql.BinaryFlag
$$ = x
}
| "VARBINARY" FieldLen
{
x := types.NewFieldType(mysql.TypeVarchar)
x.Flen = $2.(int)
x.Charset = charset.CharsetBin
x.Collate = charset.CharsetBin
x.Flag |= mysql.BinaryFlag
$$ = x
}
| BlobType
{
x := $1.(*types.FieldType)
x.Charset = charset.CharsetBin
x.Collate = charset.CharsetBin
x.Flag |= mysql.BinaryFlag
$$ = $1.(*types.FieldType)
}
| TextType OptBinary
{
x := $1.(*types.FieldType)
x.Charset = $2.(*ast.OptBinary).Charset
if $2.(*ast.OptBinary).IsBinary {
x.Flag |= mysql.BinaryFlag
}
$$ = x
}
| "ENUM" '(' StringList ')' OptCharset
{
x := types.NewFieldType(mysql.TypeEnum)
x.Elems = $3.([]string)
x.Charset = $5.(string)
$$ = x
}
| "SET" '(' StringList ')' OptCharset
{
x := types.NewFieldType(mysql.TypeSet)
x.Elems = $3.([]string)
x.Charset = $5.(string)
$$ = x
}
| "JSON"
{
x := types.NewFieldType(mysql.TypeJSON)
x.Decimal = 0
x.Charset = charset.CharsetBin
x.Collate = charset.CollationBin
$$ = x
}
NationalOpt:
{}
| "NATIONAL"
Varchar:
"NATIONAL" "VARCHAR"
| "VARCHAR"
| "NVARCHAR"
BlobType:
"TINYBLOB"
{
x := types.NewFieldType(mysql.TypeTinyBlob)
$$ = x
}
| "BLOB" OptFieldLen
{
x := types.NewFieldType(mysql.TypeBlob)
x.Flen = $2.(int)
$$ = x
}
| "MEDIUMBLOB"
{
x := types.NewFieldType(mysql.TypeMediumBlob)
$$ = x
}
| "LONGBLOB"
{
x := types.NewFieldType(mysql.TypeLongBlob)
$$ = x
}
TextType:
"TINYTEXT"
{
x := types.NewFieldType(mysql.TypeTinyBlob)
$$ = x
}
| "TEXT" OptFieldLen
{
x := types.NewFieldType(mysql.TypeBlob)
x.Flen = $2.(int)
$$ = x
}
| "MEDIUMTEXT"
{
x := types.NewFieldType(mysql.TypeMediumBlob)
$$ = x
}
| "LONGTEXT"
{
x := types.NewFieldType(mysql.TypeLongBlob)
$$ = x
}
| "LONG" "VARCHAR"
{
x := types.NewFieldType(mysql.TypeMediumBlob)
$$ = x
}
| "GEOMETRY"
{
x := types.NewFieldType(mysql.TypeGeometry)
$$ = x
}
DateAndTimeType:
"DATE"
{
x := types.NewFieldType(mysql.TypeDate)
$$ = x
}
| "DATETIME" OptFieldLen
{
x := types.NewFieldType(mysql.TypeDatetime)
x.Flen = mysql.MaxDatetimeWidthNoFsp
x.Decimal = $2.(int)
if x.Decimal > 0 {
x.Flen = x.Flen + 1 + x.Decimal
}
$$ = x
}
| "TIMESTAMP" OptFieldLen
{
x := types.NewFieldType(mysql.TypeTimestamp)
x.Flen = mysql.MaxDatetimeWidthNoFsp
x.Decimal = $2.(int)
if x.Decimal > 0 {
x.Flen = x.Flen + 1 + x.Decimal
}
$$ = x
}
| "TIME" OptFieldLen
{
x := types.NewFieldType(mysql.TypeDuration)
x.Flen = mysql.MaxDurationWidthNoFsp
x.Decimal = $2.(int)
if x.Decimal > 0 {
x.Flen = x.Flen + 1 + x.Decimal
}
$$ = x
}
| "YEAR" OptFieldLen FieldOpts
{
x := types.NewFieldType(mysql.TypeYear)
x.Flen = $2.(int)
if x.Flen != types.UnspecifiedLength && x.Flen != 4 {
// yylex.AppendError(ErrInvalidYearColumnLength.GenWithStackByArgs())
yylex.AppendError(fmt.Errorf("Supports only YEAR or YEAR(4) column."))
return -1
}
$$ = x
}
FieldLen:
'(' LengthNum ')'
{
$$ = int($2.(uint64))
}
OptFieldLen:
{
$$ = types.UnspecifiedLength
}
| FieldLen
{
$$ = $1.(int)
}
FieldOpt:
"UNSIGNED"
{
$$ = &ast.TypeOpt{IsUnsigned: true}
}
| "SIGNED"
{
$$ = &ast.TypeOpt{IsUnsigned: false}
}
| "ZEROFILL"
{
$$ = &ast.TypeOpt{IsZerofill: true, IsUnsigned: true}
}
FieldOpts:
{
$$ = []*ast.TypeOpt{}
}
| FieldOpts FieldOpt
{
$$ = append($1.([]*ast.TypeOpt), $2.(*ast.TypeOpt))
}
FloatOpt:
{
$$ = &ast.FloatOpt{Flen: types.UnspecifiedLength, Decimal: types.UnspecifiedLength}
}
| FieldLen
{
$$ = &ast.FloatOpt{Flen: $1.(int), Decimal: types.UnspecifiedLength}
}
| Precision
{
$$ = $1.(*ast.FloatOpt)
}
Precision:
'(' LengthNum ',' LengthNum ')'
{
$$ = &ast.FloatOpt{Flen: int($2.(uint64)), Decimal: int($4.(uint64))}
}
OptBinMod:
{
$$ = false
}
| "BINARY"
{
$$ = true
}
OptBinary:
{
$$ = &ast.OptBinary{
IsBinary: false,
Charset: "",
}
}
| "BINARY" OptCharset
{
$$ = &ast.OptBinary{
IsBinary: true,
Charset: $2.(string),
}
}
| CharsetKw CharsetName OptBinMod
{
$$ = &ast.OptBinary{
IsBinary: $3.(bool),
Charset: $2.(string),
}
}
OptCharset:
{
$$ = ""
}
| CharsetKw CharsetName
{
$$ = $2.(string)
}
CharsetKw:
"CHARACTER" "SET"
| "CHARSET"
| "CHAR" "SET"
OptCollate:
{
$$ = ""
}
| "COLLATE" CollationName
{
$$ = $2.(string)
}
StringList:
stringLit
{
$$ = []string{$1}
}
| StringList ',' stringLit
{
$$ = append($1.([]string), $3)
}
StringName:
stringLit
{
$$ = $1
}
| Identifier
{
$$ = $1
}
/***********************************************************************************
* Update Statement
* See https://dev.mysql.com/doc/refman/5.7/en/update.html
***********************************************************************************/
UpdateStmt:
"UPDATE" TableOptimizerHints PriorityOpt IgnoreOptional TableRef "SET" AssignmentList WhereClauseOptional OrderByOptional LimitClause
{
var refs *ast.Join
if x, ok := $5.(*ast.Join); ok {
refs = x
} else {
refs = &ast.Join{Left: $5.(ast.ResultSetNode)}
}
st := &ast.UpdateStmt{
Priority: $3.(mysql.PriorityEnum),
TableRefs: &ast.TableRefsClause{TableRefs: refs},
List: $7.([]*ast.Assignment),
IgnoreErr: $4.(bool),
}
if $2 != nil {
st.TableHints = $2.([]*ast.TableOptimizerHint)
}
if $8 != nil {
st.Where = $8.(ast.ExprNode)
}
if $9 != nil {
st.Order = $9.(*ast.OrderByClause)
}
if $10 != nil {
st.Limit = $10.(*ast.Limit)
}
$$ = st
}
| "UPDATE" TableOptimizerHints PriorityOpt IgnoreOptional TableRefs "SET" AssignmentList WhereClauseOptional
{
st := &ast.UpdateStmt{
Priority: $3.(mysql.PriorityEnum),
TableRefs: &ast.TableRefsClause{TableRefs: $5.(*ast.Join)},
List: $7.([]*ast.Assignment),
IgnoreErr: $4.(bool),
}
if $2 != nil {
st.TableHints = $2.([]*ast.TableOptimizerHint)
}
if $8 != nil {
st.Where = $8.(ast.ExprNode)
}
$$ = st
}
UseStmt:
"USE" DBName
{
$$ = &ast.UseStmt{DBName: $2.(string)}
}
WhereClause:
"WHERE" Expression
{
$$ = $2
}
WhereClauseOptional:
{
$$ = nil
}
| WhereClause
{
$$ = $1
}
CommaOpt:
{}
| ','
{}
/************************************************************************************
* Account Management Statements
* https://dev.mysql.com/doc/refman/5.7/en/account-management-sql.html
************************************************************************************/
CreateUserStmt:
"CREATE" "USER" IfNotExists UserSpecList
{
// See https://dev.mysql.com/doc/refman/5.7/en/create-user.html
$$ = &ast.CreateUserStmt{
IfNotExists: $3.(bool),
Specs: $4.([]*ast.UserSpec),
}
}
/* See http://dev.mysql.com/doc/refman/5.7/en/alter-user.html */
AlterUserStmt:
"ALTER" "USER" IfExists UserSpecList
{
$$ = &ast.AlterUserStmt{
IfExists: $3.(bool),
Specs: $4.([]*ast.UserSpec),
}
}
| "ALTER" "USER" IfExists "USER" '(' ')' "IDENTIFIED" "BY" AuthString
{
auth := &ast.AuthOption {
AuthString: $9.(string),
ByAuthString: true,
}
$$ = &ast.AlterUserStmt{
IfExists: $3.(bool),
CurrentAuth: auth,
}
}
UserSpec:
Username AuthOption
{
userSpec := &ast.UserSpec{
User: $1.(*auth.UserIdentity),
}
if $2 != nil {
userSpec.AuthOpt = $2.(*ast.AuthOption)
}
$$ = userSpec
}
UserSpecList:
UserSpec
{
$$ = []*ast.UserSpec{$1.(*ast.UserSpec)}
}
| UserSpecList ',' UserSpec
{
$$ = append($1.([]*ast.UserSpec), $3.(*ast.UserSpec))
}
AuthOption:
{
$$ = nil
}
| "IDENTIFIED" "BY" AuthString
{
$$ = &ast.AuthOption {
AuthString: $3.(string),
ByAuthString: true,
}
}
| "IDENTIFIED" "WITH" StringName
{
$$ = nil
}
| "IDENTIFIED" "WITH" StringName "BY" AuthString
{
$$ = &ast.AuthOption {
AuthString: $5.(string),
ByAuthString: true,
}
}
| "IDENTIFIED" "WITH" StringName "AS" HashString
{
$$ = &ast.AuthOption{
HashString: $5.(string),
}
}
| "IDENTIFIED" "BY" "PASSWORD" HashString
{
$$ = &ast.AuthOption{
HashString: $4.(string),
}
}
HashString:
stringLit
{
$$ = $1
}
/*************************************************************************************
* Grant statement
* See https://dev.mysql.com/doc/refman/5.7/en/grant.html
*************************************************************************************/
GrantStmt:
"GRANT" PrivElemList "ON" ObjectType PrivLevel "TO" UserSpecList WithGrantOptionOpt
{
$$ = &ast.GrantStmt{
Privs: $2.([]*ast.PrivElem),
ObjectType: $4.(ast.ObjectTypeType),
Level: $5.(*ast.GrantLevel),
Users: $7.([]*ast.UserSpec),
WithGrant: $8.(bool),
}
}
WithGrantOptionOpt:
{
$$ = false
}
| "WITH" "GRANT" "OPTION"
{
$$ = true
}
| "WITH" "MAX_QUERIES_PER_HOUR" NUM
{
$$ = false
}
| "WITH" "MAX_UPDATES_PER_HOUR" NUM
{
$$ = false
}
| "WITH" "MAX_CONNECTIONS_PER_HOUR" NUM
{
$$ = false
}
| "WITH" "MAX_USER_CONNECTIONS" NUM
{
$$ = false
}
PrivElem:
PrivType
{
$$ = &ast.PrivElem{
Priv: $1.(mysql.PrivilegeType),
}
}
| PrivType '(' ColumnNameList ')'
{
$$ = &ast.PrivElem{
Priv: $1.(mysql.PrivilegeType),
Cols: $3.([]*ast.ColumnName),
}
}
PrivElemList:
PrivElem
{
$$ = []*ast.PrivElem{$1.(*ast.PrivElem)}
}
| PrivElemList ',' PrivElem
{
$$ = append($1.([]*ast.PrivElem), $3.(*ast.PrivElem))
}
PrivType:
"ALL"
{
$$ = mysql.AllPriv
}
| "ALL" "PRIVILEGES"
{
$$ = mysql.AllPriv
}
| "ALTER"
{
$$ = mysql.AlterPriv
}
| "CREATE"
{
$$ = mysql.CreatePriv
}
| "CREATE" "USER"
{
$$ = mysql.CreateUserPriv
}
| "TRIGGER"
{
$$ = mysql.TriggerPriv
}
| "DELETE"
{
$$ = mysql.DeletePriv
}
| "DROP"
{
$$ = mysql.DropPriv
}
| "PROCESS"
{
$$ = mysql.ProcessPriv
}
| "EXECUTE"
{
$$ = mysql.ExecutePriv
}
| "INDEX"
{
$$ = mysql.IndexPriv
}
| "INSERT"
{
$$ = mysql.InsertPriv
}
| "SELECT"
{
$$ = mysql.SelectPriv
}
| "SUPER"
{
$$ = mysql.SuperPriv
}
| "SHOW" "DATABASES"
{
$$ = mysql.ShowDBPriv
}
| "UPDATE"
{
$$ = mysql.UpdatePriv
}
| "GRANT" "OPTION"
{
$$ = mysql.GrantPriv
}
| "REFERENCES"
{
$$ = mysql.ReferencesPriv
}
| "REPLICATION" "SLAVE"
{
$$ = mysql.PrivilegeType(0)
}
| "REPLICATION" "CLIENT"
{
$$ = mysql.PrivilegeType(0)
}
| "USAGE"
{
$$ = mysql.PrivilegeType(0)
}
| "RELOAD"
{
$$ = mysql.PrivilegeType(0)
}
| "CREATE" "TEMPORARY" "TABLES"
{
$$ = mysql.PrivilegeType(0)
}
| "LOCK" "TABLES"
{
$$ = mysql.PrivilegeType(0)
}
| "CREATE" "VIEW"
{
$$ = mysql.PrivilegeType(0)
}
| "SHOW" "VIEW"
{
$$ = mysql.PrivilegeType(0)
}
| "CREATE" "ROUTINE"
{
$$ = mysql.PrivilegeType(0)
}
| "ALTER" "ROUTINE"
{
$$ = mysql.PrivilegeType(0)
}
| "EVENT"
{
$$ = mysql.PrivilegeType(0)
}
ObjectType:
{
$$ = ast.ObjectTypeNone
}
| "TABLE"
{
$$ = ast.ObjectTypeTable
}
PrivLevel:
'*'
{
$$ = &ast.GrantLevel {
Level: ast.GrantLevelDB,
}
}
| '*' '.' '*'
{
$$ = &ast.GrantLevel {
Level: ast.GrantLevelGlobal,
}
}
| Identifier '.' '*'
{
$$ = &ast.GrantLevel {
Level: ast.GrantLevelDB,
DBName: $1,
}
}
| Identifier '.' Identifier
{
$$ = &ast.GrantLevel {
Level: ast.GrantLevelTable,
DBName: $1,
TableName: $3,
}
}
| Identifier
{
$$ = &ast.GrantLevel {
Level: ast.GrantLevelTable,
TableName: $1,
}
}
/**************************************RevokeStmt*******************************************
* See https://dev.mysql.com/doc/refman/5.7/en/revoke.html
*******************************************************************************************/
RevokeStmt:
"REVOKE" PrivElemList "ON" ObjectType PrivLevel "FROM" UserSpecList
{
$$ = &ast.RevokeStmt{
Privs: $2.([]*ast.PrivElem),
ObjectType: $4.(ast.ObjectTypeType),
Level: $5.(*ast.GrantLevel),
Users: $7.([]*ast.UserSpec),
}
}
/**************************************LoadDataStmt*****************************************
* See https://dev.mysql.com/doc/refman/5.7/en/load-data.html
*******************************************************************************************/
LoadDataStmt:
"LOAD" "DATA" LocalOpt "INFILE" stringLit "INTO" "TABLE" TableName CharsetOpt Fields Lines IgnoreLines ColumnNameListOptWithBrackets
{
x := &ast.LoadDataStmt{
Path: $5,
Table: $8.(*ast.TableName),
Columns: $13.([]*ast.ColumnName),
IgnoreLines:$12.(uint64),
}
if $3 != nil {
x.IsLocal = true
}
if $10 != nil {
x.FieldsInfo = $10.(*ast.FieldsClause)
}
if $11 != nil {
x.LinesInfo = $11.(*ast.LinesClause)
}
$$ = x
}
IgnoreLines:
{
$$ = uint64(0)
}
| "IGNORE" NUM "LINES"
{
$$ = getUint64FromNUM($2)
}
CharsetOpt:
{}
| "CHARACTER" "SET" CharsetName
LocalOpt:
{
$$ = nil
}
| "LOCAL"
{
$$ = $1
}
Fields:
{
escape := "\\"
$$ = &ast.FieldsClause{
Terminated: "\t",
Escaped: escape[0],
}
}
| FieldsOrColumns FieldsTerminated Enclosed Escaped
{
escape := $4.(string)
if escape != "\\" && len(escape) > 1 {
yylex.Errorf("Incorrect arguments %s to ESCAPE", escape)
return 1
}
var enclosed byte
str := $3.(string)
if len(str) > 1 {
yylex.Errorf("Incorrect arguments %s to ENCLOSED", escape)
return 1
}else if len(str) != 0 {
enclosed = str[0]
}
var escaped byte
if len(escape) > 0 {
escaped = escape[0]
}
$$ = &ast.FieldsClause{
Terminated: $2.(string),
Enclosed: enclosed,
Escaped: escaped,
}
}
FieldsOrColumns:
"FIELDS" | "COLUMNS"
FieldsTerminated:
{
$$ = "\t"
}
| "TERMINATED" "BY" stringLit
{
$$ = $3
}
Enclosed:
{
$$ = ""
}
| "ENCLOSED" "BY" stringLit
{
$$ = $3
}
Escaped:
{
$$ = "\\"
}
| "ESCAPED" "BY" stringLit
{
$$ = $3
}
Lines:
{
$$ = &ast.LinesClause{Terminated: "\n"}
}
| "LINES" Starting LinesTerminated
{
$$ = &ast.LinesClause{Starting: $2.(string), Terminated: $3.(string)}
}
Starting:
{
$$ = ""
}
| "STARTING" "BY" stringLit
{
$$ = $3
}
LinesTerminated:
{
$$ = "\n"
}
| "TERMINATED" "BY" stringLit
{
$$ = $3
}
/*********************************************************************
* Lock/Unlock Tables
* See http://dev.mysql.com/doc/refman/5.7/en/lock-tables.html
* All the statement leaves empty. This is used to prevent mysqldump error.
*********************************************************************/
UnlockTablesStmt:
"UNLOCK" TablesTerminalSym {}
LockTablesStmt:
"LOCK" TablesTerminalSym TableLockList
{}
TablesTerminalSym:
"TABLES"
| "TABLE"
TableLock:
TableName LockType
LockType:
"READ"
| "READ" "LOCAL"
| "WRITE"
TableLockList:
TableLock
| TableLockList ',' TableLock
/********************************************************************
* Kill Statement
* See https://dev.mysql.com/doc/refman/5.7/en/kill.html
*******************************************************************/
KillStmt:
KillOrKillTiDB NUM
{
$$ = &ast.KillStmt{
ConnectionID: getUint64FromNUM($2),
TiDBExtension: $1.(bool),
}
}
| KillOrKillTiDB "CONNECTION" NUM
{
$$ = &ast.KillStmt{
ConnectionID: getUint64FromNUM($3),
TiDBExtension: $1.(bool),
}
}
| KillOrKillTiDB "QUERY" NUM
{
$$ = &ast.KillStmt{
ConnectionID: getUint64FromNUM($3),
Query: true,
TiDBExtension: $1.(bool),
}
}
KillOrKillTiDB:
"KILL"
{
$$ = false
}
/* KILL TIDB is a special grammar extension in TiDB, it can be used only when
the client connect to TiDB directly, not proxied under LVS. */
| "KILL" "TIDB"
{
$$ = true
}
/*******************************************************************************************/
LoadStatsStmt:
"LOAD" "STATS" stringLit
{
$$ = &ast.LoadStatsStmt{
Path: $3,
}
}
%%

Комментарий ( 0 )

Вы можете оставить комментарий после Вход в систему

1
https://gitlife.ru/oschina-mirror/hanchuanchuan-goInception.git
git@gitlife.ru:oschina-mirror/hanchuanchuan-goInception.git
oschina-mirror
hanchuanchuan-goInception
hanchuanchuan-goInception
v1.2.2