test_organization_events.py 233 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179
  1. import math
  2. import uuid
  3. from datetime import timedelta, timezone
  4. from unittest import mock
  5. import pytest
  6. from django.test import override_settings
  7. from django.urls import reverse
  8. from django.utils import timezone as django_timezone
  9. from snuba_sdk.column import Column
  10. from snuba_sdk.function import Function
  11. from sentry.discover.models import TeamKeyTransaction
  12. from sentry.issues.grouptype import ProfileFileIOGroupType
  13. from sentry.models.group import GroupStatus
  14. from sentry.models.projectteam import ProjectTeam
  15. from sentry.models.releaseprojectenvironment import ReleaseStages
  16. from sentry.models.transaction_threshold import (
  17. ProjectTransactionThreshold,
  18. ProjectTransactionThresholdOverride,
  19. TransactionMetric,
  20. )
  21. from sentry.search.events import constants
  22. from sentry.testutils.cases import (
  23. APITestCase,
  24. PerformanceIssueTestCase,
  25. ProfilesSnubaTestCase,
  26. SnubaTestCase,
  27. )
  28. from sentry.testutils.helpers import parse_link_header
  29. from sentry.testutils.helpers.datetime import before_now, freeze_time, iso_format
  30. from sentry.testutils.silo import region_silo_test
  31. from sentry.testutils.skips import requires_not_arm64
  32. from sentry.types.group import GroupSubStatus
  33. from sentry.utils import json
  34. from sentry.utils.samples import load_data
  35. from tests.sentry.issues.test_utils import SearchIssueTestMixin
  36. MAX_QUERYABLE_TRANSACTION_THRESHOLDS = 1
  37. class OrganizationEventsEndpointTestBase(APITestCase, SnubaTestCase):
  38. viewname = "sentry-api-0-organization-events"
  39. referrer = "api.organization-events"
  40. def setUp(self):
  41. super().setUp()
  42. self.nine_mins_ago = before_now(minutes=9)
  43. self.ten_mins_ago = before_now(minutes=10)
  44. self.ten_mins_ago_iso = iso_format(self.ten_mins_ago)
  45. self.eleven_mins_ago = before_now(minutes=11)
  46. self.eleven_mins_ago_iso = iso_format(self.eleven_mins_ago)
  47. self.transaction_data = load_data("transaction", timestamp=self.ten_mins_ago)
  48. self.features = {}
  49. def client_get(self, *args, **kwargs):
  50. return self.client.get(*args, **kwargs)
  51. def reverse_url(self):
  52. return reverse(
  53. self.viewname,
  54. kwargs={"organization_slug": self.organization.slug},
  55. )
  56. def do_request(self, query, features=None, **kwargs):
  57. if features is None:
  58. features = {"organizations:discover-basic": True}
  59. features.update(self.features)
  60. self.login_as(user=self.user)
  61. with self.feature(features):
  62. return self.client_get(self.reverse_url(), query, format="json", **kwargs)
  63. def load_data(self, platform="transaction", timestamp=None, duration=None, **kwargs):
  64. if timestamp is None:
  65. timestamp = self.ten_mins_ago
  66. min_age = before_now(minutes=10)
  67. if timestamp > min_age:
  68. # Sentry does some rounding of timestamps to improve cache hits in snuba.
  69. # This can result in events not being returns if the timestamps
  70. # are too recent.
  71. raise Exception(
  72. f"Please define a timestamp older than 10 minutes to avoid flakey tests. Want a timestamp before {min_age}, got: {timestamp} "
  73. )
  74. start_timestamp = None
  75. if duration is not None:
  76. start_timestamp = timestamp - duration
  77. start_timestamp = start_timestamp - timedelta(
  78. microseconds=start_timestamp.microsecond % 1000
  79. )
  80. return load_data(platform, timestamp=timestamp, start_timestamp=start_timestamp, **kwargs)
  81. @region_silo_test
  82. class OrganizationEventsEndpointTest(OrganizationEventsEndpointTestBase, PerformanceIssueTestCase):
  83. def test_no_projects(self):
  84. response = self.do_request({})
  85. assert response.status_code == 200, response.content
  86. assert response.data["data"] == []
  87. assert response.data["meta"] == {
  88. "tips": {"query": "Need at least one valid project to query."}
  89. }
  90. def test_environment_filter(self):
  91. self.create_environment(self.project, name="production")
  92. self.store_event(
  93. data={
  94. "event_id": "a" * 32,
  95. "environment": "staging",
  96. "timestamp": self.ten_mins_ago_iso,
  97. },
  98. project_id=self.project.id,
  99. )
  100. self.store_event(
  101. data={
  102. "event_id": "b" * 32,
  103. "environment": "production",
  104. "timestamp": self.ten_mins_ago_iso,
  105. },
  106. project_id=self.project.id,
  107. )
  108. query = {
  109. "field": ["id", "project.id"],
  110. "project": [self.project.id],
  111. "environment": ["staging", "production"],
  112. }
  113. response = self.do_request(query)
  114. assert response.status_code == 200
  115. assert len(response.data["data"]) == 2
  116. def test_performance_view_feature(self):
  117. self.store_event(
  118. data={
  119. "event_id": "a" * 32,
  120. "timestamp": self.ten_mins_ago_iso,
  121. "fingerprint": ["group1"],
  122. },
  123. project_id=self.project.id,
  124. )
  125. query = {"field": ["id", "project.id"], "project": [self.project.id]}
  126. response = self.do_request(query)
  127. assert response.status_code == 200
  128. assert len(response.data["data"]) == 1
  129. def test_multi_project_feature_gate_rejection(self):
  130. team = self.create_team(organization=self.organization, members=[self.user])
  131. project = self.create_project(organization=self.organization, teams=[team])
  132. project2 = self.create_project(organization=self.organization, teams=[team])
  133. query = {"field": ["id", "project.id"], "project": [project.id, project2.id]}
  134. response = self.do_request(query)
  135. assert response.status_code == 400
  136. assert "events from multiple projects" in response.data["detail"]
  137. def test_multi_project_feature_gate_replays(self):
  138. team = self.create_team(organization=self.organization, members=[self.user])
  139. project = self.create_project(organization=self.organization, teams=[team])
  140. project2 = self.create_project(organization=self.organization, teams=[team])
  141. query = {"field": ["id", "project.id"], "project": [project.id, project2.id]}
  142. response = self.do_request(query, **{"HTTP_X-Sentry-Replay-Request": "1"})
  143. assert response.status_code == 200
  144. def test_invalid_search_terms(self):
  145. self.create_project()
  146. query = {"field": ["id"], "query": "hi \n there"}
  147. response = self.do_request(query)
  148. assert response.status_code == 400, response.content
  149. assert (
  150. response.data["detail"]
  151. == "Parse error at 'hi \n ther' (column 4). This is commonly caused by unmatched parentheses. Enclose any text in double quotes."
  152. )
  153. def test_invalid_trace_span(self):
  154. self.create_project()
  155. query = {"field": ["id"], "query": "trace.span:invalid"}
  156. response = self.do_request(query)
  157. assert response.status_code == 400, response.content
  158. assert (
  159. response.data["detail"]
  160. == "trace.span must be a valid 16 character hex (containing only digits, or a-f characters)"
  161. )
  162. query = {"field": ["id"], "query": "trace.parent_span:invalid"}
  163. response = self.do_request(query)
  164. assert response.status_code == 400, response.content
  165. assert (
  166. response.data["detail"]
  167. == "trace.parent_span must be a valid 16 character hex (containing only digits, or a-f characters)"
  168. )
  169. query = {"field": ["id"], "query": "trace.span:*"}
  170. response = self.do_request(query)
  171. assert response.status_code == 400, response.content
  172. assert (
  173. response.data["detail"] == "Wildcard conditions are not permitted on `trace.span` field"
  174. )
  175. query = {"field": ["id"], "query": "trace.parent_span:*"}
  176. response = self.do_request(query)
  177. assert response.status_code == 400, response.content
  178. assert (
  179. response.data["detail"]
  180. == "Wildcard conditions are not permitted on `trace.parent_span` field"
  181. )
  182. def test_has_trace_context(self):
  183. self.store_event(
  184. data={
  185. "event_id": "a" * 32,
  186. "message": "how to make fast",
  187. "timestamp": self.ten_mins_ago_iso,
  188. "contexts": {
  189. "trace": {
  190. "span_id": "a" * 16,
  191. "trace_id": "b" * 32,
  192. },
  193. },
  194. },
  195. project_id=self.project.id,
  196. )
  197. query = {"field": ["id", "trace.parent_span"], "query": "has:trace.span"}
  198. response = self.do_request(query)
  199. assert response.status_code == 200, response.content
  200. assert len(response.data["data"]) == 1
  201. assert response.data["data"][0]["id"] == "a" * 32
  202. query = {"field": ["id"], "query": "has:trace.parent_span"}
  203. response = self.do_request(query)
  204. assert response.status_code == 200, response.content
  205. assert len(response.data["data"]) == 0
  206. def test_not_has_trace_context(self):
  207. self.store_event(
  208. data={
  209. "event_id": "a" * 32,
  210. "message": "how to make fast",
  211. "timestamp": self.ten_mins_ago_iso,
  212. "contexts": {
  213. "trace": {
  214. "span_id": "a" * 16,
  215. "trace_id": "b" * 32,
  216. },
  217. },
  218. },
  219. project_id=self.project.id,
  220. )
  221. query = {"field": ["id", "trace.parent_span"], "query": "!has:trace.span"}
  222. response = self.do_request(query)
  223. assert response.status_code == 200, response.content
  224. assert len(response.data["data"]) == 0
  225. query = {"field": ["id"], "query": "!has:trace.parent_span"}
  226. response = self.do_request(query)
  227. assert response.status_code == 200, response.content
  228. assert len(response.data["data"]) == 1
  229. assert response.data["data"][0]["id"] == "a" * 32
  230. def test_parent_span_id_in_context(self):
  231. self.store_event(
  232. data={
  233. "event_id": "a" * 32,
  234. "message": "how to make fast",
  235. "timestamp": self.ten_mins_ago_iso,
  236. "contexts": {
  237. "trace": {
  238. "span_id": "a" * 16,
  239. "trace_id": "b" * 32,
  240. "parent_span_id": "c" * 16,
  241. },
  242. },
  243. },
  244. project_id=self.project.id,
  245. )
  246. query = {"field": ["id"], "query": f"trace.parent_span:{'c' * 16}"}
  247. response = self.do_request(query)
  248. assert response.status_code == 200, response.content
  249. assert len(response.data["data"]) == 1
  250. assert response.data["data"][0]["id"] == "a" * 32
  251. def test_out_of_retention(self):
  252. self.create_project()
  253. with self.options({"system.event-retention-days": 10}):
  254. query = {
  255. "field": ["id", "timestamp"],
  256. "orderby": ["-timestamp", "-id"],
  257. "start": iso_format(before_now(days=20)),
  258. "end": iso_format(before_now(days=15)),
  259. }
  260. response = self.do_request(query)
  261. assert response.status_code == 400, response.content
  262. assert response.data["detail"] == "Invalid date range. Please try a more recent date range."
  263. def test_raw_data(self):
  264. self.store_event(
  265. data={
  266. "event_id": "a" * 32,
  267. "environment": "staging",
  268. "timestamp": self.eleven_mins_ago_iso,
  269. "user": {"ip_address": "127.0.0.1", "email": "foo@example.com"},
  270. },
  271. project_id=self.project.id,
  272. )
  273. self.store_event(
  274. data={
  275. "event_id": "b" * 32,
  276. "environment": "staging",
  277. "timestamp": self.ten_mins_ago_iso,
  278. "user": {"ip_address": "127.0.0.1", "email": "foo@example.com"},
  279. },
  280. project_id=self.project.id,
  281. )
  282. query = {
  283. "field": ["id", "project.id", "user.email", "user.ip", "timestamp"],
  284. "orderby": "-timestamp",
  285. }
  286. response = self.do_request(query)
  287. assert response.status_code == 200, response.content
  288. data = response.data["data"]
  289. assert len(data) == 2
  290. assert data[0]["id"] == "b" * 32
  291. assert data[0]["project.id"] == self.project.id
  292. assert data[0]["user.email"] == "foo@example.com"
  293. assert "project.name" not in data[0], "project.id does not auto select name"
  294. assert "project" not in data[0]
  295. meta = response.data["meta"]
  296. field_meta = meta["fields"]
  297. assert field_meta["id"] == "string"
  298. assert field_meta["user.email"] == "string"
  299. assert field_meta["user.ip"] == "string"
  300. assert field_meta["timestamp"] == "date"
  301. def test_project_name(self):
  302. self.store_event(
  303. data={
  304. "event_id": "a" * 32,
  305. "environment": "staging",
  306. "timestamp": self.ten_mins_ago_iso,
  307. },
  308. project_id=self.project.id,
  309. )
  310. query = {"field": ["project.name", "environment"]}
  311. response = self.do_request(query)
  312. assert response.status_code == 200, response.content
  313. assert len(response.data["data"]) == 1
  314. assert response.data["data"][0]["project.name"] == self.project.slug
  315. assert "project.id" not in response.data["data"][0]
  316. assert response.data["data"][0]["environment"] == "staging"
  317. def test_project_without_name(self):
  318. self.store_event(
  319. data={
  320. "event_id": "a" * 32,
  321. "environment": "staging",
  322. "timestamp": self.ten_mins_ago_iso,
  323. },
  324. project_id=self.project.id,
  325. )
  326. query = {"field": ["project", "environment"]}
  327. response = self.do_request(query)
  328. assert response.status_code == 200, response.content
  329. assert len(response.data["data"]) == 1
  330. assert response.data["data"][0]["project"] == self.project.slug
  331. assert response.data["meta"]["fields"]["project"] == "string"
  332. assert "project.id" not in response.data["data"][0]
  333. assert response.data["data"][0]["environment"] == "staging"
  334. def test_project_in_query(self):
  335. self.store_event(
  336. data={
  337. "event_id": "a" * 32,
  338. "environment": "staging",
  339. "timestamp": self.ten_mins_ago_iso,
  340. },
  341. project_id=self.project.id,
  342. )
  343. query = {
  344. "field": ["project", "count()"],
  345. "query": f'project:"{self.project.slug}"',
  346. "statsPeriod": "14d",
  347. }
  348. response = self.do_request(query)
  349. assert response.status_code == 200, response.content
  350. assert len(response.data["data"]) == 1
  351. assert response.data["data"][0]["project"] == self.project.slug
  352. assert "project.id" not in response.data["data"][0]
  353. def test_project_in_query_not_in_header(self):
  354. project = self.create_project()
  355. other_project = self.create_project()
  356. self.store_event(
  357. data={
  358. "event_id": "a" * 32,
  359. "environment": "staging",
  360. "timestamp": self.ten_mins_ago_iso,
  361. },
  362. project_id=project.id,
  363. )
  364. query = {
  365. "field": ["project", "count()"],
  366. "query": 'project:"%s"' % project.slug,
  367. "statsPeriod": "14d",
  368. "project": other_project.id,
  369. }
  370. response = self.do_request(query)
  371. assert response.status_code == 400, response.content
  372. assert (
  373. response.data["detail"]
  374. == f"Invalid query. Project(s) {project.slug} do not exist or are not actively selected."
  375. )
  376. def test_project_in_query_does_not_exist(self):
  377. self.create_project()
  378. query = {"field": ["project", "count()"], "query": "project:morty", "statsPeriod": "14d"}
  379. response = self.do_request(query)
  380. assert response.status_code == 400, response.content
  381. assert (
  382. response.data["detail"]
  383. == "Invalid query. Project(s) morty do not exist or are not actively selected."
  384. )
  385. def test_not_project_in_query_but_in_header(self):
  386. team = self.create_team(organization=self.organization, members=[self.user])
  387. project = self.create_project(organization=self.organization, teams=[team])
  388. project2 = self.create_project(organization=self.organization, teams=[team])
  389. self.store_event(
  390. data={
  391. "event_id": "a" * 32,
  392. "timestamp": self.ten_mins_ago_iso,
  393. "fingerprint": ["group1"],
  394. },
  395. project_id=project.id,
  396. )
  397. self.store_event(
  398. data={
  399. "event_id": "b" * 32,
  400. "timestamp": self.ten_mins_ago_iso,
  401. "fingerprint": ["group2"],
  402. },
  403. project_id=project2.id,
  404. )
  405. query = {
  406. "field": ["id", "project.id"],
  407. "project": [project.id],
  408. "query": f"!project:{project2.slug}",
  409. }
  410. response = self.do_request(query)
  411. assert response.status_code == 200
  412. assert response.data["data"] == [{"id": "a" * 32, "project.id": project.id}]
  413. def test_not_project_in_query_with_all_projects(self):
  414. team = self.create_team(organization=self.organization, members=[self.user])
  415. project = self.create_project(organization=self.organization, teams=[team])
  416. project2 = self.create_project(organization=self.organization, teams=[team])
  417. self.store_event(
  418. data={
  419. "event_id": "a" * 32,
  420. "timestamp": self.ten_mins_ago_iso,
  421. "fingerprint": ["group1"],
  422. },
  423. project_id=project.id,
  424. )
  425. self.store_event(
  426. data={
  427. "event_id": "b" * 32,
  428. "timestamp": self.ten_mins_ago_iso,
  429. "fingerprint": ["group2"],
  430. },
  431. project_id=project2.id,
  432. )
  433. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  434. query = {
  435. "field": ["id", "project.id"],
  436. "project": [-1],
  437. "query": f"!project:{project2.slug}",
  438. }
  439. response = self.do_request(query, features=features)
  440. assert response.status_code == 200
  441. assert response.data["data"] == [{"id": "a" * 32, "project.id": project.id}]
  442. def test_project_condition_used_for_automatic_filters(self):
  443. self.store_event(
  444. data={
  445. "event_id": "a" * 32,
  446. "environment": "staging",
  447. "timestamp": self.ten_mins_ago_iso,
  448. },
  449. project_id=self.project.id,
  450. )
  451. query = {
  452. "field": ["project", "count()"],
  453. "query": f'project:"{self.project.slug}"',
  454. "statsPeriod": "14d",
  455. }
  456. response = self.do_request(query)
  457. assert response.status_code == 200, response.content
  458. assert len(response.data["data"]) == 1
  459. assert response.data["data"][0]["project"] == self.project.slug
  460. assert "project.id" not in response.data["data"][0]
  461. def test_auto_insert_project_name_when_event_id_present(self):
  462. self.store_event(
  463. data={
  464. "event_id": "a" * 32,
  465. "environment": "staging",
  466. "timestamp": self.ten_mins_ago_iso,
  467. },
  468. project_id=self.project.id,
  469. )
  470. query = {"field": ["id"], "statsPeriod": "1h"}
  471. response = self.do_request(query)
  472. assert response.status_code == 200, response.content
  473. assert response.data["data"] == [{"project.name": self.project.slug, "id": "a" * 32}]
  474. def test_auto_insert_project_name_when_event_id_present_with_aggregate(self):
  475. self.store_event(
  476. data={
  477. "event_id": "a" * 32,
  478. "environment": "staging",
  479. "timestamp": self.ten_mins_ago_iso,
  480. },
  481. project_id=self.project.id,
  482. )
  483. query = {"field": ["id", "count()"], "statsPeriod": "1h"}
  484. response = self.do_request(query)
  485. assert response.status_code == 200, response.content
  486. assert response.data["data"] == [
  487. {"project.name": self.project.slug, "id": "a" * 32, "count()": 1}
  488. ]
  489. def test_performance_short_group_id(self):
  490. event = self.create_performance_issue()
  491. query = {
  492. "field": ["count()"],
  493. "statsPeriod": "1h",
  494. "query": f"project:{event.group.project.slug} issue:{event.group.qualified_short_id}",
  495. "dataset": "issuePlatform",
  496. }
  497. response = self.do_request(query)
  498. assert response.status_code == 200, response.content
  499. assert response.data["data"][0]["count()"] == 1
  500. def test_multiple_performance_short_group_ids_filter(self):
  501. event1 = self.create_performance_issue()
  502. event2 = self.create_performance_issue()
  503. query = {
  504. "field": ["count()"],
  505. "statsPeriod": "1h",
  506. "query": f"project:{event1.group.project.slug} issue:[{event1.group.qualified_short_id},{event2.group.qualified_short_id}]",
  507. "dataset": "issuePlatform",
  508. }
  509. response = self.do_request(query)
  510. assert response.status_code == 200, response.content
  511. assert response.data["data"][0]["count()"] == 2
  512. def test_event_id_with_in_search(self):
  513. self.store_event(
  514. data={
  515. "event_id": "a" * 32,
  516. "environment": "staging1",
  517. "timestamp": self.ten_mins_ago_iso,
  518. },
  519. project_id=self.project.id,
  520. )
  521. self.store_event(
  522. data={
  523. "event_id": "b" * 32,
  524. "environment": "staging2",
  525. "timestamp": self.ten_mins_ago_iso,
  526. },
  527. project_id=self.project.id,
  528. )
  529. # Should not show up
  530. self.store_event(
  531. data={
  532. "event_id": "c" * 32,
  533. "environment": "staging3",
  534. "timestamp": self.ten_mins_ago_iso,
  535. },
  536. project_id=self.project.id,
  537. )
  538. query = {
  539. "field": ["id", "environment"],
  540. "statsPeriod": "1h",
  541. "query": f"id:[{'a' * 32}, {'b' * 32}]",
  542. "orderby": "environment",
  543. }
  544. response = self.do_request(query)
  545. assert response.status_code == 200, response.content
  546. assert len(response.data["data"]) == 2
  547. assert response.data["data"][0]["id"] == "a" * 32
  548. assert response.data["data"][1]["id"] == "b" * 32
  549. def test_user_search(self):
  550. self.transaction_data["user"] = {
  551. "email": "foo@example.com",
  552. "id": "123",
  553. "ip_address": "127.0.0.1",
  554. "username": "foo",
  555. }
  556. self.store_event(self.transaction_data, project_id=self.project.id)
  557. fields = {
  558. "email": "user.email",
  559. "id": "user.id",
  560. "ip_address": "user.ip",
  561. "username": "user.username",
  562. }
  563. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  564. for key, value in self.transaction_data["user"].items():
  565. field = fields[key]
  566. query = {
  567. "field": ["project", "user"],
  568. "query": f"{field}:{value}",
  569. "statsPeriod": "14d",
  570. }
  571. response = self.do_request(query, features=features)
  572. assert response.status_code == 200, response.content
  573. assert len(response.data["data"]) == 1
  574. assert response.data["data"][0]["project"] == self.project.slug
  575. assert response.data["data"][0]["user"] == "id:123"
  576. def test_has_user(self):
  577. self.store_event(self.transaction_data, project_id=self.project.id)
  578. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  579. for value in self.transaction_data["user"].values():
  580. query = {"field": ["project", "user"], "query": "has:user", "statsPeriod": "14d"}
  581. response = self.do_request(query, features=features)
  582. assert response.status_code == 200, response.content
  583. assert len(response.data["data"]) == 1
  584. assert response.data["data"][0]["user"] == "ip:{}".format(
  585. self.transaction_data["user"]["ip_address"]
  586. )
  587. def test_team_param_no_access(self):
  588. org = self.create_organization(
  589. owner=self.user, # use other user as owner
  590. name="foo",
  591. flags=0, # disable default allow_joinleave
  592. )
  593. project = self.create_project(name="baz", organization=org)
  594. user = self.create_user()
  595. self.login_as(user=user, superuser=False)
  596. team = self.create_team(organization=org, name="Team Bar")
  597. project.add_team(team)
  598. self.store_event(
  599. data={
  600. "event_id": "a" * 32,
  601. "timestamp": self.ten_mins_ago_iso,
  602. "fingerprint": ["group1"],
  603. },
  604. project_id=project.id,
  605. )
  606. query = {"field": ["id", "project.id"], "project": [project.id], "team": [team.id]}
  607. response = self.do_request(query)
  608. assert response.status_code == 403, response.content
  609. assert response.data["detail"] == "You do not have permission to perform this action."
  610. def test_team_is_nan(self):
  611. query = {"field": ["id"], "project": [self.project.id], "team": [math.nan]}
  612. response = self.do_request(query)
  613. assert response.status_code == 400, response.content
  614. assert response.data["detail"] == "Invalid Team ID: nan"
  615. def test_comparison_operators_on_numeric_field(self):
  616. event = self.store_event(
  617. {"timestamp": iso_format(before_now(minutes=1))}, project_id=self.project.id
  618. )
  619. query = {"field": ["issue"], "query": f"issue.id:>{event.group.id - 1}"}
  620. response = self.do_request(query)
  621. assert response.status_code == 200, response.content
  622. assert len(response.data["data"]) == 1
  623. assert response.data["data"][0]["issue"] == event.group.qualified_short_id
  624. query = {"field": ["issue"], "query": f"issue.id:>{event.group.id}"}
  625. response = self.do_request(query)
  626. assert response.status_code == 200, response.content
  627. assert len(response.data["data"]) == 0
  628. def test_negation_on_numeric_field_excludes_issue(self):
  629. event = self.store_event({"timestamp": self.ten_mins_ago_iso}, project_id=self.project.id)
  630. query = {"field": ["issue"], "query": f"issue.id:{event.group.id}"}
  631. response = self.do_request(query)
  632. assert response.status_code == 200, response.content
  633. assert len(response.data["data"]) == 1
  634. assert response.data["data"][0]["issue"] == event.group.qualified_short_id
  635. query = {"field": ["issue"], "query": f"!issue.id:{event.group.id}"}
  636. response = self.do_request(query)
  637. assert response.status_code == 200, response.content
  638. assert len(response.data["data"]) == 0
  639. def test_negation_on_numeric_in_filter_excludes_issue(self):
  640. event = self.store_event({"timestamp": self.ten_mins_ago_iso}, project_id=self.project.id)
  641. query = {"field": ["issue"], "query": f"issue.id:[{event.group.id}]"}
  642. response = self.do_request(query)
  643. assert response.status_code == 200, response.content
  644. assert len(response.data["data"]) == 1
  645. assert response.data["data"][0]["issue"] == event.group.qualified_short_id
  646. query = {"field": ["issue"], "query": f"!issue.id:[{event.group.id}]"}
  647. response = self.do_request(query)
  648. assert response.status_code == 200, response.content
  649. assert len(response.data["data"]) == 0
  650. def test_negation_on_duration_filter_excludes_transaction(self):
  651. event = self.store_event(self.transaction_data, project_id=self.project.id)
  652. duration = int(event.data.get("timestamp") - event.data.get("start_timestamp")) * 1000
  653. query = {"field": ["transaction"], "query": f"transaction.duration:{duration}"}
  654. response = self.do_request(query)
  655. assert response.status_code == 200, response.content
  656. assert len(response.data["data"]) == 1
  657. assert response.data["data"][0]["id"] == event.event_id
  658. query = {"field": ["transaction"], "query": f"!transaction.duration:{duration}"}
  659. response = self.do_request(query)
  660. assert response.status_code == 200, response.content
  661. assert len(response.data["data"]) == 0
  662. def test_has_issue(self):
  663. event = self.store_event({"timestamp": self.ten_mins_ago_iso}, project_id=self.project.id)
  664. self.store_event(self.transaction_data, project_id=self.project.id)
  665. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  666. # should only show 1 event of type default
  667. query = {"field": ["project", "issue"], "query": "has:issue", "statsPeriod": "14d"}
  668. response = self.do_request(query, features=features)
  669. assert response.status_code == 200, response.content
  670. assert len(response.data["data"]) == 1
  671. assert response.data["data"][0]["issue"] == event.group.qualified_short_id
  672. # should only show 1 event of type default
  673. query = {
  674. "field": ["project", "issue"],
  675. "query": "event.type:default has:issue",
  676. "statsPeriod": "14d",
  677. }
  678. response = self.do_request(query, features=features)
  679. assert response.status_code == 200, response.content
  680. assert len(response.data["data"]) == 1
  681. assert response.data["data"][0]["issue"] == event.group.qualified_short_id
  682. # should show no results because no the default event has an issue
  683. query = {
  684. "field": ["project", "issue"],
  685. "query": "event.type:default !has:issue",
  686. "statsPeriod": "14d",
  687. }
  688. response = self.do_request(query, features=features)
  689. assert response.status_code == 200, response.content
  690. assert len(response.data["data"]) == 0
  691. # should show no results because no transactions have issues
  692. query = {
  693. "field": ["project", "issue"],
  694. "query": "event.type:transaction has:issue",
  695. "statsPeriod": "14d",
  696. }
  697. response = self.do_request(query, features=features)
  698. assert response.status_code == 200, response.content
  699. assert len(response.data["data"]) == 0
  700. # should only show 1 event of type transaction since they don't have issues
  701. query = {
  702. "field": ["project", "issue"],
  703. "query": "event.type:transaction !has:issue",
  704. "statsPeriod": "14d",
  705. }
  706. response = self.do_request(query, features=features)
  707. assert response.status_code == 200, response.content
  708. assert len(response.data["data"]) == 1
  709. assert response.data["data"][0]["issue"] == "unknown"
  710. @pytest.mark.skip("Cannot look up group_id of transaction events")
  711. def test_unknown_issue(self):
  712. event = self.store_event({"timestamp": self.ten_mins_ago_iso}, project_id=self.project.id)
  713. self.store_event(self.transaction_data, project_id=self.project.id)
  714. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  715. query = {"field": ["project", "issue"], "query": "issue:unknown", "statsPeriod": "14d"}
  716. response = self.do_request(query, features=features)
  717. assert response.status_code == 200, response.content
  718. assert len(response.data["data"]) == 1
  719. assert response.data["data"][0]["issue"] == "unknown"
  720. query = {"field": ["project", "issue"], "query": "!issue:unknown", "statsPeriod": "14d"}
  721. response = self.do_request(query, features=features)
  722. assert response.status_code == 200, response.content
  723. assert len(response.data["data"]) == 1
  724. assert response.data["data"][0]["issue"] == event.group.qualified_short_id
  725. def test_negative_user_search(self):
  726. user_data = {"email": "foo@example.com", "id": "123", "username": "foo"}
  727. # Load an event with data that shouldn't match
  728. data = self.transaction_data.copy()
  729. data["transaction"] = "/transactions/nomatch"
  730. event_user = user_data.copy()
  731. event_user["id"] = "undefined"
  732. data["user"] = event_user
  733. self.store_event(data, project_id=self.project.id)
  734. # Load a matching event
  735. data = self.transaction_data.copy()
  736. data["transaction"] = "/transactions/matching"
  737. data["user"] = user_data
  738. self.store_event(data, project_id=self.project.id)
  739. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  740. query = {
  741. "field": ["project", "user"],
  742. "query": '!user:"id:undefined"',
  743. "statsPeriod": "14d",
  744. }
  745. response = self.do_request(query, features=features)
  746. assert response.status_code == 200, response.content
  747. assert len(response.data["data"]) == 1
  748. assert response.data["data"][0]["user"] == "id:{}".format(user_data["id"])
  749. assert "user.email" not in response.data["data"][0]
  750. assert "user.id" not in response.data["data"][0]
  751. def test_not_project_in_query(self):
  752. project1 = self.create_project()
  753. project2 = self.create_project()
  754. self.store_event(
  755. data={
  756. "event_id": "a" * 32,
  757. "environment": "staging",
  758. "timestamp": self.ten_mins_ago_iso,
  759. },
  760. project_id=project1.id,
  761. )
  762. self.store_event(
  763. data={
  764. "event_id": "b" * 32,
  765. "environment": "staging",
  766. "timestamp": self.ten_mins_ago_iso,
  767. },
  768. project_id=project2.id,
  769. )
  770. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  771. query = {
  772. "field": ["project", "count()"],
  773. "query": '!project:"%s"' % project1.slug,
  774. "statsPeriod": "14d",
  775. }
  776. response = self.do_request(query, features=features)
  777. assert response.status_code == 200, response.content
  778. assert len(response.data["data"]) == 1
  779. assert response.data["data"][0]["project"] == project2.slug
  780. assert "project.id" not in response.data["data"][0]
  781. def test_error_handled_condition(self):
  782. prototype = self.load_data(platform="android-ndk")
  783. events = (
  784. ("a" * 32, "not handled", False),
  785. ("b" * 32, "was handled", True),
  786. ("c" * 32, "undefined", None),
  787. )
  788. for event in events:
  789. prototype["event_id"] = event[0]
  790. prototype["message"] = event[1]
  791. prototype["exception"]["values"][0]["value"] = event[1]
  792. prototype["exception"]["values"][0]["mechanism"]["handled"] = event[2]
  793. prototype["timestamp"] = self.ten_mins_ago_iso
  794. self.store_event(data=prototype, project_id=self.project.id)
  795. with self.feature("organizations:discover-basic"):
  796. query = {
  797. "field": ["message", "error.handled"],
  798. "query": "error.handled:0",
  799. "orderby": "message",
  800. }
  801. response = self.do_request(query)
  802. assert response.status_code == 200, response.data
  803. assert 1 == len(response.data["data"])
  804. assert 0 == response.data["data"][0]["error.handled"]
  805. with self.feature("organizations:discover-basic"):
  806. query = {
  807. "field": ["message", "error.handled"],
  808. "query": "error.handled:1",
  809. "orderby": "message",
  810. }
  811. response = self.do_request(query)
  812. assert response.status_code == 200, response.data
  813. assert 2 == len(response.data["data"])
  814. assert 1 == response.data["data"][0]["error.handled"]
  815. assert 1 == response.data["data"][1]["error.handled"]
  816. def test_error_unhandled_condition(self):
  817. prototype = self.load_data(platform="android-ndk")
  818. events = (
  819. ("a" * 32, "not handled", False),
  820. ("b" * 32, "was handled", True),
  821. ("c" * 32, "undefined", None),
  822. )
  823. for event in events:
  824. prototype["event_id"] = event[0]
  825. prototype["message"] = event[1]
  826. prototype["exception"]["values"][0]["value"] = event[1]
  827. prototype["exception"]["values"][0]["mechanism"]["handled"] = event[2]
  828. prototype["timestamp"] = self.ten_mins_ago_iso
  829. self.store_event(data=prototype, project_id=self.project.id)
  830. with self.feature("organizations:discover-basic"):
  831. query = {
  832. "field": ["message", "error.unhandled", "error.handled"],
  833. "query": "error.unhandled:true",
  834. "orderby": "message",
  835. }
  836. response = self.do_request(query)
  837. assert response.status_code == 200, response.data
  838. assert 1 == len(response.data["data"])
  839. assert 0 == response.data["data"][0]["error.handled"]
  840. assert 1 == response.data["data"][0]["error.unhandled"]
  841. with self.feature("organizations:discover-basic"):
  842. query = {
  843. "field": ["message", "error.handled", "error.unhandled"],
  844. "query": "error.unhandled:false",
  845. "orderby": "message",
  846. }
  847. response = self.do_request(query)
  848. assert response.status_code == 200, response.data
  849. assert 2 == len(response.data["data"])
  850. assert 1 == response.data["data"][0]["error.handled"]
  851. assert 0 == response.data["data"][0]["error.unhandled"]
  852. assert 1 == response.data["data"][1]["error.handled"]
  853. assert 0 == response.data["data"][1]["error.unhandled"]
  854. def test_groupby_error_handled_and_unhandled(self):
  855. prototype = self.load_data(platform="android-ndk")
  856. events = (
  857. ("a" * 32, "not handled", False),
  858. ("b" * 32, "was handled", True),
  859. ("c" * 32, "undefined", None),
  860. )
  861. for event in events:
  862. prototype["event_id"] = event[0]
  863. prototype["message"] = event[1]
  864. prototype["exception"]["values"][0]["value"] = event[1]
  865. prototype["exception"]["values"][0]["mechanism"]["handled"] = event[2]
  866. prototype["timestamp"] = self.ten_mins_ago_iso
  867. self.store_event(data=prototype, project_id=self.project.id)
  868. with self.feature("organizations:discover-basic"):
  869. query = {
  870. "field": ["error.handled", "count()"],
  871. "query": "event.type:error",
  872. }
  873. response = self.do_request(query)
  874. assert response.status_code == 200, response.data
  875. assert 2 == len(response.data["data"])
  876. assert 0 == response.data["data"][0]["error.handled"]
  877. assert 1 == response.data["data"][0]["count()"]
  878. assert 1 == response.data["data"][1]["error.handled"]
  879. assert 2 == response.data["data"][1]["count()"]
  880. with self.feature("organizations:discover-basic"):
  881. query = {
  882. "field": ["error.unhandled", "count()"],
  883. "query": "event.type:error",
  884. }
  885. response = self.do_request(query)
  886. assert response.status_code == 200, response.data
  887. assert 2 == len(response.data["data"])
  888. assert 0 == response.data["data"][0]["error.unhandled"]
  889. assert 2 == response.data["data"][0]["count()"]
  890. assert 1 == response.data["data"][1]["error.unhandled"]
  891. assert 1 == response.data["data"][1]["count()"]
  892. def test_error_main_thread_condition(self):
  893. prototype = self.load_data(platform="android-ndk")
  894. prototype["timestamp"] = self.ten_mins_ago_iso
  895. self.store_event(data=prototype, project_id=self.project.id)
  896. with self.feature("organizations:discover-basic"):
  897. query = {
  898. "field": ["id", "project.id"],
  899. "query": "error.main_thread:true",
  900. "project": [self.project.id],
  901. }
  902. response = self.do_request(query)
  903. assert response.status_code == 200, response.data
  904. assert 1 == len(response.data["data"])
  905. with self.feature("organizations:discover-basic"):
  906. query = {
  907. "field": ["id", "project.id"],
  908. "query": "error.main_thread:false",
  909. "project": [self.project.id],
  910. }
  911. response = self.do_request(query)
  912. assert response.status_code == 200, response.data
  913. assert 0 == len(response.data["data"])
  914. def test_implicit_groupby(self):
  915. self.store_event(
  916. data={
  917. "event_id": "a" * 32,
  918. "timestamp": self.eleven_mins_ago_iso,
  919. "fingerprint": ["group_1"],
  920. },
  921. project_id=self.project.id,
  922. )
  923. event1 = self.store_event(
  924. data={
  925. "event_id": "b" * 32,
  926. "timestamp": self.ten_mins_ago_iso,
  927. "fingerprint": ["group_1"],
  928. },
  929. project_id=self.project.id,
  930. )
  931. event2 = self.store_event(
  932. data={
  933. "event_id": "c" * 32,
  934. "timestamp": self.ten_mins_ago_iso,
  935. "fingerprint": ["group_2"],
  936. },
  937. project_id=self.project.id,
  938. )
  939. query = {"field": ["count(id)", "project.id", "issue.id"], "orderby": "issue.id"}
  940. response = self.do_request(query)
  941. assert response.status_code == 200, response.content
  942. assert len(response.data["data"]) == 2
  943. data = response.data["data"]
  944. assert data[0] == {
  945. "project.id": self.project.id,
  946. "issue.id": event1.group_id,
  947. "count(id)": 2,
  948. }
  949. assert data[1] == {
  950. "project.id": self.project.id,
  951. "issue.id": event2.group_id,
  952. "count(id)": 1,
  953. }
  954. meta = response.data["meta"]["fields"]
  955. assert meta["count(id)"] == "integer"
  956. def test_orderby(self):
  957. self.store_event(
  958. data={"event_id": "a" * 32, "timestamp": self.eleven_mins_ago_iso},
  959. project_id=self.project.id,
  960. )
  961. self.store_event(
  962. data={"event_id": "b" * 32, "timestamp": self.ten_mins_ago_iso},
  963. project_id=self.project.id,
  964. )
  965. self.store_event(
  966. data={"event_id": "c" * 32, "timestamp": self.ten_mins_ago_iso},
  967. project_id=self.project.id,
  968. )
  969. query = {"field": ["id", "timestamp"], "orderby": ["-timestamp", "-id"]}
  970. response = self.do_request(query)
  971. assert response.status_code == 200, response.content
  972. data = response.data["data"]
  973. assert data[0]["id"] == "c" * 32
  974. assert data[1]["id"] == "b" * 32
  975. assert data[2]["id"] == "a" * 32
  976. def test_sort_title(self):
  977. self.store_event(
  978. data={"event_id": "a" * 32, "message": "zlast", "timestamp": self.eleven_mins_ago_iso},
  979. project_id=self.project.id,
  980. )
  981. self.store_event(
  982. data={"event_id": "b" * 32, "message": "second", "timestamp": self.ten_mins_ago_iso},
  983. project_id=self.project.id,
  984. )
  985. self.store_event(
  986. data={"event_id": "c" * 32, "message": "first", "timestamp": self.ten_mins_ago_iso},
  987. project_id=self.project.id,
  988. )
  989. query = {"field": ["id", "title"], "sort": "title"}
  990. response = self.do_request(query)
  991. assert response.status_code == 200, response.content
  992. data = response.data["data"]
  993. assert data[0]["id"] == "c" * 32
  994. assert data[1]["id"] == "b" * 32
  995. assert data[2]["id"] == "a" * 32
  996. def test_sort_invalid(self):
  997. self.create_project()
  998. query = {"field": ["id"], "sort": "garbage"}
  999. response = self.do_request(query)
  1000. assert response.status_code == 400
  1001. assert "sort by" in response.data["detail"]
  1002. def test_latest_release_alias(self):
  1003. event1 = self.store_event(
  1004. data={"event_id": "a" * 32, "timestamp": self.eleven_mins_ago_iso, "release": "0.8"},
  1005. project_id=self.project.id,
  1006. )
  1007. query = {"field": ["issue.id", "release"], "query": "release:latest"}
  1008. response = self.do_request(query)
  1009. assert response.status_code == 200, response.content
  1010. data = response.data["data"]
  1011. assert data[0]["issue.id"] == event1.group_id
  1012. assert data[0]["release"] == "0.8"
  1013. event2 = self.store_event(
  1014. data={"event_id": "a" * 32, "timestamp": self.ten_mins_ago_iso, "release": "0.9"},
  1015. project_id=self.project.id,
  1016. )
  1017. query = {"field": ["issue.id", "release"], "query": "release:latest"}
  1018. response = self.do_request(query)
  1019. assert response.status_code == 200, response.content
  1020. data = response.data["data"]
  1021. assert data[0]["issue.id"] == event2.group_id
  1022. assert data[0]["release"] == "0.9"
  1023. def test_semver(self):
  1024. release_1 = self.create_release(version="test@1.2.3")
  1025. release_2 = self.create_release(version="test@1.2.4")
  1026. release_3 = self.create_release(version="test@1.2.5")
  1027. release_1_e_1 = self.store_event(
  1028. data={"release": release_1.version, "timestamp": self.ten_mins_ago_iso},
  1029. project_id=self.project.id,
  1030. ).event_id
  1031. release_1_e_2 = self.store_event(
  1032. data={"release": release_1.version, "timestamp": self.ten_mins_ago_iso},
  1033. project_id=self.project.id,
  1034. ).event_id
  1035. release_2_e_1 = self.store_event(
  1036. data={"release": release_2.version, "timestamp": self.ten_mins_ago_iso},
  1037. project_id=self.project.id,
  1038. ).event_id
  1039. release_2_e_2 = self.store_event(
  1040. data={"release": release_2.version, "timestamp": self.ten_mins_ago_iso},
  1041. project_id=self.project.id,
  1042. ).event_id
  1043. release_3_e_1 = self.store_event(
  1044. data={"release": release_3.version, "timestamp": self.ten_mins_ago_iso},
  1045. project_id=self.project.id,
  1046. ).event_id
  1047. release_3_e_2 = self.store_event(
  1048. data={"release": release_3.version, "timestamp": self.ten_mins_ago_iso},
  1049. project_id=self.project.id,
  1050. ).event_id
  1051. query = {"field": ["id"], "query": f"{constants.SEMVER_ALIAS}:>1.2.3"}
  1052. response = self.do_request(query)
  1053. assert response.status_code == 200, response.content
  1054. assert {r["id"] for r in response.data["data"]} == {
  1055. release_2_e_1,
  1056. release_2_e_2,
  1057. release_3_e_1,
  1058. release_3_e_2,
  1059. }
  1060. query = {"field": ["id"], "query": f"{constants.SEMVER_ALIAS}:>=1.2.3"}
  1061. response = self.do_request(query)
  1062. assert response.status_code == 200, response.content
  1063. assert {r["id"] for r in response.data["data"]} == {
  1064. release_1_e_1,
  1065. release_1_e_2,
  1066. release_2_e_1,
  1067. release_2_e_2,
  1068. release_3_e_1,
  1069. release_3_e_2,
  1070. }
  1071. query = {"field": ["id"], "query": f"{constants.SEMVER_ALIAS}:<1.2.4"}
  1072. response = self.do_request(query)
  1073. assert response.status_code == 200, response.content
  1074. assert {r["id"] for r in response.data["data"]} == {
  1075. release_1_e_1,
  1076. release_1_e_2,
  1077. }
  1078. query = {"field": ["id"], "query": f"{constants.SEMVER_ALIAS}:1.2.3"}
  1079. response = self.do_request(query)
  1080. assert response.status_code == 200, response.content
  1081. assert {r["id"] for r in response.data["data"]} == {
  1082. release_1_e_1,
  1083. release_1_e_2,
  1084. }
  1085. query = {"field": ["id"], "query": f"!{constants.SEMVER_ALIAS}:1.2.3"}
  1086. response = self.do_request(query)
  1087. assert response.status_code == 200, response.content
  1088. assert {r["id"] for r in response.data["data"]} == {
  1089. release_2_e_1,
  1090. release_2_e_2,
  1091. release_3_e_1,
  1092. release_3_e_2,
  1093. }
  1094. def test_release_stage(self):
  1095. replaced_release = self.create_release(
  1096. version="replaced_release",
  1097. environments=[self.environment],
  1098. adopted=django_timezone.now(),
  1099. unadopted=django_timezone.now(),
  1100. )
  1101. adopted_release = self.create_release(
  1102. version="adopted_release",
  1103. environments=[self.environment],
  1104. adopted=django_timezone.now(),
  1105. )
  1106. self.create_release(version="not_adopted_release", environments=[self.environment])
  1107. adopted_release_e_1 = self.store_event(
  1108. data={
  1109. "release": adopted_release.version,
  1110. "timestamp": self.ten_mins_ago_iso,
  1111. "environment": self.environment.name,
  1112. },
  1113. project_id=self.project.id,
  1114. ).event_id
  1115. adopted_release_e_2 = self.store_event(
  1116. data={
  1117. "release": adopted_release.version,
  1118. "timestamp": self.ten_mins_ago_iso,
  1119. "environment": self.environment.name,
  1120. },
  1121. project_id=self.project.id,
  1122. ).event_id
  1123. replaced_release_e_1 = self.store_event(
  1124. data={
  1125. "release": replaced_release.version,
  1126. "timestamp": self.ten_mins_ago_iso,
  1127. "environment": self.environment.name,
  1128. },
  1129. project_id=self.project.id,
  1130. ).event_id
  1131. replaced_release_e_2 = self.store_event(
  1132. data={
  1133. "release": replaced_release.version,
  1134. "timestamp": self.ten_mins_ago_iso,
  1135. "environment": self.environment.name,
  1136. },
  1137. project_id=self.project.id,
  1138. ).event_id
  1139. query = {
  1140. "field": ["id"],
  1141. "query": f"{constants.RELEASE_STAGE_ALIAS}:{ReleaseStages.ADOPTED.value}",
  1142. "environment": [self.environment.name],
  1143. }
  1144. response = self.do_request(query)
  1145. assert response.status_code == 200, response.content
  1146. assert {r["id"] for r in response.data["data"]} == {
  1147. adopted_release_e_1,
  1148. adopted_release_e_2,
  1149. }
  1150. query = {
  1151. "field": ["id"],
  1152. "query": f"!{constants.RELEASE_STAGE_ALIAS}:{ReleaseStages.LOW_ADOPTION.value}",
  1153. "environment": [self.environment.name],
  1154. }
  1155. response = self.do_request(query)
  1156. assert response.status_code == 200, response.content
  1157. assert {r["id"] for r in response.data["data"]} == {
  1158. adopted_release_e_1,
  1159. adopted_release_e_2,
  1160. replaced_release_e_1,
  1161. replaced_release_e_2,
  1162. }
  1163. query = {
  1164. "field": ["id"],
  1165. "query": f"{constants.RELEASE_STAGE_ALIAS}:[{ReleaseStages.ADOPTED.value}, {ReleaseStages.REPLACED.value}]",
  1166. "environment": [self.environment.name],
  1167. }
  1168. response = self.do_request(query)
  1169. assert response.status_code == 200, response.content
  1170. assert {r["id"] for r in response.data["data"]} == {
  1171. adopted_release_e_1,
  1172. adopted_release_e_2,
  1173. replaced_release_e_1,
  1174. replaced_release_e_2,
  1175. }
  1176. def test_semver_package(self):
  1177. release_1 = self.create_release(version="test@1.2.3")
  1178. release_2 = self.create_release(version="test2@1.2.4")
  1179. release_1_e_1 = self.store_event(
  1180. data={"release": release_1.version, "timestamp": self.ten_mins_ago_iso},
  1181. project_id=self.project.id,
  1182. ).event_id
  1183. release_1_e_2 = self.store_event(
  1184. data={"release": release_1.version, "timestamp": self.ten_mins_ago_iso},
  1185. project_id=self.project.id,
  1186. ).event_id
  1187. release_2_e_1 = self.store_event(
  1188. data={"release": release_2.version, "timestamp": self.ten_mins_ago_iso},
  1189. project_id=self.project.id,
  1190. ).event_id
  1191. query = {"field": ["id"], "query": f"{constants.SEMVER_PACKAGE_ALIAS}:test"}
  1192. response = self.do_request(query)
  1193. assert response.status_code == 200, response.content
  1194. assert {r["id"] for r in response.data["data"]} == {
  1195. release_1_e_1,
  1196. release_1_e_2,
  1197. }
  1198. query = {"field": ["id"], "query": f"{constants.SEMVER_PACKAGE_ALIAS}:test2"}
  1199. response = self.do_request(query)
  1200. assert response.status_code == 200, response.content
  1201. assert {r["id"] for r in response.data["data"]} == {
  1202. release_2_e_1,
  1203. }
  1204. def test_semver_build(self):
  1205. release_1 = self.create_release(version="test@1.2.3+123")
  1206. release_2 = self.create_release(version="test2@1.2.4+124")
  1207. release_1_e_1 = self.store_event(
  1208. data={"release": release_1.version, "timestamp": self.ten_mins_ago_iso},
  1209. project_id=self.project.id,
  1210. ).event_id
  1211. release_1_e_2 = self.store_event(
  1212. data={"release": release_1.version, "timestamp": self.ten_mins_ago_iso},
  1213. project_id=self.project.id,
  1214. ).event_id
  1215. release_2_e_1 = self.store_event(
  1216. data={"release": release_2.version, "timestamp": self.ten_mins_ago_iso},
  1217. project_id=self.project.id,
  1218. ).event_id
  1219. query = {"field": ["id"], "query": f"{constants.SEMVER_BUILD_ALIAS}:123"}
  1220. response = self.do_request(query)
  1221. assert response.status_code == 200, response.content
  1222. assert {r["id"] for r in response.data["data"]} == {
  1223. release_1_e_1,
  1224. release_1_e_2,
  1225. }
  1226. query = {"field": ["id"], "query": f"{constants.SEMVER_BUILD_ALIAS}:124"}
  1227. response = self.do_request(query)
  1228. assert response.status_code == 200, response.content
  1229. assert {r["id"] for r in response.data["data"]} == {
  1230. release_2_e_1,
  1231. }
  1232. query = {"field": ["id"], "query": f"!{constants.SEMVER_BUILD_ALIAS}:124"}
  1233. response = self.do_request(query)
  1234. assert response.status_code == 200, response.content
  1235. assert {r["id"] for r in response.data["data"]} == {
  1236. release_1_e_1,
  1237. release_1_e_2,
  1238. }
  1239. def test_aliased_fields(self):
  1240. event1 = self.store_event(
  1241. data={
  1242. "event_id": "a" * 32,
  1243. "timestamp": self.ten_mins_ago_iso,
  1244. "fingerprint": ["group_1"],
  1245. "user": {"email": "foo@example.com"},
  1246. },
  1247. project_id=self.project.id,
  1248. )
  1249. event2 = self.store_event(
  1250. data={
  1251. "event_id": "b" * 32,
  1252. "timestamp": self.ten_mins_ago_iso,
  1253. "fingerprint": ["group_2"],
  1254. "user": {"email": "foo@example.com"},
  1255. },
  1256. project_id=self.project.id,
  1257. )
  1258. self.store_event(
  1259. data={
  1260. "event_id": "c" * 32,
  1261. "timestamp": self.ten_mins_ago_iso,
  1262. "fingerprint": ["group_2"],
  1263. "user": {"email": "bar@example.com"},
  1264. },
  1265. project_id=self.project.id,
  1266. )
  1267. query = {"field": ["issue.id", "count(id)", "count_unique(user)"], "orderby": "issue.id"}
  1268. response = self.do_request(query)
  1269. assert response.status_code == 200, response.content
  1270. assert len(response.data["data"]) == 2
  1271. data = response.data["data"]
  1272. assert data[0]["issue.id"] == event1.group_id
  1273. assert data[0]["count(id)"] == 1
  1274. assert data[0]["count_unique(user)"] == 1
  1275. assert "projectid" not in data[0]
  1276. assert "project.id" not in data[0]
  1277. assert data[1]["issue.id"] == event2.group_id
  1278. assert data[1]["count(id)"] == 2
  1279. assert data[1]["count_unique(user)"] == 2
  1280. def test_aggregate_field_with_dotted_param(self):
  1281. event1 = self.store_event(
  1282. data={
  1283. "event_id": "a" * 32,
  1284. "timestamp": self.ten_mins_ago_iso,
  1285. "fingerprint": ["group_1"],
  1286. "user": {"id": "123", "email": "foo@example.com"},
  1287. },
  1288. project_id=self.project.id,
  1289. )
  1290. event2 = self.store_event(
  1291. data={
  1292. "event_id": "b" * 32,
  1293. "timestamp": self.ten_mins_ago_iso,
  1294. "fingerprint": ["group_2"],
  1295. "user": {"id": "123", "email": "foo@example.com"},
  1296. },
  1297. project_id=self.project.id,
  1298. )
  1299. self.store_event(
  1300. data={
  1301. "event_id": "c" * 32,
  1302. "timestamp": self.ten_mins_ago_iso,
  1303. "fingerprint": ["group_2"],
  1304. "user": {"id": "456", "email": "bar@example.com"},
  1305. },
  1306. project_id=self.project.id,
  1307. )
  1308. query = {
  1309. "field": ["issue.id", "issue_title", "count(id)", "count_unique(user.email)"],
  1310. "orderby": "issue.id",
  1311. }
  1312. response = self.do_request(query)
  1313. assert response.status_code == 200, response.content
  1314. assert len(response.data["data"]) == 2
  1315. data = response.data["data"]
  1316. assert data[0]["issue.id"] == event1.group_id
  1317. assert data[0]["count(id)"] == 1
  1318. assert data[0]["count_unique(user.email)"] == 1
  1319. assert "projectid" not in data[0]
  1320. assert "project.id" not in data[0]
  1321. assert data[1]["issue.id"] == event2.group_id
  1322. assert data[1]["count(id)"] == 2
  1323. assert data[1]["count_unique(user.email)"] == 2
  1324. def test_failure_rate_alias_field(self):
  1325. data = self.transaction_data.copy()
  1326. data["transaction"] = "/failure_rate/success"
  1327. self.store_event(data, project_id=self.project.id)
  1328. data = self.transaction_data.copy()
  1329. data["transaction"] = "/failure_rate/unknown"
  1330. data["contexts"]["trace"]["status"] = "unknown_error"
  1331. self.store_event(data, project_id=self.project.id)
  1332. for i in range(6):
  1333. data = self.transaction_data.copy()
  1334. data["transaction"] = f"/failure_rate/{i}"
  1335. data["contexts"]["trace"]["status"] = "unauthenticated"
  1336. self.store_event(data, project_id=self.project.id)
  1337. query = {"field": ["failure_rate()"], "query": "event.type:transaction"}
  1338. response = self.do_request(query)
  1339. assert response.status_code == 200, response.content
  1340. assert len(response.data["data"]) == 1
  1341. data = response.data["data"]
  1342. assert data[0]["failure_rate()"] == 0.75
  1343. def test_count_miserable_alias_field(self):
  1344. events = [
  1345. ("one", 300),
  1346. ("one", 300),
  1347. ("two", 3000),
  1348. ("two", 3000),
  1349. ("three", 300),
  1350. ("three", 3000),
  1351. ]
  1352. for idx, event in enumerate(events):
  1353. data = self.load_data(
  1354. timestamp=before_now(minutes=(10 + idx)),
  1355. duration=timedelta(milliseconds=event[1]),
  1356. )
  1357. data["event_id"] = f"{idx}" * 32
  1358. data["transaction"] = f"/count_miserable/horribilis/{idx}"
  1359. data["user"] = {"email": f"{event[0]}@example.com"}
  1360. self.store_event(data, project_id=self.project.id)
  1361. query = {"field": ["count_miserable(user, 300)"], "query": "event.type:transaction"}
  1362. response = self.do_request(query)
  1363. assert response.status_code == 200, response.content
  1364. assert len(response.data["data"]) == 1
  1365. data = response.data["data"]
  1366. assert data[0]["count_miserable(user, 300)"] == 2
  1367. @mock.patch(
  1368. "sentry.search.events.fields.MAX_QUERYABLE_TRANSACTION_THRESHOLDS",
  1369. MAX_QUERYABLE_TRANSACTION_THRESHOLDS,
  1370. )
  1371. @mock.patch(
  1372. "sentry.search.events.datasets.discover.MAX_QUERYABLE_TRANSACTION_THRESHOLDS",
  1373. MAX_QUERYABLE_TRANSACTION_THRESHOLDS,
  1374. )
  1375. def test_too_many_transaction_thresholds(self):
  1376. project_transaction_thresholds = []
  1377. project_ids = []
  1378. for i in range(MAX_QUERYABLE_TRANSACTION_THRESHOLDS + 1):
  1379. project = self.create_project(name=f"bulk_txn_{i}")
  1380. project_ids.append(project.id)
  1381. project_transaction_thresholds.append(
  1382. ProjectTransactionThreshold(
  1383. organization=self.organization,
  1384. project=project,
  1385. threshold=400,
  1386. metric=TransactionMetric.LCP.value,
  1387. )
  1388. )
  1389. ProjectTransactionThreshold.objects.bulk_create(project_transaction_thresholds)
  1390. query = {
  1391. "field": [
  1392. "transaction",
  1393. "count_miserable(user)",
  1394. ],
  1395. "query": "event.type:transaction",
  1396. "project": project_ids,
  1397. }
  1398. response = self.do_request(
  1399. query,
  1400. features={
  1401. "organizations:discover-basic": True,
  1402. "organizations:global-views": True,
  1403. },
  1404. )
  1405. assert response.status_code == 400
  1406. assert (
  1407. response.data["detail"]
  1408. == "Exceeded 1 configured transaction thresholds limit, try with fewer Projects."
  1409. )
  1410. def test_count_miserable_new_alias_field(self):
  1411. ProjectTransactionThreshold.objects.create(
  1412. project=self.project,
  1413. organization=self.project.organization,
  1414. threshold=400,
  1415. metric=TransactionMetric.DURATION.value,
  1416. )
  1417. events = [
  1418. ("one", 400),
  1419. ("one", 400),
  1420. ("two", 3000),
  1421. ("two", 3000),
  1422. ("three", 300),
  1423. ("three", 3000),
  1424. ]
  1425. for idx, event in enumerate(events):
  1426. data = self.load_data(
  1427. timestamp=before_now(minutes=(10 + idx)),
  1428. duration=timedelta(milliseconds=event[1]),
  1429. )
  1430. data["event_id"] = f"{idx}" * 32
  1431. data["transaction"] = f"/count_miserable/horribilis/{event[0]}"
  1432. data["user"] = {"email": f"{idx}@example.com"}
  1433. self.store_event(data, project_id=self.project.id)
  1434. query = {
  1435. "field": [
  1436. "transaction",
  1437. "count_miserable(user)",
  1438. ],
  1439. "query": "event.type:transaction",
  1440. "project": [self.project.id],
  1441. "sort": "count_miserable_user",
  1442. }
  1443. response = self.do_request(
  1444. query,
  1445. )
  1446. assert response.status_code == 200, response.content
  1447. assert len(response.data["data"]) == 3
  1448. data = response.data["data"]
  1449. assert data[0]["count_miserable(user)"] == 0
  1450. assert data[1]["count_miserable(user)"] == 1
  1451. assert data[2]["count_miserable(user)"] == 2
  1452. query["query"] = "event.type:transaction count_miserable(user):>0"
  1453. response = self.do_request(
  1454. query,
  1455. )
  1456. assert response.status_code == 200, response.content
  1457. assert len(response.data["data"]) == 2
  1458. data = response.data["data"]
  1459. assert abs(data[0]["count_miserable(user)"]) == 1
  1460. assert abs(data[1]["count_miserable(user)"]) == 2
  1461. def test_user_misery_denominator(self):
  1462. """This is to test against a bug where the denominator of misery(total unique users) was wrong
  1463. This is because the total unique users for a LCP misery should only count users that have had a txn with lcp,
  1464. and not count all transactions (ie. uniq_if(transaction has lcp) not just uniq())
  1465. """
  1466. ProjectTransactionThreshold.objects.create(
  1467. project=self.project,
  1468. organization=self.project.organization,
  1469. threshold=600,
  1470. metric=TransactionMetric.LCP.value,
  1471. )
  1472. lcps = [
  1473. 400,
  1474. 400,
  1475. 300,
  1476. 3000,
  1477. 3000,
  1478. 3000,
  1479. ]
  1480. for idx, lcp in enumerate(lcps):
  1481. data = self.load_data(
  1482. timestamp=before_now(minutes=(10 + idx)),
  1483. )
  1484. data["event_id"] = f"{idx}" * 32
  1485. data["transaction"] = "/misery/new/"
  1486. data["user"] = {"email": f"{idx}@example.com"}
  1487. data["measurements"] = {
  1488. "lcp": {"value": lcp},
  1489. }
  1490. self.store_event(data, project_id=self.project.id)
  1491. # Shouldn't count towards misery
  1492. data = self.load_data(timestamp=self.ten_mins_ago, duration=timedelta(milliseconds=0))
  1493. data["transaction"] = "/misery/new/"
  1494. data["user"] = {"email": "7@example.com"}
  1495. data["measurements"] = {}
  1496. self.store_event(data, project_id=self.project.id)
  1497. query = {
  1498. "field": [
  1499. "transaction",
  1500. "user_misery()",
  1501. ],
  1502. "query": "event.type:transaction",
  1503. "project": [self.project.id],
  1504. "sort": "-user_misery",
  1505. }
  1506. response = self.do_request(
  1507. query,
  1508. )
  1509. assert response.status_code == 200, response.content
  1510. assert len(response.data["data"]) == 1
  1511. data = response.data["data"]
  1512. # (3 frustrated + 5.8875) / (6 + 117.75)
  1513. assert abs(data[0]["user_misery()"] - 0.071818) < 0.0001
  1514. def test_user_misery_alias_field(self):
  1515. events = [
  1516. ("one", 300),
  1517. ("one", 300),
  1518. ("two", 3000),
  1519. ("two", 3000),
  1520. ("three", 300),
  1521. ("three", 3000),
  1522. ]
  1523. for idx, event in enumerate(events):
  1524. data = self.load_data(
  1525. timestamp=before_now(minutes=(10 + idx)),
  1526. duration=timedelta(milliseconds=event[1]),
  1527. )
  1528. data["event_id"] = f"{idx}" * 32
  1529. data["transaction"] = f"/user_misery/{idx}"
  1530. data["user"] = {"email": f"{event[0]}@example.com"}
  1531. self.store_event(data, project_id=self.project.id)
  1532. query = {"field": ["user_misery(300)"], "query": "event.type:transaction"}
  1533. response = self.do_request(query)
  1534. assert response.status_code == 200, response.content
  1535. assert len(response.data["data"]) == 1
  1536. data = response.data["data"]
  1537. assert abs(data[0]["user_misery(300)"] - 0.0653) < 0.0001
  1538. def test_apdex_denominator_correct(self):
  1539. """This is to test against a bug where the denominator of apdex(total count) was wrong
  1540. This is because the total_count for a LCP apdex should only count transactions that have lcp, and not count
  1541. all transactions (ie. count_if(transaction has lcp) not just count())
  1542. """
  1543. ProjectTransactionThreshold.objects.create(
  1544. project=self.project,
  1545. organization=self.project.organization,
  1546. threshold=600,
  1547. metric=TransactionMetric.LCP.value,
  1548. )
  1549. lcps = [
  1550. 400,
  1551. 400,
  1552. 300,
  1553. 800,
  1554. 3000,
  1555. 3000,
  1556. 3000,
  1557. ]
  1558. for idx, lcp in enumerate(lcps):
  1559. data = self.load_data(
  1560. timestamp=before_now(minutes=(10 + idx)),
  1561. )
  1562. data["event_id"] = f"{idx}" * 32
  1563. data["transaction"] = "/apdex/new/"
  1564. data["user"] = {"email": f"{idx}@example.com"}
  1565. data["measurements"] = {
  1566. "lcp": {"value": lcp},
  1567. }
  1568. self.store_event(data, project_id=self.project.id)
  1569. # Shouldn't count towards apdex
  1570. data = self.load_data(
  1571. timestamp=self.ten_mins_ago,
  1572. duration=timedelta(milliseconds=0),
  1573. )
  1574. data["transaction"] = "/apdex/new/"
  1575. data["user"] = {"email": "7@example.com"}
  1576. data["measurements"] = {}
  1577. self.store_event(data, project_id=self.project.id)
  1578. query = {
  1579. "field": [
  1580. "transaction",
  1581. "apdex()",
  1582. ],
  1583. "query": "event.type:transaction",
  1584. "project": [self.project.id],
  1585. "sort": "-apdex",
  1586. }
  1587. response = self.do_request(
  1588. query,
  1589. )
  1590. assert response.status_code == 200, response.content
  1591. assert len(response.data["data"]) == 1
  1592. data = response.data["data"]
  1593. # 3 satisfied + 1 tolerated => 3.5/7
  1594. assert data[0]["apdex()"] == 0.5
  1595. def test_apdex_new_alias_field(self):
  1596. ProjectTransactionThreshold.objects.create(
  1597. project=self.project,
  1598. organization=self.project.organization,
  1599. threshold=400,
  1600. metric=TransactionMetric.DURATION.value,
  1601. )
  1602. events = [
  1603. ("one", 400),
  1604. ("one", 400),
  1605. ("two", 3000),
  1606. ("two", 3000),
  1607. ("three", 300),
  1608. ("three", 3000),
  1609. ]
  1610. for idx, event in enumerate(events):
  1611. data = self.load_data(
  1612. timestamp=before_now(minutes=(10 + idx)),
  1613. duration=timedelta(milliseconds=event[1]),
  1614. )
  1615. data["event_id"] = f"{idx}" * 32
  1616. data["transaction"] = f"/apdex/new/{event[0]}"
  1617. data["user"] = {"email": f"{idx}@example.com"}
  1618. self.store_event(data, project_id=self.project.id)
  1619. query = {
  1620. "field": [
  1621. "transaction",
  1622. "apdex()",
  1623. ],
  1624. "query": "event.type:transaction",
  1625. "project": [self.project.id],
  1626. "sort": "-apdex",
  1627. }
  1628. response = self.do_request(
  1629. query,
  1630. )
  1631. assert response.status_code == 200, response.content
  1632. assert len(response.data["data"]) == 3
  1633. data = response.data["data"]
  1634. assert data[0]["apdex()"] == 1.0
  1635. assert data[1]["apdex()"] == 0.5
  1636. assert data[2]["apdex()"] == 0.0
  1637. query["query"] = "event.type:transaction apdex():>0.50"
  1638. response = self.do_request(
  1639. query,
  1640. )
  1641. assert response.status_code == 200, response.content
  1642. assert len(response.data["data"]) == 1
  1643. data = response.data["data"]
  1644. assert data[0]["apdex()"] == 1.0
  1645. def test_user_misery_alias_field_with_project_threshold(self):
  1646. ProjectTransactionThreshold.objects.create(
  1647. project=self.project,
  1648. organization=self.project.organization,
  1649. threshold=400,
  1650. metric=TransactionMetric.DURATION.value,
  1651. )
  1652. events = [
  1653. ("one", 400),
  1654. ("one", 400),
  1655. ("two", 3000),
  1656. ("two", 3000),
  1657. ("three", 300),
  1658. ("three", 3000),
  1659. ]
  1660. for idx, event in enumerate(events):
  1661. data = self.load_data(
  1662. timestamp=before_now(minutes=(10 + idx)),
  1663. duration=timedelta(milliseconds=event[1]),
  1664. )
  1665. data["event_id"] = f"{idx}" * 32
  1666. data["transaction"] = f"/count_miserable/horribilis/{event[0]}"
  1667. data["user"] = {"email": f"{idx}@example.com"}
  1668. self.store_event(data, project_id=self.project.id)
  1669. query = {
  1670. "field": [
  1671. "transaction",
  1672. "user_misery()",
  1673. ],
  1674. "orderby": "user_misery()",
  1675. "query": "event.type:transaction",
  1676. "project": [self.project.id],
  1677. }
  1678. response = self.do_request(query)
  1679. assert response.status_code == 200, response.content
  1680. assert len(response.data["data"]) == 3
  1681. data = response.data["data"]
  1682. assert data[0]["user_misery()"] == pytest.approx(0.04916, rel=1e-3)
  1683. assert data[1]["user_misery()"] == pytest.approx(0.05751, rel=1e-3)
  1684. assert data[2]["user_misery()"] == pytest.approx(0.06586, rel=1e-3)
  1685. query["query"] = "event.type:transaction user_misery():>0.050"
  1686. response = self.do_request(
  1687. query,
  1688. )
  1689. assert response.status_code == 200, response.content
  1690. assert len(response.data["data"]) == 2
  1691. data = response.data["data"]
  1692. assert data[0]["user_misery()"] == pytest.approx(0.05751, rel=1e-3)
  1693. assert data[1]["user_misery()"] == pytest.approx(0.06586, rel=1e-3)
  1694. def test_user_misery_alias_field_with_transaction_threshold(self):
  1695. events = [
  1696. ("one", 300),
  1697. ("two", 300),
  1698. ("one", 3000),
  1699. ("two", 3000),
  1700. ("three", 400),
  1701. ("four", 4000),
  1702. ]
  1703. for idx, event in enumerate(events):
  1704. data = self.load_data(
  1705. timestamp=before_now(minutes=(10 + idx)),
  1706. duration=timedelta(milliseconds=event[1]),
  1707. )
  1708. data["event_id"] = f"{idx}" * 32
  1709. data["transaction"] = f"/count_miserable/horribilis/{idx}"
  1710. data["user"] = {"email": f"{event[0]}@example.com"}
  1711. self.store_event(data, project_id=self.project.id)
  1712. if idx % 2:
  1713. ProjectTransactionThresholdOverride.objects.create(
  1714. transaction=f"/count_miserable/horribilis/{idx}",
  1715. project=self.project,
  1716. organization=self.project.organization,
  1717. threshold=100 * idx,
  1718. metric=TransactionMetric.DURATION.value,
  1719. )
  1720. query = {
  1721. "field": [
  1722. "transaction",
  1723. "user_misery()",
  1724. ],
  1725. "query": "event.type:transaction",
  1726. "orderby": "transaction",
  1727. "project": [self.project.id],
  1728. }
  1729. response = self.do_request(
  1730. query,
  1731. )
  1732. assert response.status_code == 200, response.content
  1733. expected = [
  1734. ("/count_miserable/horribilis/0", ["duration", 300], 0.049578),
  1735. ("/count_miserable/horribilis/1", ["duration", 100], 0.049578),
  1736. ("/count_miserable/horribilis/2", ["duration", 300], 0.058),
  1737. ("/count_miserable/horribilis/3", ["duration", 300], 0.058),
  1738. ("/count_miserable/horribilis/4", ["duration", 300], 0.049578),
  1739. ("/count_miserable/horribilis/5", ["duration", 500], 0.058),
  1740. ]
  1741. assert len(response.data["data"]) == 6
  1742. data = response.data["data"]
  1743. for i, record in enumerate(expected):
  1744. name, threshold_config, misery = record
  1745. assert data[i]["transaction"] == name
  1746. assert data[i]["project_threshold_config"] == threshold_config
  1747. assert data[i]["user_misery()"] == pytest.approx(misery, rel=1e-3)
  1748. query["query"] = "event.type:transaction user_misery():>0.050"
  1749. response = self.do_request(
  1750. query,
  1751. )
  1752. assert response.status_code == 200, response.content
  1753. assert len(response.data["data"]) == 3
  1754. data = response.data["data"]
  1755. assert data[0]["user_misery()"] == pytest.approx(0.058, rel=1e-3)
  1756. assert data[1]["user_misery()"] == pytest.approx(0.058, rel=1e-3)
  1757. assert data[2]["user_misery()"] == pytest.approx(0.058, rel=1e-3)
  1758. def test_user_misery_alias_field_with_transaction_threshold_and_project_threshold(self):
  1759. project = self.create_project()
  1760. ProjectTransactionThreshold.objects.create(
  1761. project=project,
  1762. organization=project.organization,
  1763. threshold=100,
  1764. metric=TransactionMetric.DURATION.value,
  1765. )
  1766. events = [
  1767. ("one", 300),
  1768. ("two", 300),
  1769. ("one", 3000),
  1770. ("two", 3000),
  1771. ("three", 400),
  1772. ("four", 4000),
  1773. ]
  1774. for idx, event in enumerate(events):
  1775. data = self.load_data(
  1776. timestamp=before_now(minutes=(10 + idx)),
  1777. duration=timedelta(milliseconds=event[1]),
  1778. )
  1779. data["event_id"] = f"{idx}" * 32
  1780. data["transaction"] = f"/count_miserable/horribilis/{idx}"
  1781. data["user"] = {"email": f"{event[0]}@example.com"}
  1782. self.store_event(data, project_id=project.id)
  1783. if idx % 2:
  1784. ProjectTransactionThresholdOverride.objects.create(
  1785. transaction=f"/count_miserable/horribilis/{idx}",
  1786. project=project,
  1787. organization=project.organization,
  1788. threshold=100 * idx,
  1789. metric=TransactionMetric.DURATION.value,
  1790. )
  1791. project2 = self.create_project()
  1792. data = self.load_data()
  1793. data["transaction"] = "/count_miserable/horribilis/project2"
  1794. data["user"] = {"email": "project2@example.com"}
  1795. self.store_event(data, project_id=project2.id)
  1796. query = {
  1797. "field": [
  1798. "transaction",
  1799. "user_misery()",
  1800. ],
  1801. "query": "event.type:transaction",
  1802. "orderby": "transaction",
  1803. "project": [project.id, project2.id],
  1804. }
  1805. response = self.do_request(
  1806. query,
  1807. features={
  1808. "organizations:discover-basic": True,
  1809. "organizations:global-views": True,
  1810. },
  1811. )
  1812. assert response.status_code == 200, response.content
  1813. expected = [
  1814. (
  1815. "/count_miserable/horribilis/0",
  1816. ["duration", 100],
  1817. 0.049578,
  1818. ), # Uses project threshold
  1819. ("/count_miserable/horribilis/1", ["duration", 100], 0.049578), # Uses txn threshold
  1820. ("/count_miserable/horribilis/2", ["duration", 100], 0.058), # Uses project threshold
  1821. ("/count_miserable/horribilis/3", ["duration", 300], 0.058), # Uses txn threshold
  1822. (
  1823. "/count_miserable/horribilis/4",
  1824. ["duration", 100],
  1825. 0.049578,
  1826. ), # Uses project threshold
  1827. ("/count_miserable/horribilis/5", ["duration", 500], 0.058), # Uses txn threshold
  1828. ("/count_miserable/horribilis/project2", ["duration", 300], 0.058), # Uses fallback
  1829. ]
  1830. assert len(response.data["data"]) == 7
  1831. data = response.data["data"]
  1832. for i, record in enumerate(expected):
  1833. name, threshold_config, misery = record
  1834. assert data[i]["transaction"] == name
  1835. assert data[i]["project_threshold_config"] == threshold_config
  1836. assert data[i]["user_misery()"] == pytest.approx(misery, rel=1e-3)
  1837. query["query"] = "event.type:transaction user_misery():>0.050"
  1838. response = self.do_request(
  1839. query,
  1840. features={
  1841. "organizations:discover-basic": True,
  1842. "organizations:global-views": True,
  1843. },
  1844. )
  1845. assert response.status_code == 200, response.content
  1846. assert len(response.data["data"]) == 4
  1847. def test_aggregation(self):
  1848. self.store_event(
  1849. data={
  1850. "event_id": "a" * 32,
  1851. "timestamp": self.ten_mins_ago_iso,
  1852. "fingerprint": ["group_1"],
  1853. "user": {"email": "foo@example.com"},
  1854. "environment": "prod",
  1855. "tags": {"sub_customer.is-Enterprise-42": "1"},
  1856. },
  1857. project_id=self.project.id,
  1858. )
  1859. self.store_event(
  1860. data={
  1861. "event_id": "b" * 32,
  1862. "timestamp": self.ten_mins_ago_iso,
  1863. "fingerprint": ["group_2"],
  1864. "user": {"email": "foo@example.com"},
  1865. "environment": "staging",
  1866. "tags": {"sub_customer.is-Enterprise-42": "1"},
  1867. },
  1868. project_id=self.project.id,
  1869. )
  1870. self.store_event(
  1871. data={
  1872. "event_id": "c" * 32,
  1873. "timestamp": self.ten_mins_ago_iso,
  1874. "fingerprint": ["group_2"],
  1875. "user": {"email": "foo@example.com"},
  1876. "environment": "prod",
  1877. "tags": {"sub_customer.is-Enterprise-42": "0"},
  1878. },
  1879. project_id=self.project.id,
  1880. )
  1881. self.store_event(
  1882. data={
  1883. "event_id": "d" * 32,
  1884. "timestamp": self.ten_mins_ago_iso,
  1885. "fingerprint": ["group_2"],
  1886. "user": {"email": "foo@example.com"},
  1887. "environment": "prod",
  1888. "tags": {"sub_customer.is-Enterprise-42": "1"},
  1889. },
  1890. project_id=self.project.id,
  1891. )
  1892. query = {
  1893. "field": ["sub_customer.is-Enterprise-42", "count(sub_customer.is-Enterprise-42)"],
  1894. "orderby": "sub_customer.is-Enterprise-42",
  1895. }
  1896. response = self.do_request(query)
  1897. assert response.status_code == 200, response.content
  1898. assert len(response.data["data"]) == 2
  1899. data = response.data["data"]
  1900. assert data[0]["count(sub_customer.is-Enterprise-42)"] == 1
  1901. assert data[1]["count(sub_customer.is-Enterprise-42)"] == 3
  1902. def test_aggregation_comparison(self):
  1903. self.store_event(
  1904. data={
  1905. "event_id": "a" * 32,
  1906. "timestamp": self.ten_mins_ago_iso,
  1907. "fingerprint": ["group_1"],
  1908. "user": {"email": "foo@example.com"},
  1909. },
  1910. project_id=self.project.id,
  1911. )
  1912. event = self.store_event(
  1913. data={
  1914. "event_id": "b" * 32,
  1915. "timestamp": self.ten_mins_ago_iso,
  1916. "fingerprint": ["group_2"],
  1917. "user": {"email": "foo@example.com"},
  1918. },
  1919. project_id=self.project.id,
  1920. )
  1921. self.store_event(
  1922. data={
  1923. "event_id": "c" * 32,
  1924. "timestamp": self.ten_mins_ago_iso,
  1925. "fingerprint": ["group_2"],
  1926. "user": {"email": "bar@example.com"},
  1927. },
  1928. project_id=self.project.id,
  1929. )
  1930. self.store_event(
  1931. data={
  1932. "event_id": "d" * 32,
  1933. "timestamp": self.ten_mins_ago_iso,
  1934. "fingerprint": ["group_3"],
  1935. "user": {"email": "bar@example.com"},
  1936. },
  1937. project_id=self.project.id,
  1938. )
  1939. self.store_event(
  1940. data={
  1941. "event_id": "e" * 32,
  1942. "timestamp": self.ten_mins_ago_iso,
  1943. "fingerprint": ["group_3"],
  1944. "user": {"email": "bar@example.com"},
  1945. },
  1946. project_id=self.project.id,
  1947. )
  1948. query = {
  1949. "field": ["issue.id", "count(id)", "count_unique(user)"],
  1950. "query": "count(id):>1 count_unique(user):>1",
  1951. "orderby": "issue.id",
  1952. }
  1953. response = self.do_request(query)
  1954. assert response.status_code == 200, response.content
  1955. assert len(response.data["data"]) == 1
  1956. data = response.data["data"]
  1957. assert data[0]["issue.id"] == event.group_id
  1958. assert data[0]["count(id)"] == 2
  1959. assert data[0]["count_unique(user)"] == 2
  1960. def test_aggregation_alias_comparison(self):
  1961. data = self.load_data(
  1962. timestamp=self.ten_mins_ago,
  1963. duration=timedelta(seconds=5),
  1964. )
  1965. data["transaction"] = "/aggregates/1"
  1966. self.store_event(data, project_id=self.project.id)
  1967. data = self.load_data(
  1968. timestamp=self.ten_mins_ago,
  1969. duration=timedelta(seconds=3),
  1970. )
  1971. data["transaction"] = "/aggregates/2"
  1972. event = self.store_event(data, project_id=self.project.id)
  1973. query = {
  1974. "field": ["transaction", "p95()"],
  1975. "query": "event.type:transaction p95():<4000",
  1976. "orderby": ["transaction"],
  1977. }
  1978. response = self.do_request(query)
  1979. assert response.status_code == 200, response.content
  1980. assert len(response.data["data"]) == 1
  1981. data = response.data["data"]
  1982. assert data[0]["transaction"] == event.transaction
  1983. assert data[0]["p95()"] == 3000
  1984. def test_auto_aggregations(self):
  1985. data = self.load_data(
  1986. timestamp=self.ten_mins_ago,
  1987. duration=timedelta(seconds=5),
  1988. )
  1989. data["transaction"] = "/aggregates/1"
  1990. self.store_event(data, project_id=self.project.id)
  1991. data = self.load_data(
  1992. timestamp=self.ten_mins_ago,
  1993. duration=timedelta(seconds=3),
  1994. )
  1995. data["transaction"] = "/aggregates/2"
  1996. event = self.store_event(data, project_id=self.project.id)
  1997. query = {
  1998. "field": ["transaction", "p75()"],
  1999. "query": "event.type:transaction p95():<4000",
  2000. "orderby": ["transaction"],
  2001. }
  2002. response = self.do_request(query)
  2003. assert response.status_code == 200, response.content
  2004. assert len(response.data["data"]) == 1
  2005. data = response.data["data"]
  2006. assert data[0]["transaction"] == event.transaction
  2007. query = {
  2008. "field": ["transaction"],
  2009. "query": "event.type:transaction p95():<4000",
  2010. "orderby": ["transaction"],
  2011. }
  2012. response = self.do_request(query)
  2013. assert response.status_code == 400, response.content
  2014. def test_aggregation_comparison_with_conditions(self):
  2015. self.store_event(
  2016. data={
  2017. "event_id": "a" * 32,
  2018. "timestamp": self.ten_mins_ago_iso,
  2019. "fingerprint": ["group_1"],
  2020. "user": {"email": "foo@example.com"},
  2021. "environment": "prod",
  2022. },
  2023. project_id=self.project.id,
  2024. )
  2025. self.store_event(
  2026. data={
  2027. "event_id": "b" * 32,
  2028. "timestamp": self.ten_mins_ago_iso,
  2029. "fingerprint": ["group_2"],
  2030. "user": {"email": "foo@example.com"},
  2031. "environment": "staging",
  2032. },
  2033. project_id=self.project.id,
  2034. )
  2035. event = self.store_event(
  2036. data={
  2037. "event_id": "c" * 32,
  2038. "timestamp": self.ten_mins_ago_iso,
  2039. "fingerprint": ["group_2"],
  2040. "user": {"email": "foo@example.com"},
  2041. "environment": "prod",
  2042. },
  2043. project_id=self.project.id,
  2044. )
  2045. self.store_event(
  2046. data={
  2047. "event_id": "d" * 32,
  2048. "timestamp": self.ten_mins_ago_iso,
  2049. "fingerprint": ["group_2"],
  2050. "user": {"email": "foo@example.com"},
  2051. "environment": "prod",
  2052. },
  2053. project_id=self.project.id,
  2054. )
  2055. query = {
  2056. "field": ["issue.id", "count(id)"],
  2057. "query": "count(id):>1 user.email:foo@example.com environment:prod",
  2058. "orderby": "issue.id",
  2059. }
  2060. response = self.do_request(query)
  2061. assert response.status_code == 200, response.content
  2062. assert len(response.data["data"]) == 1
  2063. data = response.data["data"]
  2064. assert data[0]["issue.id"] == event.group_id
  2065. assert data[0]["count(id)"] == 2
  2066. def test_aggregation_date_comparison_with_conditions(self):
  2067. event = self.store_event(
  2068. data={
  2069. "event_id": "a" * 32,
  2070. "timestamp": self.ten_mins_ago_iso,
  2071. "fingerprint": ["group_1"],
  2072. "user": {"email": "foo@example.com"},
  2073. "environment": "prod",
  2074. },
  2075. project_id=self.project.id,
  2076. )
  2077. self.store_event(
  2078. data={
  2079. "event_id": "b" * 32,
  2080. "timestamp": self.ten_mins_ago_iso,
  2081. "fingerprint": ["group_2"],
  2082. "user": {"email": "foo@example.com"},
  2083. "environment": "staging",
  2084. },
  2085. project_id=self.project.id,
  2086. )
  2087. self.store_event(
  2088. data={
  2089. "event_id": "c" * 32,
  2090. "timestamp": self.ten_mins_ago_iso,
  2091. "fingerprint": ["group_2"],
  2092. "user": {"email": "foo@example.com"},
  2093. "environment": "prod",
  2094. },
  2095. project_id=self.project.id,
  2096. )
  2097. self.store_event(
  2098. data={
  2099. "event_id": "d" * 32,
  2100. "timestamp": self.ten_mins_ago_iso,
  2101. "fingerprint": ["group_2"],
  2102. "user": {"email": "foo@example.com"},
  2103. "environment": "prod",
  2104. },
  2105. project_id=self.project.id,
  2106. )
  2107. query = {
  2108. "field": ["issue.id", "max(timestamp)"],
  2109. "query": "max(timestamp):>1 user.email:foo@example.com environment:prod",
  2110. "orderby": "issue.id",
  2111. }
  2112. response = self.do_request(query)
  2113. assert response.status_code == 200, response.content
  2114. assert len(response.data["data"]) == 2
  2115. assert response.data["meta"]["fields"]["max(timestamp)"] == "date"
  2116. data = response.data["data"]
  2117. assert data[0]["issue.id"] == event.group_id
  2118. def test_percentile_function(self):
  2119. data = self.load_data(
  2120. timestamp=self.ten_mins_ago,
  2121. duration=timedelta(seconds=5),
  2122. )
  2123. data["transaction"] = "/aggregates/1"
  2124. event1 = self.store_event(data, project_id=self.project.id)
  2125. data = self.load_data(
  2126. timestamp=self.ten_mins_ago,
  2127. duration=timedelta(seconds=3),
  2128. )
  2129. data["transaction"] = "/aggregates/2"
  2130. event2 = self.store_event(data, project_id=self.project.id)
  2131. query = {
  2132. "field": ["transaction", "percentile(transaction.duration, 0.95)"],
  2133. "query": "event.type:transaction",
  2134. "orderby": ["transaction"],
  2135. }
  2136. response = self.do_request(query)
  2137. assert response.status_code == 200, response.content
  2138. assert len(response.data["data"]) == 2
  2139. data = response.data["data"]
  2140. assert data[0]["transaction"] == event1.transaction
  2141. assert data[0]["percentile(transaction.duration, 0.95)"] == 5000
  2142. assert data[1]["transaction"] == event2.transaction
  2143. assert data[1]["percentile(transaction.duration, 0.95)"] == 3000
  2144. def test_percentile_function_as_condition(self):
  2145. data = self.load_data(
  2146. timestamp=self.ten_mins_ago,
  2147. duration=timedelta(seconds=5),
  2148. )
  2149. data["transaction"] = "/aggregates/1"
  2150. event1 = self.store_event(data, project_id=self.project.id)
  2151. data = self.load_data(
  2152. timestamp=self.ten_mins_ago,
  2153. duration=timedelta(seconds=3),
  2154. )
  2155. data["transaction"] = "/aggregates/2"
  2156. self.store_event(data, project_id=self.project.id)
  2157. query = {
  2158. "field": ["transaction", "percentile(transaction.duration, 0.95)"],
  2159. "query": "event.type:transaction percentile(transaction.duration, 0.95):>4000",
  2160. "orderby": ["transaction"],
  2161. }
  2162. response = self.do_request(query)
  2163. assert response.status_code == 200, response.content
  2164. assert len(response.data["data"]) == 1
  2165. data = response.data["data"]
  2166. assert data[0]["transaction"] == event1.transaction
  2167. assert data[0]["percentile(transaction.duration, 0.95)"] == 5000
  2168. def test_epm_function(self):
  2169. data = self.load_data(
  2170. timestamp=self.ten_mins_ago,
  2171. duration=timedelta(seconds=5),
  2172. )
  2173. data["transaction"] = "/aggregates/1"
  2174. event1 = self.store_event(data, project_id=self.project.id)
  2175. data = self.load_data(
  2176. timestamp=self.ten_mins_ago,
  2177. duration=timedelta(seconds=3),
  2178. )
  2179. data["transaction"] = "/aggregates/2"
  2180. event2 = self.store_event(data, project_id=self.project.id)
  2181. query = {
  2182. "field": ["transaction", "epm()"],
  2183. "query": "event.type:transaction",
  2184. "orderby": ["transaction"],
  2185. "start": self.eleven_mins_ago_iso,
  2186. "end": iso_format(self.nine_mins_ago),
  2187. }
  2188. response = self.do_request(query)
  2189. assert response.status_code == 200, response.content
  2190. assert len(response.data["data"]) == 2
  2191. data = response.data["data"]
  2192. assert data[0]["transaction"] == event1.transaction
  2193. assert data[0]["epm()"] == 0.5
  2194. assert data[1]["transaction"] == event2.transaction
  2195. assert data[1]["epm()"] == 0.5
  2196. meta = response.data["meta"]
  2197. assert meta["fields"]["epm()"] == "rate"
  2198. assert meta["units"]["epm()"] == "1/minute"
  2199. def test_nonexistent_fields(self):
  2200. self.store_event(
  2201. data={
  2202. "event_id": "a" * 32,
  2203. "message": "how to make fast",
  2204. "timestamp": self.ten_mins_ago_iso,
  2205. },
  2206. project_id=self.project.id,
  2207. )
  2208. query = {"field": ["issue_world.id"]}
  2209. response = self.do_request(query)
  2210. assert response.status_code == 200, response.content
  2211. assert response.data["data"][0]["issue_world.id"] == ""
  2212. def test_no_requested_fields_or_grouping(self):
  2213. self.store_event(
  2214. data={
  2215. "event_id": "a" * 32,
  2216. "message": "how to make fast",
  2217. "timestamp": self.ten_mins_ago_iso,
  2218. },
  2219. project_id=self.project.id,
  2220. )
  2221. query = {"query": "test"}
  2222. response = self.do_request(query)
  2223. assert response.status_code == 400, response.content
  2224. assert response.data["detail"] == "No columns selected"
  2225. def test_condition_on_aggregate_misses(self):
  2226. self.store_event(
  2227. data={
  2228. "event_id": "c" * 32,
  2229. "timestamp": self.ten_mins_ago_iso,
  2230. "fingerprint": ["group_2"],
  2231. "user": {"email": "bar@example.com"},
  2232. },
  2233. project_id=self.project.id,
  2234. )
  2235. query = {"field": ["issue.id"], "query": "event_count:>0", "orderby": "issue.id"}
  2236. response = self.do_request(query)
  2237. assert response.status_code == 200, response.content
  2238. assert len(response.data["data"]) == 0
  2239. def test_next_prev_link_headers(self):
  2240. events = [("a", "group_1"), ("b", "group_2"), ("c", "group_2"), ("d", "group_2")]
  2241. for e in events:
  2242. self.store_event(
  2243. data={
  2244. "event_id": e[0] * 32,
  2245. "timestamp": self.ten_mins_ago_iso,
  2246. "fingerprint": [e[1]],
  2247. "user": {"email": "foo@example.com"},
  2248. "tags": {"language": "C++"},
  2249. },
  2250. project_id=self.project.id,
  2251. )
  2252. query = {
  2253. "field": ["count(id)", "issue.id", "context.key"],
  2254. "sort": "-count_id",
  2255. "query": "language:C++",
  2256. }
  2257. response = self.do_request(query)
  2258. assert response.status_code == 200, response.content
  2259. links = parse_link_header(response["Link"])
  2260. for link in links:
  2261. assert "field=issue.id" in link
  2262. assert "field=count%28id%29" in link
  2263. assert "field=context.key" in link
  2264. assert "sort=-count_id" in link
  2265. assert "query=language%3AC%2B%2B" in link
  2266. assert len(response.data["data"]) == 2
  2267. data = response.data["data"]
  2268. assert data[0]["count(id)"] == 3
  2269. assert data[1]["count(id)"] == 1
  2270. def test_empty_count_query(self):
  2271. event = self.store_event(
  2272. data={
  2273. "event_id": "a" * 32,
  2274. "timestamp": self.ten_mins_ago_iso,
  2275. "fingerprint": ["1123581321"],
  2276. "user": {"email": "foo@example.com"},
  2277. "tags": {"language": "C++"},
  2278. },
  2279. project_id=self.project.id,
  2280. )
  2281. query = {
  2282. "field": ["count()"],
  2283. "query": f"issue.id:{event.group_id} timestamp:>{self.ten_mins_ago_iso}",
  2284. "statsPeriod": "14d",
  2285. }
  2286. response = self.do_request(query)
  2287. assert response.status_code == 200, response.content
  2288. data = response.data["data"]
  2289. assert len(data) == 1
  2290. assert data[0]["count()"] == 0
  2291. def test_stack_wildcard_condition(self):
  2292. data = self.load_data(platform="javascript")
  2293. data["timestamp"] = self.ten_mins_ago_iso
  2294. self.store_event(data=data, project_id=self.project.id)
  2295. query = {"field": ["stack.filename", "message"], "query": "stack.filename:*.js"}
  2296. response = self.do_request(query)
  2297. assert response.status_code == 200, response.content
  2298. assert len(response.data["data"]) == 1
  2299. assert response.data["meta"]["fields"]["message"] == "string"
  2300. def test_email_wildcard_condition(self):
  2301. data = self.load_data(platform="javascript")
  2302. data["timestamp"] = self.ten_mins_ago_iso
  2303. self.store_event(data=data, project_id=self.project.id)
  2304. query = {"field": ["stack.filename", "message"], "query": "user.email:*@example.org"}
  2305. response = self.do_request(query)
  2306. assert response.status_code == 200, response.content
  2307. assert len(response.data["data"]) == 1
  2308. assert response.data["meta"]["fields"]["message"] == "string"
  2309. def test_release_wildcard_condition(self):
  2310. release = self.create_release(version="test@1.2.3+123")
  2311. self.store_event(
  2312. data={"release": release.version, "timestamp": self.ten_mins_ago_iso},
  2313. project_id=self.project.id,
  2314. )
  2315. query = {"field": ["stack.filename", "release"], "query": "release:test*"}
  2316. response = self.do_request(query)
  2317. assert response.status_code == 200, response.content
  2318. assert len(response.data["data"]) == 1
  2319. assert response.data["data"][0]["release"] == release.version
  2320. def test_transaction_event_type(self):
  2321. self.store_event(data=self.transaction_data, project_id=self.project.id)
  2322. query = {
  2323. "field": ["transaction", "transaction.duration", "transaction.status"],
  2324. "query": "event.type:transaction",
  2325. }
  2326. response = self.do_request(query)
  2327. assert response.status_code == 200, response.content
  2328. assert len(response.data["data"]) == 1
  2329. assert response.data["meta"]["fields"]["transaction.duration"] == "duration"
  2330. assert response.data["meta"]["fields"]["transaction.status"] == "string"
  2331. assert response.data["meta"]["units"]["transaction.duration"] == "millisecond"
  2332. assert response.data["data"][0]["transaction.status"] == "ok"
  2333. def test_trace_columns(self):
  2334. self.store_event(data=self.transaction_data, project_id=self.project.id)
  2335. query = {"field": ["trace"], "query": "event.type:transaction"}
  2336. response = self.do_request(query)
  2337. assert response.status_code == 200, response.content
  2338. assert len(response.data["data"]) == 1
  2339. assert response.data["meta"]["fields"]["trace"] == "string"
  2340. assert (
  2341. response.data["data"][0]["trace"]
  2342. == self.transaction_data["contexts"]["trace"]["trace_id"]
  2343. )
  2344. def test_issue_in_columns(self):
  2345. project1 = self.create_project()
  2346. project2 = self.create_project()
  2347. event1 = self.store_event(
  2348. data={
  2349. "event_id": "a" * 32,
  2350. "transaction": "/example",
  2351. "message": "how to make fast",
  2352. "timestamp": self.ten_mins_ago_iso,
  2353. "fingerprint": ["group_1"],
  2354. },
  2355. project_id=project1.id,
  2356. )
  2357. event2 = self.store_event(
  2358. data={
  2359. "event_id": "b" * 32,
  2360. "transaction": "/example",
  2361. "message": "how to make fast",
  2362. "timestamp": self.ten_mins_ago_iso,
  2363. "fingerprint": ["group_1"],
  2364. },
  2365. project_id=project2.id,
  2366. )
  2367. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2368. query = {"field": ["id", "issue"], "orderby": ["id"]}
  2369. response = self.do_request(query, features=features)
  2370. assert response.status_code == 200, response.content
  2371. data = response.data["data"]
  2372. assert len(data) == 2
  2373. assert data[0]["id"] == event1.event_id
  2374. assert data[0]["issue.id"] == event1.group_id
  2375. assert data[0]["issue"] == event1.group.qualified_short_id
  2376. assert data[1]["id"] == event2.event_id
  2377. assert data[1]["issue.id"] == event2.group_id
  2378. assert data[1]["issue"] == event2.group.qualified_short_id
  2379. def test_issue_in_search_and_columns(self):
  2380. project1 = self.create_project()
  2381. project2 = self.create_project()
  2382. event1 = self.store_event(
  2383. data={
  2384. "event_id": "a" * 32,
  2385. "transaction": "/example",
  2386. "message": "how to make fast",
  2387. "timestamp": self.ten_mins_ago_iso,
  2388. "fingerprint": ["group_1"],
  2389. },
  2390. project_id=project1.id,
  2391. )
  2392. self.store_event(
  2393. data={
  2394. "event_id": "b" * 32,
  2395. "transaction": "/example",
  2396. "message": "how to make fast",
  2397. "timestamp": self.ten_mins_ago_iso,
  2398. "fingerprint": ["group_1"],
  2399. },
  2400. project_id=project2.id,
  2401. )
  2402. tests = [
  2403. ("issue", "issue:%s" % event1.group.qualified_short_id),
  2404. ("issue.id", "issue:%s" % event1.group.qualified_short_id),
  2405. ("issue", "issue.id:%s" % event1.group_id),
  2406. ("issue.id", "issue.id:%s" % event1.group_id),
  2407. ]
  2408. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2409. for testdata in tests:
  2410. query = {"field": [testdata[0]], "query": testdata[1]}
  2411. response = self.do_request(query, features=features)
  2412. assert response.status_code == 200, response.content
  2413. data = response.data["data"]
  2414. assert len(data) == 1
  2415. assert data[0]["id"] == event1.event_id
  2416. assert data[0]["issue.id"] == event1.group_id
  2417. if testdata[0] == "issue":
  2418. assert data[0]["issue"] == event1.group.qualified_short_id
  2419. else:
  2420. assert data[0].get("issue", None) is None
  2421. def test_issue_negation(self):
  2422. project1 = self.create_project()
  2423. project2 = self.create_project()
  2424. event1 = self.store_event(
  2425. data={
  2426. "event_id": "a" * 32,
  2427. "transaction": "/example",
  2428. "message": "how to make fast",
  2429. "timestamp": self.ten_mins_ago_iso,
  2430. "fingerprint": ["group_1"],
  2431. },
  2432. project_id=project1.id,
  2433. )
  2434. event2 = self.store_event(
  2435. data={
  2436. "event_id": "b" * 32,
  2437. "transaction": "/example",
  2438. "message": "go really fast plz",
  2439. "timestamp": self.ten_mins_ago_iso,
  2440. "fingerprint": ["group_2"],
  2441. },
  2442. project_id=project2.id,
  2443. )
  2444. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2445. query = {
  2446. "field": ["title", "issue.id"],
  2447. "query": f"!issue:{event1.group.qualified_short_id}",
  2448. }
  2449. response = self.do_request(query, features=features)
  2450. assert response.status_code == 200, response.content
  2451. data = response.data["data"]
  2452. assert len(data) == 1
  2453. assert data[0]["title"] == event2.title
  2454. assert data[0]["issue.id"] == event2.group_id
  2455. def test_search_for_nonexistent_issue(self):
  2456. self.store_event(
  2457. data={
  2458. "event_id": "a" * 32,
  2459. "transaction": "/example",
  2460. "message": "how to make fast",
  2461. "timestamp": self.ten_mins_ago_iso,
  2462. "fingerprint": ["group_1"],
  2463. },
  2464. project_id=self.project.id,
  2465. )
  2466. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2467. query = {"field": ["count()"], "query": "issue.id:112358"}
  2468. response = self.do_request(query, features=features)
  2469. assert response.status_code == 200, response.content
  2470. data = response.data["data"]
  2471. assert len(data) == 1
  2472. assert data[0]["count()"] == 0
  2473. def test_issue_alias_inside_aggregate(self):
  2474. self.store_event(
  2475. data={
  2476. "event_id": "a" * 32,
  2477. "transaction": "/example",
  2478. "message": "how to make fast",
  2479. "timestamp": self.ten_mins_ago_iso,
  2480. "fingerprint": ["group_1"],
  2481. },
  2482. project_id=self.project.id,
  2483. )
  2484. self.store_event(
  2485. data={
  2486. "event_id": "b" * 32,
  2487. "transaction": "/example",
  2488. "message": "how to make fast",
  2489. "timestamp": self.ten_mins_ago_iso,
  2490. "fingerprint": ["group_2"],
  2491. },
  2492. project_id=self.project.id,
  2493. )
  2494. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2495. query = {
  2496. "field": ["project", "count(id)", "count_unique(issue.id)", "count_unique(issue)"],
  2497. "sort": "-count(id)",
  2498. "statsPeriod": "24h",
  2499. }
  2500. response = self.do_request(query, features=features)
  2501. assert response.status_code == 200, response.content
  2502. data = response.data["data"]
  2503. assert len(data) == 1
  2504. assert data[0]["count(id)"] == 2
  2505. assert data[0]["count_unique(issue.id)"] == 2
  2506. assert data[0]["count_unique(issue)"] == 2
  2507. def test_project_alias_inside_aggregate(self):
  2508. project1 = self.create_project()
  2509. project2 = self.create_project()
  2510. self.store_event(
  2511. data={
  2512. "event_id": "a" * 32,
  2513. "transaction": "/example",
  2514. "message": "how to make fast",
  2515. "timestamp": self.ten_mins_ago_iso,
  2516. "fingerprint": ["group_1"],
  2517. },
  2518. project_id=project1.id,
  2519. )
  2520. self.store_event(
  2521. data={
  2522. "event_id": "b" * 32,
  2523. "transaction": "/example",
  2524. "message": "how to make fast",
  2525. "timestamp": self.ten_mins_ago_iso,
  2526. "fingerprint": ["group_2"],
  2527. },
  2528. project_id=project2.id,
  2529. )
  2530. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2531. query = {
  2532. "field": [
  2533. "event.type",
  2534. "count(id)",
  2535. "count_unique(project.id)",
  2536. "count_unique(project)",
  2537. ],
  2538. "sort": "-count(id)",
  2539. "statsPeriod": "24h",
  2540. }
  2541. response = self.do_request(query, features=features)
  2542. assert response.status_code == 200, response.content
  2543. data = response.data["data"]
  2544. assert len(data) == 1
  2545. assert data[0]["count(id)"] == 2
  2546. assert data[0]["count_unique(project.id)"] == 2
  2547. assert data[0]["count_unique(project)"] == 2
  2548. def test_user_display(self):
  2549. project1 = self.create_project()
  2550. project2 = self.create_project()
  2551. self.store_event(
  2552. data={
  2553. "event_id": "a" * 32,
  2554. "transaction": "/example",
  2555. "message": "how to make fast",
  2556. "timestamp": self.ten_mins_ago_iso,
  2557. "user": {"email": "cathy@example.com"},
  2558. },
  2559. project_id=project1.id,
  2560. )
  2561. self.store_event(
  2562. data={
  2563. "event_id": "b" * 32,
  2564. "transaction": "/example",
  2565. "message": "how to make fast",
  2566. "timestamp": self.ten_mins_ago_iso,
  2567. "user": {"username": "catherine"},
  2568. },
  2569. project_id=project2.id,
  2570. )
  2571. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2572. query = {
  2573. "field": ["event.type", "user.display"],
  2574. "query": "user.display:cath*",
  2575. "statsPeriod": "24h",
  2576. }
  2577. response = self.do_request(query, features=features)
  2578. assert response.status_code == 200, response.content
  2579. data = response.data["data"]
  2580. assert len(data) == 2
  2581. result = {r["user.display"] for r in data}
  2582. assert result == {"catherine", "cathy@example.com"}
  2583. def test_user_display_with_aggregates(self):
  2584. self.store_event(
  2585. data={
  2586. "event_id": "a" * 32,
  2587. "transaction": "/example",
  2588. "message": "how to make fast",
  2589. "timestamp": self.ten_mins_ago_iso,
  2590. "user": {"email": "cathy@example.com"},
  2591. },
  2592. project_id=self.project.id,
  2593. )
  2594. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2595. query = {
  2596. "field": ["event.type", "user.display", "count_unique(title)"],
  2597. "statsPeriod": "24h",
  2598. }
  2599. response = self.do_request(query, features=features)
  2600. assert response.status_code == 200, response.content
  2601. data = response.data["data"]
  2602. assert len(data) == 1
  2603. result = {r["user.display"] for r in data}
  2604. assert result == {"cathy@example.com"}
  2605. query = {"field": ["event.type", "count_unique(user.display)"], "statsPeriod": "24h"}
  2606. response = self.do_request(query, features=features)
  2607. assert response.status_code == 200, response.content
  2608. data = response.data["data"]
  2609. assert len(data) == 1
  2610. assert data[0]["count_unique(user.display)"] == 1
  2611. def test_orderby_user_display(self):
  2612. project1 = self.create_project()
  2613. project2 = self.create_project()
  2614. self.store_event(
  2615. data={
  2616. "event_id": "a" * 32,
  2617. "transaction": "/example",
  2618. "message": "how to make fast",
  2619. "timestamp": self.ten_mins_ago_iso,
  2620. "user": {"email": "cathy@example.com"},
  2621. },
  2622. project_id=project1.id,
  2623. )
  2624. self.store_event(
  2625. data={
  2626. "event_id": "b" * 32,
  2627. "transaction": "/example",
  2628. "message": "how to make fast",
  2629. "timestamp": self.ten_mins_ago_iso,
  2630. "user": {"username": "catherine"},
  2631. },
  2632. project_id=project2.id,
  2633. )
  2634. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2635. query = {
  2636. "field": ["event.type", "user.display"],
  2637. "query": "user.display:cath*",
  2638. "statsPeriod": "24h",
  2639. "orderby": "-user.display",
  2640. }
  2641. response = self.do_request(query, features=features)
  2642. assert response.status_code == 200, response.content
  2643. data = response.data["data"]
  2644. assert len(data) == 2
  2645. result = [r["user.display"] for r in data]
  2646. # because we're ordering by `-user.display`, we expect the results in reverse sorted order
  2647. assert result == ["cathy@example.com", "catherine"]
  2648. def test_orderby_user_display_with_aggregates(self):
  2649. project1 = self.create_project()
  2650. project2 = self.create_project()
  2651. self.store_event(
  2652. data={
  2653. "event_id": "a" * 32,
  2654. "transaction": "/example",
  2655. "message": "how to make fast",
  2656. "timestamp": self.ten_mins_ago_iso,
  2657. "user": {"email": "cathy@example.com"},
  2658. },
  2659. project_id=project1.id,
  2660. )
  2661. self.store_event(
  2662. data={
  2663. "event_id": "b" * 32,
  2664. "transaction": "/example",
  2665. "message": "how to make fast",
  2666. "timestamp": self.ten_mins_ago_iso,
  2667. "user": {"username": "catherine"},
  2668. },
  2669. project_id=project2.id,
  2670. )
  2671. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2672. query = {
  2673. "field": ["event.type", "user.display", "count_unique(title)"],
  2674. "query": "user.display:cath*",
  2675. "statsPeriod": "24h",
  2676. "orderby": "user.display",
  2677. }
  2678. response = self.do_request(query, features=features)
  2679. assert response.status_code == 200, response.content
  2680. data = response.data["data"]
  2681. assert len(data) == 2
  2682. result = [r["user.display"] for r in data]
  2683. # because we're ordering by `user.display`, we expect the results in sorted order
  2684. assert result == ["catherine", "cathy@example.com"]
  2685. def test_any_field_alias(self):
  2686. day_ago = before_now(days=1).replace(hour=10, minute=11, second=12, microsecond=13)
  2687. self.store_event(
  2688. data={
  2689. "event_id": "a" * 32,
  2690. "transaction": "/example",
  2691. "message": "how to make fast",
  2692. "timestamp": iso_format(day_ago),
  2693. "user": {"email": "cathy@example.com"},
  2694. },
  2695. project_id=self.project.id,
  2696. )
  2697. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2698. query = {
  2699. "field": [
  2700. "event.type",
  2701. "any(user.display)",
  2702. "any(timestamp.to_day)",
  2703. "any(timestamp.to_hour)",
  2704. ],
  2705. "statsPeriod": "7d",
  2706. }
  2707. response = self.do_request(query, features=features)
  2708. assert response.status_code == 200, response.content
  2709. data = response.data["data"]
  2710. assert len(data) == 1
  2711. result = {r["any(user.display)"] for r in data}
  2712. assert result == {"cathy@example.com"}
  2713. result = {r["any(timestamp.to_day)"][:19] for r in data}
  2714. assert result == {iso_format(day_ago.replace(hour=0, minute=0, second=0, microsecond=0))}
  2715. result = {r["any(timestamp.to_hour)"][:19] for r in data}
  2716. assert result == {iso_format(day_ago.replace(minute=0, second=0, microsecond=0))}
  2717. def test_field_aliases_in_conflicting_functions(self):
  2718. self.store_event(
  2719. data={
  2720. "event_id": "a" * 32,
  2721. "transaction": "/example",
  2722. "message": "how to make fast",
  2723. "timestamp": iso_format(
  2724. before_now(days=1).replace(hour=10, minute=11, second=12, microsecond=13)
  2725. ),
  2726. "user": {"email": "cathy@example.com"},
  2727. },
  2728. project_id=self.project.id,
  2729. )
  2730. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2731. field_aliases = ["user.display", "timestamp.to_day", "timestamp.to_hour"]
  2732. for alias in field_aliases:
  2733. query = {
  2734. "field": [alias, f"any({alias})"],
  2735. "statsPeriod": "7d",
  2736. }
  2737. response = self.do_request(query, features=features)
  2738. assert response.status_code == 400, response.content
  2739. assert (
  2740. response.data["detail"]
  2741. == f"A single field cannot be used both inside and outside a function in the same query. To use {alias} you must first remove the function(s): any({alias})"
  2742. )
  2743. @pytest.mark.skip(
  2744. """
  2745. For some reason ClickHouse errors when there are two of the same string literals
  2746. (in this case the empty string "") in a query and one is in the prewhere clause.
  2747. Does not affect production or ClickHouse versions > 20.4.
  2748. """
  2749. )
  2750. def test_has_message(self):
  2751. event = self.store_event(
  2752. {"timestamp": self.ten_mins_ago_iso, "message": "a"}, project_id=self.project.id
  2753. )
  2754. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2755. query = {"field": ["project", "message"], "query": "has:message", "statsPeriod": "14d"}
  2756. response = self.do_request(query, features=features)
  2757. assert response.status_code == 200, response.content
  2758. assert len(response.data["data"]) == 1
  2759. assert response.data["data"][0]["message"] == event.message
  2760. query = {"field": ["project", "message"], "query": "!has:message", "statsPeriod": "14d"}
  2761. response = self.do_request(query, features=features)
  2762. assert response.status_code == 200, response.content
  2763. assert len(response.data["data"]) == 0
  2764. def test_has_transaction_status(self):
  2765. self.store_event(self.transaction_data, project_id=self.project.id)
  2766. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2767. query = {
  2768. "field": ["event.type", "count(id)"],
  2769. "query": "event.type:transaction has:transaction.status",
  2770. "sort": "-count(id)",
  2771. "statsPeriod": "24h",
  2772. }
  2773. response = self.do_request(query, features=features)
  2774. assert response.status_code == 200, response.content
  2775. data = response.data["data"]
  2776. assert len(data) == 1
  2777. assert data[0]["count(id)"] == 1
  2778. @pytest.mark.xfail(reason="Started failing on ClickHouse 21.8")
  2779. def test_not_has_transaction_status(self):
  2780. self.store_event(self.transaction_data, project_id=self.project.id)
  2781. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2782. query = {
  2783. "field": ["event.type", "count(id)"],
  2784. "query": "event.type:transaction !has:transaction.status",
  2785. "sort": "-count(id)",
  2786. "statsPeriod": "24h",
  2787. }
  2788. response = self.do_request(query, features=features)
  2789. assert response.status_code == 200, response.content
  2790. data = response.data["data"]
  2791. assert len(data) == 1
  2792. assert data[0]["count(id)"] == 0
  2793. def test_tag_that_looks_like_aggregation(self):
  2794. data = {
  2795. "message": "Failure state",
  2796. "timestamp": self.ten_mins_ago_iso,
  2797. "tags": {"count_diff": 99},
  2798. }
  2799. self.store_event(data, project_id=self.project.id)
  2800. query = {
  2801. "field": ["message", "count_diff", "count()"],
  2802. "query": "",
  2803. "project": [self.project.id],
  2804. "statsPeriod": "24h",
  2805. }
  2806. response = self.do_request(query)
  2807. assert response.status_code == 200, response.content
  2808. meta = response.data["meta"]["fields"]
  2809. assert "string" == meta["count_diff"], "tags should not be counted as integers"
  2810. assert "string" == meta["message"]
  2811. assert "integer" == meta["count()"]
  2812. assert 1 == len(response.data["data"])
  2813. data = response.data["data"][0]
  2814. assert "99" == data["count_diff"]
  2815. assert "Failure state" == data["message"]
  2816. assert 1 == data["count()"]
  2817. def test_aggregate_negation(self):
  2818. data = self.load_data(
  2819. timestamp=self.ten_mins_ago,
  2820. duration=timedelta(seconds=5),
  2821. )
  2822. self.store_event(data, project_id=self.project.id)
  2823. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2824. query = {
  2825. "field": ["event.type", "count()"],
  2826. "query": "event.type:transaction count():1",
  2827. "statsPeriod": "24h",
  2828. }
  2829. response = self.do_request(query, features=features)
  2830. assert response.status_code == 200, response.content
  2831. data = response.data["data"]
  2832. assert len(data) == 1
  2833. query = {
  2834. "field": ["event.type", "count()"],
  2835. "query": "event.type:transaction !count():1",
  2836. "statsPeriod": "24h",
  2837. }
  2838. response = self.do_request(query, features=features)
  2839. assert response.status_code == 200, response.content
  2840. data = response.data["data"]
  2841. assert len(data) == 0
  2842. def test_all_aggregates_in_columns(self):
  2843. data = self.load_data(
  2844. timestamp=self.eleven_mins_ago,
  2845. duration=timedelta(seconds=5),
  2846. )
  2847. data["transaction"] = "/failure_rate/1"
  2848. self.store_event(data, project_id=self.project.id)
  2849. data = self.load_data(
  2850. timestamp=self.ten_mins_ago,
  2851. duration=timedelta(seconds=5),
  2852. )
  2853. data["transaction"] = "/failure_rate/1"
  2854. data["contexts"]["trace"]["status"] = "unauthenticated"
  2855. event = self.store_event(data, project_id=self.project.id)
  2856. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  2857. query = {
  2858. "field": [
  2859. "event.type",
  2860. "p50()",
  2861. "p75()",
  2862. "p95()",
  2863. "p99()",
  2864. "p100()",
  2865. "percentile(transaction.duration, 0.99)",
  2866. "apdex(300)",
  2867. "count_miserable(user, 300)",
  2868. "user_misery(300)",
  2869. "failure_rate()",
  2870. ],
  2871. "query": "event.type:transaction",
  2872. }
  2873. response = self.do_request(query, features=features)
  2874. assert response.status_code == 200, response.content
  2875. meta = response.data["meta"]["fields"]
  2876. units = response.data["meta"]["units"]
  2877. assert meta["p50()"] == "duration"
  2878. assert meta["p75()"] == "duration"
  2879. assert meta["p95()"] == "duration"
  2880. assert meta["p99()"] == "duration"
  2881. assert meta["p100()"] == "duration"
  2882. assert meta["percentile(transaction.duration, 0.99)"] == "duration"
  2883. assert meta["apdex(300)"] == "number"
  2884. assert meta["failure_rate()"] == "percentage"
  2885. assert meta["user_misery(300)"] == "number"
  2886. assert meta["count_miserable(user, 300)"] == "integer"
  2887. assert units["p50()"] == "millisecond"
  2888. assert units["p75()"] == "millisecond"
  2889. assert units["p95()"] == "millisecond"
  2890. assert units["p99()"] == "millisecond"
  2891. assert units["p100()"] == "millisecond"
  2892. assert units["percentile(transaction.duration, 0.99)"] == "millisecond"
  2893. data = response.data["data"]
  2894. assert len(data) == 1
  2895. assert data[0]["p50()"] == 5000
  2896. assert data[0]["p75()"] == 5000
  2897. assert data[0]["p95()"] == 5000
  2898. assert data[0]["p99()"] == 5000
  2899. assert data[0]["p100()"] == 5000
  2900. assert data[0]["percentile(transaction.duration, 0.99)"] == 5000
  2901. assert data[0]["apdex(300)"] == 0.0
  2902. assert data[0]["count_miserable(user, 300)"] == 1
  2903. assert data[0]["user_misery(300)"] == 0.058
  2904. assert data[0]["failure_rate()"] == 0.5
  2905. features = {
  2906. "organizations:discover-basic": True,
  2907. "organizations:global-views": True,
  2908. }
  2909. query = {
  2910. "field": [
  2911. "event.type",
  2912. "p50()",
  2913. "p75()",
  2914. "p95()",
  2915. "p99()",
  2916. "p100()",
  2917. "percentile(transaction.duration, 0.99)",
  2918. "apdex(300)",
  2919. "apdex()",
  2920. "count_miserable(user, 300)",
  2921. "user_misery(300)",
  2922. "failure_rate()",
  2923. "count_miserable(user)",
  2924. "user_misery()",
  2925. ],
  2926. "query": "event.type:transaction",
  2927. "project": [self.project.id],
  2928. }
  2929. response = self.do_request(query, features=features)
  2930. assert response.status_code == 200, response.content
  2931. meta = response.data["meta"]["fields"]
  2932. units = response.data["meta"]["units"]
  2933. assert meta["p50()"] == "duration"
  2934. assert meta["p75()"] == "duration"
  2935. assert meta["p95()"] == "duration"
  2936. assert meta["p99()"] == "duration"
  2937. assert meta["p100()"] == "duration"
  2938. assert meta["percentile(transaction.duration, 0.99)"] == "duration"
  2939. assert meta["apdex(300)"] == "number"
  2940. assert meta["apdex()"] == "number"
  2941. assert meta["failure_rate()"] == "percentage"
  2942. assert meta["user_misery(300)"] == "number"
  2943. assert meta["count_miserable(user, 300)"] == "integer"
  2944. assert meta["project_threshold_config"] == "string"
  2945. assert meta["user_misery()"] == "number"
  2946. assert meta["count_miserable(user)"] == "integer"
  2947. assert units["p50()"] == "millisecond"
  2948. assert units["p75()"] == "millisecond"
  2949. assert units["p95()"] == "millisecond"
  2950. assert units["p99()"] == "millisecond"
  2951. assert units["p100()"] == "millisecond"
  2952. assert units["percentile(transaction.duration, 0.99)"] == "millisecond"
  2953. data = response.data["data"]
  2954. assert len(data) == 1
  2955. assert data[0]["p50()"] == 5000
  2956. assert data[0]["p75()"] == 5000
  2957. assert data[0]["p95()"] == 5000
  2958. assert data[0]["p99()"] == 5000
  2959. assert data[0]["p100()"] == 5000
  2960. assert data[0]["percentile(transaction.duration, 0.99)"] == 5000
  2961. assert data[0]["apdex(300)"] == 0.0
  2962. assert data[0]["apdex()"] == 0.0
  2963. assert data[0]["count_miserable(user, 300)"] == 1
  2964. assert data[0]["user_misery(300)"] == 0.058
  2965. assert data[0]["failure_rate()"] == 0.5
  2966. assert data[0]["project_threshold_config"] == ["duration", 300]
  2967. assert data[0]["user_misery()"] == 0.058
  2968. assert data[0]["count_miserable(user)"] == 1
  2969. query = {
  2970. "field": ["event.type", "last_seen()", "latest_event()"],
  2971. "query": "event.type:transaction",
  2972. }
  2973. response = self.do_request(query, features=features)
  2974. assert response.status_code == 200, response.content
  2975. data = response.data["data"]
  2976. assert len(data) == 1
  2977. assert self.ten_mins_ago_iso[:-5] in data[0]["last_seen()"]
  2978. assert data[0]["latest_event()"] == event.event_id
  2979. query = {
  2980. "field": [
  2981. "event.type",
  2982. "count()",
  2983. "count(id)",
  2984. "count_unique(project)",
  2985. "min(transaction.duration)",
  2986. "max(transaction.duration)",
  2987. "avg(transaction.duration)",
  2988. "stddev(transaction.duration)",
  2989. "var(transaction.duration)",
  2990. "cov(transaction.duration, transaction.duration)",
  2991. "corr(transaction.duration, transaction.duration)",
  2992. "linear_regression(transaction.duration, transaction.duration)",
  2993. "sum(transaction.duration)",
  2994. ],
  2995. "query": "event.type:transaction",
  2996. }
  2997. response = self.do_request(query, features=features)
  2998. assert response.status_code == 200, response.content
  2999. data = response.data["data"]
  3000. assert len(data) == 1
  3001. assert data[0]["count()"] == 2
  3002. assert data[0]["count(id)"] == 2
  3003. assert data[0]["count_unique(project)"] == 1
  3004. assert data[0]["min(transaction.duration)"] == 5000
  3005. assert data[0]["max(transaction.duration)"] == 5000
  3006. assert data[0]["avg(transaction.duration)"] == 5000
  3007. assert data[0]["stddev(transaction.duration)"] == 0.0
  3008. assert data[0]["var(transaction.duration)"] == 0.0
  3009. assert data[0]["cov(transaction.duration, transaction.duration)"] == 0.0
  3010. assert data[0]["corr(transaction.duration, transaction.duration)"] == 0.0
  3011. assert data[0]["linear_regression(transaction.duration, transaction.duration)"] == [0, 0]
  3012. assert data[0]["sum(transaction.duration)"] == 10000
  3013. @requires_not_arm64
  3014. def test_null_user_misery_returns_zero(self):
  3015. self.transaction_data["user"] = None
  3016. self.transaction_data["transaction"] = "/no_users/1"
  3017. self.store_event(self.transaction_data, project_id=self.project.id)
  3018. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3019. query = {
  3020. "field": ["user_misery(300)"],
  3021. "query": "event.type:transaction",
  3022. }
  3023. response = self.do_request(query, features=features)
  3024. assert response.status_code == 200, response.content
  3025. meta = response.data["meta"]["fields"]
  3026. assert meta["user_misery(300)"] == "number"
  3027. data = response.data["data"]
  3028. assert data[0]["user_misery(300)"] == 0
  3029. @requires_not_arm64
  3030. def test_null_user_misery_new_returns_zero(self):
  3031. self.transaction_data["user"] = None
  3032. self.transaction_data["transaction"] = "/no_users/1"
  3033. self.store_event(self.transaction_data, project_id=self.project.id)
  3034. features = {
  3035. "organizations:discover-basic": True,
  3036. }
  3037. query = {
  3038. "field": ["user_misery()"],
  3039. "query": "event.type:transaction",
  3040. }
  3041. response = self.do_request(query, features=features)
  3042. assert response.status_code == 200, response.content
  3043. meta = response.data["meta"]["fields"]
  3044. assert meta["user_misery()"] == "number"
  3045. data = response.data["data"]
  3046. assert data[0]["user_misery()"] == 0
  3047. def test_all_aggregates_in_query(self):
  3048. data = self.load_data(
  3049. timestamp=self.eleven_mins_ago,
  3050. duration=timedelta(seconds=5),
  3051. )
  3052. data["transaction"] = "/failure_rate/1"
  3053. self.store_event(data, project_id=self.project.id)
  3054. data = self.load_data(
  3055. timestamp=self.ten_mins_ago,
  3056. duration=timedelta(seconds=5),
  3057. )
  3058. data["transaction"] = "/failure_rate/2"
  3059. data["contexts"]["trace"]["status"] = "unauthenticated"
  3060. self.store_event(data, project_id=self.project.id)
  3061. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3062. query = {
  3063. "field": [
  3064. "event.type",
  3065. "p50()",
  3066. "p75()",
  3067. "p95()",
  3068. "percentile(transaction.duration, 0.99)",
  3069. "p100()",
  3070. ],
  3071. "query": "event.type:transaction p50():>100 p75():>1000 p95():>1000 p100():>1000 percentile(transaction.duration, 0.99):>1000",
  3072. }
  3073. response = self.do_request(query, features=features)
  3074. assert response.status_code == 200, response.content
  3075. data = response.data["data"]
  3076. assert len(data) == 1
  3077. assert data[0]["p50()"] == 5000
  3078. assert data[0]["p75()"] == 5000
  3079. assert data[0]["p95()"] == 5000
  3080. assert data[0]["p100()"] == 5000
  3081. assert data[0]["percentile(transaction.duration, 0.99)"] == 5000
  3082. query = {
  3083. "field": [
  3084. "event.type",
  3085. "apdex(300)",
  3086. "count_miserable(user, 300)",
  3087. "user_misery(300)",
  3088. "failure_rate()",
  3089. ],
  3090. "query": "event.type:transaction apdex(300):>-1.0 failure_rate():>0.25",
  3091. }
  3092. response = self.do_request(query, features=features)
  3093. assert response.status_code == 200, response.content
  3094. data = response.data["data"]
  3095. assert len(data) == 1
  3096. assert data[0]["apdex(300)"] == 0.0
  3097. assert data[0]["count_miserable(user, 300)"] == 1
  3098. assert data[0]["user_misery(300)"] == 0.058
  3099. assert data[0]["failure_rate()"] == 0.5
  3100. query = {
  3101. "field": ["event.type", "last_seen()", "latest_event()"],
  3102. "query": "event.type:transaction last_seen():>1990-12-01T00:00:00",
  3103. }
  3104. response = self.do_request(query, features=features)
  3105. assert response.status_code == 200, response.content
  3106. data = response.data["data"]
  3107. assert len(data) == 1
  3108. query = {
  3109. "field": ["event.type", "count()", "count(id)", "count_unique(transaction)"],
  3110. "query": "event.type:transaction count():>1 count(id):>1 count_unique(transaction):>1",
  3111. }
  3112. response = self.do_request(query, features=features)
  3113. assert response.status_code == 200, response.content
  3114. data = response.data["data"]
  3115. assert len(data) == 1
  3116. assert data[0]["count()"] == 2
  3117. assert data[0]["count(id)"] == 2
  3118. assert data[0]["count_unique(transaction)"] == 2
  3119. query = {
  3120. "field": [
  3121. "event.type",
  3122. "min(transaction.duration)",
  3123. "max(transaction.duration)",
  3124. "avg(transaction.duration)",
  3125. "sum(transaction.duration)",
  3126. "stddev(transaction.duration)",
  3127. "var(transaction.duration)",
  3128. "cov(transaction.duration, transaction.duration)",
  3129. "corr(transaction.duration, transaction.duration)",
  3130. ],
  3131. "query": " ".join(
  3132. [
  3133. "event.type:transaction",
  3134. "min(transaction.duration):>1000",
  3135. "max(transaction.duration):>1000",
  3136. "avg(transaction.duration):>1000",
  3137. "sum(transaction.duration):>1000",
  3138. "stddev(transaction.duration):>=0.0",
  3139. "var(transaction.duration):>=0.0",
  3140. "cov(transaction.duration, transaction.duration):>=0.0",
  3141. # correlation is nan because variance is 0
  3142. # "corr(transaction.duration, transaction.duration):>=0.0",
  3143. ]
  3144. ),
  3145. }
  3146. response = self.do_request(query, features=features)
  3147. assert response.status_code == 200, response.content
  3148. data = response.data["data"]
  3149. assert len(data) == 1
  3150. assert data[0]["min(transaction.duration)"] == 5000
  3151. assert data[0]["max(transaction.duration)"] == 5000
  3152. assert data[0]["avg(transaction.duration)"] == 5000
  3153. assert data[0]["sum(transaction.duration)"] == 10000
  3154. assert data[0]["stddev(transaction.duration)"] == 0.0
  3155. assert data[0]["var(transaction.duration)"] == 0.0
  3156. assert data[0]["cov(transaction.duration, transaction.duration)"] == 0.0
  3157. assert data[0]["corr(transaction.duration, transaction.duration)"] == 0.0
  3158. query = {
  3159. "field": ["event.type", "apdex(400)"],
  3160. "query": "event.type:transaction apdex(400):0",
  3161. }
  3162. response = self.do_request(query, features=features)
  3163. assert response.status_code == 200, response.content
  3164. data = response.data["data"]
  3165. assert len(data) == 1
  3166. assert data[0]["apdex(400)"] == 0
  3167. def test_functions_in_orderby(self):
  3168. data = self.load_data(
  3169. timestamp=self.eleven_mins_ago,
  3170. duration=timedelta(seconds=5),
  3171. )
  3172. data["transaction"] = "/failure_rate/1"
  3173. self.store_event(data, project_id=self.project.id)
  3174. data = self.load_data(
  3175. timestamp=self.ten_mins_ago,
  3176. duration=timedelta(seconds=5),
  3177. )
  3178. data["transaction"] = "/failure_rate/2"
  3179. data["contexts"]["trace"]["status"] = "unauthenticated"
  3180. event = self.store_event(data, project_id=self.project.id)
  3181. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3182. query = {
  3183. "field": ["event.type", "p75()"],
  3184. "sort": "-p75",
  3185. "query": "event.type:transaction",
  3186. }
  3187. response = self.do_request(query, features=features)
  3188. assert response.status_code == 200, response.content
  3189. data = response.data["data"]
  3190. assert len(data) == 1
  3191. assert data[0]["p75()"] == 5000
  3192. query = {
  3193. "field": ["event.type", "percentile(transaction.duration, 0.99)"],
  3194. "sort": "-percentile_transaction_duration_0_99",
  3195. "query": "event.type:transaction",
  3196. }
  3197. response = self.do_request(query, features=features)
  3198. assert response.status_code == 200, response.content
  3199. data = response.data["data"]
  3200. assert len(data) == 1
  3201. assert data[0]["percentile(transaction.duration, 0.99)"] == 5000
  3202. query = {
  3203. "field": ["event.type", "apdex(300)"],
  3204. "sort": "-apdex(300)",
  3205. "query": "event.type:transaction",
  3206. }
  3207. response = self.do_request(query, features=features)
  3208. assert response.status_code == 200, response.content
  3209. data = response.data["data"]
  3210. assert len(data) == 1
  3211. assert data[0]["apdex(300)"] == 0.0
  3212. query = {
  3213. "field": ["event.type", "latest_event()"],
  3214. "query": "event.type:transaction",
  3215. "sort": "latest_event",
  3216. }
  3217. response = self.do_request(query, features=features)
  3218. assert response.status_code == 200, response.content
  3219. data = response.data["data"]
  3220. assert len(data) == 1
  3221. assert data[0]["latest_event()"] == event.event_id
  3222. query = {
  3223. "field": ["event.type", "count_unique(transaction)"],
  3224. "query": "event.type:transaction",
  3225. "sort": "-count_unique_transaction",
  3226. }
  3227. response = self.do_request(query, features=features)
  3228. assert response.status_code == 200, response.content
  3229. data = response.data["data"]
  3230. assert len(data) == 1
  3231. assert data[0]["count_unique(transaction)"] == 2
  3232. query = {
  3233. "field": ["event.type", "min(transaction.duration)"],
  3234. "query": "event.type:transaction",
  3235. "sort": "-min_transaction_duration",
  3236. }
  3237. response = self.do_request(query, features=features)
  3238. assert response.status_code == 200, response.content
  3239. data = response.data["data"]
  3240. assert len(data) == 1
  3241. assert data[0]["min(transaction.duration)"] == 5000
  3242. def test_issue_alias_in_aggregate(self):
  3243. self.store_event(
  3244. data={
  3245. "event_id": "a" * 32,
  3246. "timestamp": self.eleven_mins_ago_iso,
  3247. "fingerprint": ["group_1"],
  3248. },
  3249. project_id=self.project.id,
  3250. )
  3251. self.store_event(
  3252. data={
  3253. "event_id": "b" * 32,
  3254. "timestamp": self.ten_mins_ago_iso,
  3255. "fingerprint": ["group_2"],
  3256. },
  3257. project_id=self.project.id,
  3258. )
  3259. query = {"field": ["event.type", "count_unique(issue)"], "query": "count_unique(issue):>1"}
  3260. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3261. response = self.do_request(query, features=features)
  3262. assert response.status_code == 200, response.content
  3263. data = response.data["data"]
  3264. assert len(data) == 1
  3265. assert data[0]["count_unique(issue)"] == 2
  3266. def test_deleted_issue_in_results(self):
  3267. event1 = self.store_event(
  3268. data={
  3269. "event_id": "a" * 32,
  3270. "timestamp": self.eleven_mins_ago_iso,
  3271. "fingerprint": ["group_1"],
  3272. },
  3273. project_id=self.project.id,
  3274. )
  3275. event2 = self.store_event(
  3276. data={
  3277. "event_id": "b" * 32,
  3278. "timestamp": self.ten_mins_ago_iso,
  3279. "fingerprint": ["group_2"],
  3280. },
  3281. project_id=self.project.id,
  3282. )
  3283. event2.group.delete()
  3284. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3285. query = {"field": ["issue", "count()"], "sort": "issue.id"}
  3286. response = self.do_request(query, features=features)
  3287. assert response.status_code == 200, response.content
  3288. data = response.data["data"]
  3289. assert len(data) == 2
  3290. assert data[0]["issue"] == event1.group.qualified_short_id
  3291. assert data[1]["issue"] == "unknown"
  3292. def test_last_seen_negative_duration(self):
  3293. self.store_event(
  3294. data={
  3295. "event_id": "f" * 32,
  3296. "timestamp": self.ten_mins_ago_iso,
  3297. "fingerprint": ["group_1"],
  3298. },
  3299. project_id=self.project.id,
  3300. )
  3301. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3302. query = {"field": ["id", "last_seen()"], "query": "last_seen():-30d"}
  3303. response = self.do_request(query, features=features)
  3304. assert response.status_code == 200, response.content
  3305. data = response.data["data"]
  3306. assert len(data) == 1
  3307. assert data[0]["id"] == "f" * 32
  3308. def test_last_seen_aggregate_condition(self):
  3309. self.store_event(
  3310. data={
  3311. "event_id": "f" * 32,
  3312. "timestamp": self.ten_mins_ago_iso,
  3313. "fingerprint": ["group_1"],
  3314. },
  3315. project_id=self.project.id,
  3316. )
  3317. query = {
  3318. "field": ["id", "last_seen()"],
  3319. "query": f"last_seen():>{iso_format(before_now(days=30))}",
  3320. }
  3321. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3322. response = self.do_request(query, features=features)
  3323. assert response.status_code == 200, response.content
  3324. data = response.data["data"]
  3325. assert len(data) == 1
  3326. assert data[0]["id"] == "f" * 32
  3327. def test_conditional_filter(self):
  3328. for v in ["a", "b"]:
  3329. self.store_event(
  3330. data={
  3331. "event_id": v * 32,
  3332. "timestamp": self.ten_mins_ago_iso,
  3333. "fingerprint": ["group_1"],
  3334. },
  3335. project_id=self.project.id,
  3336. )
  3337. query = {
  3338. "field": ["id"],
  3339. "query": "id:{} OR id:{}".format("a" * 32, "b" * 32),
  3340. "orderby": "id",
  3341. }
  3342. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3343. response = self.do_request(query, features=features)
  3344. assert response.status_code == 200, response.content
  3345. data = response.data["data"]
  3346. assert len(data) == 2
  3347. assert data[0]["id"] == "a" * 32
  3348. assert data[1]["id"] == "b" * 32
  3349. def test_aggregation_comparison_with_conditional_filter(self):
  3350. self.store_event(
  3351. data={
  3352. "event_id": "a" * 32,
  3353. "timestamp": self.ten_mins_ago_iso,
  3354. "fingerprint": ["group_1"],
  3355. "user": {"email": "foo@example.com"},
  3356. "environment": "prod",
  3357. },
  3358. project_id=self.project.id,
  3359. )
  3360. self.store_event(
  3361. data={
  3362. "event_id": "b" * 32,
  3363. "timestamp": self.ten_mins_ago_iso,
  3364. "fingerprint": ["group_2"],
  3365. "user": {"email": "foo@example.com"},
  3366. "environment": "staging",
  3367. },
  3368. project_id=self.project.id,
  3369. )
  3370. event = self.store_event(
  3371. data={
  3372. "event_id": "c" * 32,
  3373. "timestamp": self.ten_mins_ago_iso,
  3374. "fingerprint": ["group_2"],
  3375. "user": {"email": "foo@example.com"},
  3376. "environment": "prod",
  3377. },
  3378. project_id=self.project.id,
  3379. )
  3380. self.store_event(
  3381. data={
  3382. "event_id": "d" * 32,
  3383. "timestamp": self.ten_mins_ago_iso,
  3384. "fingerprint": ["group_2"],
  3385. "user": {"email": "foo@example.com"},
  3386. "environment": "canary",
  3387. },
  3388. project_id=self.project.id,
  3389. )
  3390. query = {
  3391. "field": ["issue.id", "count(id)"],
  3392. "query": "count(id):>1 user.email:foo@example.com AND (environment:prod OR environment:staging)",
  3393. "orderby": "issue.id",
  3394. }
  3395. response = self.do_request(query)
  3396. assert response.status_code == 200, response.content
  3397. assert len(response.data["data"]) == 1
  3398. data = response.data["data"]
  3399. assert data[0]["issue.id"] == event.group_id
  3400. assert data[0]["count(id)"] == 2
  3401. def run_test_in_query(self, query, expected_events, expected_negative_events=None):
  3402. params = {
  3403. "field": ["id"],
  3404. "query": query,
  3405. "orderby": "id",
  3406. }
  3407. response = self.do_request(
  3408. params, {"organizations:discover-basic": True, "organizations:global-views": True}
  3409. )
  3410. assert response.status_code == 200, response.content
  3411. assert [row["id"] for row in response.data["data"]] == [e.event_id for e in expected_events]
  3412. if expected_negative_events is not None:
  3413. params["query"] = f"!{query}"
  3414. response = self.do_request(
  3415. params,
  3416. {"organizations:discover-basic": True, "organizations:global-views": True},
  3417. )
  3418. assert response.status_code == 200, response.content
  3419. assert [row["id"] for row in response.data["data"]] == [
  3420. e.event_id for e in expected_negative_events
  3421. ]
  3422. def test_in_query_events(self):
  3423. project_1 = self.create_project()
  3424. event_1 = self.store_event(
  3425. data={
  3426. "event_id": "a" * 32,
  3427. "timestamp": self.ten_mins_ago_iso,
  3428. "fingerprint": ["group_1"],
  3429. "message": "group1",
  3430. "user": {"email": "hello@example.com"},
  3431. "environment": "prod",
  3432. "tags": {"random": "123"},
  3433. "release": "1.0",
  3434. },
  3435. project_id=project_1.id,
  3436. )
  3437. project_2 = self.create_project()
  3438. event_2 = self.store_event(
  3439. data={
  3440. "event_id": "b" * 32,
  3441. "timestamp": self.ten_mins_ago_iso,
  3442. "fingerprint": ["group_2"],
  3443. "message": "group2",
  3444. "user": {"email": "bar@example.com"},
  3445. "environment": "staging",
  3446. "tags": {"random": "456"},
  3447. "stacktrace": {"frames": [{"filename": "src/app/group2.py"}]},
  3448. "release": "1.2",
  3449. },
  3450. project_id=project_2.id,
  3451. )
  3452. project_3 = self.create_project()
  3453. event_3 = self.store_event(
  3454. data={
  3455. "event_id": "c" * 32,
  3456. "timestamp": self.ten_mins_ago_iso,
  3457. "fingerprint": ["group_3"],
  3458. "message": "group3",
  3459. "user": {"email": "foo@example.com"},
  3460. "environment": "canary",
  3461. "tags": {"random": "789"},
  3462. },
  3463. project_id=project_3.id,
  3464. )
  3465. self.run_test_in_query("environment:[prod, staging]", [event_1, event_2], [event_3])
  3466. self.run_test_in_query("environment:[staging]", [event_2], [event_1, event_3])
  3467. self.run_test_in_query(
  3468. "user.email:[foo@example.com, hello@example.com]", [event_1, event_3], [event_2]
  3469. )
  3470. self.run_test_in_query("user.email:[foo@example.com]", [event_3], [event_1, event_2])
  3471. self.run_test_in_query(
  3472. "user.display:[foo@example.com, hello@example.com]", [event_1, event_3], [event_2]
  3473. )
  3474. self.run_test_in_query(
  3475. 'message:["group2 src/app/group2.py in ?", group1]', [event_1, event_2], [event_3]
  3476. )
  3477. self.run_test_in_query(
  3478. f"issue.id:[{event_1.group_id},{event_2.group_id}]", [event_1, event_2]
  3479. )
  3480. self.run_test_in_query(
  3481. f"issue:[{event_1.group.qualified_short_id},{event_2.group.qualified_short_id}]",
  3482. [event_1, event_2],
  3483. )
  3484. self.run_test_in_query(
  3485. f"issue:[{event_1.group.qualified_short_id},{event_2.group.qualified_short_id}, unknown]",
  3486. [event_1, event_2],
  3487. )
  3488. self.run_test_in_query(f"project_id:[{project_3.id},{project_2.id}]", [event_2, event_3])
  3489. self.run_test_in_query(
  3490. f"project.name:[{project_3.slug},{project_2.slug}]", [event_2, event_3]
  3491. )
  3492. self.run_test_in_query("random:[789,456]", [event_2, event_3], [event_1])
  3493. self.run_test_in_query("tags[random]:[789,456]", [event_2, event_3], [event_1])
  3494. self.run_test_in_query("release:[1.0,1.2]", [event_1, event_2], [event_3])
  3495. def test_in_query_events_stack(self):
  3496. test_js = self.store_event(
  3497. self.load_data(
  3498. platform="javascript",
  3499. timestamp=self.ten_mins_ago,
  3500. duration=timedelta(seconds=5),
  3501. ),
  3502. project_id=self.project.id,
  3503. )
  3504. test_java = self.store_event(
  3505. self.load_data(
  3506. platform="java",
  3507. timestamp=self.ten_mins_ago,
  3508. duration=timedelta(seconds=5),
  3509. ),
  3510. project_id=self.project.id,
  3511. )
  3512. self.run_test_in_query(
  3513. "stack.filename:[../../sentry/scripts/views.js]", [test_js], [test_java]
  3514. )
  3515. def test_in_query_transactions(self):
  3516. data = self.transaction_data.copy()
  3517. data["event_id"] = "a" * 32
  3518. data["contexts"]["trace"]["status"] = "ok"
  3519. transaction_1 = self.store_event(data, project_id=self.project.id)
  3520. data = self.transaction_data.copy()
  3521. data["event_id"] = "b" * 32
  3522. data["contexts"]["trace"]["status"] = "aborted"
  3523. transaction_2 = self.store_event(data, project_id=self.project.id)
  3524. data = self.transaction_data.copy()
  3525. data["event_id"] = "c" * 32
  3526. data["contexts"]["trace"]["status"] = "already_exists"
  3527. transaction_3 = self.store_event(data, project_id=self.project.id)
  3528. self.run_test_in_query(
  3529. "transaction.status:[aborted, already_exists]",
  3530. [transaction_2, transaction_3],
  3531. [transaction_1],
  3532. )
  3533. def test_messed_up_function_values(self):
  3534. # TODO (evanh): It would be nice if this surfaced an error to the user.
  3535. # The problem: The && causes the parser to treat that term not as a bad
  3536. # function call but a valid raw search with parens in it. It's not trivial
  3537. # to change the parser to recognize "bad function values" and surface them.
  3538. for v in ["a", "b"]:
  3539. self.store_event(
  3540. data={
  3541. "event_id": v * 32,
  3542. "timestamp": self.ten_mins_ago_iso,
  3543. "fingerprint": ["group_1"],
  3544. },
  3545. project_id=self.project.id,
  3546. )
  3547. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  3548. query = {
  3549. "field": [
  3550. "transaction",
  3551. "project",
  3552. "epm()",
  3553. "p50()",
  3554. "p95()",
  3555. "failure_rate()",
  3556. "apdex(300)",
  3557. "count_unique(user)",
  3558. "user_misery(300)",
  3559. "count_miserable(user, 300)",
  3560. ],
  3561. "query": "failure_rate():>0.003&& users:>10 event.type:transaction",
  3562. "sort": "-failure_rate",
  3563. "statsPeriod": "24h",
  3564. }
  3565. response = self.do_request(query, features=features)
  3566. assert response.status_code == 200, response.content
  3567. data = response.data["data"]
  3568. assert len(data) == 0
  3569. def test_context_fields_between_datasets(self):
  3570. event_data = self.load_data(platform="android")
  3571. transaction_data = self.load_data()
  3572. event_data["spans"] = transaction_data["spans"]
  3573. event_data["contexts"]["trace"] = transaction_data["contexts"]["trace"]
  3574. event_data["type"] = "transaction"
  3575. event_data["transaction"] = "/failure_rate/1"
  3576. event_data["timestamp"] = iso_format(self.ten_mins_ago)
  3577. event_data["start_timestamp"] = iso_format(before_now(minutes=10, seconds=5))
  3578. event_data["user"]["geo"] = {"country_code": "US", "region": "CA", "city": "San Francisco"}
  3579. self.store_event(event_data, project_id=self.project.id)
  3580. event_data["type"] = "error"
  3581. self.store_event(event_data, project_id=self.project.id)
  3582. fields = [
  3583. "os.build",
  3584. "os.kernel_version",
  3585. "device.arch",
  3586. # TODO: battery level is not consistent across both datasets
  3587. # "device.battery_level",
  3588. "device.brand",
  3589. "device.charging",
  3590. "device.locale",
  3591. "device.model_id",
  3592. "device.name",
  3593. "device.online",
  3594. "device.orientation",
  3595. "device.simulator",
  3596. "device.uuid",
  3597. ]
  3598. data = [
  3599. {"field": fields + ["location", "count()"], "query": "event.type:error"},
  3600. {"field": fields + ["duration", "count()"], "query": "event.type:transaction"},
  3601. ]
  3602. for datum in data:
  3603. response = self.do_request(datum)
  3604. assert response.status_code == 200, response.content
  3605. assert len(response.data["data"]) == 1, datum
  3606. results = response.data["data"]
  3607. assert results[0]["count()"] == 1, datum
  3608. for field in fields:
  3609. key, value = field.split(".", 1)
  3610. expected = str(event_data["contexts"][key][value])
  3611. assert results[0][field] == expected, field + str(datum)
  3612. def test_http_fields_between_datasets(self):
  3613. event_data = self.load_data(platform="android")
  3614. transaction_data = self.load_data()
  3615. event_data["spans"] = transaction_data["spans"]
  3616. event_data["contexts"]["trace"] = transaction_data["contexts"]["trace"]
  3617. event_data["type"] = "transaction"
  3618. event_data["transaction"] = "/failure_rate/1"
  3619. event_data["timestamp"] = iso_format(self.ten_mins_ago)
  3620. event_data["start_timestamp"] = iso_format(before_now(minutes=10, seconds=5))
  3621. event_data["user"]["geo"] = {"country_code": "US", "region": "CA", "city": "San Francisco"}
  3622. event_data["request"] = transaction_data["request"]
  3623. self.store_event(event_data, project_id=self.project.id)
  3624. event_data["type"] = "error"
  3625. self.store_event(event_data, project_id=self.project.id)
  3626. fields = ["http.method", "http.referer", "http.url"]
  3627. expected = ["GET", "fixtures.transaction", "http://countries:8010/country_by_code/"]
  3628. data = [
  3629. {"field": fields + ["location", "count()"], "query": "event.type:error"},
  3630. {"field": fields + ["duration", "count()"], "query": "event.type:transaction"},
  3631. ]
  3632. for datum in data:
  3633. response = self.do_request(datum)
  3634. assert response.status_code == 200, response.content
  3635. assert len(response.data["data"]) == 1, datum
  3636. results = response.data["data"]
  3637. assert results[0]["count()"] == 1, datum
  3638. for (field, exp) in zip(fields, expected):
  3639. assert results[0][field] == exp, field + str(datum)
  3640. def test_failure_count_alias_field(self):
  3641. data = self.transaction_data.copy()
  3642. data["transaction"] = "/failure_count/success"
  3643. self.store_event(data, project_id=self.project.id)
  3644. data = self.transaction_data.copy()
  3645. data["transaction"] = "/failure_count/unknown"
  3646. data["contexts"]["trace"]["status"] = "unknown_error"
  3647. self.store_event(data, project_id=self.project.id)
  3648. for i in range(6):
  3649. data = self.transaction_data.copy()
  3650. data["transaction"] = f"/failure_count/{i}"
  3651. data["contexts"]["trace"]["status"] = "unauthenticated"
  3652. self.store_event(data, project_id=self.project.id)
  3653. query = {"field": ["count()", "failure_count()"], "query": "event.type:transaction"}
  3654. response = self.do_request(query)
  3655. assert response.status_code == 200, response.content
  3656. assert len(response.data["data"]) == 1
  3657. data = response.data["data"]
  3658. assert data[0]["count()"] == 8
  3659. assert data[0]["failure_count()"] == 6
  3660. @mock.patch("sentry.utils.snuba.quantize_time")
  3661. def test_quantize_dates(self, mock_quantize):
  3662. self.create_project()
  3663. mock_quantize.return_value = before_now(days=1).replace(tzinfo=timezone.utc)
  3664. # Don't quantize short time periods
  3665. query = {"statsPeriod": "1h", "query": "", "field": ["id", "timestamp"]}
  3666. self.do_request(query)
  3667. # Don't quantize absolute date periods
  3668. self.do_request(query)
  3669. query = {
  3670. "start": iso_format(before_now(days=20)),
  3671. "end": iso_format(before_now(days=15)),
  3672. "query": "",
  3673. "field": ["id", "timestamp"],
  3674. }
  3675. self.do_request(query)
  3676. assert len(mock_quantize.mock_calls) == 0
  3677. # Quantize long date periods
  3678. query = {"field": ["id", "timestamp"], "statsPeriod": "90d", "query": ""}
  3679. self.do_request(query)
  3680. assert len(mock_quantize.mock_calls) == 2
  3681. def test_limit_number_of_fields(self):
  3682. self.create_project()
  3683. for i in range(1, 25):
  3684. response = self.do_request({"field": ["id"] * i})
  3685. if i <= 20:
  3686. assert response.status_code == 200
  3687. else:
  3688. assert response.status_code == 400
  3689. assert (
  3690. response.data["detail"]
  3691. == "You can view up to 20 fields at a time. Please delete some and try again."
  3692. )
  3693. def test_percentile_function_meta_types(self):
  3694. self.store_event(self.transaction_data, project_id=self.project.id)
  3695. query = {
  3696. "field": [
  3697. "transaction",
  3698. "percentile(transaction.duration, 0.95)",
  3699. "percentile(measurements.fp, 0.95)",
  3700. "percentile(measurements.fcp, 0.95)",
  3701. "percentile(measurements.lcp, 0.95)",
  3702. "percentile(measurements.fid, 0.95)",
  3703. "percentile(measurements.ttfb, 0.95)",
  3704. "percentile(measurements.ttfb.requesttime, 0.95)",
  3705. "percentile(measurements.cls, 0.95)",
  3706. "percentile(measurements.foo, 0.95)",
  3707. "percentile(measurements.bar, 0.95)",
  3708. ],
  3709. "query": "",
  3710. "orderby": ["transaction"],
  3711. }
  3712. response = self.do_request(query)
  3713. assert response.status_code == 200, response.content
  3714. meta = response.data["meta"]["fields"]
  3715. assert meta["percentile(transaction.duration, 0.95)"] == "duration"
  3716. assert meta["percentile(measurements.fp, 0.95)"] == "duration"
  3717. assert meta["percentile(measurements.fcp, 0.95)"] == "duration"
  3718. assert meta["percentile(measurements.lcp, 0.95)"] == "duration"
  3719. assert meta["percentile(measurements.fid, 0.95)"] == "duration"
  3720. assert meta["percentile(measurements.ttfb, 0.95)"] == "duration"
  3721. assert meta["percentile(measurements.ttfb.requesttime, 0.95)"] == "duration"
  3722. assert meta["percentile(measurements.cls, 0.95)"] == "number"
  3723. assert meta["percentile(measurements.foo, 0.95)"] == "number"
  3724. assert meta["percentile(measurements.bar, 0.95)"] == "number"
  3725. units = response.data["meta"]["units"]
  3726. assert units["percentile(transaction.duration, 0.95)"] == "millisecond"
  3727. assert units["percentile(measurements.fp, 0.95)"] == "millisecond"
  3728. assert units["percentile(measurements.fcp, 0.95)"] == "millisecond"
  3729. assert units["percentile(measurements.lcp, 0.95)"] == "millisecond"
  3730. assert units["percentile(measurements.fid, 0.95)"] == "millisecond"
  3731. assert units["percentile(measurements.ttfb, 0.95)"] == "millisecond"
  3732. assert units["percentile(measurements.ttfb.requesttime, 0.95)"] == "millisecond"
  3733. def test_count_at_least_query(self):
  3734. self.store_event(self.transaction_data, self.project.id)
  3735. response = self.do_request({"field": "count_at_least(measurements.fcp, 0)"})
  3736. assert response.status_code == 200
  3737. assert len(response.data["data"]) == 1
  3738. assert response.data["data"][0]["count_at_least(measurements.fcp, 0)"] == 1
  3739. # a value that's a little bigger than the stored fcp
  3740. fcp = int(self.transaction_data["measurements"]["fcp"]["value"] + 1)
  3741. response = self.do_request({"field": f"count_at_least(measurements.fcp, {fcp})"})
  3742. assert response.status_code == 200
  3743. assert len(response.data["data"]) == 1
  3744. assert response.data["data"][0][f"count_at_least(measurements.fcp, {fcp})"] == 0
  3745. def test_measurements_query(self):
  3746. self.store_event(self.transaction_data, self.project.id)
  3747. query = {
  3748. "field": [
  3749. "measurements.fp",
  3750. "measurements.fcp",
  3751. "measurements.lcp",
  3752. "measurements.fid",
  3753. ]
  3754. }
  3755. response = self.do_request(query)
  3756. assert response.status_code == 200, response.content
  3757. assert len(response.data["data"]) == 1
  3758. for field in query["field"]:
  3759. measure = field.split(".", 1)[1]
  3760. assert (
  3761. response.data["data"][0][field]
  3762. == self.transaction_data["measurements"][measure]["value"]
  3763. )
  3764. query = {
  3765. "field": [
  3766. "measurements.fP",
  3767. "measurements.Fcp",
  3768. "measurements.LcP",
  3769. "measurements.FID",
  3770. ]
  3771. }
  3772. response = self.do_request(query)
  3773. assert response.status_code == 200, response.content
  3774. assert len(response.data["data"]) == 1
  3775. for field in query["field"]:
  3776. measure = field.split(".", 1)[1].lower()
  3777. assert (
  3778. response.data["data"][0][field]
  3779. == self.transaction_data["measurements"][measure]["value"]
  3780. )
  3781. def test_measurements_aggregations(self):
  3782. self.store_event(self.transaction_data, self.project.id)
  3783. # should try all the potential aggregates
  3784. # Skipped tests for stddev and var since sampling one data point
  3785. # results in nan.
  3786. query = {
  3787. "field": [
  3788. "percentile(measurements.fcp, 0.5)",
  3789. "count_unique(measurements.fcp)",
  3790. "min(measurements.fcp)",
  3791. "max(measurements.fcp)",
  3792. "avg(measurements.fcp)",
  3793. "sum(measurements.fcp)",
  3794. ],
  3795. }
  3796. response = self.do_request(query)
  3797. assert response.status_code == 200, response.content
  3798. assert len(response.data["data"]) == 1
  3799. assert (
  3800. response.data["data"][0]["percentile(measurements.fcp, 0.5)"]
  3801. == self.transaction_data["measurements"]["fcp"]["value"]
  3802. )
  3803. assert response.data["data"][0]["count_unique(measurements.fcp)"] == 1
  3804. assert (
  3805. response.data["data"][0]["min(measurements.fcp)"]
  3806. == self.transaction_data["measurements"]["fcp"]["value"]
  3807. )
  3808. assert (
  3809. response.data["data"][0]["max(measurements.fcp)"]
  3810. == self.transaction_data["measurements"]["fcp"]["value"]
  3811. )
  3812. assert (
  3813. response.data["data"][0]["avg(measurements.fcp)"]
  3814. == self.transaction_data["measurements"]["fcp"]["value"]
  3815. )
  3816. assert (
  3817. response.data["data"][0]["sum(measurements.fcp)"]
  3818. == self.transaction_data["measurements"]["fcp"]["value"]
  3819. )
  3820. def get_measurement_condition_response(self, query_str, field):
  3821. query = {
  3822. "field": ["transaction", "count()"] + (field if field else []),
  3823. "query": query_str,
  3824. }
  3825. response = self.do_request(query)
  3826. assert response.status_code == 200, response.content
  3827. return response
  3828. def assert_measurement_condition_without_results(self, query_str, field=None):
  3829. response = self.get_measurement_condition_response(query_str, field)
  3830. assert len(response.data["data"]) == 0
  3831. def assert_measurement_condition_with_results(self, query_str, field=None):
  3832. response = self.get_measurement_condition_response(query_str, field)
  3833. assert len(response.data["data"]) == 1
  3834. assert response.data["data"][0]["transaction"] == self.transaction_data["metadata"]["title"]
  3835. assert response.data["data"][0]["count()"] == 1
  3836. def test_measurements_conditions(self):
  3837. self.store_event(self.transaction_data, self.project.id)
  3838. fcp = self.transaction_data["measurements"]["fcp"]["value"]
  3839. # equality condition
  3840. # We use json dumps here to ensure precision when converting from float to str
  3841. # This is necessary because equality on floating point values need to be precise
  3842. self.assert_measurement_condition_with_results(f"measurements.fcp:{json.dumps(fcp)}")
  3843. # greater than condition
  3844. self.assert_measurement_condition_with_results(f"measurements.fcp:>{fcp - 1}")
  3845. self.assert_measurement_condition_without_results(f"measurements.fcp:>{fcp + 1}")
  3846. # less than condition
  3847. self.assert_measurement_condition_with_results(f"measurements.fcp:<{fcp + 1}")
  3848. self.assert_measurement_condition_without_results(f"measurements.fcp:<{fcp - 1}")
  3849. # has condition
  3850. self.assert_measurement_condition_with_results("has:measurements.fcp")
  3851. self.assert_measurement_condition_without_results("!has:measurements.fcp")
  3852. def test_measurements_aggregation_conditions(self):
  3853. self.store_event(self.transaction_data, self.project.id)
  3854. fcp = self.transaction_data["measurements"]["fcp"]["value"]
  3855. functions = [
  3856. "percentile(measurements.fcp, 0.5)",
  3857. "min(measurements.fcp)",
  3858. "max(measurements.fcp)",
  3859. "avg(measurements.fcp)",
  3860. "sum(measurements.fcp)",
  3861. ]
  3862. for function in functions:
  3863. self.assert_measurement_condition_with_results(
  3864. f"{function}:>{fcp - 1}", field=[function]
  3865. )
  3866. self.assert_measurement_condition_without_results(
  3867. f"{function}:>{fcp + 1}", field=[function]
  3868. )
  3869. self.assert_measurement_condition_with_results(
  3870. f"{function}:<{fcp + 1}", field=[function]
  3871. )
  3872. self.assert_measurement_condition_without_results(
  3873. f"{function}:<{fcp - 1}", field=[function]
  3874. )
  3875. count_unique = "count_unique(measurements.fcp)"
  3876. self.assert_measurement_condition_with_results(f"{count_unique}:1", field=[count_unique])
  3877. self.assert_measurement_condition_without_results(f"{count_unique}:0", field=[count_unique])
  3878. def test_compare_numeric_aggregate(self):
  3879. self.store_event(self.transaction_data, self.project.id)
  3880. query = {
  3881. "field": [
  3882. "p75(measurements.fcp)",
  3883. "compare_numeric_aggregate(p75_measurements_fcp,greater,0)",
  3884. ],
  3885. }
  3886. response = self.do_request(query)
  3887. assert response.status_code == 200, response.content
  3888. assert len(response.data["data"]) == 1
  3889. assert (
  3890. response.data["data"][0]["compare_numeric_aggregate(p75_measurements_fcp,greater,0)"]
  3891. == 1
  3892. )
  3893. query = {
  3894. "field": ["p75()", "compare_numeric_aggregate(p75,equals,0)"],
  3895. }
  3896. response = self.do_request(query)
  3897. assert response.status_code == 200, response.content
  3898. assert len(response.data["data"]) == 1
  3899. assert response.data["data"][0]["compare_numeric_aggregate(p75,equals,0)"] == 0
  3900. def test_no_team_key_transactions(self):
  3901. transactions = [
  3902. "/blah_transaction/",
  3903. "/foo_transaction/",
  3904. "/zoo_transaction/",
  3905. ]
  3906. for transaction in transactions:
  3907. self.transaction_data["transaction"] = transaction
  3908. self.store_event(self.transaction_data, self.project.id)
  3909. query = {
  3910. "team": "myteams",
  3911. "project": [self.project.id],
  3912. # use the order by to ensure the result order
  3913. "orderby": "transaction",
  3914. "field": [
  3915. "team_key_transaction",
  3916. "transaction",
  3917. "transaction.status",
  3918. "project",
  3919. "epm()",
  3920. "failure_rate()",
  3921. "percentile(transaction.duration, 0.95)",
  3922. ],
  3923. }
  3924. response = self.do_request(query)
  3925. assert response.status_code == 200, response.content
  3926. data = response.data["data"]
  3927. assert len(data) == 3
  3928. assert data[0]["team_key_transaction"] == 0
  3929. assert data[0]["transaction"] == "/blah_transaction/"
  3930. assert data[1]["team_key_transaction"] == 0
  3931. assert data[1]["transaction"] == "/foo_transaction/"
  3932. assert data[2]["team_key_transaction"] == 0
  3933. assert data[2]["transaction"] == "/zoo_transaction/"
  3934. def test_team_key_transactions_my_teams(self):
  3935. team1 = self.create_team(organization=self.organization, name="Team A")
  3936. self.create_team_membership(team1, user=self.user)
  3937. self.project.add_team(team1)
  3938. team2 = self.create_team(organization=self.organization, name="Team B")
  3939. self.project.add_team(team2)
  3940. transactions = ["/blah_transaction/"]
  3941. key_transactions = [
  3942. (team1, "/foo_transaction/"),
  3943. (team2, "/zoo_transaction/"),
  3944. ]
  3945. for transaction in transactions:
  3946. self.transaction_data["transaction"] = transaction
  3947. self.store_event(self.transaction_data, self.project.id)
  3948. for team, transaction in key_transactions:
  3949. self.transaction_data["transaction"] = transaction
  3950. self.store_event(self.transaction_data, self.project.id)
  3951. TeamKeyTransaction.objects.create(
  3952. organization=self.organization,
  3953. transaction=transaction,
  3954. project_team=ProjectTeam.objects.get(project=self.project, team=team),
  3955. )
  3956. query = {
  3957. "team": "myteams",
  3958. "project": [self.project.id],
  3959. "field": [
  3960. "team_key_transaction",
  3961. "transaction",
  3962. "transaction.status",
  3963. "project",
  3964. "epm()",
  3965. "failure_rate()",
  3966. "percentile(transaction.duration, 0.95)",
  3967. ],
  3968. }
  3969. query["orderby"] = ["team_key_transaction", "transaction"]
  3970. response = self.do_request(query)
  3971. assert response.status_code == 200, response.content
  3972. data = response.data["data"]
  3973. assert len(data) == 3
  3974. assert data[0]["team_key_transaction"] == 0
  3975. assert data[0]["transaction"] == "/blah_transaction/"
  3976. assert data[1]["team_key_transaction"] == 0
  3977. assert data[1]["transaction"] == "/zoo_transaction/"
  3978. assert data[2]["team_key_transaction"] == 1
  3979. assert data[2]["transaction"] == "/foo_transaction/"
  3980. # not specifying any teams should use my teams
  3981. query = {
  3982. "project": [self.project.id],
  3983. "field": [
  3984. "team_key_transaction",
  3985. "transaction",
  3986. "transaction.status",
  3987. "project",
  3988. "epm()",
  3989. "failure_rate()",
  3990. "percentile(transaction.duration, 0.95)",
  3991. ],
  3992. }
  3993. query["orderby"] = ["team_key_transaction", "transaction"]
  3994. response = self.do_request(query)
  3995. assert response.status_code == 200, response.content
  3996. data = response.data["data"]
  3997. assert len(data) == 3
  3998. assert data[0]["team_key_transaction"] == 0
  3999. assert data[0]["transaction"] == "/blah_transaction/"
  4000. assert data[1]["team_key_transaction"] == 0
  4001. assert data[1]["transaction"] == "/zoo_transaction/"
  4002. assert data[2]["team_key_transaction"] == 1
  4003. assert data[2]["transaction"] == "/foo_transaction/"
  4004. def test_team_key_transactions_orderby(self):
  4005. team1 = self.create_team(organization=self.organization, name="Team A")
  4006. team2 = self.create_team(organization=self.organization, name="Team B")
  4007. transactions = ["/blah_transaction/"]
  4008. key_transactions = [
  4009. (team1, "/foo_transaction/"),
  4010. (team2, "/zoo_transaction/"),
  4011. ]
  4012. for transaction in transactions:
  4013. self.transaction_data["transaction"] = transaction
  4014. self.store_event(self.transaction_data, self.project.id)
  4015. for team, transaction in key_transactions:
  4016. self.create_team_membership(team, user=self.user)
  4017. self.project.add_team(team)
  4018. self.transaction_data["transaction"] = transaction
  4019. self.store_event(self.transaction_data, self.project.id)
  4020. TeamKeyTransaction.objects.create(
  4021. organization=self.organization,
  4022. transaction=transaction,
  4023. project_team=ProjectTeam.objects.get(project=self.project, team=team),
  4024. )
  4025. query = {
  4026. "team": "myteams",
  4027. "project": [self.project.id],
  4028. "field": [
  4029. "team_key_transaction",
  4030. "transaction",
  4031. "transaction.status",
  4032. "project",
  4033. "epm()",
  4034. "failure_rate()",
  4035. "percentile(transaction.duration, 0.95)",
  4036. ],
  4037. }
  4038. # test ascending order
  4039. query["orderby"] = ["team_key_transaction", "transaction"]
  4040. response = self.do_request(query)
  4041. assert response.status_code == 200, response.content
  4042. data = response.data["data"]
  4043. assert len(data) == 3
  4044. assert data[0]["team_key_transaction"] == 0
  4045. assert data[0]["transaction"] == "/blah_transaction/"
  4046. assert data[1]["team_key_transaction"] == 1
  4047. assert data[1]["transaction"] == "/foo_transaction/"
  4048. assert data[2]["team_key_transaction"] == 1
  4049. assert data[2]["transaction"] == "/zoo_transaction/"
  4050. # test descending order
  4051. query["orderby"] = ["-team_key_transaction", "-transaction"]
  4052. response = self.do_request(query)
  4053. assert response.status_code == 200, response.content
  4054. data = response.data["data"]
  4055. assert len(data) == 3
  4056. assert data[0]["team_key_transaction"] == 1
  4057. assert data[0]["transaction"] == "/zoo_transaction/"
  4058. assert data[1]["team_key_transaction"] == 1
  4059. assert data[1]["transaction"] == "/foo_transaction/"
  4060. assert data[2]["team_key_transaction"] == 0
  4061. assert data[2]["transaction"] == "/blah_transaction/"
  4062. def test_team_key_transactions_query(self):
  4063. team1 = self.create_team(organization=self.organization, name="Team A")
  4064. team2 = self.create_team(organization=self.organization, name="Team B")
  4065. transactions = ["/blah_transaction/"]
  4066. key_transactions = [
  4067. (team1, "/foo_transaction/"),
  4068. (team2, "/zoo_transaction/"),
  4069. ]
  4070. for transaction in transactions:
  4071. self.transaction_data["transaction"] = transaction
  4072. self.store_event(self.transaction_data, self.project.id)
  4073. for team, transaction in key_transactions:
  4074. self.create_team_membership(team, user=self.user)
  4075. self.project.add_team(team)
  4076. self.transaction_data["transaction"] = transaction
  4077. self.store_event(self.transaction_data, self.project.id)
  4078. TeamKeyTransaction.objects.create(
  4079. organization=self.organization,
  4080. project_team=ProjectTeam.objects.get(
  4081. project=self.project,
  4082. team=team,
  4083. ),
  4084. transaction=transaction,
  4085. )
  4086. query = {
  4087. "team": "myteams",
  4088. "project": [self.project.id],
  4089. # use the order by to ensure the result order
  4090. "orderby": "transaction",
  4091. "field": [
  4092. "team_key_transaction",
  4093. "transaction",
  4094. "transaction.status",
  4095. "project",
  4096. "epm()",
  4097. "failure_rate()",
  4098. "percentile(transaction.duration, 0.95)",
  4099. ],
  4100. }
  4101. # key transactions
  4102. query["query"] = "has:team_key_transaction"
  4103. response = self.do_request(query)
  4104. assert response.status_code == 200, response.content
  4105. data = response.data["data"]
  4106. assert len(data) == 2
  4107. assert data[0]["team_key_transaction"] == 1
  4108. assert data[0]["transaction"] == "/foo_transaction/"
  4109. assert data[1]["team_key_transaction"] == 1
  4110. assert data[1]["transaction"] == "/zoo_transaction/"
  4111. # key transactions
  4112. query["query"] = "team_key_transaction:true"
  4113. response = self.do_request(query)
  4114. assert response.status_code == 200, response.content
  4115. data = response.data["data"]
  4116. assert len(data) == 2
  4117. assert data[0]["team_key_transaction"] == 1
  4118. assert data[0]["transaction"] == "/foo_transaction/"
  4119. assert data[1]["team_key_transaction"] == 1
  4120. assert data[1]["transaction"] == "/zoo_transaction/"
  4121. # not key transactions
  4122. query["query"] = "!has:team_key_transaction"
  4123. response = self.do_request(query)
  4124. assert response.status_code == 200, response.content
  4125. data = response.data["data"]
  4126. assert len(data) == 1
  4127. assert data[0]["team_key_transaction"] == 0
  4128. assert data[0]["transaction"] == "/blah_transaction/"
  4129. # not key transactions
  4130. query["query"] = "team_key_transaction:false"
  4131. response = self.do_request(query)
  4132. assert response.status_code == 200, response.content
  4133. data = response.data["data"]
  4134. assert len(data) == 1
  4135. assert data[0]["team_key_transaction"] == 0
  4136. assert data[0]["transaction"] == "/blah_transaction/"
  4137. def test_too_many_team_key_transactions(self):
  4138. MAX_QUERYABLE_TEAM_KEY_TRANSACTIONS = 1
  4139. with mock.patch(
  4140. "sentry.search.events.fields.MAX_QUERYABLE_TEAM_KEY_TRANSACTIONS",
  4141. MAX_QUERYABLE_TEAM_KEY_TRANSACTIONS,
  4142. ):
  4143. team = self.create_team(organization=self.organization, name="Team A")
  4144. self.create_team_membership(team, user=self.user)
  4145. self.project.add_team(team)
  4146. project_team = ProjectTeam.objects.get(project=self.project, team=team)
  4147. for i in range(MAX_QUERYABLE_TEAM_KEY_TRANSACTIONS + 1):
  4148. transaction = f"transaction-{team.id}-{i}"
  4149. self.transaction_data["transaction"] = transaction
  4150. self.store_event(self.transaction_data, self.project.id)
  4151. TeamKeyTransaction.objects.bulk_create(
  4152. [
  4153. TeamKeyTransaction(
  4154. organization=self.organization,
  4155. project_team=project_team,
  4156. transaction=f"transaction-{team.id}-{i}",
  4157. )
  4158. for i in range(MAX_QUERYABLE_TEAM_KEY_TRANSACTIONS + 1)
  4159. ]
  4160. )
  4161. query = {
  4162. "team": "myteams",
  4163. "project": [self.project.id],
  4164. "orderby": "transaction",
  4165. "field": [
  4166. "team_key_transaction",
  4167. "transaction",
  4168. "transaction.status",
  4169. "project",
  4170. "epm()",
  4171. "failure_rate()",
  4172. "percentile(transaction.duration, 0.95)",
  4173. ],
  4174. }
  4175. response = self.do_request(query)
  4176. assert response.status_code == 200, response.content
  4177. data = response.data["data"]
  4178. assert len(data) == 2
  4179. assert (
  4180. sum(row["team_key_transaction"] for row in data)
  4181. == MAX_QUERYABLE_TEAM_KEY_TRANSACTIONS
  4182. )
  4183. def test_no_pagination_param(self):
  4184. self.store_event(
  4185. data={
  4186. "event_id": "a" * 32,
  4187. "timestamp": self.ten_mins_ago_iso,
  4188. "fingerprint": ["group1"],
  4189. },
  4190. project_id=self.project.id,
  4191. )
  4192. query = {"field": ["id", "project.id"], "project": [self.project.id], "noPagination": True}
  4193. response = self.do_request(query)
  4194. assert response.status_code == 200
  4195. assert len(response.data["data"]) == 1
  4196. assert "Link" not in response
  4197. def test_nan_result(self):
  4198. query = {"field": ["apdex(300)"], "project": [self.project.id], "query": f"id:{'0' * 32}"}
  4199. response = self.do_request(query)
  4200. assert response.status_code == 200
  4201. assert len(response.data["data"]) == 1
  4202. assert response.data["data"][0]["apdex(300)"] == 0
  4203. def test_equation_simple(self):
  4204. event_data = self.load_data(
  4205. timestamp=self.ten_mins_ago,
  4206. )
  4207. event_data["breakdowns"]["span_ops"]["ops.http"]["value"] = 1500
  4208. self.store_event(data=event_data, project_id=self.project.id)
  4209. query = {
  4210. "field": ["spans.http", "equation|spans.http / 3"],
  4211. "project": [self.project.id],
  4212. "query": "event.type:transaction",
  4213. }
  4214. response = self.do_request(
  4215. query,
  4216. {
  4217. "organizations:discover-basic": True,
  4218. },
  4219. )
  4220. assert response.status_code == 200, response.content
  4221. assert len(response.data["data"]) == 1
  4222. assert (
  4223. response.data["data"][0]["equation|spans.http / 3"]
  4224. == event_data["breakdowns"]["span_ops"]["ops.http"]["value"] / 3
  4225. )
  4226. assert response.data["meta"]["fields"]["equation|spans.http / 3"] == "number"
  4227. def test_equation_sort(self):
  4228. event_data = self.transaction_data.copy()
  4229. event_data["breakdowns"] = {"span_ops": {"ops.http": {"value": 1500}}}
  4230. self.store_event(data=event_data, project_id=self.project.id)
  4231. event_data2 = self.transaction_data.copy()
  4232. event_data2["breakdowns"] = {"span_ops": {"ops.http": {"value": 2000}}}
  4233. self.store_event(data=event_data2, project_id=self.project.id)
  4234. query = {
  4235. "field": ["spans.http", "equation|spans.http / 3"],
  4236. "project": [self.project.id],
  4237. "orderby": "equation|spans.http / 3",
  4238. "query": "event.type:transaction",
  4239. }
  4240. response = self.do_request(
  4241. query,
  4242. {
  4243. "organizations:discover-basic": True,
  4244. },
  4245. )
  4246. assert response.status_code == 200, response.content
  4247. assert len(response.data["data"]) == 2
  4248. assert (
  4249. response.data["data"][0]["equation|spans.http / 3"]
  4250. == event_data["breakdowns"]["span_ops"]["ops.http"]["value"] / 3
  4251. )
  4252. assert (
  4253. response.data["data"][1]["equation|spans.http / 3"]
  4254. == event_data2["breakdowns"]["span_ops"]["ops.http"]["value"] / 3
  4255. )
  4256. def test_equation_operation_limit(self):
  4257. query = {
  4258. "field": ["spans.http", f"equation|spans.http{' * 2' * 11}"],
  4259. "project": [self.project.id],
  4260. "query": "event.type:transaction",
  4261. }
  4262. response = self.do_request(
  4263. query,
  4264. {
  4265. "organizations:discover-basic": True,
  4266. },
  4267. )
  4268. assert response.status_code == 400
  4269. @mock.patch("sentry.api.bases.organization_events.MAX_FIELDS", 2)
  4270. def test_equation_field_limit(self):
  4271. query = {
  4272. "field": ["spans.http", "transaction.duration", "equation|5 * 2"],
  4273. "project": [self.project.id],
  4274. "query": "event.type:transaction",
  4275. }
  4276. response = self.do_request(
  4277. query,
  4278. {
  4279. "organizations:discover-basic": True,
  4280. },
  4281. )
  4282. assert response.status_code == 400
  4283. def test_count_if(self):
  4284. unicode_phrase1 = "\u716e\u6211\u66f4\u591a\u7684\u98df\u7269\uff0c\u6211\u9913\u4e86"
  4285. for i in range(5):
  4286. data = self.load_data(
  4287. timestamp=before_now(minutes=(10 + i)),
  4288. duration=timedelta(milliseconds=100 if i < 3 else 200),
  4289. )
  4290. data["tags"] = {
  4291. "sub_customer.is-Enterprise-42": "yes" if i == 0 else "no",
  4292. "unicode-phrase": unicode_phrase1 if i == 0 else "no",
  4293. }
  4294. self.store_event(data, project_id=self.project.id)
  4295. query = {
  4296. "field": [
  4297. "count_if(transaction.duration, less, 150)",
  4298. "count_if(transaction.duration, greater, 150)",
  4299. "count_if(sub_customer.is-Enterprise-42, equals, yes)",
  4300. "count_if(sub_customer.is-Enterprise-42, notEquals, yes)",
  4301. f"count_if(unicode-phrase, equals, {unicode_phrase1})",
  4302. ],
  4303. "project": [self.project.id],
  4304. }
  4305. response = self.do_request(query)
  4306. assert response.status_code == 200
  4307. assert len(response.data["data"]) == 1
  4308. assert response.data["data"][0]["count_if(transaction.duration, less, 150)"] == 3
  4309. assert response.data["data"][0]["count_if(transaction.duration, greater, 150)"] == 2
  4310. assert response.data["data"][0]["count_if(sub_customer.is-Enterprise-42, equals, yes)"] == 1
  4311. assert (
  4312. response.data["data"][0]["count_if(sub_customer.is-Enterprise-42, notEquals, yes)"] == 4
  4313. )
  4314. assert response.data["data"][0][f"count_if(unicode-phrase, equals, {unicode_phrase1})"] == 1
  4315. def test_count_if_array_field(self):
  4316. data = self.load_data(platform="javascript")
  4317. data["timestamp"] = self.ten_mins_ago_iso
  4318. self.store_event(data=data, project_id=self.project.id)
  4319. query = {
  4320. "field": [
  4321. "count_if(stack.filename, equals, raven.js)",
  4322. ],
  4323. "project": [self.project.id],
  4324. }
  4325. response = self.do_request(query)
  4326. assert response.status_code == 200, response.content
  4327. assert len(response.data["data"]) == 1
  4328. assert response.data["data"][0]["count_if(stack.filename, equals, raven.js)"] == 1
  4329. def test_count_if_measurements_cls(self):
  4330. data = self.transaction_data.copy()
  4331. data["measurements"] = {"cls": {"value": 0.5}}
  4332. self.store_event(data, project_id=self.project.id)
  4333. data["measurements"] = {"cls": {"value": 0.1}}
  4334. self.store_event(data, project_id=self.project.id)
  4335. query = {
  4336. "field": [
  4337. "count_if(measurements.cls, greater, 0.05)",
  4338. "count_if(measurements.cls, less, 0.3)",
  4339. ],
  4340. "project": [self.project.id],
  4341. }
  4342. response = self.do_request(query)
  4343. assert response.status_code == 200
  4344. assert len(response.data["data"]) == 1
  4345. assert response.data["data"][0]["count_if(measurements.cls, greater, 0.05)"] == 2
  4346. assert response.data["data"][0]["count_if(measurements.cls, less, 0.3)"] == 1
  4347. def test_count_if_filter(self):
  4348. for i in range(5):
  4349. data = self.load_data(
  4350. timestamp=before_now(minutes=(10 + i)),
  4351. duration=timedelta(milliseconds=100 if i < 3 else 200),
  4352. )
  4353. data["tags"] = {"sub_customer.is-Enterprise-42": "yes" if i == 0 else "no"}
  4354. self.store_event(data, project_id=self.project.id)
  4355. query = {
  4356. "field": [
  4357. "count_if(transaction.duration, less, 150)",
  4358. ],
  4359. "query": "count_if(transaction.duration, less, 150):>2",
  4360. "project": [self.project.id],
  4361. }
  4362. response = self.do_request(query)
  4363. assert response.status_code == 200
  4364. assert len(response.data["data"]) == 1
  4365. assert response.data["data"][0]["count_if(transaction.duration, less, 150)"] == 3
  4366. query = {
  4367. "field": [
  4368. "count_if(transaction.duration, less, 150)",
  4369. ],
  4370. "query": "count_if(transaction.duration, less, 150):<2",
  4371. "project": [self.project.id],
  4372. }
  4373. response = self.do_request(query)
  4374. assert response.status_code == 200
  4375. assert len(response.data["data"]) == 0
  4376. def test_filters_with_escaped_asterisk(self):
  4377. self.transaction_data["transaction"] = r"/:a*/:b-:c(\d\.\e+)"
  4378. self.store_event(self.transaction_data, project_id=self.project.id)
  4379. query = {
  4380. "field": ["transaction", "transaction.duration"],
  4381. # make sure to escape the asterisk so it's not treated as a wildcard
  4382. "query": r'transaction:"/:a\*/:b-:c(\d\.\e+)"',
  4383. "project": [self.project.id],
  4384. }
  4385. response = self.do_request(query)
  4386. assert response.status_code == 200
  4387. assert len(response.data["data"]) == 1
  4388. def test_filters_with_back_slashes(self):
  4389. self.transaction_data["transaction"] = r"a\b\c@d"
  4390. self.store_event(self.transaction_data, project_id=self.project.id)
  4391. query = {
  4392. "field": ["transaction", "transaction.duration"],
  4393. "query": r'transaction:"a\b\c@d"',
  4394. "project": [self.project.id],
  4395. }
  4396. response = self.do_request(query)
  4397. assert response.status_code == 200
  4398. assert len(response.data["data"]) == 1
  4399. def test_mobile_measurements(self):
  4400. self.transaction_data["measurements"]["frames_total"] = {"value": 100}
  4401. self.transaction_data["measurements"]["frames_slow"] = {"value": 10}
  4402. self.transaction_data["measurements"]["frames_frozen"] = {"value": 5}
  4403. self.transaction_data["measurements"]["stall_count"] = {"value": 2}
  4404. self.transaction_data["measurements"]["stall_total_time"] = {"value": 12}
  4405. self.transaction_data["measurements"]["stall_longest_time"] = {"value": 7}
  4406. self.store_event(self.transaction_data, project_id=self.project.id)
  4407. query = {
  4408. "field": [
  4409. "measurements.frames_total",
  4410. "measurements.frames_slow",
  4411. "measurements.frames_frozen",
  4412. "measurements.frames_slow_rate",
  4413. "measurements.frames_frozen_rate",
  4414. "measurements.stall_count",
  4415. "measurements.stall_total_time",
  4416. "measurements.stall_longest_time",
  4417. "measurements.stall_percentage",
  4418. ],
  4419. "query": "",
  4420. "project": [self.project.id],
  4421. }
  4422. response = self.do_request(query)
  4423. assert response.status_code == 200
  4424. data = response.data["data"]
  4425. assert len(data) == 1
  4426. assert data[0]["measurements.frames_total"] == 100
  4427. assert data[0]["measurements.frames_slow"] == 10
  4428. assert data[0]["measurements.frames_frozen"] == 5
  4429. assert data[0]["measurements.frames_slow_rate"] == 0.1
  4430. assert data[0]["measurements.frames_frozen_rate"] == 0.05
  4431. assert data[0]["measurements.stall_count"] == 2
  4432. assert data[0]["measurements.stall_total_time"] == 12
  4433. assert data[0]["measurements.stall_longest_time"] == 7
  4434. assert data[0]["measurements.stall_percentage"] == 0.004
  4435. meta = response.data["meta"]["fields"]
  4436. assert meta["measurements.frames_total"] == "number"
  4437. assert meta["measurements.frames_slow"] == "number"
  4438. assert meta["measurements.frames_frozen"] == "number"
  4439. assert meta["measurements.frames_slow_rate"] == "percentage"
  4440. assert meta["measurements.frames_frozen_rate"] == "percentage"
  4441. assert meta["measurements.stall_count"] == "number"
  4442. assert meta["measurements.stall_total_time"] == "number"
  4443. assert meta["measurements.stall_longest_time"] == "number"
  4444. assert meta["measurements.stall_percentage"] == "percentage"
  4445. query = {
  4446. "field": [
  4447. "p75(measurements.frames_slow_rate)",
  4448. "p75(measurements.frames_frozen_rate)",
  4449. "percentile(measurements.frames_slow_rate,0.5)",
  4450. "percentile(measurements.frames_frozen_rate,0.5)",
  4451. "p75(measurements.stall_percentage)",
  4452. "percentile(measurements.stall_percentage,0.5)",
  4453. ],
  4454. "query": "",
  4455. "project": [self.project.id],
  4456. }
  4457. response = self.do_request(query)
  4458. assert response.status_code == 200
  4459. data = response.data["data"]
  4460. assert len(data) == 1
  4461. assert data[0]["p75(measurements.frames_slow_rate)"] == 0.1
  4462. assert data[0]["p75(measurements.frames_frozen_rate)"] == 0.05
  4463. assert data[0]["p75(measurements.stall_percentage)"] == 0.004
  4464. assert data[0]["percentile(measurements.frames_slow_rate,0.5)"] == 0.1
  4465. assert data[0]["percentile(measurements.frames_frozen_rate,0.5)"] == 0.05
  4466. assert data[0]["percentile(measurements.stall_percentage,0.5)"] == 0.004
  4467. meta = response.data["meta"]["fields"]
  4468. assert meta["p75(measurements.frames_slow_rate)"] == "percentage"
  4469. assert meta["p75(measurements.frames_frozen_rate)"] == "percentage"
  4470. assert meta["p75(measurements.stall_percentage)"] == "percentage"
  4471. assert meta["percentile(measurements.frames_slow_rate,0.5)"] == "percentage"
  4472. assert meta["percentile(measurements.stall_percentage,0.5)"] == "percentage"
  4473. def test_project_auto_fields(self):
  4474. self.store_event(
  4475. data={
  4476. "event_id": "a" * 32,
  4477. "environment": "staging",
  4478. "timestamp": self.ten_mins_ago_iso,
  4479. },
  4480. project_id=self.project.id,
  4481. )
  4482. query = {"field": ["environment"]}
  4483. response = self.do_request(query)
  4484. assert response.status_code == 200, response.content
  4485. assert len(response.data["data"]) == 1
  4486. assert response.data["data"][0]["environment"] == "staging"
  4487. assert response.data["data"][0]["project.name"] == self.project.slug
  4488. def test_timestamp_different_from_params(self):
  4489. fifteen_days_ago = iso_format(before_now(days=15))
  4490. fifteen_days_later = iso_format(before_now(days=-15))
  4491. for query_text in [
  4492. f"timestamp:<{fifteen_days_ago}",
  4493. f"timestamp:<={fifteen_days_ago}",
  4494. f"timestamp:>{fifteen_days_later}",
  4495. f"timestamp:>={fifteen_days_later}",
  4496. ]:
  4497. query = {
  4498. "field": ["count()"],
  4499. "query": query_text,
  4500. "statsPeriod": "14d",
  4501. "project": self.project.id,
  4502. }
  4503. response = self.do_request(query)
  4504. assert response.status_code == 400, query_text
  4505. @mock.patch("sentry.search.events.builder.discover.raw_snql_query")
  4506. def test_removes_unnecessary_default_project_and_transaction_thresholds(self, mock_snql_query):
  4507. mock_snql_query.side_effect = [{"meta": {}, "data": []}]
  4508. ProjectTransactionThreshold.objects.create(
  4509. project=self.project,
  4510. organization=self.organization,
  4511. # these are the default values that we use
  4512. threshold=constants.DEFAULT_PROJECT_THRESHOLD,
  4513. metric=TransactionMetric.DURATION.value,
  4514. )
  4515. ProjectTransactionThresholdOverride.objects.create(
  4516. transaction="transaction",
  4517. project=self.project,
  4518. organization=self.organization,
  4519. # these are the default values that we use
  4520. threshold=constants.DEFAULT_PROJECT_THRESHOLD,
  4521. metric=TransactionMetric.DURATION.value,
  4522. )
  4523. query = {
  4524. "field": ["apdex()", "user_misery()"],
  4525. "query": "event.type:transaction",
  4526. "project": [self.project.id],
  4527. }
  4528. response = self.do_request(
  4529. query,
  4530. features={
  4531. "organizations:discover-basic": True,
  4532. "organizations:global-views": True,
  4533. },
  4534. )
  4535. assert response.status_code == 200, response.content
  4536. assert mock_snql_query.call_count == 1
  4537. assert (
  4538. Function("tuple", ["duration", 300], "project_threshold_config")
  4539. in mock_snql_query.call_args_list[0][0][0].query.select
  4540. )
  4541. @mock.patch("sentry.search.events.builder.discover.raw_snql_query")
  4542. def test_removes_unnecessary_default_project_and_transaction_thresholds_keeps_others(
  4543. self, mock_snql_query
  4544. ):
  4545. mock_snql_query.side_effect = [{"meta": {}, "data": []}]
  4546. ProjectTransactionThreshold.objects.create(
  4547. project=self.project,
  4548. organization=self.organization,
  4549. # these are the default values that we use
  4550. threshold=constants.DEFAULT_PROJECT_THRESHOLD,
  4551. metric=TransactionMetric.DURATION.value,
  4552. )
  4553. ProjectTransactionThresholdOverride.objects.create(
  4554. transaction="transaction",
  4555. project=self.project,
  4556. organization=self.organization,
  4557. # these are the default values that we use
  4558. threshold=constants.DEFAULT_PROJECT_THRESHOLD,
  4559. metric=TransactionMetric.DURATION.value,
  4560. )
  4561. project = self.create_project()
  4562. ProjectTransactionThreshold.objects.create(
  4563. project=project,
  4564. organization=self.organization,
  4565. threshold=100,
  4566. metric=TransactionMetric.LCP.value,
  4567. )
  4568. ProjectTransactionThresholdOverride.objects.create(
  4569. transaction="transaction",
  4570. project=project,
  4571. organization=self.organization,
  4572. threshold=200,
  4573. metric=TransactionMetric.LCP.value,
  4574. )
  4575. query = {
  4576. "field": ["apdex()", "user_misery()"],
  4577. "query": "event.type:transaction",
  4578. "project": [self.project.id, project.id],
  4579. }
  4580. response = self.do_request(
  4581. query,
  4582. features={
  4583. "organizations:discover-basic": True,
  4584. "organizations:global-views": True,
  4585. },
  4586. )
  4587. assert response.status_code == 200, response.content
  4588. assert mock_snql_query.call_count == 1
  4589. project_threshold_override_config_index = Function(
  4590. "indexOf",
  4591. [
  4592. # only 1 transaction override is present here
  4593. # because the other use to the default values
  4594. [(Function("toUInt64", [project.id]), "transaction")],
  4595. (Column("project_id"), Column("transaction")),
  4596. ],
  4597. "project_threshold_override_config_index",
  4598. )
  4599. project_threshold_config_index = Function(
  4600. "indexOf",
  4601. [
  4602. # only 1 project override is present here
  4603. # because the other use to the default values
  4604. [Function("toUInt64", [project.id])],
  4605. Column("project_id"),
  4606. ],
  4607. "project_threshold_config_index",
  4608. )
  4609. assert (
  4610. Function(
  4611. "if",
  4612. [
  4613. Function("equals", [project_threshold_override_config_index, 0]),
  4614. Function(
  4615. "if",
  4616. [
  4617. Function("equals", [project_threshold_config_index, 0]),
  4618. ("duration", 300),
  4619. Function(
  4620. "arrayElement", [[("lcp", 100)], project_threshold_config_index]
  4621. ),
  4622. ],
  4623. ),
  4624. Function(
  4625. "arrayElement",
  4626. [[("lcp", 200)], project_threshold_override_config_index],
  4627. ),
  4628. ],
  4629. "project_threshold_config",
  4630. )
  4631. in mock_snql_query.call_args_list[0][0][0].query.select
  4632. )
  4633. def test_count_web_vitals(self):
  4634. # Good
  4635. self.transaction_data["measurements"] = {
  4636. "lcp": {"value": constants.VITAL_THRESHOLDS["lcp"]["meh"] - 100},
  4637. }
  4638. self.store_event(self.transaction_data, self.project.id)
  4639. # Meh
  4640. self.transaction_data["measurements"] = {
  4641. "lcp": {"value": constants.VITAL_THRESHOLDS["lcp"]["meh"] + 100},
  4642. }
  4643. self.store_event(self.transaction_data, self.project.id)
  4644. self.store_event(self.transaction_data, self.project.id)
  4645. query = {
  4646. "field": [
  4647. "count_web_vitals(measurements.lcp, poor)",
  4648. "count_web_vitals(measurements.lcp, meh)",
  4649. "count_web_vitals(measurements.lcp, good)",
  4650. ]
  4651. }
  4652. response = self.do_request(query)
  4653. assert response.status_code == 200, response.content
  4654. assert len(response.data["data"]) == 1
  4655. assert response.data["data"][0] == {
  4656. "count_web_vitals(measurements.lcp, poor)": 0,
  4657. "count_web_vitals(measurements.lcp, meh)": 2,
  4658. "count_web_vitals(measurements.lcp, good)": 1,
  4659. }
  4660. def test_count_web_vitals_invalid_vital(self):
  4661. query = {
  4662. "field": [
  4663. "count_web_vitals(measurements.foo, poor)",
  4664. ],
  4665. "project": [self.project.id],
  4666. }
  4667. response = self.do_request(query)
  4668. assert response.status_code == 400, response.content
  4669. query = {
  4670. "field": [
  4671. "count_web_vitals(tags[lcp], poor)",
  4672. ],
  4673. "project": [self.project.id],
  4674. }
  4675. response = self.do_request(query)
  4676. assert response.status_code == 400, response.content
  4677. query = {
  4678. "field": [
  4679. "count_web_vitals(transaction.duration, poor)",
  4680. ],
  4681. "project": [self.project.id],
  4682. }
  4683. response = self.do_request(query)
  4684. assert response.status_code == 400, response.content
  4685. query = {
  4686. "field": [
  4687. "count_web_vitals(measurements.lcp, bad)",
  4688. ],
  4689. "project": [self.project.id],
  4690. }
  4691. response = self.do_request(query)
  4692. assert response.status_code == 400, response.content
  4693. def test_tag_that_looks_like_aggregate(self):
  4694. data = self.load_data()
  4695. data["tags"] = {"p95": "<5k"}
  4696. self.store_event(data, project_id=self.project.id)
  4697. query = {
  4698. "field": ["p95"],
  4699. "query": "p95:<5k",
  4700. "project": [self.project.id],
  4701. }
  4702. response = self.do_request(query)
  4703. assert response.status_code == 200, response.content
  4704. data = response.data["data"]
  4705. assert len(data) == 1
  4706. assert data[0]["p95"] == "<5k"
  4707. def test_chained_or_query_meta_tip(self):
  4708. query = {
  4709. "field": ["transaction"],
  4710. "query": "transaction:a OR transaction:b",
  4711. "project": [self.project.id],
  4712. }
  4713. response = self.do_request(query)
  4714. assert response.status_code == 200, response.content
  4715. meta = response.data["meta"]
  4716. assert meta["tips"] == {
  4717. "query": "Did you know you can replace chained or conditions like `field:a OR field:b OR field:c` with `field:[a,b,c]`",
  4718. "columns": None,
  4719. }
  4720. @override_settings(SENTRY_SELF_HOSTED=False)
  4721. def test_no_ratelimit(self):
  4722. query = {
  4723. "field": ["transaction"],
  4724. "project": [self.project.id],
  4725. }
  4726. with freeze_time("2000-01-01"):
  4727. for _ in range(15):
  4728. self.do_request(query)
  4729. response = self.do_request(query)
  4730. assert response.status_code == 200, response.content
  4731. def test_transaction_source(self):
  4732. query = {
  4733. "field": ["transaction"],
  4734. "query": "transaction.source:task",
  4735. "project": [self.project.id],
  4736. }
  4737. response = self.do_request(query)
  4738. assert response.status_code == 200, response.content
  4739. def test_readable_device_name(self):
  4740. data = self.load_data()
  4741. data["tags"] = {"device": "iPhone14,3"}
  4742. self.store_event(data, project_id=self.project.id)
  4743. query = {
  4744. "field": ["device"],
  4745. "query": "",
  4746. "project": [self.project.id],
  4747. "readable": True,
  4748. }
  4749. response = self.do_request(query)
  4750. assert response.status_code == 200, response.content
  4751. data = response.data["data"]
  4752. assert len(data) == 1
  4753. assert data[0]["device"] == "iPhone14,3"
  4754. assert data[0]["readable"] == "iPhone 13 Pro Max"
  4755. def test_http_status_code(self):
  4756. project1 = self.create_project()
  4757. project2 = self.create_project()
  4758. self.store_event(
  4759. data={
  4760. "event_id": "a" * 32,
  4761. "transaction": "/example",
  4762. "message": "how to make fast",
  4763. "timestamp": self.ten_mins_ago_iso,
  4764. "tags": {"http.status_code": "200"},
  4765. },
  4766. project_id=project1.id,
  4767. )
  4768. self.store_event(
  4769. data={
  4770. "event_id": "b" * 32,
  4771. "transaction": "/example",
  4772. "message": "how to make fast",
  4773. "timestamp": self.ten_mins_ago_iso,
  4774. "contexts": {"response": {"status_code": 400}},
  4775. },
  4776. project_id=project2.id,
  4777. )
  4778. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4779. query = {
  4780. "field": ["event.type", "http.status_code"],
  4781. "query": "",
  4782. "statsPeriod": "24h",
  4783. }
  4784. response = self.do_request(query, features=features)
  4785. assert response.status_code == 200, response.content
  4786. data = response.data["data"]
  4787. assert len(data) == 2
  4788. result = {r["http.status_code"] for r in data}
  4789. assert result == {"200", "400"}
  4790. def test_http_status_code_context_priority(self):
  4791. project1 = self.create_project()
  4792. self.store_event(
  4793. data={
  4794. "event_id": "a" * 32,
  4795. "transaction": "/example",
  4796. "message": "how to make fast",
  4797. "timestamp": self.ten_mins_ago_iso,
  4798. "tags": {"http.status_code": "200"},
  4799. "contexts": {"response": {"status_code": 400}},
  4800. },
  4801. project_id=project1.id,
  4802. )
  4803. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4804. query = {
  4805. "field": ["event.type", "http.status_code"],
  4806. "query": "",
  4807. "statsPeriod": "24h",
  4808. }
  4809. response = self.do_request(query, features=features)
  4810. assert response.status_code == 200, response.content
  4811. data = response.data["data"]
  4812. assert len(data) == 1
  4813. assert data[0]["http.status_code"] == "400"
  4814. def test_total_count(self):
  4815. project1 = self.create_project()
  4816. for i in range(3):
  4817. self.store_event(data=self.load_data(platform="javascript"), project_id=project1.id)
  4818. self.store_event(
  4819. data={
  4820. "event_id": "a" * 32,
  4821. "transaction": "/example",
  4822. "message": "how to make fast",
  4823. "timestamp": self.ten_mins_ago_iso,
  4824. },
  4825. project_id=project1.id,
  4826. )
  4827. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4828. query = {
  4829. "field": ["transaction", "total.count", "count()"],
  4830. "query": "!transaction:/example",
  4831. "statsPeriod": "24h",
  4832. }
  4833. response = self.do_request(query, features=features)
  4834. assert response.status_code == 200, response.content
  4835. data = response.data["data"]
  4836. assert len(data) == 1
  4837. assert data[0]["total.count"] == 3
  4838. def test_total_count_by_itself(self):
  4839. project1 = self.create_project()
  4840. for i in range(3):
  4841. self.store_event(data=self.load_data(platform="javascript"), project_id=project1.id)
  4842. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4843. query = {
  4844. "field": ["total.count"],
  4845. "statsPeriod": "24h",
  4846. }
  4847. response = self.do_request(query, features=features)
  4848. assert response.status_code == 400, response.content
  4849. def test_total_count_equation(self):
  4850. project1 = self.create_project()
  4851. for i in range(3):
  4852. self.store_event(data=self.load_data(platform="javascript"), project_id=project1.id)
  4853. self.store_event(
  4854. data={
  4855. "event_id": "a" * 32,
  4856. "transaction": "/example",
  4857. "message": "how to make fast",
  4858. "timestamp": self.ten_mins_ago_iso,
  4859. },
  4860. project_id=project1.id,
  4861. )
  4862. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4863. query = {
  4864. "field": ["transaction", "count()", "total.count", "equation|count()/total.count"],
  4865. "query": "",
  4866. "orderby": "equation|count()/total.count",
  4867. "statsPeriod": "24h",
  4868. }
  4869. response = self.do_request(query, features=features)
  4870. assert response.status_code == 200, response.content
  4871. data = response.data["data"]
  4872. assert len(data) == 2
  4873. assert data[0]["equation|count()/total.count"] == 0.25
  4874. assert data[1]["equation|count()/total.count"] == 0.75
  4875. def test_total_count_filter(self):
  4876. project1 = self.create_project()
  4877. for i in range(3):
  4878. self.store_event(data=self.load_data(platform="javascript"), project_id=project1.id)
  4879. self.store_event(
  4880. data={
  4881. "event_id": "a" * 32,
  4882. "transaction": "/example",
  4883. "message": "how to make fast",
  4884. "timestamp": self.ten_mins_ago_iso,
  4885. "tags": {"total.count": ">45"},
  4886. },
  4887. project_id=project1.id,
  4888. )
  4889. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4890. query = {
  4891. "field": ["transaction", "count()", "total.count"],
  4892. "query": "total.count:>45",
  4893. "orderby": "count()",
  4894. "statsPeriod": "24h",
  4895. }
  4896. response = self.do_request(query, features=features)
  4897. assert response.status_code == 200, response.content
  4898. data = response.data["data"]
  4899. assert len(data) == 1
  4900. assert data[0]["total.count"] == 1
  4901. def test_total_sum_transaction_duration_equation(self):
  4902. for i in range(3):
  4903. data = self.load_data(
  4904. timestamp=self.eleven_mins_ago,
  4905. duration=timedelta(seconds=5),
  4906. )
  4907. data["transaction"] = "/endpoint/1"
  4908. self.store_event(data, project_id=self.project.id)
  4909. data = self.load_data(
  4910. timestamp=self.ten_mins_ago,
  4911. duration=timedelta(seconds=5),
  4912. )
  4913. data["transaction"] = "/endpoint/2"
  4914. self.store_event(data, project_id=self.project.id)
  4915. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4916. query = {
  4917. "field": [
  4918. "transaction",
  4919. "sum(transaction.duration)",
  4920. "total.transaction_duration",
  4921. "equation|sum(transaction.duration)/total.transaction_duration",
  4922. ],
  4923. "query": "",
  4924. "orderby": "-equation|sum(transaction.duration)/total.transaction_duration",
  4925. "statsPeriod": "24h",
  4926. }
  4927. response = self.do_request(query, features=features)
  4928. assert response.status_code == 200, response.content
  4929. data = response.data["data"]
  4930. assert len(data) == 2
  4931. assert data[0]["equation|sum(transaction.duration)/total.transaction_duration"] == 0.75
  4932. assert data[1]["equation|sum(transaction.duration)/total.transaction_duration"] == 0.25
  4933. def test_device_class(self):
  4934. project1 = self.create_project()
  4935. for i in range(3):
  4936. self.store_event(
  4937. data={
  4938. "event_id": "a" * 32,
  4939. "transaction": "/example",
  4940. "message": "how to make fast",
  4941. "timestamp": self.ten_mins_ago_iso,
  4942. "tags": {"device.class": f"{i + 1}"},
  4943. },
  4944. project_id=project1.id,
  4945. )
  4946. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4947. query = {
  4948. "field": ["device.class"],
  4949. "statsPeriod": "24h",
  4950. }
  4951. response = self.do_request(query, features=features)
  4952. assert response.status_code == 200, response.content
  4953. data = response.data["data"]
  4954. assert len(data) == 3
  4955. result = (*map(lambda columns: columns["device.class"], data),)
  4956. assert "low" in result
  4957. assert "medium" in result
  4958. assert "high" in result
  4959. def test_device_class_filter_low(self):
  4960. project1 = self.create_project()
  4961. for i in range(3):
  4962. self.store_event(data=self.load_data(platform="javascript"), project_id=project1.id)
  4963. self.store_event(
  4964. data={
  4965. "event_id": "a" * 32,
  4966. "transaction": "/example",
  4967. "message": "how to make fast",
  4968. "timestamp": self.ten_mins_ago_iso,
  4969. "tags": {"device.class": "1"},
  4970. },
  4971. project_id=project1.id,
  4972. )
  4973. features = {"organizations:discover-basic": True, "organizations:global-views": True}
  4974. query = {
  4975. "field": ["device.class", "count()"],
  4976. "query": "device.class:low",
  4977. "orderby": "count()",
  4978. "statsPeriod": "24h",
  4979. }
  4980. response = self.do_request(query, features=features)
  4981. assert response.status_code == 200, response.content
  4982. data = response.data["data"]
  4983. assert len(data) == 1
  4984. assert data[0]["count()"] == 1
  4985. def test_group_id_as_custom_tag(self):
  4986. project1 = self.create_project()
  4987. self.store_event(
  4988. data={
  4989. "event_id": "a" * 32,
  4990. "message": "poof",
  4991. "timestamp": self.ten_mins_ago_iso,
  4992. "user": {"email": self.user.email},
  4993. "tags": {"group_id": "this should just get returned"},
  4994. },
  4995. project_id=project1.id,
  4996. )
  4997. query = {
  4998. "field": ["group_id"],
  4999. "query": "",
  5000. "orderby": "group_id",
  5001. "statsPeriod": "24h",
  5002. }
  5003. response = self.do_request(query)
  5004. assert response.status_code == 200, response.content
  5005. assert response.data["data"][0]["group_id"] == "this should just get returned"
  5006. def test_floored_epm(self):
  5007. for _ in range(5):
  5008. data = self.load_data(
  5009. timestamp=self.ten_mins_ago,
  5010. duration=timedelta(seconds=5),
  5011. )
  5012. data["transaction"] = "/aggregates/1"
  5013. event1 = self.store_event(data, project_id=self.project.id)
  5014. query = {
  5015. "field": ["transaction", "floored_epm()", "epm()"],
  5016. "query": "event.type:transaction",
  5017. "orderby": ["transaction"],
  5018. "start": self.eleven_mins_ago_iso,
  5019. "end": iso_format(self.nine_mins_ago),
  5020. }
  5021. response = self.do_request(query)
  5022. assert response.status_code == 200, response.content
  5023. assert len(response.data["data"]) == 1
  5024. data = response.data["data"]
  5025. assert data[0]["transaction"] == event1.transaction
  5026. assert data[0]["floored_epm()"] == 1
  5027. assert data[0]["epm()"] == 2.5
  5028. def test_floored_epm_more_events(self):
  5029. for _ in range(25):
  5030. data = self.load_data(
  5031. timestamp=self.ten_mins_ago,
  5032. duration=timedelta(seconds=5),
  5033. )
  5034. data["transaction"] = "/aggregates/1"
  5035. event1 = self.store_event(data, project_id=self.project.id)
  5036. query = {
  5037. "field": ["transaction", "floored_epm()", "epm()"],
  5038. "query": "event.type:transaction",
  5039. "orderby": ["transaction"],
  5040. "start": self.eleven_mins_ago_iso,
  5041. "end": iso_format(self.nine_mins_ago),
  5042. }
  5043. response = self.do_request(query)
  5044. assert response.status_code == 200, response.content
  5045. assert len(response.data["data"]) == 1
  5046. data = response.data["data"]
  5047. assert data[0]["transaction"] == event1.transaction
  5048. assert data[0]["epm()"] == 12.5
  5049. assert data[0]["floored_epm()"] == 10
  5050. @region_silo_test
  5051. class OrganizationEventsProfilesDatasetEndpointTest(OrganizationEventsEndpointTestBase):
  5052. @mock.patch("sentry.search.events.builder.discover.raw_snql_query")
  5053. def test_profiles_dataset_simple(self, mock_snql_query):
  5054. mock_snql_query.side_effect = [
  5055. {
  5056. "data": [
  5057. {
  5058. "project": self.project.id,
  5059. "transaction": "foo",
  5060. "last_seen": "2022-10-20T16:41:22+00:00",
  5061. "latest_event": "a" * 32,
  5062. "count": 1,
  5063. "count_unique_transaction": 1,
  5064. "percentile_profile_duration_0_25": 1,
  5065. "p50_profile_duration": 1,
  5066. "p75_profile_duration": 1,
  5067. "p95_profile_duration": 1,
  5068. "p99_profile_duration": 1,
  5069. "p100_profile_duration": 1,
  5070. "min_profile_duration": 1,
  5071. "max_profile_duration": 1,
  5072. "avg_profile_duration": 1,
  5073. "sum_profile_duration": 1,
  5074. },
  5075. ],
  5076. "meta": [
  5077. {
  5078. "name": "project",
  5079. "type": "UInt64",
  5080. },
  5081. {
  5082. "name": "transaction",
  5083. "type": "LowCardinality(String)",
  5084. },
  5085. {
  5086. "name": "last_seen",
  5087. "type": "DateTime",
  5088. },
  5089. {
  5090. "name": "latest_event",
  5091. "type": "String",
  5092. },
  5093. {
  5094. "name": "count",
  5095. "type": "UInt64",
  5096. },
  5097. {
  5098. "name": "count_unique_transaction",
  5099. "type": "UInt64",
  5100. },
  5101. {
  5102. "name": "percentile_profile_duration_0_25",
  5103. "type": "Float64",
  5104. },
  5105. *[
  5106. {
  5107. "name": f"{fn}_profile_duration",
  5108. "type": "Float64",
  5109. }
  5110. for fn in ["p50", "p75", "p95", "p99", "p100", "min", "max", "avg", "sum"]
  5111. ],
  5112. ],
  5113. },
  5114. ]
  5115. fields = [
  5116. "project",
  5117. "transaction",
  5118. "last_seen()",
  5119. "latest_event()",
  5120. "count()",
  5121. "count_unique(transaction)",
  5122. "percentile(profile.duration, 0.25)",
  5123. "p50(profile.duration)",
  5124. "p75(profile.duration)",
  5125. "p95(profile.duration)",
  5126. "p99(profile.duration)",
  5127. "p100(profile.duration)",
  5128. "min(profile.duration)",
  5129. "max(profile.duration)",
  5130. "avg(profile.duration)",
  5131. "sum(profile.duration)",
  5132. ]
  5133. query = {
  5134. "field": fields,
  5135. "project": [self.project.id],
  5136. "dataset": "profiles",
  5137. }
  5138. response = self.do_request(query, features={"organizations:profiling": True})
  5139. assert response.status_code == 200, response.content
  5140. # making sure the response keys are in the form we expect and not aliased
  5141. data_keys = {key for row in response.data["data"] for key in row}
  5142. field_keys = {key for key in response.data["meta"]["fields"]}
  5143. unit_keys = {key for key in response.data["meta"]["units"]}
  5144. assert set(fields) == data_keys
  5145. assert set(fields) == field_keys
  5146. assert set(fields) == unit_keys
  5147. @region_silo_test
  5148. class OrganizationEventsProfileFunctionsDatasetEndpointTest(
  5149. OrganizationEventsEndpointTestBase, ProfilesSnubaTestCase
  5150. ):
  5151. def test_functions_dataset_simple(self):
  5152. self.store_functions(
  5153. [
  5154. {
  5155. "self_times_ns": [100 for _ in range(100)],
  5156. "package": "foo",
  5157. "function": "bar",
  5158. "in_app": True,
  5159. },
  5160. ],
  5161. project=self.project,
  5162. timestamp=before_now(hours=3),
  5163. )
  5164. self.store_functions(
  5165. [
  5166. {
  5167. "self_times_ns": [150 for _ in range(100)],
  5168. "package": "foo",
  5169. "function": "bar",
  5170. "in_app": True,
  5171. },
  5172. ],
  5173. project=self.project,
  5174. timestamp=before_now(hours=1),
  5175. )
  5176. mid = before_now(hours=2)
  5177. fields = [
  5178. "transaction",
  5179. "project",
  5180. "function",
  5181. "package",
  5182. "is_application",
  5183. "platform.name",
  5184. "environment",
  5185. "release",
  5186. "count()",
  5187. "examples()",
  5188. "p50()",
  5189. "p75()",
  5190. "p95()",
  5191. "p99()",
  5192. "avg()",
  5193. "sum()",
  5194. f"regression_score(function.duration, 0.95, {int(mid.timestamp())})",
  5195. ]
  5196. response = self.do_request(
  5197. {
  5198. "field": fields,
  5199. "statsPeriod": "4h",
  5200. "project": [self.project.id],
  5201. "dataset": "profileFunctions",
  5202. },
  5203. features={"organizations:profiling": True},
  5204. )
  5205. assert response.status_code == 200, response.content
  5206. # making sure the response keys are in the form we expect and not aliased
  5207. data_keys = {key for row in response.data["data"] for key in row}
  5208. field_keys = {key for key in response.data["meta"]["fields"]}
  5209. unit_keys = {key for key in response.data["meta"]["units"]}
  5210. assert set(fields) == data_keys
  5211. assert set(fields) == field_keys
  5212. assert set(fields) == unit_keys
  5213. assert response.data["meta"]["units"] == {
  5214. "transaction": None,
  5215. "project": None,
  5216. "function": None,
  5217. "package": None,
  5218. "is_application": None,
  5219. "platform.name": None,
  5220. "environment": None,
  5221. "release": None,
  5222. "count()": None,
  5223. "examples()": None,
  5224. "p50()": "nanosecond",
  5225. "p75()": "nanosecond",
  5226. "p95()": "nanosecond",
  5227. "p99()": "nanosecond",
  5228. "avg()": "nanosecond",
  5229. "sum()": "nanosecond",
  5230. f"regression_score(function.duration, 0.95, {int(mid.timestamp())})": None,
  5231. }
  5232. @region_silo_test
  5233. class OrganizationEventsIssuePlatformDatasetEndpointTest(
  5234. OrganizationEventsEndpointTestBase, SearchIssueTestMixin, PerformanceIssueTestCase
  5235. ):
  5236. def test_performance_issue_id_filter(self):
  5237. event = self.create_performance_issue()
  5238. query = {
  5239. "field": ["count()"],
  5240. "statsPeriod": "2h",
  5241. "query": f"issue.id:{event.group.id}",
  5242. "dataset": "issuePlatform",
  5243. }
  5244. response = self.do_request(query)
  5245. assert response.status_code == 200, response.content
  5246. assert response.data["data"][0]["count()"] == 1
  5247. def test_generic_issue_ids_filter(self):
  5248. user_data = {
  5249. "id": self.user.id,
  5250. "username": "user",
  5251. "email": "hellboy@bar.com",
  5252. "ip_address": "127.0.0.1",
  5253. }
  5254. event, _, group_info = self.store_search_issue(
  5255. self.project.id,
  5256. self.user.id,
  5257. [f"{ProfileFileIOGroupType.type_id}-group1"],
  5258. "prod",
  5259. before_now(hours=1).replace(tzinfo=timezone.utc),
  5260. user=user_data,
  5261. )
  5262. event, _, group_info = self.store_search_issue(
  5263. self.project.id,
  5264. self.user.id,
  5265. [f"{ProfileFileIOGroupType.type_id}-group2"],
  5266. "prod",
  5267. before_now(hours=1).replace(tzinfo=timezone.utc),
  5268. user=user_data,
  5269. )
  5270. assert group_info is not None
  5271. query = {
  5272. "field": ["title", "release", "environment", "user.display", "timestamp"],
  5273. "statsPeriod": "90d",
  5274. "query": f"issue.id:{group_info.group.id}",
  5275. "dataset": "issuePlatform",
  5276. }
  5277. with self.feature(["organizations:profiling"]):
  5278. response = self.do_request(query)
  5279. assert response.status_code == 200, response.content
  5280. assert len(response.data["data"]) == 1
  5281. assert response.data["data"][0]["title"] == group_info.group.title
  5282. assert response.data["data"][0]["environment"] == "prod"
  5283. assert response.data["data"][0]["user.display"] == user_data["email"]
  5284. assert response.data["data"][0]["timestamp"] == event.timestamp
  5285. query = {
  5286. "field": ["title", "release", "environment", "user.display", "timestamp"],
  5287. "statsPeriod": "90d",
  5288. "query": f"issue:{group_info.group.qualified_short_id}",
  5289. "dataset": "issuePlatform",
  5290. }
  5291. with self.feature(["organizations:profiling"]):
  5292. response = self.do_request(query)
  5293. assert response.status_code == 200, response.content
  5294. assert len(response.data["data"]) == 1
  5295. assert response.data["data"][0]["title"] == group_info.group.title
  5296. assert response.data["data"][0]["environment"] == "prod"
  5297. assert response.data["data"][0]["user.display"] == user_data["email"]
  5298. assert response.data["data"][0]["timestamp"] == event.timestamp
  5299. def test_performance_short_group_id(self):
  5300. event = self.create_performance_issue()
  5301. query = {
  5302. "field": ["count()"],
  5303. "statsPeriod": "1h",
  5304. "query": f"project:{event.group.project.slug} issue:{event.group.qualified_short_id}",
  5305. "dataset": "issuePlatform",
  5306. }
  5307. response = self.do_request(query)
  5308. assert response.status_code == 200, response.content
  5309. assert response.data["data"][0]["count()"] == 1
  5310. def test_multiple_performance_short_group_ids_filter(self):
  5311. event1 = self.create_performance_issue()
  5312. event2 = self.create_performance_issue()
  5313. query = {
  5314. "field": ["count()"],
  5315. "statsPeriod": "1h",
  5316. "query": f"project:{event1.group.project.slug} issue:[{event1.group.qualified_short_id},{event2.group.qualified_short_id}]",
  5317. "dataset": "issuePlatform",
  5318. }
  5319. response = self.do_request(query)
  5320. assert response.status_code == 200, response.content
  5321. assert response.data["data"][0]["count()"] == 2
  5322. def test_user_display_issue_platform(self):
  5323. project1 = self.create_project()
  5324. user_data = {
  5325. "id": self.user.id,
  5326. "username": "user",
  5327. "email": "hellboy@bar.com",
  5328. "ip_address": "127.0.0.1",
  5329. }
  5330. _, _, group_info = self.store_search_issue(
  5331. project1.id,
  5332. 1,
  5333. ["group1-fingerprint"],
  5334. None,
  5335. before_now(hours=1).replace(tzinfo=timezone.utc),
  5336. user=user_data,
  5337. )
  5338. assert group_info is not None
  5339. features = {
  5340. "organizations:discover-basic": True,
  5341. "organizations:global-views": True,
  5342. "organizations:profiling": True,
  5343. }
  5344. query = {
  5345. "field": ["user.display"],
  5346. "query": f"user.display:hell* issue.id:{group_info.group.id}",
  5347. "statsPeriod": "24h",
  5348. "dataset": "issuePlatform",
  5349. }
  5350. response = self.do_request(query, features=features)
  5351. assert response.status_code == 200, response.content
  5352. data = response.data["data"]
  5353. assert len(data) == 1
  5354. result = {r["user.display"] for r in data}
  5355. assert result == {user_data["email"]}
  5356. def test_all_events_fields(self):
  5357. user_data = {
  5358. "id": self.user.id,
  5359. "username": "user",
  5360. "email": "hellboy@bar.com",
  5361. "ip_address": "127.0.0.1",
  5362. }
  5363. replay_id = str(uuid.uuid4())
  5364. profile_id = str(uuid.uuid4())
  5365. event = self.create_performance_issue(
  5366. contexts={
  5367. "trace": {
  5368. "trace_id": str(uuid.uuid4().hex),
  5369. "span_id": "933e5c9a8e464da9",
  5370. "type": "trace",
  5371. },
  5372. "replay": {"replay_id": replay_id},
  5373. "profile": {"profile_id": profile_id},
  5374. },
  5375. user_data=user_data,
  5376. )
  5377. query = {
  5378. "field": [
  5379. "id",
  5380. "transaction",
  5381. "title",
  5382. "release",
  5383. "environment",
  5384. "user.display",
  5385. "device",
  5386. "os",
  5387. "url",
  5388. "runtime",
  5389. "replayId",
  5390. "profile.id",
  5391. "transaction.duration",
  5392. "timestamp",
  5393. ],
  5394. "statsPeriod": "1h",
  5395. "query": f"project:{event.group.project.slug} issue:{event.group.qualified_short_id}",
  5396. "dataset": "issuePlatform",
  5397. }
  5398. response = self.do_request(query)
  5399. assert response.status_code == 200, response.content
  5400. data = response.data["data"][0]
  5401. assert data == {
  5402. "id": event.event_id,
  5403. "transaction": event.transaction,
  5404. "project.name": event.project.name.lower(),
  5405. "title": event.group.title,
  5406. "release": event.release,
  5407. "environment": event.get_environment().name,
  5408. "user.display": user_data["email"],
  5409. "device": "Mac",
  5410. "os": "",
  5411. "url": event.interfaces.data["request"].full_url,
  5412. "runtime": dict(event.get_raw_data()["tags"])["runtime"],
  5413. "replayId": replay_id.replace("-", ""),
  5414. "profile.id": profile_id.replace("-", ""),
  5415. "transaction.duration": 3000,
  5416. "timestamp": event.datetime.replace(microsecond=0).isoformat(),
  5417. }
  5418. class OrganizationEventsErrorsDatasetEndpointTest(OrganizationEventsEndpointTestBase):
  5419. def test_status(self):
  5420. with self.options({"issues.group_attributes.send_kafka": True}):
  5421. self.store_event(
  5422. data={
  5423. "event_id": "a" * 32,
  5424. "timestamp": self.ten_mins_ago_iso,
  5425. "fingerprint": ["group1"],
  5426. },
  5427. project_id=self.project.id,
  5428. ).group
  5429. group_2 = self.store_event(
  5430. data={
  5431. "event_id": "b" * 32,
  5432. "timestamp": self.ten_mins_ago_iso,
  5433. "fingerprint": ["group2"],
  5434. },
  5435. project_id=self.project.id,
  5436. ).group
  5437. group_3 = self.store_event(
  5438. data={
  5439. "event_id": "c" * 32,
  5440. "timestamp": self.ten_mins_ago_iso,
  5441. "fingerprint": ["group3"],
  5442. },
  5443. project_id=self.project.id,
  5444. ).group
  5445. query = {
  5446. "field": ["count()"],
  5447. "statsPeriod": "2h",
  5448. "query": "status:unresolved",
  5449. "dataset": "errors",
  5450. }
  5451. response = self.do_request(query)
  5452. assert response.status_code == 200, response.content
  5453. assert response.data["data"][0]["count()"] == 3
  5454. group_2.status = GroupStatus.IGNORED
  5455. group_2.substatus = GroupSubStatus.FOREVER
  5456. group_2.save(update_fields=["status", "substatus"])
  5457. group_3.status = GroupStatus.IGNORED
  5458. group_3.substatus = GroupSubStatus.FOREVER
  5459. group_3.save(update_fields=["status", "substatus"])
  5460. # XXX: Snuba caches query results, so change the time period so that the query
  5461. # changes enough to bust the cache.
  5462. query["statsPeriod"] = "3h"
  5463. response = self.do_request(query)
  5464. assert response.status_code == 200, response.content
  5465. assert response.data["data"][0]["count()"] == 1
  5466. def test_is_status(self):
  5467. with self.options({"issues.group_attributes.send_kafka": True}):
  5468. self.store_event(
  5469. data={
  5470. "event_id": "a" * 32,
  5471. "timestamp": self.ten_mins_ago_iso,
  5472. "fingerprint": ["group1"],
  5473. },
  5474. project_id=self.project.id,
  5475. ).group
  5476. group_2 = self.store_event(
  5477. data={
  5478. "event_id": "b" * 32,
  5479. "timestamp": self.ten_mins_ago_iso,
  5480. "fingerprint": ["group2"],
  5481. },
  5482. project_id=self.project.id,
  5483. ).group
  5484. group_3 = self.store_event(
  5485. data={
  5486. "event_id": "c" * 32,
  5487. "timestamp": self.ten_mins_ago_iso,
  5488. "fingerprint": ["group3"],
  5489. },
  5490. project_id=self.project.id,
  5491. ).group
  5492. query = {
  5493. "field": ["count()"],
  5494. "statsPeriod": "2h",
  5495. "query": "is:unresolved",
  5496. "dataset": "errors",
  5497. }
  5498. response = self.do_request(query)
  5499. assert response.status_code == 200, response.content
  5500. assert response.data["data"][0]["count()"] == 3
  5501. group_2.status = GroupStatus.IGNORED
  5502. group_2.substatus = GroupSubStatus.FOREVER
  5503. group_2.save(update_fields=["status", "substatus"])
  5504. group_3.status = GroupStatus.IGNORED
  5505. group_3.substatus = GroupSubStatus.FOREVER
  5506. group_3.save(update_fields=["status", "substatus"])
  5507. # XXX: Snuba caches query results, so change the time period so that the query
  5508. # changes enough to bust the cache.
  5509. query["statsPeriod"] = "3h"
  5510. response = self.do_request(query)
  5511. assert response.status_code == 200, response.content
  5512. assert response.data["data"][0]["count()"] == 1
  5513. def test_short_group_id(self):
  5514. group_1 = self.store_event(
  5515. data={
  5516. "event_id": "a" * 32,
  5517. "timestamp": self.ten_mins_ago_iso,
  5518. "fingerprint": ["group1"],
  5519. },
  5520. project_id=self.project.id,
  5521. ).group
  5522. query = {
  5523. "field": ["count()"],
  5524. "statsPeriod": "1h",
  5525. "query": f"project:{group_1.project.slug} issue:{group_1.qualified_short_id}",
  5526. "dataset": "errors",
  5527. }
  5528. response = self.do_request(query)
  5529. assert response.status_code == 200, response.content
  5530. assert response.data["data"][0]["count()"] == 1
  5531. def test_user_display(self):
  5532. group_1 = self.store_event(
  5533. data={
  5534. "event_id": "a" * 32,
  5535. "timestamp": self.ten_mins_ago_iso,
  5536. "fingerprint": ["group1"],
  5537. "user": {
  5538. "email": "hellboy@bar.com",
  5539. },
  5540. },
  5541. project_id=self.project.id,
  5542. ).group
  5543. features = {
  5544. "organizations:discover-basic": True,
  5545. "organizations:global-views": True,
  5546. }
  5547. query = {
  5548. "field": ["user.display"],
  5549. "query": f"user.display:hell* issue.id:{group_1.id}",
  5550. "statsPeriod": "24h",
  5551. "dataset": "errors",
  5552. }
  5553. response = self.do_request(query, features=features)
  5554. assert response.status_code == 200, response.content
  5555. data = response.data["data"]
  5556. assert len(data) == 1
  5557. result = {r["user.display"] for r in data}
  5558. assert result == {"hellboy@bar.com"}