getReasonGroupName.ts 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. import startCase from 'lodash/startCase';
  2. import {Outcome} from 'sentry/types/core';
  3. // List of Relay's current invalid reasons - https://github.com/getsentry/relay/blob/89a8dd7caaad1f126e1cacced0d73bb50fcd4f5a/relay-server/src/services/outcome.rs#L333
  4. enum DiscardReason {
  5. DUPLICATE = 'duplicate',
  6. PROJECT_ID = 'project_id',
  7. AUTH_VERSION = 'auth_version',
  8. AUTH_CLIENT = 'auth_client',
  9. NO_DATA = 'no_data',
  10. DISALLOWED_METHOD = 'disallowed_method',
  11. CONTENT_TYPE = 'content_type',
  12. INVALID_MULTIPART = 'invalid_multipart',
  13. INVALID_MSGPACK = 'invalid_msgpack',
  14. INVALID_JSON = 'invalid_json',
  15. INVALID_ENVELOPE = 'invalid_envelope',
  16. TIMESTAMP = 'timestamp',
  17. DUPLICATE_ITEM = 'duplicate_item',
  18. INVALID_TRANSACTION = 'invalid_transaction',
  19. INVALID_SPAN = 'invalid_span',
  20. INVALID_REPLAY = 'invalid_replay',
  21. INVALID_REPLAY_RECORDING = 'invalid_replay_recording',
  22. INVALID_REPLAY_VIDEO = 'invalid_replay_video',
  23. PAYLOAD = 'payload',
  24. INVALID_COMPRESSION = 'invalid_compression',
  25. TOO_LARGE = 'too_large',
  26. MISSING_MINIDUMP_UPLOAD = 'missing_minidump_upload',
  27. INVALID_MINIDUMP = 'invalid_minidump',
  28. SECURITY_REPORT = 'security_report',
  29. SECURITY_REPORT_TYPE = 'security_report_type',
  30. PROCESS_UNREAL = 'process_unreal',
  31. CORS = 'cors',
  32. NO_EVENT_PAYLOAD = 'no_event_payload',
  33. EMPTY_ENVELOPE = 'empty_envelope',
  34. INVALID_REPLAY_NO_PAYLOAD = 'invalid_replay_no_payload',
  35. TRANSACTION_SAMPLED = 'transaction_sampled',
  36. INTERNAL = 'internal',
  37. MULTI_PROJECT_ID = 'multi_project_id',
  38. PROJECT_STATE = 'project_state',
  39. PROJECT_STATE_PII = 'project_state_pii',
  40. INVALID_REPLAY_PII_SCRUBBER_FAILED = 'invalid_replay_pii_scrubber_failed',
  41. FEATURE_DISABLED = 'feature_disabled',
  42. }
  43. // List of Client Discard Reason according to the Client Report's doc - https://develop.sentry.dev/sdk/client-reports/#envelope-item-payload
  44. enum ClientDiscardReason {
  45. QUEUE_OVERFLOW = 'queue_overflow',
  46. CACHE_OVERFLOW = 'cache_overflow',
  47. RATELIMIT_BACKOFF = 'ratelimit_backoff',
  48. NETWORK_ERROR = 'network_error',
  49. SAMPLE_RATE = 'sample_rate',
  50. BEFORE_SEND = 'before_send',
  51. EVENT_PROCESSSOR = 'event_processor',
  52. SEND_ERROR = 'send_error',
  53. INTERNAL_SDK_ERROR = 'internal_sdk_error',
  54. INSUFFICIENT_DATA = 'insufficient_data',
  55. BACKPRESSURE = 'backpressure',
  56. }
  57. enum RateLimitedReason {
  58. PROJECT_QUOTA = 'project_quota',
  59. ORG_QUOTA = 'org_quota',
  60. KEY_QUOTA = 'key_quota',
  61. SPIKE_PROTECTION = 'spike_protection',
  62. SMART_RATE_LIMIT = 'smart_rate_limit',
  63. }
  64. // Invalid reasons should not be exposed directly, but instead in the following groups:
  65. const invalidReasonsGroup: Record<string, DiscardReason[]> = {
  66. duplicate: [DiscardReason.DUPLICATE],
  67. project_missing: [DiscardReason.PROJECT_ID],
  68. invalid_request: [
  69. DiscardReason.AUTH_VERSION,
  70. DiscardReason.AUTH_CLIENT,
  71. DiscardReason.NO_DATA,
  72. DiscardReason.DISALLOWED_METHOD,
  73. DiscardReason.CONTENT_TYPE,
  74. DiscardReason.INVALID_MULTIPART,
  75. DiscardReason.INVALID_MSGPACK,
  76. DiscardReason.INVALID_JSON,
  77. DiscardReason.INVALID_ENVELOPE,
  78. DiscardReason.TIMESTAMP,
  79. DiscardReason.DUPLICATE_ITEM,
  80. ],
  81. invalid_data: [
  82. DiscardReason.INVALID_TRANSACTION,
  83. DiscardReason.INVALID_SPAN,
  84. DiscardReason.INVALID_REPLAY,
  85. DiscardReason.INVALID_REPLAY_RECORDING,
  86. DiscardReason.INVALID_REPLAY_VIDEO,
  87. ],
  88. payload: [DiscardReason.PAYLOAD, DiscardReason.INVALID_COMPRESSION],
  89. too_large: [DiscardReason.TOO_LARGE],
  90. minidump: [DiscardReason.MISSING_MINIDUMP_UPLOAD, DiscardReason.INVALID_MINIDUMP],
  91. security_report: [DiscardReason.SECURITY_REPORT, DiscardReason.SECURITY_REPORT_TYPE],
  92. unreal: [DiscardReason.PROCESS_UNREAL],
  93. disallowed_domain: [DiscardReason.CORS],
  94. empty: [
  95. DiscardReason.NO_EVENT_PAYLOAD,
  96. DiscardReason.EMPTY_ENVELOPE,
  97. DiscardReason.INVALID_REPLAY_NO_PAYLOAD,
  98. ],
  99. sampling: [DiscardReason.TRANSACTION_SAMPLED],
  100. };
  101. function getInvalidReasonGroupName(reason: DiscardReason): string {
  102. for (const [group, reasons] of Object.entries(invalidReasonsGroup)) {
  103. if (reasons.includes(reason)) {
  104. return group;
  105. }
  106. }
  107. return 'internal';
  108. }
  109. function getRateLimitedReasonGroupName(reason: RateLimitedReason | string): string {
  110. if (reason.endsWith('_usage_exceeded')) {
  111. return 'quota';
  112. }
  113. if (reason.endsWith('_disabled')) {
  114. return 'disabled';
  115. }
  116. switch (reason) {
  117. case RateLimitedReason.ORG_QUOTA:
  118. case RateLimitedReason.PROJECT_QUOTA:
  119. return 'global limit';
  120. case RateLimitedReason.KEY_QUOTA:
  121. return 'DSN limit';
  122. case RateLimitedReason.SPIKE_PROTECTION:
  123. case RateLimitedReason.SMART_RATE_LIMIT:
  124. return 'spike protection';
  125. default:
  126. return 'internal';
  127. }
  128. }
  129. function getClientDiscardReasonGroupName(reason: ClientDiscardReason): string {
  130. switch (reason) {
  131. case ClientDiscardReason.QUEUE_OVERFLOW:
  132. case ClientDiscardReason.CACHE_OVERFLOW:
  133. case ClientDiscardReason.RATELIMIT_BACKOFF:
  134. case ClientDiscardReason.NETWORK_ERROR:
  135. case ClientDiscardReason.SAMPLE_RATE:
  136. case ClientDiscardReason.BEFORE_SEND:
  137. case ClientDiscardReason.EVENT_PROCESSSOR:
  138. case ClientDiscardReason.SEND_ERROR:
  139. case ClientDiscardReason.INTERNAL_SDK_ERROR:
  140. case ClientDiscardReason.INSUFFICIENT_DATA:
  141. case ClientDiscardReason.BACKPRESSURE:
  142. return reason;
  143. default:
  144. return 'other';
  145. }
  146. }
  147. export function getReasonGroupName(outcome: string | number, reason: string): string {
  148. switch (outcome) {
  149. case Outcome.INVALID:
  150. return getInvalidReasonGroupName(reason as DiscardReason);
  151. case Outcome.CARDINALITY_LIMITED:
  152. case Outcome.RATE_LIMITED:
  153. case Outcome.ABUSE:
  154. return getRateLimitedReasonGroupName(reason as RateLimitedReason);
  155. case Outcome.FILTERED:
  156. return startCase(reason);
  157. case Outcome.CLIENT_DISCARD:
  158. return getClientDiscardReasonGroupName(reason as ClientDiscardReason);
  159. default:
  160. return String(reason);
  161. }
  162. }