test_plugin.py 85 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717
  1. import zipfile
  2. from io import BytesIO
  3. from uuid import uuid4
  4. import pytest
  5. from django.core.files.uploadedfile import SimpleUploadedFile
  6. from django.urls import reverse
  7. from sentry.models.debugfile import ProjectDebugFile
  8. from sentry.models.files.file import File
  9. from sentry.stacktraces.processing import find_stacktraces_in_data
  10. from sentry.testutils.cases import TransactionTestCase
  11. from sentry.testutils.helpers.datetime import before_now
  12. from sentry.testutils.relay import RelayStoreHelper
  13. from sentry.testutils.skips import requires_symbolicator
  14. from sentry.utils import json
  15. PROGUARD_UUID = "6dc7fdb0-d2fb-4c8e-9d6b-bb1aa98929b1"
  16. PROGUARD_SOURCE = b"""\
  17. org.slf4j.helpers.Util$ClassContextSecurityManager -> org.a.b.g$a:
  18. 65:65:void <init>() -> <init>
  19. 67:67:java.lang.Class[] getClassContext() -> a
  20. 69:69:java.lang.Class[] getExtraClassContext() -> a
  21. 68:68:java.lang.Class[] getContext() -> a
  22. 65:65:void <init>(org.slf4j.helpers.Util$1) -> <init>
  23. org.slf4j.helpers.Util$ClassContext -> org.a.b.g$b:
  24. 65:65:void <init>() -> <init>
  25. """
  26. PROGUARD_INLINE_UUID = "d748e578-b3d1-5be5-b0e5-a42e8c9bf8e0"
  27. PROGUARD_INLINE_SOURCE = b"""\
  28. # compiler: R8
  29. # compiler_version: 2.0.74
  30. # min_api: 16
  31. # pg_map_id: 5b46fdc
  32. # common_typos_disable
  33. $r8$backportedMethods$utility$Objects$2$equals -> a:
  34. boolean equals(java.lang.Object,java.lang.Object) -> a
  35. $r8$twr$utility -> b:
  36. void $closeResource(java.lang.Throwable,java.lang.Object) -> a
  37. android.support.v4.app.RemoteActionCompatParcelizer -> android.support.v4.app.RemoteActionCompatParcelizer:
  38. 1:1:void <init>():11:11 -> <init>
  39. io.sentry.sample.-$$Lambda$r3Avcbztes2hicEObh02jjhQqd4 -> e.a.c.a:
  40. io.sentry.sample.MainActivity f$0 -> b
  41. io.sentry.sample.MainActivity -> io.sentry.sample.MainActivity:
  42. 1:1:void <init>():15:15 -> <init>
  43. 1:1:boolean onCreateOptionsMenu(android.view.Menu):60:60 -> onCreateOptionsMenu
  44. 1:1:boolean onOptionsItemSelected(android.view.MenuItem):69:69 -> onOptionsItemSelected
  45. 2:2:boolean onOptionsItemSelected(android.view.MenuItem):76:76 -> onOptionsItemSelected
  46. 1:1:void bar():54:54 -> t
  47. 1:1:void foo():44 -> t
  48. 1:1:void onClickHandler(android.view.View):40 -> t
  49. """
  50. PROGUARD_BUG_UUID = "071207ac-b491-4a74-957c-2c94fd9594f2"
  51. PROGUARD_BUG_SOURCE = b"x"
  52. JVM_DEBUG_ID = "6dc7fdb0-d2fb-4c8e-9d6b-bb1aa98929b1"
  53. JVM_SOURCE = b"""\
  54. package io.sentry.samples
  55. import android.content.Intent
  56. import android.os.Bundle
  57. import androidx.activity.ComponentActivity
  58. class MainActivity : ComponentActivity() {
  59. override fun onCreate(savedInstanceState: Bundle?) {
  60. super.onCreate(savedInstanceState)
  61. setContentView(R.layout.activity_main)
  62. InnerClass().whoops()
  63. val list = findViewById<RecyclerView>(R.id.list)
  64. list.layoutManager = LinearLayoutManager(this)
  65. list.adapter = TrackAdapter()
  66. }
  67. class InnerClass {
  68. fun whoops() {
  69. AnotherInnerClass().whoops2()
  70. }
  71. }
  72. class AnotherInnerClass {
  73. fun whoops2() {
  74. AdditionalInnerClass().whoops3()
  75. }
  76. }
  77. class AdditionalInnerClass {
  78. fun whoops3() {
  79. OneMoreInnerClass().whoops4()
  80. }
  81. }
  82. class OneMoreInnerClass {
  83. fun whoops4() {
  84. throw RuntimeException("whoops")
  85. }
  86. }
  87. }
  88. """
  89. PROGUARD_SOURCE_LOOKUP_UUID = "05d96b1c-1786-477c-8615-d3cf83e027c7"
  90. PROGUARD_SOURCE_LOOKUP_SOURCE = b"""\
  91. io.sentry.samples.instrumentation.ui.EditActivity -> io.sentry.samples.instrumentation.ui.EditActivity:
  92. # {"id":"sourceFile","fileName":"EditActivity.kt"}
  93. int $r8$clinit -> a
  94. 0:65535:void <init>():15:15 -> <init>
  95. 1:6:void onCreate(android.os.Bundle):18:18 -> onCreate
  96. 7:12:void onCreate(android.os.Bundle):19:19 -> onCreate
  97. 13:22:void onCreate(android.os.Bundle):21:21 -> onCreate
  98. 23:32:void onCreate(android.os.Bundle):22:22 -> onCreate
  99. 33:42:void onCreate(android.os.Bundle):23:23 -> onCreate
  100. 43:49:void onCreate(android.os.Bundle):24:24 -> onCreate
  101. 50:71:void onCreate(android.os.Bundle):26:26 -> onCreate
  102. 72:73:java.lang.String io.sentry.samples.instrumentation.data.Track.getName():46:46 -> onCreate
  103. 72:73:void onCreate(android.os.Bundle):28 -> onCreate
  104. 74:76:void onCreate(android.os.Bundle):28:28 -> onCreate
  105. 77:78:java.lang.String io.sentry.samples.instrumentation.data.Track.getComposer():48:48 -> onCreate
  106. 77:78:void onCreate(android.os.Bundle):29 -> onCreate
  107. 79:81:void onCreate(android.os.Bundle):29:29 -> onCreate
  108. 82:83:long io.sentry.samples.instrumentation.data.Track.getMillis():51:51 -> onCreate
  109. 82:83:void onCreate(android.os.Bundle):30 -> onCreate
  110. 84:90:void onCreate(android.os.Bundle):30:30 -> onCreate
  111. 91:92:float io.sentry.samples.instrumentation.data.Track.getPrice():53:53 -> onCreate
  112. 91:92:void onCreate(android.os.Bundle):31 -> onCreate
  113. 93:102:void onCreate(android.os.Bundle):31:31 -> onCreate
  114. 103:119:void onCreate(android.os.Bundle):34:34 -> onCreate
  115. io.sentry.samples.instrumentation.ui.EditActivity$$ExternalSyntheticLambda0 -> io.sentry.samples.instrumentation.ui.g:
  116. # {"id":"sourceFile","fileName":"R8$$SyntheticClass"}
  117. # {"id":"com.android.tools.r8.synthesized"}
  118. android.widget.EditText io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.f$4 -> e
  119. android.widget.EditText io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.f$3 -> d
  120. android.widget.EditText io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.f$2 -> c
  121. io.sentry.samples.instrumentation.data.Track io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.f$0 -> a
  122. android.widget.EditText io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.f$1 -> b
  123. io.sentry.samples.instrumentation.ui.EditActivity io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.f$5 -> f
  124. void io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.<init>(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity) -> <init>
  125. # {"id":"com.android.tools.r8.synthesized"}
  126. 19:21:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):0:0 -> onMenuItemClick
  127. 19:21:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  128. # {"id":"com.android.tools.r8.synthesized"}
  129. 22:35:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):35:35 -> onMenuItemClick
  130. 22:35:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  131. # {"id":"com.android.tools.r8.synthesized"}
  132. 36:44:void io.sentry.samples.instrumentation.ui.AnotherClassInSameFile$AnotherInnerClass.helloOtherInner():26:26 -> onMenuItemClick
  133. 36:44:void io.sentry.samples.instrumentation.ui.AnotherClassInSameFile.otherFun():21 -> onMenuItemClick
  134. 36:44:void io.sentry.samples.instrumentation.ui.AnotherClassInSameFile.helloOther():17 -> onMenuItemClick
  135. 36:44:void io.sentry.samples.instrumentation.ui.SomeService$InnerClassOfSomeService.helloInner():10 -> onMenuItemClick
  136. 36:44:void io.sentry.samples.instrumentation.ui.SomeService.helloThere():5 -> onMenuItemClick
  137. 36:44:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):37 -> onMenuItemClick
  138. 36:44:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  139. # {"id":"com.android.tools.r8.synthesized"}
  140. 45:58:io.sentry.protocol.SentryId io.sentry.Sentry.captureException(java.lang.Throwable):433:433 -> onMenuItemClick
  141. 45:58:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):39 -> onMenuItemClick
  142. 45:58:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  143. # {"id":"com.android.tools.r8.synthesized"}
  144. 59:68:io.sentry.ITransaction io.sentry.Sentry.startTransaction(java.lang.String,java.lang.String,boolean):697:697 -> onMenuItemClick
  145. 59:68:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):42 -> onMenuItemClick
  146. 59:68:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  147. # {"id":"com.android.tools.r8.synthesized"}
  148. 69:71:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):42:42 -> onMenuItemClick
  149. 69:71:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  150. # {"id":"com.android.tools.r8.synthesized"}
  151. 72:79:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):48:48 -> onMenuItemClick
  152. 72:79:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  153. # {"id":"com.android.tools.r8.synthesized"}
  154. 80:87:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):49:49 -> onMenuItemClick
  155. 80:87:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  156. # {"id":"com.android.tools.r8.synthesized"}
  157. 88:95:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):50:50 -> onMenuItemClick
  158. 88:95:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  159. # {"id":"com.android.tools.r8.synthesized"}
  160. 96:103:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):51:51 -> onMenuItemClick
  161. 96:103:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  162. # {"id":"com.android.tools.r8.synthesized"}
  163. 104:125:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):52:52 -> onMenuItemClick
  164. 104:125:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  165. # {"id":"com.android.tools.r8.synthesized"}
  166. 126:142:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):53:53 -> onMenuItemClick
  167. 126:142:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  168. # {"id":"com.android.tools.r8.synthesized"}
  169. 143:164:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):54:54 -> onMenuItemClick
  170. 143:164:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  171. # {"id":"com.android.tools.r8.synthesized"}
  172. 165:175:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):64:64 -> onMenuItemClick
  173. 165:175:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  174. # {"id":"com.android.tools.r8.synthesized"}
  175. 176:187:void io.sentry.samples.instrumentation.ui.EditActivity.addNewTrack(java.lang.String,java.lang.String,long,float):84:84 -> onMenuItemClick
  176. 176:187:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):64 -> onMenuItemClick
  177. 176:187:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  178. # {"id":"com.android.tools.r8.synthesized"}
  179. 188:190:void io.sentry.samples.instrumentation.data.Track.<init>(long,java.lang.String,java.lang.Long,java.lang.String,java.lang.Long,java.lang.Long,long,java.lang.Long,float,int,kotlin.jvm.internal.DefaultConstructorMarker):43:43 -> onMenuItemClick
  180. 188:190:void io.sentry.samples.instrumentation.ui.EditActivity.addNewTrack(java.lang.String,java.lang.String,long,float):84 -> onMenuItemClick
  181. 188:190:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):64 -> onMenuItemClick
  182. 188:190:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  183. # {"id":"com.android.tools.r8.synthesized"}
  184. 191:198:void io.sentry.samples.instrumentation.ui.EditActivity.addNewTrack(java.lang.String,java.lang.String,long,float):94:94 -> onMenuItemClick
  185. 191:198:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):64 -> onMenuItemClick
  186. 191:198:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  187. # {"id":"com.android.tools.r8.synthesized"}
  188. 199:211:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):66:66 -> onMenuItemClick
  189. 199:211:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  190. # {"id":"com.android.tools.r8.synthesized"}
  191. 212:227:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):67:67 -> onMenuItemClick
  192. 212:227:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  193. # {"id":"com.android.tools.r8.synthesized"}
  194. 228:238:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):69:69 -> onMenuItemClick
  195. 228:238:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  196. # {"id":"com.android.tools.r8.synthesized"}
  197. 239:250:io.sentry.samples.instrumentation.data.Track io.sentry.samples.instrumentation.data.Track.copy$default(io.sentry.samples.instrumentation.data.Track,long,java.lang.String,java.lang.Long,java.lang.String,java.lang.Long,java.lang.Long,long,java.lang.Long,float,int,java.lang.Object):0:0 -> onMenuItemClick
  198. 239:250:void io.sentry.samples.instrumentation.ui.EditActivity.update(io.sentry.samples.instrumentation.data.Track,java.lang.String,java.lang.String,long,float):100 -> onMenuItemClick
  199. 239:250:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):69 -> onMenuItemClick
  200. 239:250:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  201. # {"id":"com.android.tools.r8.synthesized"}
  202. 251:265:io.sentry.samples.instrumentation.data.Track io.sentry.samples.instrumentation.data.Track.copy(long,java.lang.String,java.lang.Long,java.lang.String,java.lang.Long,java.lang.Long,long,java.lang.Long,float):0:0 -> onMenuItemClick
  203. 251:265:io.sentry.samples.instrumentation.data.Track io.sentry.samples.instrumentation.data.Track.copy$default(io.sentry.samples.instrumentation.data.Track,long,java.lang.String,java.lang.Long,java.lang.String,java.lang.Long,java.lang.Long,long,java.lang.Long,float,int,java.lang.Object):0 -> onMenuItemClick
  204. 251:265:void io.sentry.samples.instrumentation.ui.EditActivity.update(io.sentry.samples.instrumentation.data.Track,java.lang.String,java.lang.String,long,float):100 -> onMenuItemClick
  205. 251:265:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):69 -> onMenuItemClick
  206. 251:265:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  207. # {"id":"com.android.tools.r8.synthesized"}
  208. 266:273:void io.sentry.samples.instrumentation.ui.EditActivity.update(io.sentry.samples.instrumentation.data.Track,java.lang.String,java.lang.String,long,float):106:106 -> onMenuItemClick
  209. 266:273:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):69 -> onMenuItemClick
  210. 266:273:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  211. # {"id":"com.android.tools.r8.synthesized"}
  212. 274:286:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):71:71 -> onMenuItemClick
  213. 274:286:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  214. # {"id":"com.android.tools.r8.synthesized"}
  215. 287:301:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):72:72 -> onMenuItemClick
  216. 287:301:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  217. # {"id":"com.android.tools.r8.synthesized"}
  218. 302:306:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):74:74 -> onMenuItemClick
  219. 302:306:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  220. # {"id":"com.android.tools.r8.synthesized"}
  221. 307:312:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):75:75 -> onMenuItemClick
  222. 307:312:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  223. # {"id":"com.android.tools.r8.synthesized"}
  224. 313:316:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):56:56 -> onMenuItemClick
  225. 313:316:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  226. # {"id":"com.android.tools.r8.synthesized"}
  227. 317:320:boolean io.sentry.samples.instrumentation.ui.EditActivity.onCreate$lambda-1(io.sentry.samples.instrumentation.data.Track,android.widget.EditText,android.widget.EditText,android.widget.EditText,android.widget.EditText,io.sentry.samples.instrumentation.ui.EditActivity,android.view.MenuItem):61:61 -> onMenuItemClick
  228. 317:320:boolean io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0.onMenuItemClick(android.view.MenuItem):0 -> onMenuItemClick
  229. # {"id":"com.android.tools.r8.synthesized"}
  230. """
  231. EDIT_ACTIVITY_SOURCE = b"""\
  232. package io.sentry.samples.instrumentation.ui
  233. import android.os.Bundle
  234. import android.widget.EditText
  235. import android.widget.Toast
  236. import androidx.activity.ComponentActivity
  237. import androidx.appcompat.widget.Toolbar
  238. import io.sentry.Sentry
  239. import io.sentry.SpanStatus
  240. import io.sentry.samples.instrumentation.R
  241. import io.sentry.samples.instrumentation.SampleApp
  242. import io.sentry.samples.instrumentation.data.Track
  243. import kotlinx.coroutines.runBlocking
  244. class EditActivity : ComponentActivity() {
  245. override fun onCreate(savedInstanceState: Bundle?) {
  246. super.onCreate(savedInstanceState)
  247. setContentView(R.layout.activity_edit)
  248. val nameInput = findViewById<EditText>(R.id.track_name)
  249. val composerInput = findViewById<EditText>(R.id.track_composer)
  250. val durationInput = findViewById<EditText>(R.id.track_duration)
  251. val unitPriceInput = findViewById<EditText>(R.id.track_unit_price)
  252. val originalTrack: Track? = intent.getSerializableExtra(TRACK_EXTRA_KEY) as? Track
  253. originalTrack?.run {
  254. nameInput.setText(name)
  255. composerInput.setText(composer)
  256. durationInput.setText(millis.toString())
  257. unitPriceInput.setText(price.toString())
  258. }
  259. findViewById<Toolbar>(R.id.toolbar).setOnMenuItemClickListener {
  260. if (it.itemId == R.id.action_save) {
  261. try {
  262. SomeService().helloThere()
  263. } catch (e: Exception) {
  264. Sentry.captureException(e)
  265. }
  266. val transaction = Sentry.startTransaction(
  267. "Track Interaction",
  268. if (originalTrack == null) "ui.action.add" else "ui.action.edit",
  269. true
  270. )
  271. val name = nameInput.text.toString()
  272. val composer = composerInput.text.toString()
  273. val duration = durationInput.text.toString()
  274. val unitPrice = unitPriceInput.text.toString()
  275. if (name.isEmpty() || composer.isEmpty() ||
  276. duration.isEmpty() || duration.toLongOrNull() == null ||
  277. unitPrice.isEmpty() || unitPrice.toFloatOrNull() == null
  278. ) {
  279. Toast.makeText(
  280. this,
  281. "Some of the inputs are empty or have wrong format " +
  282. "(duration/unitprice not a number)",
  283. Toast.LENGTH_LONG
  284. ).show()
  285. } else {
  286. if (originalTrack == null) {
  287. addNewTrack(name, composer, duration.toLong(), unitPrice.toFloat())
  288. val createCount = SampleApp.analytics.getInt("create_count", 0) + 1
  289. SampleApp.analytics.edit().putInt("create_count", createCount).apply()
  290. } else {
  291. originalTrack.update(name, composer, duration.toLong(), unitPrice.toFloat())
  292. val editCount = SampleApp.analytics.getInt("edit_count", 0) + 1
  293. SampleApp.analytics.edit().putInt("edit_count", editCount).apply()
  294. }
  295. transaction.finish(SpanStatus.OK)
  296. finish()
  297. }
  298. return@setOnMenuItemClickListener true
  299. }
  300. return@setOnMenuItemClickListener false
  301. }
  302. }
  303. private fun addNewTrack(name: String, composer: String, duration: Long, unitPrice: Float) {
  304. val newTrack = Track(
  305. name = name,
  306. albumId = null,
  307. composer = composer,
  308. mediaTypeId = null,
  309. genreId = null,
  310. millis = duration,
  311. bytes = null,
  312. price = unitPrice
  313. )
  314. runBlocking {
  315. SampleApp.database.tracksDao().insert(newTrack)
  316. }
  317. }
  318. private fun Track.update(name: String, composer: String, duration: Long, unitPrice: Float) {
  319. val updatedTrack = copy(
  320. name = name,
  321. composer = composer,
  322. millis = duration,
  323. price = unitPrice
  324. )
  325. runBlocking {
  326. SampleApp.database.tracksDao().update(updatedTrack)
  327. }
  328. }
  329. companion object {
  330. const val TRACK_EXTRA_KEY = "EditActivity.Track"
  331. }
  332. }
  333. """
  334. SOME_SERVICE_SOURCE = b"""\
  335. package io.sentry.samples.instrumentation.ui
  336. class SomeService {
  337. fun helloThere() {
  338. InnerClassOfSomeService().helloInner()
  339. }
  340. class InnerClassOfSomeService {
  341. fun helloInner() {
  342. AnotherClassInSameFile().helloOther()
  343. }
  344. }
  345. }
  346. class AnotherClassInSameFile {
  347. fun helloOther() {
  348. otherFun()
  349. }
  350. private fun otherFun() {
  351. AnotherInnerClass().helloOtherInner()
  352. }
  353. class AnotherInnerClass {
  354. fun helloOtherInner() {
  355. throw RuntimeException("thrown on purpose to test ProGuard Android source context")
  356. }
  357. }
  358. }
  359. """
  360. @pytest.mark.django_db(transaction=True)
  361. class BasicResolvingIntegrationTest(RelayStoreHelper, TransactionTestCase):
  362. @pytest.fixture(autouse=True)
  363. def initialize(self, set_sentry_option, live_server):
  364. with set_sentry_option("system.url-prefix", live_server.url):
  365. # Run test case
  366. yield
  367. def upload_proguard_mapping(self, uuid, mapping_file_content):
  368. url = reverse(
  369. "sentry-api-0-dsym-files",
  370. kwargs={
  371. "organization_id_or_slug": self.project.organization.slug,
  372. "project_id_or_slug": self.project.slug,
  373. },
  374. )
  375. self.login_as(user=self.user)
  376. out = BytesIO()
  377. f = zipfile.ZipFile(out, "w")
  378. f.writestr("proguard/%s.txt" % uuid, mapping_file_content)
  379. f.writestr("ignored-file.txt", b"This is just some stuff")
  380. f.close()
  381. response = self.client.post(
  382. url,
  383. {
  384. "file": SimpleUploadedFile(
  385. "symbols.zip", out.getvalue(), content_type="application/zip"
  386. )
  387. },
  388. format="multipart",
  389. )
  390. assert response.status_code == 201, response.content
  391. assert len(response.json()) == 1
  392. @requires_symbolicator
  393. @pytest.mark.symbolicator
  394. def test_basic_resolving(self):
  395. self.upload_proguard_mapping(PROGUARD_UUID, PROGUARD_SOURCE)
  396. event_data = {
  397. "user": {"ip_address": "31.172.207.97"},
  398. "extra": {},
  399. "project": self.project.id,
  400. "platform": "java",
  401. "debug_meta": {"images": [{"type": "proguard", "uuid": PROGUARD_UUID}]},
  402. "exception": {
  403. "values": [
  404. {
  405. "stacktrace": {
  406. "frames": [
  407. {
  408. "function": "a",
  409. "abs_path": None,
  410. "module": "org.a.b.g$a",
  411. "filename": None,
  412. "lineno": 67,
  413. },
  414. {
  415. "function": "a",
  416. "abs_path": None,
  417. "module": "org.a.b.g$a",
  418. "filename": None,
  419. "lineno": 69,
  420. },
  421. ]
  422. },
  423. "module": "org.a.b",
  424. "type": "g$a",
  425. "value": "Attempt to invoke virtual method 'org.a.b.g$a.a' on a null object reference",
  426. }
  427. ]
  428. },
  429. "timestamp": before_now(seconds=1).isoformat(),
  430. }
  431. event = self.post_and_retrieve_event(event_data)
  432. if not self.use_relay():
  433. # We measure the number of queries after an initial post,
  434. # because there are many queries polluting the array
  435. # before the actual "processing" happens (like, auth_user)
  436. with self.assertWriteQueries(
  437. {
  438. "nodestore_node": 2,
  439. "sentry_eventuser": 1,
  440. "sentry_groupedmessage": 1,
  441. "sentry_userreport": 1,
  442. }
  443. ):
  444. self.post_and_retrieve_event(event_data)
  445. exc = event.interfaces["exception"].values[0]
  446. bt = exc.stacktrace
  447. frames = bt.frames
  448. assert exc.type == "Util$ClassContextSecurityManager"
  449. assert (
  450. exc.value
  451. == "Attempt to invoke virtual method 'org.slf4j.helpers.Util$ClassContextSecurityManager.getExtraClassContext' on a null object reference"
  452. )
  453. assert exc.module == "org.slf4j.helpers"
  454. assert frames[0].function == "getClassContext"
  455. assert frames[0].module == "org.slf4j.helpers.Util$ClassContextSecurityManager"
  456. assert frames[1].function == "getExtraClassContext"
  457. assert frames[1].module == "org.slf4j.helpers.Util$ClassContextSecurityManager"
  458. assert event.culprit == (
  459. "org.slf4j.helpers.Util$ClassContextSecurityManager " "in getExtraClassContext"
  460. )
  461. @requires_symbolicator
  462. @pytest.mark.symbolicator
  463. def test_resolving_does_not_fail_when_no_value(self):
  464. self.upload_proguard_mapping(PROGUARD_UUID, PROGUARD_SOURCE)
  465. event_data = {
  466. "user": {"ip_address": "31.172.207.97"},
  467. "extra": {},
  468. "project": self.project.id,
  469. "platform": "java",
  470. "debug_meta": {"images": [{"type": "proguard", "uuid": PROGUARD_UUID}]},
  471. "exception": {
  472. "values": [
  473. {
  474. "stacktrace": {
  475. "frames": [
  476. {
  477. "function": "a",
  478. "abs_path": None,
  479. "module": "org.a.b.g$a",
  480. "filename": None,
  481. "lineno": 67,
  482. },
  483. {
  484. "function": "a",
  485. "abs_path": None,
  486. "module": "org.a.b.g$a",
  487. "filename": None,
  488. "lineno": 69,
  489. },
  490. ]
  491. },
  492. "module": "org.a.b",
  493. "type": "g$a",
  494. }
  495. ]
  496. },
  497. "timestamp": before_now(seconds=1).isoformat(),
  498. }
  499. event = self.post_and_retrieve_event(event_data)
  500. if not self.use_relay():
  501. # We measure the number of queries after an initial post,
  502. # because there are many queries polluting the array
  503. # before the actual "processing" happens (like, auth_user)
  504. with self.assertWriteQueries(
  505. {
  506. "nodestore_node": 2,
  507. "sentry_eventuser": 1,
  508. "sentry_groupedmessage": 1,
  509. "sentry_userreport": 1,
  510. }
  511. ):
  512. self.post_and_retrieve_event(event_data)
  513. metrics = event.data["_metrics"]
  514. assert not metrics.get("flag.processing.error")
  515. @requires_symbolicator
  516. @pytest.mark.symbolicator
  517. def test_resolving_does_not_fail_when_no_module_or_function(self):
  518. self.upload_proguard_mapping(PROGUARD_UUID, PROGUARD_SOURCE)
  519. event_data = {
  520. "user": {"ip_address": "31.172.207.97"},
  521. "extra": {},
  522. "project": self.project.id,
  523. "platform": "java",
  524. "debug_meta": {"images": [{"type": "proguard", "uuid": PROGUARD_UUID}]},
  525. "exception": {
  526. "values": [
  527. {
  528. "stacktrace": {
  529. "frames": [
  530. {
  531. "function": "a",
  532. "abs_path": None,
  533. "module": "org.a.b.g$a",
  534. "filename": None,
  535. "lineno": 67,
  536. },
  537. {
  538. "function": "a",
  539. "abs_path": None,
  540. "module": "org.a.b.g$a",
  541. "filename": None,
  542. "lineno": 69,
  543. },
  544. {
  545. "function": "__start_thread",
  546. "package": "/apex/com.android.art/lib64/libart.so",
  547. "lineno": 196,
  548. "in_app": False,
  549. },
  550. {
  551. "package": "/apex/com.android.art/lib64/libart.so",
  552. "lineno": 214,
  553. "in_app": False,
  554. },
  555. ]
  556. },
  557. "module": "org.a.b",
  558. "type": "g$a",
  559. "value": "Attempt to invoke virtual method 'org.a.b.g$a.a' on a null object reference",
  560. }
  561. ]
  562. },
  563. "timestamp": before_now(seconds=1).isoformat(),
  564. }
  565. event = self.post_and_retrieve_event(event_data)
  566. if not self.use_relay():
  567. # We measure the number of queries after an initial post,
  568. # because there are many queries polluting the array
  569. # before the actual "processing" happens (like, auth_user)
  570. with self.assertWriteQueries(
  571. {
  572. "nodestore_node": 2,
  573. "sentry_eventuser": 1,
  574. "sentry_groupedmessage": 1,
  575. "sentry_userreport": 1,
  576. }
  577. ):
  578. self.post_and_retrieve_event(event_data)
  579. metrics = event.data["_metrics"]
  580. assert not metrics.get("flag.processing.error")
  581. @requires_symbolicator
  582. @pytest.mark.symbolicator
  583. def test_sets_inapp_after_resolving(self):
  584. self.upload_proguard_mapping(PROGUARD_UUID, PROGUARD_SOURCE)
  585. version = "org.slf4j@1.2.3"
  586. env_name = "some_env"
  587. event = self.store_event(
  588. data={"release": version, "environment": env_name}, project_id=self.project.id
  589. )
  590. event_data = {
  591. "user": {"ip_address": "31.172.207.97"},
  592. "extra": {},
  593. "release": "org.slf4j@1.2.3",
  594. "project": self.project.id,
  595. "platform": "java",
  596. "debug_meta": {"images": [{"type": "proguard", "uuid": PROGUARD_UUID}]},
  597. "exception": {
  598. "values": [
  599. {
  600. "stacktrace": {
  601. "frames": [
  602. {
  603. "function": "a",
  604. "abs_path": None,
  605. "module": "org.a.b.g$a",
  606. "filename": None,
  607. "lineno": 67,
  608. },
  609. {
  610. "function": "a",
  611. "abs_path": None,
  612. "module": "org.a.b.g$a",
  613. "filename": None,
  614. "lineno": 69,
  615. "in_app": False,
  616. },
  617. {
  618. "function": "a",
  619. "abs_path": None,
  620. "module": "org.a.b.g$a",
  621. "filename": None,
  622. "lineno": 68,
  623. "in_app": True,
  624. },
  625. {
  626. "function": "init",
  627. "abs_path": None,
  628. "module": "com.android.Zygote",
  629. "filename": None,
  630. "lineno": 62,
  631. },
  632. {
  633. "function": "a",
  634. "abs_path": None,
  635. "module": "org.a.b.g$b",
  636. "filename": None,
  637. "lineno": 70,
  638. },
  639. ]
  640. },
  641. "module": "org.a.b",
  642. "type": "g$a",
  643. "value": "Attempt to invoke virtual method 'org.a.b.g$a.a' on a null object reference",
  644. }
  645. ]
  646. },
  647. "timestamp": before_now(seconds=1).isoformat(),
  648. }
  649. event = self.post_and_retrieve_event(event_data)
  650. if not self.use_relay():
  651. # We measure the number of queries after an initial post,
  652. # because there are many queries polluting the array
  653. # before the actual "processing" happens (like, auth_user)
  654. with self.assertWriteQueries(
  655. {
  656. "nodestore_node": 2,
  657. "sentry_eventuser": 1,
  658. "sentry_groupedmessage": 1,
  659. "sentry_userreport": 1,
  660. }
  661. ):
  662. self.post_and_retrieve_event(event_data)
  663. exc = event.interfaces["exception"].values[0]
  664. bt = exc.stacktrace
  665. frames = bt.frames
  666. assert exc.module == "org.slf4j.helpers"
  667. assert frames[0].in_app is True
  668. assert frames[1].in_app is False
  669. assert frames[2].in_app is True
  670. assert frames[3].in_app is False
  671. assert frames[4].in_app is True
  672. @requires_symbolicator
  673. @pytest.mark.symbolicator
  674. def test_resolving_inline(self):
  675. self.upload_proguard_mapping(PROGUARD_INLINE_UUID, PROGUARD_INLINE_SOURCE)
  676. event_data = {
  677. "user": {"ip_address": "31.172.207.97"},
  678. "extra": {},
  679. "project": self.project.id,
  680. "platform": "java",
  681. "debug_meta": {"images": [{"type": "proguard", "uuid": PROGUARD_INLINE_UUID}]},
  682. "exception": {
  683. "values": [
  684. {
  685. "stacktrace": {
  686. "frames": [
  687. {
  688. "function": "onClick",
  689. "abs_path": None,
  690. "module": "e.a.c.a",
  691. "filename": None,
  692. "lineno": 2,
  693. },
  694. {
  695. "function": "t",
  696. "abs_path": None,
  697. "module": "io.sentry.sample.MainActivity",
  698. "filename": "MainActivity.java",
  699. "lineno": 1,
  700. },
  701. ]
  702. },
  703. "module": "org.a.b",
  704. "type": "g$a",
  705. "value": "Oh no",
  706. }
  707. ]
  708. },
  709. "timestamp": before_now(seconds=1).isoformat(),
  710. }
  711. event = self.post_and_retrieve_event(event_data)
  712. if not self.use_relay():
  713. # We measure the number of queries after an initial post,
  714. # because there are many queries polluting the array
  715. # before the actual "processing" happens (like, auth_user)
  716. with self.assertWriteQueries(
  717. {
  718. "nodestore_node": 2,
  719. "sentry_eventuser": 1,
  720. "sentry_groupedmessage": 1,
  721. "sentry_userreport": 1,
  722. }
  723. ):
  724. self.post_and_retrieve_event(event_data)
  725. exc = event.interfaces["exception"].values[0]
  726. bt = exc.stacktrace
  727. frames = bt.frames
  728. assert len(frames) == 4
  729. assert frames[0].function == "onClick"
  730. assert frames[0].module == "io.sentry.sample.-$$Lambda$r3Avcbztes2hicEObh02jjhQqd4"
  731. assert frames[1].filename == "MainActivity.java"
  732. assert frames[1].module == "io.sentry.sample.MainActivity"
  733. assert frames[1].function == "onClickHandler"
  734. assert frames[1].lineno == 40
  735. assert frames[2].function == "foo"
  736. assert frames[2].lineno == 44
  737. assert frames[3].function == "bar"
  738. assert frames[3].lineno == 54
  739. assert frames[3].filename == "MainActivity.java"
  740. assert frames[3].module == "io.sentry.sample.MainActivity"
  741. @requires_symbolicator
  742. @pytest.mark.symbolicator
  743. def test_resolving_inline_with_native_frames(self):
  744. self.upload_proguard_mapping(PROGUARD_INLINE_UUID, PROGUARD_INLINE_SOURCE)
  745. event_data = {
  746. "user": {"ip_address": "31.172.207.97"},
  747. "extra": {},
  748. "project": self.project.id,
  749. "platform": "java",
  750. "debug_meta": {"images": [{"type": "proguard", "uuid": PROGUARD_INLINE_UUID}]},
  751. "exception": {
  752. "values": [
  753. {
  754. "stacktrace": {
  755. "frames": [
  756. {
  757. "function": "onClick",
  758. "abs_path": None,
  759. "module": "e.a.c.a",
  760. "filename": None,
  761. "lineno": 2,
  762. },
  763. {
  764. "function": "t",
  765. "abs_path": None,
  766. "module": "io.sentry.sample.MainActivity",
  767. "filename": "MainActivity.java",
  768. "lineno": 1,
  769. },
  770. {
  771. "abs_path": "Thread.java",
  772. "filename": "Thread.java",
  773. "function": "sleep",
  774. "lineno": 450,
  775. "lock": {
  776. "address": "0x0ddc1f22",
  777. "class_name": "Object",
  778. "package_name": "java.lang",
  779. "type:": 1,
  780. },
  781. "module": "java.lang.Thread",
  782. },
  783. {
  784. "function": "__start_thread",
  785. "package": "/apex/com.android.art/lib64/libart.so",
  786. "lineno": 196,
  787. "in_app": False,
  788. },
  789. ]
  790. },
  791. "module": "org.a.b",
  792. "type": "g$a",
  793. "value": "Oh no",
  794. }
  795. ]
  796. },
  797. "timestamp": before_now(seconds=1).isoformat(),
  798. }
  799. event = self.post_and_retrieve_event(event_data)
  800. if not self.use_relay():
  801. # We measure the number of queries after an initial post,
  802. # because there are many queries polluting the array
  803. # before the actual "processing" happens (like, auth_user)
  804. with self.assertWriteQueries(
  805. {
  806. "nodestore_node": 2,
  807. "sentry_eventuser": 1,
  808. "sentry_groupedmessage": 1,
  809. "sentry_userreport": 1,
  810. }
  811. ):
  812. self.post_and_retrieve_event(event_data)
  813. exc = event.interfaces["exception"].values[0]
  814. bt = exc.stacktrace
  815. frames = bt.frames
  816. assert len(frames) == 6
  817. assert frames[0].function == "onClick"
  818. assert frames[0].module == "io.sentry.sample.-$$Lambda$r3Avcbztes2hicEObh02jjhQqd4"
  819. assert frames[1].filename == "MainActivity.java"
  820. assert frames[1].module == "io.sentry.sample.MainActivity"
  821. assert frames[1].function == "onClickHandler"
  822. assert frames[1].lineno == 40
  823. assert frames[2].function == "foo"
  824. assert frames[2].lineno == 44
  825. assert frames[3].function == "bar"
  826. assert frames[3].lineno == 54
  827. assert frames[3].filename == "MainActivity.java"
  828. assert frames[3].module == "io.sentry.sample.MainActivity"
  829. assert frames[4].function == "sleep"
  830. assert frames[4].lineno == 450
  831. assert frames[4].filename == "Thread.java"
  832. assert frames[4].module == "java.lang.Thread"
  833. assert frames[5].function == "__start_thread"
  834. assert frames[5].package == "/apex/com.android.art/lib64/libart.so"
  835. @requires_symbolicator
  836. @pytest.mark.symbolicator
  837. def test_error_on_resolving(self):
  838. url = reverse(
  839. "sentry-api-0-dsym-files",
  840. kwargs={
  841. "organization_id_or_slug": self.project.organization.slug,
  842. "project_id_or_slug": self.project.slug,
  843. },
  844. )
  845. self.login_as(user=self.user)
  846. out = BytesIO()
  847. f = zipfile.ZipFile(out, "w")
  848. f.writestr("proguard/%s.txt" % PROGUARD_BUG_UUID, PROGUARD_BUG_SOURCE)
  849. f.close()
  850. response = self.client.post(
  851. url,
  852. {
  853. "file": SimpleUploadedFile(
  854. "symbols.zip", out.getvalue(), content_type="application/zip"
  855. )
  856. },
  857. format="multipart",
  858. )
  859. assert response.status_code == 201, response.content
  860. assert len(response.json()) == 1
  861. event_data = {
  862. "user": {"ip_address": "31.172.207.97"},
  863. "extra": {},
  864. "project": self.project.id,
  865. "platform": "java",
  866. "debug_meta": {"images": [{"type": "proguard", "uuid": PROGUARD_BUG_UUID}]},
  867. "exception": {
  868. "values": [
  869. {
  870. "stacktrace": {
  871. "frames": [
  872. {
  873. "function": "a",
  874. "abs_path": None,
  875. "module": "org.a.b.g$a",
  876. "filename": None,
  877. "lineno": 67,
  878. },
  879. {
  880. "function": "a",
  881. "abs_path": None,
  882. "module": "org.a.b.g$a",
  883. "filename": None,
  884. "lineno": 69,
  885. },
  886. ]
  887. },
  888. "type": "RuntimeException",
  889. "value": "Oh no",
  890. }
  891. ]
  892. },
  893. "timestamp": before_now(seconds=1).isoformat(),
  894. }
  895. event = self.post_and_retrieve_event(event_data)
  896. assert len(event.data["errors"]) == 1
  897. error = event.data["errors"][0]
  898. assert error["mapping_uuid"] == "071207ac-b491-4a74-957c-2c94fd9594f2"
  899. assert error["type"] == "proguard_missing_lineno"
  900. def upload_jvm_bundle(self, debug_id, source_files):
  901. files = {}
  902. for source_file in source_files:
  903. files[f"files/_/_/{source_file}"] = {"url": f"~/{source_file}"}
  904. manifest = {
  905. "org": self.project.organization.slug,
  906. "project": self.project.slug,
  907. "debug_id": debug_id,
  908. "files": files,
  909. }
  910. file_like = BytesIO(b"SYSB")
  911. with zipfile.ZipFile(file_like, "a") as zip:
  912. for path, contents in source_files.items():
  913. zip.writestr(f"files/_/_/{path}", contents)
  914. zip.writestr("manifest.json", json.dumps(manifest))
  915. file_like.seek(0)
  916. file = File.objects.create(
  917. name="bundle.zip",
  918. type="sourcebundle",
  919. headers={"Content-Type": "application/x-sentry-bundle+zip"},
  920. )
  921. file.putfile(file_like)
  922. ProjectDebugFile.objects.create(
  923. project_id=self.project.id,
  924. debug_id=debug_id,
  925. file=file,
  926. )
  927. @requires_symbolicator
  928. @pytest.mark.symbolicator
  929. def test_basic_source_lookup(self):
  930. debug_id = str(uuid4())
  931. self.upload_jvm_bundle(debug_id, {"io/sentry/samples/MainActivity.jvm": JVM_SOURCE})
  932. event_data = {
  933. "user": {"ip_address": "31.172.207.97"},
  934. "extra": {},
  935. "project": self.project.id,
  936. "platform": "java",
  937. "debug_meta": {"images": [{"type": "jvm", "debug_id": debug_id}]},
  938. "exception": {
  939. "values": [
  940. {
  941. "stacktrace": {
  942. "frames": [
  943. {
  944. "function": "otherMethod",
  945. "abs_path": "OtherActivity.java",
  946. "module": "OtherActivity",
  947. "filename": "OtherActivity.java",
  948. "lineno": 100,
  949. },
  950. {
  951. "function": "differentMethod",
  952. "abs_path": "DifferentActivity",
  953. "module": "DifferentActivity",
  954. "filename": "DifferentActivity",
  955. "lineno": 200,
  956. },
  957. {
  958. "function": "onCreate",
  959. "abs_path": None,
  960. "module": "io.sentry.samples.MainActivity",
  961. "filename": None,
  962. "lineno": 11,
  963. },
  964. {
  965. "function": "whoops",
  966. "abs_path": "MainActivity.kt",
  967. "module": "io.sentry.samples.MainActivity$InnerClass",
  968. "filename": "MainActivity.kt",
  969. "lineno": 20,
  970. },
  971. {
  972. "function": "whoops2",
  973. "abs_path": None,
  974. "module": "io.sentry.samples.MainActivity$AnotherInnerClass",
  975. "filename": None,
  976. "lineno": 26,
  977. },
  978. {
  979. "function": "whoops3",
  980. "abs_path": "MainActivity.kt",
  981. "module": "io.sentry.samples.MainActivity$AdditionalInnerClass",
  982. "filename": "MainActivity.kt",
  983. "lineno": 32,
  984. },
  985. {
  986. "function": "whoops4",
  987. "abs_path": "SourceFile",
  988. "module": "io.sentry.samples.MainActivity$OneMoreInnerClass",
  989. "filename": "SourceFile",
  990. "lineno": 38,
  991. },
  992. ]
  993. },
  994. "module": "io.sentry.samples",
  995. "type": "RuntimeException",
  996. "value": "whoops",
  997. }
  998. ]
  999. },
  1000. "timestamp": before_now(seconds=1).isoformat(),
  1001. }
  1002. event = self.post_and_retrieve_event(event_data)
  1003. if not self.use_relay():
  1004. # We measure the number of queries after an initial post,
  1005. # because there are many queries polluting the array
  1006. # before the actual "processing" happens (like, auth_user)
  1007. with self.assertWriteQueries(
  1008. {
  1009. "nodestore_node": 2,
  1010. "sentry_eventuser": 1,
  1011. "sentry_groupedmessage": 1,
  1012. "sentry_userreport": 1,
  1013. }
  1014. ):
  1015. self.post_and_retrieve_event(event_data)
  1016. exc = event.interfaces["exception"].values[0]
  1017. bt = exc.stacktrace
  1018. frames = bt.frames
  1019. assert exc.type == "RuntimeException"
  1020. assert exc.value == "whoops"
  1021. assert exc.module == "io.sentry.samples"
  1022. assert frames[0].function == "otherMethod"
  1023. assert frames[0].module == "OtherActivity"
  1024. assert frames[0].lineno == 100
  1025. assert frames[0].context_line is None
  1026. assert frames[0].pre_context is None
  1027. assert frames[0].post_context is None
  1028. assert frames[1].function == "differentMethod"
  1029. assert frames[1].module == "DifferentActivity"
  1030. assert frames[1].lineno == 200
  1031. assert frames[1].context_line is None
  1032. assert frames[1].pre_context is None
  1033. assert frames[1].post_context is None
  1034. assert frames[2].function == "onCreate"
  1035. assert frames[2].module == "io.sentry.samples.MainActivity"
  1036. assert frames[2].lineno == 11
  1037. assert frames[2].context_line == " InnerClass().whoops()"
  1038. assert frames[2].pre_context == [
  1039. "",
  1040. "class MainActivity : ComponentActivity() {",
  1041. " override fun onCreate(savedInstanceState: Bundle?) {",
  1042. " super.onCreate(savedInstanceState)",
  1043. " setContentView(R.layout.activity_main)",
  1044. ]
  1045. assert frames[2].post_context == [
  1046. "",
  1047. " val list = findViewById<RecyclerView>(R.id.list)",
  1048. " list.layoutManager = LinearLayoutManager(this)",
  1049. " list.adapter = TrackAdapter()",
  1050. " }",
  1051. ]
  1052. assert frames[3].function == "whoops"
  1053. assert frames[3].module == "io.sentry.samples.MainActivity$InnerClass"
  1054. assert frames[3].lineno == 20
  1055. assert frames[3].context_line == " AnotherInnerClass().whoops2()"
  1056. assert frames[3].pre_context == [
  1057. " list.adapter = TrackAdapter()",
  1058. " }",
  1059. "",
  1060. " class InnerClass {",
  1061. " fun whoops() {",
  1062. ]
  1063. assert frames[3].post_context == [
  1064. " }",
  1065. " }",
  1066. "",
  1067. " class AnotherInnerClass {",
  1068. " fun whoops2() {",
  1069. ]
  1070. assert frames[4].function == "whoops2"
  1071. assert frames[4].module == "io.sentry.samples.MainActivity$AnotherInnerClass"
  1072. assert frames[4].lineno == 26
  1073. assert frames[4].context_line == " AdditionalInnerClass().whoops3()"
  1074. assert frames[4].pre_context == [
  1075. " }",
  1076. " }",
  1077. "",
  1078. " class AnotherInnerClass {",
  1079. " fun whoops2() {",
  1080. ]
  1081. assert frames[4].post_context == [
  1082. " }",
  1083. " }",
  1084. "",
  1085. " class AdditionalInnerClass {",
  1086. " fun whoops3() {",
  1087. ]
  1088. assert frames[5].function == "whoops3"
  1089. assert frames[5].module == "io.sentry.samples.MainActivity$AdditionalInnerClass"
  1090. assert frames[5].lineno == 32
  1091. assert frames[5].context_line == " OneMoreInnerClass().whoops4()"
  1092. assert frames[5].pre_context == [
  1093. " }",
  1094. " }",
  1095. "",
  1096. " class AdditionalInnerClass {",
  1097. " fun whoops3() {",
  1098. ]
  1099. assert frames[5].post_context == [
  1100. " }",
  1101. " }",
  1102. "",
  1103. " class OneMoreInnerClass {",
  1104. " fun whoops4() {",
  1105. ]
  1106. assert frames[6].function == "whoops4"
  1107. assert frames[6].module == "io.sentry.samples.MainActivity$OneMoreInnerClass"
  1108. assert frames[6].lineno == 38
  1109. assert frames[6].context_line == ' throw RuntimeException("whoops")'
  1110. assert frames[6].pre_context == [
  1111. " }",
  1112. " }",
  1113. "",
  1114. " class OneMoreInnerClass {",
  1115. " fun whoops4() {",
  1116. ]
  1117. assert frames[6].post_context == [" }", " }", "}", ""]
  1118. @requires_symbolicator
  1119. @pytest.mark.symbolicator
  1120. def test_source_lookup_with_proguard(self):
  1121. self.upload_proguard_mapping(PROGUARD_SOURCE_LOOKUP_UUID, PROGUARD_SOURCE_LOOKUP_SOURCE)
  1122. debug_id1 = str(uuid4())
  1123. self.upload_jvm_bundle(
  1124. debug_id1,
  1125. {
  1126. "io/sentry/samples/instrumentation/ui/EditActivity.jvm": EDIT_ACTIVITY_SOURCE,
  1127. },
  1128. )
  1129. debug_id2 = str(uuid4())
  1130. self.upload_jvm_bundle(
  1131. debug_id2,
  1132. {
  1133. "io/sentry/samples/instrumentation/ui/SomeService.jvm": SOME_SERVICE_SOURCE,
  1134. },
  1135. )
  1136. event_data = {
  1137. "user": {"ip_address": "31.172.207.97"},
  1138. "extra": {},
  1139. "project": self.project.id,
  1140. "platform": "java",
  1141. "debug_meta": {
  1142. "images": [
  1143. {
  1144. "type": "jvm",
  1145. "debug_id": debug_id1,
  1146. },
  1147. {
  1148. "type": "jvm",
  1149. "debug_id": debug_id2,
  1150. },
  1151. {
  1152. "type": "jvm",
  1153. "debug_id": str(uuid4()),
  1154. }, # does not exist
  1155. {"type": "proguard", "uuid": PROGUARD_SOURCE_LOOKUP_UUID},
  1156. {"type": "proguard", "uuid": str(uuid4())}, # does not exist
  1157. ]
  1158. },
  1159. "exception": {
  1160. "values": [
  1161. {
  1162. "type": "RuntimeException",
  1163. "value": "thrown on purpose to test ProGuard Android source context",
  1164. "module": "java.lang",
  1165. "thread_id": 1,
  1166. "stacktrace": {
  1167. "frames": [
  1168. {
  1169. "filename": "ZygoteInit.java",
  1170. "function": "main",
  1171. "module": "com.android.internal.os.ZygoteInit",
  1172. "lineno": 698,
  1173. "native": False,
  1174. },
  1175. {
  1176. "filename": "ZygoteInit.java",
  1177. "function": "run",
  1178. "module": "com.android.internal.os.ZygoteInit$MethodAndArgsCaller",
  1179. "lineno": 903,
  1180. "native": False,
  1181. },
  1182. {
  1183. "filename": "Method.java",
  1184. "function": "invoke",
  1185. "module": "java.lang.reflect.Method",
  1186. "lineno": 372,
  1187. "native": False,
  1188. },
  1189. {
  1190. "filename": "Method.java",
  1191. "function": "invoke",
  1192. "module": "java.lang.reflect.Method",
  1193. "native": True,
  1194. },
  1195. {
  1196. "filename": "ActivityThread.java",
  1197. "function": "main",
  1198. "module": "android.app.ActivityThread",
  1199. "lineno": 5254,
  1200. "native": False,
  1201. },
  1202. {
  1203. "filename": "Looper.java",
  1204. "function": "loop",
  1205. "module": "android.os.Looper",
  1206. "lineno": 135,
  1207. "native": False,
  1208. },
  1209. {
  1210. "filename": "Handler.java",
  1211. "function": "dispatchMessage",
  1212. "module": "android.os.Handler",
  1213. "lineno": 95,
  1214. "native": False,
  1215. },
  1216. {
  1217. "filename": "Handler.java",
  1218. "function": "handleCallback",
  1219. "module": "android.os.Handler",
  1220. "lineno": 739,
  1221. "native": False,
  1222. },
  1223. {
  1224. "filename": "View.java",
  1225. "function": "run",
  1226. "module": "android.view.View$PerformClick",
  1227. "lineno": 19866,
  1228. "native": False,
  1229. },
  1230. {
  1231. "filename": "View.java",
  1232. "function": "performClick",
  1233. "module": "android.view.View",
  1234. "lineno": 4780,
  1235. "native": False,
  1236. },
  1237. {
  1238. "filename": "ActionMenuItemView.java",
  1239. "function": "onClick",
  1240. "module": "androidx.appcompat.view.menu.ActionMenuItemView",
  1241. "lineno": 7,
  1242. "native": False,
  1243. },
  1244. {
  1245. "filename": "ActionMenuView.java",
  1246. "function": "invokeItem",
  1247. "module": "androidx.appcompat.widget.ActionMenuView",
  1248. "lineno": 4,
  1249. "native": False,
  1250. },
  1251. {
  1252. "filename": "MenuBuilder.java",
  1253. "function": "performItemAction",
  1254. "module": "androidx.appcompat.view.menu.MenuBuilder",
  1255. "lineno": 1,
  1256. "native": False,
  1257. },
  1258. {
  1259. "filename": "MenuBuilder.java",
  1260. "function": "performItemAction",
  1261. "module": "androidx.appcompat.view.menu.MenuBuilder",
  1262. "lineno": 4,
  1263. "native": False,
  1264. },
  1265. {
  1266. "filename": "MenuItemImpl.java",
  1267. "function": "invoke",
  1268. "module": "androidx.appcompat.view.menu.MenuItemImpl",
  1269. "lineno": 15,
  1270. "native": False,
  1271. },
  1272. {
  1273. "filename": "MenuBuilder.java",
  1274. "function": "dispatchMenuItemSelected",
  1275. "module": "androidx.appcompat.view.menu.MenuBuilder",
  1276. "lineno": 5,
  1277. "native": False,
  1278. },
  1279. {
  1280. "filename": "ActionMenuView.java",
  1281. "function": "onMenuItemSelected",
  1282. "module": "androidx.appcompat.widget.ActionMenuView$MenuBuilderCallback",
  1283. "lineno": 7,
  1284. "native": False,
  1285. },
  1286. {
  1287. "filename": "Toolbar.java",
  1288. "function": "onMenuItemClick",
  1289. "module": "androidx.appcompat.widget.Toolbar$1",
  1290. "lineno": 7,
  1291. "native": False,
  1292. },
  1293. {
  1294. "filename": "R8$$SyntheticClass",
  1295. "function": "onMenuItemClick",
  1296. "module": "io.sentry.samples.instrumentation.ui.g",
  1297. "lineno": 40,
  1298. "in_app": True,
  1299. "native": False,
  1300. },
  1301. ]
  1302. },
  1303. }
  1304. ]
  1305. },
  1306. "timestamp": before_now(seconds=1).isoformat(),
  1307. }
  1308. event = self.post_and_retrieve_event(event_data)
  1309. if not self.use_relay():
  1310. # We measure the number of queries after an initial post,
  1311. # because there are many queries polluting the array
  1312. # before the actual "processing" happens (like, auth_user)
  1313. with self.assertWriteQueries(
  1314. {
  1315. "nodestore_node": 2,
  1316. "sentry_eventuser": 1,
  1317. "sentry_groupedmessage": 1,
  1318. "sentry_userreport": 1,
  1319. }
  1320. ):
  1321. self.post_and_retrieve_event(event_data)
  1322. exc = event.interfaces["exception"].values[0]
  1323. bt = exc.stacktrace
  1324. frames = bt.frames
  1325. assert exc.type == "RuntimeException"
  1326. assert exc.value == "thrown on purpose to test ProGuard Android source context"
  1327. assert exc.module == "java.lang"
  1328. assert frames[18].function == "onMenuItemClick"
  1329. assert (
  1330. frames[18].module
  1331. == "io.sentry.samples.instrumentation.ui.EditActivity$$InternalSyntheticLambda$1$ebaa538726b99bb77e0f5e7c86443911af17d6e5be2b8771952ae0caa4ff2ac7$0"
  1332. )
  1333. assert frames[18].lineno == 0
  1334. assert frames[18].context_line is None
  1335. assert frames[18].pre_context is None
  1336. assert frames[18].post_context is None
  1337. assert frames[19].function == "onCreate$lambda-1"
  1338. assert frames[19].module == "io.sentry.samples.instrumentation.ui.EditActivity"
  1339. assert frames[19].lineno == 37
  1340. assert frames[19].context_line == " SomeService().helloThere()"
  1341. assert frames[19].pre_context == [
  1342. " }",
  1343. "",
  1344. " findViewById<Toolbar>(R.id.toolbar).setOnMenuItemClickListener {",
  1345. " if (it.itemId == R.id.action_save) {",
  1346. " try {",
  1347. ]
  1348. assert frames[19].post_context == [
  1349. " } catch (e: Exception) {",
  1350. " Sentry.captureException(e)",
  1351. " }",
  1352. "",
  1353. " val transaction = Sentry.startTransaction(",
  1354. ]
  1355. assert frames[20].function == "helloThere"
  1356. assert frames[20].module == "io.sentry.samples.instrumentation.ui.SomeService"
  1357. assert frames[20].lineno == 5
  1358. assert frames[20].context_line == " InnerClassOfSomeService().helloInner()"
  1359. assert frames[20].pre_context == [
  1360. "package io.sentry.samples.instrumentation.ui",
  1361. "",
  1362. "class SomeService {",
  1363. " fun helloThere() {",
  1364. ]
  1365. assert frames[20].post_context == [
  1366. " }",
  1367. "",
  1368. " class InnerClassOfSomeService {",
  1369. " fun helloInner() {",
  1370. " AnotherClassInSameFile().helloOther()",
  1371. ]
  1372. assert frames[21].function == "helloInner"
  1373. assert (
  1374. frames[21].module
  1375. == "io.sentry.samples.instrumentation.ui.SomeService$InnerClassOfSomeService"
  1376. )
  1377. assert frames[21].lineno == 10
  1378. assert frames[21].context_line == " AnotherClassInSameFile().helloOther()"
  1379. assert frames[21].pre_context == [
  1380. " InnerClassOfSomeService().helloInner()",
  1381. " }",
  1382. "",
  1383. " class InnerClassOfSomeService {",
  1384. " fun helloInner() {",
  1385. ]
  1386. assert frames[21].post_context == [
  1387. " }",
  1388. " }",
  1389. "}",
  1390. "",
  1391. "class AnotherClassInSameFile {",
  1392. ]
  1393. assert frames[22].function == "helloOther"
  1394. assert frames[22].module == "io.sentry.samples.instrumentation.ui.AnotherClassInSameFile"
  1395. assert frames[22].lineno == 17
  1396. assert frames[22].context_line is None
  1397. assert frames[22].pre_context is None
  1398. assert frames[22].post_context is None
  1399. assert frames[23].function == "otherFun"
  1400. assert frames[23].module == "io.sentry.samples.instrumentation.ui.AnotherClassInSameFile"
  1401. assert frames[23].lineno == 21
  1402. assert frames[23].context_line is None
  1403. assert frames[23].pre_context is None
  1404. assert frames[23].post_context is None
  1405. assert frames[24].function == "helloOtherInner"
  1406. assert (
  1407. frames[24].module
  1408. == "io.sentry.samples.instrumentation.ui.AnotherClassInSameFile$AnotherInnerClass"
  1409. )
  1410. assert frames[24].lineno == 26
  1411. assert frames[24].context_line is None
  1412. assert frames[24].pre_context is None
  1413. assert frames[24].post_context is None
  1414. @requires_symbolicator
  1415. @pytest.mark.symbolicator
  1416. def test_invalid_exception(self):
  1417. event_data = {
  1418. "user": {"ip_address": "31.172.207.97"},
  1419. "extra": {},
  1420. "project": self.project.id,
  1421. "platform": "java",
  1422. "debug_meta": {},
  1423. "exception": {
  1424. "values": [
  1425. {"type": "PlatformException"},
  1426. {"type": "SecurityException", "module": "java.lang"},
  1427. {"type": "RemoteException", "module": "android.os"},
  1428. ]
  1429. },
  1430. "timestamp": before_now(seconds=1).isoformat(),
  1431. }
  1432. event = self.post_and_retrieve_event(event_data)
  1433. expected = [
  1434. {"type": e.get("type", None), "module": e.get("module", None)}
  1435. for e in event_data["exception"]["values"]
  1436. ]
  1437. received = [
  1438. {"type": e.type, "module": e.module} for e in event.interfaces["exception"].values
  1439. ]
  1440. assert received == expected
  1441. def test_is_jvm_event(self):
  1442. from sentry.lang.java.utils import is_jvm_event
  1443. event = {
  1444. "user": {"ip_address": "31.172.207.97"},
  1445. "extra": {},
  1446. "project": self.project.id,
  1447. "platform": "java",
  1448. "debug_meta": {"images": [{"type": "jvm", "debug_id": PROGUARD_INLINE_UUID}]},
  1449. "exception": {
  1450. "values": [
  1451. {
  1452. "stacktrace": {
  1453. "frames": [
  1454. {
  1455. "function": "whoops4",
  1456. "abs_path": "SourceFile",
  1457. "module": "io.sentry.samples.MainActivity$OneMoreInnerClass",
  1458. "filename": "SourceFile",
  1459. "lineno": 38,
  1460. },
  1461. ]
  1462. },
  1463. "module": "io.sentry.samples",
  1464. "type": "RuntimeException",
  1465. "value": "whoops",
  1466. }
  1467. ]
  1468. },
  1469. "timestamp": before_now(seconds=1),
  1470. }
  1471. stacktraces = find_stacktraces_in_data(event)
  1472. assert is_jvm_event(event, stacktraces)
  1473. event = {
  1474. "user": {"ip_address": "31.172.207.97"},
  1475. "extra": {},
  1476. "project": self.project.id,
  1477. "debug_meta": {"images": [{"type": "jvm", "debug_id": PROGUARD_INLINE_UUID}]},
  1478. "exception": {
  1479. "values": [
  1480. {
  1481. "stacktrace": {
  1482. "frames": [
  1483. {
  1484. "function": "whoops4",
  1485. "abs_path": "SourceFile",
  1486. "module": "io.sentry.samples.MainActivity$OneMoreInnerClass",
  1487. "filename": "SourceFile",
  1488. "lineno": 38,
  1489. },
  1490. ]
  1491. },
  1492. "module": "io.sentry.samples",
  1493. "type": "RuntimeException",
  1494. "value": "whoops",
  1495. }
  1496. ]
  1497. },
  1498. "timestamp": before_now(seconds=1),
  1499. }
  1500. # has no platform
  1501. stacktraces = find_stacktraces_in_data(event)
  1502. assert is_jvm_event(event, stacktraces)
  1503. event = {
  1504. "user": {"ip_address": "31.172.207.97"},
  1505. "extra": {},
  1506. "project": self.project.id,
  1507. "platform": "java",
  1508. "debug_meta": {},
  1509. "exception": {
  1510. "values": [
  1511. {
  1512. "stacktrace": {
  1513. "frames": [
  1514. {
  1515. "function": "whoops4",
  1516. "abs_path": "SourceFile",
  1517. "module": "io.sentry.samples.MainActivity$OneMoreInnerClass",
  1518. "filename": "SourceFile",
  1519. "lineno": 38,
  1520. },
  1521. ]
  1522. },
  1523. "module": "io.sentry.samples",
  1524. "type": "RuntimeException",
  1525. "value": "whoops",
  1526. }
  1527. ]
  1528. },
  1529. "timestamp": before_now(seconds=1),
  1530. }
  1531. # has no modules
  1532. stacktraces = find_stacktraces_in_data(event)
  1533. assert is_jvm_event(event, stacktraces)
  1534. event = {
  1535. "user": {"ip_address": "31.172.207.97"},
  1536. "extra": {},
  1537. "project": self.project.id,
  1538. "debug_meta": {},
  1539. "exception": {
  1540. "values": [
  1541. {
  1542. "stacktrace": {
  1543. "frames": [
  1544. {
  1545. "platform": "java",
  1546. "function": "whoops4",
  1547. "abs_path": "SourceFile",
  1548. "module": "io.sentry.samples.MainActivity$OneMoreInnerClass",
  1549. "filename": "SourceFile",
  1550. "lineno": 38,
  1551. },
  1552. ]
  1553. },
  1554. "module": "io.sentry.samples",
  1555. "type": "RuntimeException",
  1556. "value": "whoops",
  1557. }
  1558. ]
  1559. },
  1560. "timestamp": before_now(seconds=1),
  1561. }
  1562. # has a Java frame
  1563. stacktraces = find_stacktraces_in_data(event)
  1564. assert is_jvm_event(event, stacktraces)
  1565. event = {
  1566. "user": {"ip_address": "31.172.207.97"},
  1567. "extra": {},
  1568. "project": self.project.id,
  1569. "debug_meta": {},
  1570. "exception": {
  1571. "values": [
  1572. {
  1573. "stacktrace": {
  1574. "frames": [
  1575. {
  1576. "function": "whoops4",
  1577. "abs_path": "SourceFile",
  1578. "module": "io.sentry.samples.MainActivity$OneMoreInnerClass",
  1579. "filename": "SourceFile",
  1580. "lineno": 38,
  1581. },
  1582. ]
  1583. },
  1584. "module": "io.sentry.samples",
  1585. "type": "RuntimeException",
  1586. "value": "whoops",
  1587. }
  1588. ]
  1589. },
  1590. "timestamp": before_now(seconds=1),
  1591. }
  1592. # has no platform, frame, or modules
  1593. stacktraces = find_stacktraces_in_data(event)
  1594. assert not is_jvm_event(event, stacktraces)