api.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. """Port of sentry.api.endpoints.debug_files.DifAssembleEndpoint"""
  2. import io
  3. import re
  4. import tempfile
  5. import zipfile
  6. from hashlib import sha1
  7. from asgiref.sync import sync_to_async
  8. from django.core.files import File as DjangoFile
  9. from django.shortcuts import aget_object_or_404
  10. from ninja import File as NinjaFile
  11. from ninja import Router
  12. from ninja.errors import HttpError
  13. from ninja.files import UploadedFile
  14. from symbolic import ProguardMapper
  15. from apps.files.models import File, FileBlob
  16. from apps.organizations_ext.models import Organization
  17. from apps.projects.models import Project
  18. from glitchtip.api.authentication import AuthHttpRequest
  19. from glitchtip.utils import async_call_celery_task
  20. from .models import DebugInformationFile
  21. from .schema import AssemblePayload
  22. from .tasks import DIF_STATE_CREATED, DIF_STATE_NOT_FOUND, DIF_STATE_OK, difs_assemble
  23. MAX_UPLOAD_BLOB_SIZE = 32 * 1024 * 1024 # 32MB
  24. router = Router()
  25. @router.post(
  26. "projects/{slug:organization_slug}/{slug:project_slug}/files/difs/assemble/"
  27. )
  28. async def difs_assemble_api(
  29. request: AuthHttpRequest,
  30. organization_slug: str,
  31. project_slug: str,
  32. payload: AssemblePayload,
  33. ):
  34. organization = await aget_object_or_404(
  35. Organization, slug=organization_slug.lower(), users=request.auth.user_id
  36. )
  37. await aget_object_or_404(
  38. Project, slug=project_slug.lower(), organization=organization
  39. )
  40. responses = {}
  41. files = payload.root.items()
  42. for checksum, file in files:
  43. chunks = file.chunks
  44. name = file.name
  45. debug_id = file.debug_id
  46. debug_file = await (
  47. DebugInformationFile.objects.filter(
  48. project__slug=project_slug, file__checksum=checksum
  49. )
  50. .select_related("file")
  51. .afirst()
  52. )
  53. if debug_file is not None:
  54. responses[checksum] = {
  55. "state": DIF_STATE_OK,
  56. "missingChunks": [],
  57. }
  58. continue
  59. existed_chunks = [
  60. file_blob
  61. async for file_blob in FileBlob.objects.filter(
  62. checksum__in=chunks
  63. ).values_list("checksum", flat=True)
  64. ]
  65. missing_chunks = list(set(chunks) - set(existed_chunks))
  66. if len(missing_chunks) != 0:
  67. responses[checksum] = {
  68. "state": DIF_STATE_NOT_FOUND,
  69. "missingChunks": missing_chunks,
  70. }
  71. continue
  72. responses[checksum] = {"state": DIF_STATE_CREATED, "missingChunks": []}
  73. await async_call_celery_task(
  74. difs_assemble, project_slug, name, checksum, chunks, debug_id
  75. )
  76. return responses
  77. @router.post("projects/{slug:organization_slug}/{slug:project_slug}/reprocessing/")
  78. async def project_reprocessing(
  79. request: AuthHttpRequest,
  80. organization_slug: str,
  81. project_slug: str,
  82. ):
  83. """
  84. Not implemented. It is a dummy API to keep `sentry-cli upload-dif` happy
  85. """
  86. return None
  87. def extract_proguard_id(name: str):
  88. match = re.search("proguard/([-a-fA-F0-9]+).txt", name)
  89. if match is None:
  90. return
  91. return match.group(1)
  92. def extract_proguard_metadata(proguard_file):
  93. try:
  94. mapper = ProguardMapper.open(proguard_file)
  95. if mapper is None:
  96. return
  97. metadata = {"arch": "any", "feature": "mapping"}
  98. return metadata
  99. except Exception:
  100. pass
  101. async def create_dif_from_read_only_file(proguard_file, project, proguard_id, filename):
  102. with tempfile.NamedTemporaryFile("br+") as tmp:
  103. content = proguard_file.read()
  104. tmp.write(content)
  105. tmp.flush()
  106. metadata = extract_proguard_metadata(tmp.name)
  107. if metadata is None:
  108. return None
  109. checksum = sha1(content).hexdigest()
  110. size = len(content)
  111. blob = await FileBlob.objects.filter(checksum=checksum).afirst()
  112. if blob is None:
  113. blob = FileBlob(checksum=checksum, size=size) # noqa
  114. await sync_to_async(blob.blob.save)(filename, DjangoFile(tmp))
  115. await blob.asave()
  116. fileobj = await File.objects.filter(checksum=checksum).afirst()
  117. if fileobj is None:
  118. fileobj = File()
  119. fileobj.name = filename
  120. fileobj.headers = {}
  121. fileobj.checksum = checksum
  122. fileobj.size = size
  123. fileobj.blob = blob
  124. await fileobj.asave()
  125. dif = await DebugInformationFile.objects.filter(
  126. file__checksum=checksum, project=project
  127. ).afirst()
  128. if dif is None:
  129. dif = DebugInformationFile()
  130. dif.name = filename
  131. dif.project = project
  132. dif.file = fileobj
  133. dif.data = {
  134. "arch": metadata["arch"],
  135. "debug_id": proguard_id,
  136. "symbol_type": "proguard",
  137. "features": ["mapping"],
  138. }
  139. await dif.asave()
  140. result = {
  141. "id": dif.id,
  142. "debugId": proguard_id,
  143. "cpuName": "any",
  144. "objectName": "proguard-mapping",
  145. "symbolType": "proguard",
  146. "size": size,
  147. "sha1": checksum,
  148. "data": {"features": ["mapping"]},
  149. "headers": {"Content-Type": "text/x-proguard+plain"},
  150. "dateCreated": fileobj.created,
  151. }
  152. return result
  153. @router.post("projects/{slug:organization_slug}/{slug:project_slug}/files/dsyms/")
  154. async def dsyms(
  155. request: AuthHttpRequest,
  156. organization_slug: str,
  157. project_slug: str,
  158. file: UploadedFile = NinjaFile(...),
  159. ):
  160. organization = await aget_object_or_404(
  161. Organization, slug=organization_slug.lower(), users=request.auth.user_id
  162. )
  163. # self.check_object_permissions(request, organization)
  164. project = await aget_object_or_404(
  165. Project, slug=project_slug.lower(), organization=organization
  166. )
  167. if file.size > MAX_UPLOAD_BLOB_SIZE:
  168. raise HttpError(
  169. 400,
  170. "File size too large",
  171. )
  172. content = file.read()
  173. buffer = io.BytesIO(content)
  174. if zipfile.is_zipfile(buffer) is False:
  175. raise HttpError(400, "Invalid file type uploaded")
  176. results = []
  177. with zipfile.ZipFile(buffer) as uploaded_zip_file:
  178. for filename in uploaded_zip_file.namelist():
  179. proguard_id = extract_proguard_id(filename)
  180. if proguard_id is None:
  181. raise HttpError(400, "")
  182. with uploaded_zip_file.open(filename) as proguard_file:
  183. result = await create_dif_from_read_only_file(
  184. proguard_file, project, proguard_id, filename
  185. )
  186. if result is None:
  187. raise HttpError(
  188. 400,
  189. "Invalid proguard mapping file uploaded",
  190. )
  191. results.append(result)
  192. return results