utils.tsx 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. import {ReactText} from 'react';
  2. import {browserHistory} from 'react-router';
  3. import {Location} from 'history';
  4. import {t} from 'app/locale';
  5. import {Organization, Project} from 'app/types';
  6. import EventView from 'app/utils/discover/eventView';
  7. import {
  8. formatAbbreviatedNumber,
  9. formatFloat,
  10. formatPercentage,
  11. getDuration,
  12. } from 'app/utils/formatters';
  13. import {HistogramData} from 'app/utils/performance/histogram/types';
  14. import {decodeScalar} from 'app/utils/queryString';
  15. import {MutableSearch} from 'app/utils/tokenizeSearch';
  16. import {AxisOption, getTermHelp, PERFORMANCE_TERM} from '../data';
  17. import {Rectangle} from '../transactionSummary/transactionVitals/types';
  18. import {platformToPerformanceType, PROJECT_PERFORMANCE_TYPE} from '../utils';
  19. export const LEFT_AXIS_QUERY_KEY = 'left';
  20. export const RIGHT_AXIS_QUERY_KEY = 'right';
  21. type LandingDisplay = {
  22. label: string;
  23. field: LandingDisplayField;
  24. };
  25. export enum LandingDisplayField {
  26. ALL = 'all',
  27. FRONTEND_PAGELOAD = 'frontend_pageload',
  28. FRONTEND_OTHER = 'frontend_other',
  29. BACKEND = 'backend',
  30. MOBILE = 'mobile',
  31. }
  32. export const LANDING_DISPLAYS = [
  33. {
  34. label: 'All Transactions',
  35. field: LandingDisplayField.ALL,
  36. },
  37. {
  38. label: 'Frontend (Pageload)',
  39. field: LandingDisplayField.FRONTEND_PAGELOAD,
  40. },
  41. {
  42. label: 'Frontend (Other)',
  43. field: LandingDisplayField.FRONTEND_OTHER,
  44. },
  45. {
  46. label: 'Backend',
  47. field: LandingDisplayField.BACKEND,
  48. },
  49. {
  50. label: 'Mobile',
  51. field: LandingDisplayField.MOBILE,
  52. isShown: (organization: Organization) =>
  53. organization.features.includes('performance-mobile-vitals'),
  54. },
  55. ];
  56. export function excludeTransaction(
  57. transaction: string | ReactText,
  58. props: {eventView: EventView; location: Location}
  59. ) {
  60. const {eventView, location} = props;
  61. const searchConditions = new MutableSearch(eventView.query);
  62. searchConditions.addFilterValues('!transaction', [`${transaction}`]);
  63. browserHistory.push({
  64. pathname: location.pathname,
  65. query: {
  66. ...location.query,
  67. cursor: undefined,
  68. query: searchConditions.formatString(),
  69. },
  70. });
  71. }
  72. export function getCurrentLandingDisplay(
  73. location: Location,
  74. projects: Project[],
  75. eventView?: EventView
  76. ): LandingDisplay {
  77. const landingField = decodeScalar(location?.query?.landingDisplay);
  78. const display = LANDING_DISPLAYS.find(({field}) => field === landingField);
  79. if (display) {
  80. return display;
  81. }
  82. const defaultDisplayField = getDefaultDisplayFieldForPlatform(projects, eventView);
  83. const defaultDisplay = LANDING_DISPLAYS.find(
  84. ({field}) => field === defaultDisplayField
  85. );
  86. return defaultDisplay || LANDING_DISPLAYS[0];
  87. }
  88. export function handleLandingDisplayChange(field: string, location: Location) {
  89. const newQuery = {...location.query};
  90. delete newQuery[LEFT_AXIS_QUERY_KEY];
  91. delete newQuery[RIGHT_AXIS_QUERY_KEY];
  92. // Transaction op can affect the display and show no results if it is explicitly set.
  93. const query = decodeScalar(location.query.query, '');
  94. const searchConditions = new MutableSearch(query);
  95. searchConditions.removeFilter('transaction.op');
  96. browserHistory.push({
  97. pathname: location.pathname,
  98. query: {
  99. ...newQuery,
  100. query: searchConditions.formatString(),
  101. landingDisplay: field,
  102. },
  103. });
  104. }
  105. export function getChartWidth(chartData: HistogramData, refPixelRect: Rectangle | null) {
  106. const distance = refPixelRect ? refPixelRect.point2.x - refPixelRect.point1.x : 0;
  107. const chartWidth = chartData.length * distance;
  108. return {
  109. chartWidth,
  110. };
  111. }
  112. export function getDefaultDisplayFieldForPlatform(
  113. projects: Project[],
  114. eventView?: EventView
  115. ) {
  116. if (!eventView) {
  117. return LandingDisplayField.ALL;
  118. }
  119. const projectIds = eventView.project;
  120. const performanceTypeToDisplay = {
  121. [PROJECT_PERFORMANCE_TYPE.ANY]: LandingDisplayField.ALL,
  122. [PROJECT_PERFORMANCE_TYPE.FRONTEND]: LandingDisplayField.FRONTEND_PAGELOAD,
  123. [PROJECT_PERFORMANCE_TYPE.BACKEND]: LandingDisplayField.BACKEND,
  124. [PROJECT_PERFORMANCE_TYPE.MOBILE]: LandingDisplayField.MOBILE,
  125. };
  126. const performanceType = platformToPerformanceType(projects, projectIds);
  127. const landingField =
  128. performanceTypeToDisplay[performanceType] ?? LandingDisplayField.ALL;
  129. return landingField;
  130. }
  131. type VitalCardDetail = {
  132. title: string;
  133. tooltip: string;
  134. formatter: (value: number) => string | number;
  135. };
  136. export const vitalCardDetails = (
  137. organization: Organization
  138. ): {[key: string]: VitalCardDetail | undefined} => {
  139. return {
  140. 'p75(transaction.duration)': {
  141. title: t('Duration (p75)'),
  142. tooltip: getTermHelp(organization, PERFORMANCE_TERM.P75),
  143. formatter: value => getDuration(value / 1000, value >= 1000 ? 3 : 0, true),
  144. },
  145. 'tpm()': {
  146. title: t('Throughput'),
  147. tooltip: getTermHelp(organization, PERFORMANCE_TERM.THROUGHPUT),
  148. formatter: formatAbbreviatedNumber,
  149. },
  150. 'failure_rate()': {
  151. title: t('Failure Rate'),
  152. tooltip: getTermHelp(organization, PERFORMANCE_TERM.FAILURE_RATE),
  153. formatter: value => formatPercentage(value, 2),
  154. },
  155. 'apdex()': {
  156. title: t('Apdex'),
  157. tooltip: getTermHelp(organization, PERFORMANCE_TERM.APDEX_NEW),
  158. formatter: value => formatFloat(value, 4),
  159. },
  160. 'p75(measurements.frames_slow_rate)': {
  161. title: t('Slow Frames (p75)'),
  162. tooltip: getTermHelp(organization, PERFORMANCE_TERM.SLOW_FRAMES),
  163. formatter: value => formatPercentage(value, 2),
  164. },
  165. 'p75(measurements.frames_frozen_rate)': {
  166. title: t('Frozen Frames (p75)'),
  167. tooltip: getTermHelp(organization, PERFORMANCE_TERM.FROZEN_FRAMES),
  168. formatter: value => formatPercentage(value, 2),
  169. },
  170. 'p75(measurements.app_start_cold)': {
  171. title: t('Cold Start (p75)'),
  172. tooltip: getTermHelp(organization, PERFORMANCE_TERM.APP_START_COLD),
  173. formatter: value => getDuration(value / 1000, value >= 1000 ? 3 : 0, true),
  174. },
  175. 'p75(measurements.app_start_warm)': {
  176. title: t('Warm Start (p75)'),
  177. tooltip: getTermHelp(organization, PERFORMANCE_TERM.APP_START_WARM),
  178. formatter: value => getDuration(value / 1000, value >= 1000 ? 3 : 0, true),
  179. },
  180. 'p75(measurements.stall_percentage)': {
  181. title: t('Stall Percentage (p75)'),
  182. tooltip: getTermHelp(organization, PERFORMANCE_TERM.STALL_PERCENTAGE),
  183. formatter: value => formatPercentage(value, 2),
  184. },
  185. };
  186. };
  187. export function getDisplayAxes(options: AxisOption[], location: Location) {
  188. const leftDefault = options.find(opt => opt.isLeftDefault) || options[0];
  189. const rightDefault = options.find(opt => opt.isRightDefault) || options[1];
  190. const leftAxis =
  191. options.find(opt => opt.value === location.query[LEFT_AXIS_QUERY_KEY]) || leftDefault;
  192. const rightAxis =
  193. options.find(opt => opt.value === location.query[RIGHT_AXIS_QUERY_KEY]) ||
  194. rightDefault;
  195. return {
  196. leftAxis,
  197. rightAxis,
  198. };
  199. }