content.spec.tsx 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. import {render, screen, userEvent} from 'sentry-test/reactTestingLibrary';
  2. import StackTraceContent from 'sentry/components/events/interfaces/crashContent/stackTrace/content';
  3. import {StacktraceType} from 'sentry/types/stacktrace';
  4. const eventEntryStacktrace = TestStubs.EventEntryStacktrace();
  5. const event = TestStubs.Event({entries: [eventEntryStacktrace]});
  6. const data = eventEntryStacktrace.data as Required<StacktraceType>;
  7. function renderedComponent(
  8. props: Partial<React.ComponentProps<typeof StackTraceContent>>
  9. ) {
  10. return render(
  11. <StackTraceContent
  12. data={data}
  13. className="no-exception"
  14. platform="other"
  15. event={event}
  16. newestFirst
  17. includeSystemFrames
  18. {...props}
  19. />
  20. );
  21. }
  22. describe('StackTrace', function () {
  23. it('renders', function () {
  24. const {container} = renderedComponent({});
  25. // stack trace content
  26. const stackTraceContent = screen.getByTestId('stack-trace-content');
  27. expect(stackTraceContent).toBeInTheDocument();
  28. // stack trace content has to have a platform icon and a frame list
  29. expect(stackTraceContent.children).toHaveLength(2);
  30. // platform icon
  31. expect(screen.getByTestId('platform-icon-python')).toBeInTheDocument();
  32. // frame list
  33. const frames = screen.getByTestId('frames');
  34. expect(frames.children).toHaveLength(5);
  35. expect(container).toSnapshot();
  36. });
  37. it('renders the frame in the correct order', function () {
  38. renderedComponent({});
  39. // frame - filename
  40. const frameFilenames = screen.getAllByTestId('filename');
  41. expect(frameFilenames).toHaveLength(5);
  42. expect(frameFilenames[0]).toHaveTextContent('raven/scripts/runner.py');
  43. expect(frameFilenames[1]).toHaveTextContent('raven/scripts/runner.py');
  44. expect(frameFilenames[2]).toHaveTextContent('raven/base.py');
  45. expect(frameFilenames[3]).toHaveTextContent('raven/base.py');
  46. expect(frameFilenames[4]).toHaveTextContent('raven/base.py');
  47. // frame - function
  48. const frameFunction = screen.getAllByTestId('function');
  49. expect(frameFunction).toHaveLength(5);
  50. expect(frameFunction[0]).toHaveTextContent('main');
  51. expect(frameFunction[1]).toHaveTextContent('send_test_message');
  52. expect(frameFunction[2]).toHaveTextContent('captureMessage');
  53. expect(frameFunction[3]).toHaveTextContent('capture');
  54. expect(frameFunction[4]).toHaveTextContent('build_msg');
  55. });
  56. it('collapse/expand frames by clicking anywhere in the frame element', function () {
  57. renderedComponent({});
  58. // frame list
  59. const frames = screen.getByTestId('frames');
  60. expect(frames.children).toHaveLength(5);
  61. // only one frame is expanded by default
  62. expect(screen.getByTestId('toggle-button-expanded')).toBeInTheDocument();
  63. expect(screen.getAllByTestId('toggle-button-collapsed')).toHaveLength(4);
  64. // clickable list item element
  65. const frameTitles = screen.getAllByTestId('title');
  66. // collapse the expanded frame (by default)
  67. userEvent.click(frameTitles[0]);
  68. // all frames are now collapsed
  69. expect(screen.queryByTestId('toggle-button-expanded')).not.toBeInTheDocument();
  70. expect(screen.getAllByTestId('toggle-button-collapsed')).toHaveLength(5);
  71. // expand penultimate and last frame
  72. userEvent.click(frameTitles[frameTitles.length - 2]);
  73. userEvent.click(frameTitles[frameTitles.length - 1]);
  74. // two frames are now collapsed
  75. expect(screen.getAllByTestId('toggle-button-expanded')).toHaveLength(2);
  76. expect(screen.getAllByTestId('toggle-button-collapsed')).toHaveLength(3);
  77. });
  78. it('collapse/expand frames by clicking on the toggle button', function () {
  79. renderedComponent({});
  80. // frame list
  81. const frames = screen.getByTestId('frames');
  82. expect(frames.children).toHaveLength(5);
  83. const expandedToggleButtons = screen.getByTestId('toggle-button-expanded');
  84. // only one frame is expanded by default
  85. expect(expandedToggleButtons).toBeInTheDocument();
  86. expect(screen.getAllByTestId('toggle-button-collapsed')).toHaveLength(4);
  87. // collapse the expanded frame (by default)
  88. userEvent.click(expandedToggleButtons);
  89. // all frames are now collapsed
  90. expect(screen.queryByTestId('toggle-button-expanded')).not.toBeInTheDocument();
  91. expect(screen.getAllByTestId('toggle-button-collapsed')).toHaveLength(5);
  92. const collapsedToggleButtons = screen.getAllByTestId('toggle-button-collapsed');
  93. // expand penultimate and last frame
  94. userEvent.click(collapsedToggleButtons[collapsedToggleButtons.length - 2]);
  95. userEvent.click(collapsedToggleButtons[collapsedToggleButtons.length - 1]);
  96. // two frames are now collapsed
  97. expect(screen.getAllByTestId('toggle-button-expanded')).toHaveLength(2);
  98. expect(screen.getAllByTestId('toggle-button-collapsed')).toHaveLength(3);
  99. });
  100. it('if all in_app equals false, all the frames are showing by default', function () {
  101. renderedComponent({});
  102. // frame list
  103. const frames = screen.getByTestId('frames');
  104. expect(frames.children).toHaveLength(5);
  105. });
  106. describe('if there is a frame with in_app equal to true, display only in_app frames', function () {
  107. it('displays crashed from only', function () {
  108. const dataFrames = [...data.frames];
  109. const newData = {
  110. ...data,
  111. hasSystemFrames: true,
  112. frames: [
  113. {...dataFrames[0], inApp: true},
  114. ...dataFrames.splice(1, dataFrames.length),
  115. ],
  116. };
  117. renderedComponent({
  118. data: newData,
  119. event: {...event, entries: [{...event.entries[0], stacktrace: newData.frames}]},
  120. includeSystemFrames: false,
  121. });
  122. // clickable list item element
  123. const frameTitles = screen.getAllByTestId('title');
  124. // frame list - in app only
  125. expect(frameTitles).toHaveLength(2);
  126. expect(frameTitles[0]).toHaveTextContent(
  127. 'Crashed in non-app: raven/scripts/runner.py in main at line 112'
  128. );
  129. expect(frameTitles[1]).toHaveTextContent('raven/base.py in build_msg at line 303');
  130. });
  131. it('displays called from only', function () {
  132. const dataFrames = [...data.frames];
  133. const newData = {
  134. ...data,
  135. hasSystemFrames: true,
  136. registers: {},
  137. frames: [
  138. ...dataFrames.splice(0, dataFrames.length - 1),
  139. {...dataFrames[dataFrames.length - 1], inApp: true},
  140. ],
  141. };
  142. renderedComponent({
  143. data: newData,
  144. event: {...event, entries: [{...event.entries[0], stacktrace: newData.frames}]},
  145. includeSystemFrames: false,
  146. });
  147. // clickable list item element
  148. const frameTitles = screen.getAllByTestId('title');
  149. // frame list - in app only
  150. expect(frameTitles).toHaveLength(2);
  151. expect(frameTitles[0]).toHaveTextContent(
  152. 'raven/scripts/runner.py in main at line 112'
  153. );
  154. expect(frameTitles[1]).toHaveTextContent(
  155. 'Called from: raven/scripts/runner.py in send_test_message at line 77'
  156. );
  157. });
  158. it('displays crashed from and called from', function () {
  159. const dataFrames = [...data.frames];
  160. const newData = {
  161. ...data,
  162. hasSystemFrames: true,
  163. frames: [
  164. ...dataFrames.slice(0, 1),
  165. {...dataFrames[1], inApp: true},
  166. ...dataFrames.slice(2, dataFrames.length),
  167. ],
  168. };
  169. renderedComponent({
  170. data: newData,
  171. event: {...event, entries: [{...event.entries[0], stacktrace: newData.frames}]},
  172. includeSystemFrames: false,
  173. });
  174. // clickable list item element
  175. const frameTitles = screen.getAllByTestId('title');
  176. // frame list - in app only
  177. expect(frameTitles).toHaveLength(3);
  178. expect(frameTitles[0]).toHaveTextContent(
  179. 'Crashed in non-app: raven/scripts/runner.py in main at line 112'
  180. );
  181. expect(frameTitles[1]).toHaveTextContent('raven/base.py in capture at line 459');
  182. expect(frameTitles[2]).toHaveTextContent(
  183. 'Called from: raven/base.py in build_msg at line 303'
  184. );
  185. });
  186. });
  187. });