setupDocsLoader.tsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. import {Fragment, useCallback, useEffect, useState} from 'react';
  2. import styled from '@emotion/styled';
  3. import {motion} from 'framer-motion';
  4. import {Location} from 'history';
  5. import beautify from 'js-beautify';
  6. import {Button} from 'sentry/components/button';
  7. import {CodeSnippet} from 'sentry/components/codeSnippet';
  8. import HookOrDefault from 'sentry/components/hookOrDefault';
  9. import ExternalLink from 'sentry/components/links/externalLink';
  10. import LoadingError from 'sentry/components/loadingError';
  11. import {DocumentationWrapper} from 'sentry/components/onboarding/documentationWrapper';
  12. import {
  13. ProductSelection,
  14. ProductSolution,
  15. } from 'sentry/components/onboarding/productSelection';
  16. import {PlatformKey} from 'sentry/data/platformCategories';
  17. import {IconChevron} from 'sentry/icons';
  18. import {t} from 'sentry/locale';
  19. import {Organization, Project, ProjectKey} from 'sentry/types';
  20. import {trackAnalytics} from 'sentry/utils/analytics';
  21. import {handleXhrErrorResponse} from 'sentry/utils/handleXhrErrorResponse';
  22. import {decodeList} from 'sentry/utils/queryString';
  23. import useApi from 'sentry/utils/useApi';
  24. const ProductSelectionAvailabilityHook = HookOrDefault({
  25. hookName: 'component:product-selection-availability',
  26. defaultComponent: ProductSelection,
  27. });
  28. export function SetupDocsLoader({
  29. organization,
  30. location,
  31. project,
  32. platform,
  33. close,
  34. }: {
  35. close: () => void;
  36. location: Location;
  37. organization: Organization;
  38. platform: PlatformKey | null;
  39. project: Project;
  40. }) {
  41. const api = useApi();
  42. const currentPlatform = platform ?? project?.platform ?? 'other';
  43. const [projectKey, setProjectKey] = useState<ProjectKey | null>(null);
  44. const [hasLoadingError, setHasLoadingError] = useState(false);
  45. const [projectKeyUpdateError, setProjectKeyUpdateError] = useState(false);
  46. const productsQuery =
  47. (location.query.product as ProductSolution | ProductSolution[] | undefined) ?? [];
  48. const products = decodeList(productsQuery) as ProductSolution[];
  49. const fetchData = useCallback(async () => {
  50. const keysApiUrl = `/projects/${organization.slug}/${project.slug}/keys/`;
  51. try {
  52. const loadedKeys = await api.requestPromise(keysApiUrl);
  53. if (loadedKeys.length === 0) {
  54. setHasLoadingError(true);
  55. return;
  56. }
  57. setProjectKey(loadedKeys[0]);
  58. setHasLoadingError(false);
  59. } catch (error) {
  60. setHasLoadingError(error);
  61. throw error;
  62. }
  63. }, [api, organization.slug, project.slug]);
  64. // Automatically update the products on the project key when the user changes the product selection
  65. // Note that on initial visit, this will also update the project key with the default products (=all products)
  66. // This DOES NOT take into account any initial products that may already be set on the project key - they will always be overwritten!
  67. const handleUpdateSelectedProducts = useCallback(async () => {
  68. const keyId = projectKey?.id;
  69. if (!keyId) {
  70. return;
  71. }
  72. const newDynamicSdkLoaderOptions: ProjectKey['dynamicSdkLoaderOptions'] = {
  73. hasPerformance: false,
  74. hasReplay: false,
  75. hasDebug: false,
  76. };
  77. products.forEach(product => {
  78. // eslint-disable-next-line default-case
  79. switch (product) {
  80. case ProductSolution.PERFORMANCE_MONITORING:
  81. newDynamicSdkLoaderOptions.hasPerformance = true;
  82. break;
  83. case ProductSolution.SESSION_REPLAY:
  84. newDynamicSdkLoaderOptions.hasReplay = true;
  85. break;
  86. }
  87. });
  88. try {
  89. await api.requestPromise(
  90. `/projects/${organization.slug}/${project.slug}/keys/${keyId}/`,
  91. {
  92. method: 'PUT',
  93. data: {
  94. dynamicSdkLoaderOptions: newDynamicSdkLoaderOptions,
  95. },
  96. }
  97. );
  98. setProjectKeyUpdateError(false);
  99. } catch (error) {
  100. const message = t('Unable to updated dynamic SDK loader configuration');
  101. handleXhrErrorResponse(message, error);
  102. setProjectKeyUpdateError(true);
  103. }
  104. }, [api, organization.slug, project.slug, projectKey?.id, products]);
  105. const track = useCallback(() => {
  106. if (!project?.id) {
  107. return;
  108. }
  109. trackAnalytics('onboarding.setup_loader_docs_rendered', {
  110. organization,
  111. platform: currentPlatform,
  112. project_id: project?.id,
  113. });
  114. }, [organization, currentPlatform, project?.id]);
  115. useEffect(() => {
  116. fetchData();
  117. }, [fetchData, organization.slug, project.slug]);
  118. useEffect(() => {
  119. handleUpdateSelectedProducts();
  120. }, [handleUpdateSelectedProducts, location.query.product]);
  121. useEffect(() => {
  122. track();
  123. }, [track]);
  124. return (
  125. <Fragment>
  126. <ProductSelectionAvailabilityHook
  127. organization={organization}
  128. lazyLoader
  129. skipLazyLoader={close}
  130. platform={currentPlatform}
  131. />
  132. {projectKeyUpdateError && (
  133. <LoadingError
  134. message={t('Failed to update the project key with the selected products.')}
  135. onRetry={handleUpdateSelectedProducts}
  136. />
  137. )}
  138. {!hasLoadingError ? (
  139. projectKey !== null && (
  140. <ProjectKeyInfo
  141. projectKey={projectKey}
  142. platform={platform}
  143. organization={organization}
  144. project={project}
  145. products={products}
  146. />
  147. )
  148. ) : (
  149. <LoadingError
  150. message={t('Failed to load Client Keys for the project.')}
  151. onRetry={fetchData}
  152. />
  153. )}
  154. </Fragment>
  155. );
  156. }
  157. function ProjectKeyInfo({
  158. projectKey,
  159. platform,
  160. organization,
  161. project,
  162. products,
  163. }: {
  164. organization: Organization;
  165. platform: PlatformKey | null;
  166. products: ProductSolution[];
  167. project: Project;
  168. projectKey: ProjectKey;
  169. }) {
  170. const [showOptionalConfig, setShowOptionalConfig] = useState(false);
  171. const loaderLink = projectKey.dsn.cdn;
  172. const currentPlatform = platform ?? project?.platform ?? 'other';
  173. const hasPerformance = products.includes(ProductSolution.PERFORMANCE_MONITORING);
  174. const hasSessionReplay = products.includes(ProductSolution.SESSION_REPLAY);
  175. const configCodeSnippet = beautify.html(
  176. `<script>
  177. Sentry.onLoad(function() {
  178. Sentry.init({${
  179. !(hasPerformance || hasSessionReplay)
  180. ? `
  181. // You can add any additional configuration here`
  182. : ''
  183. }${
  184. hasPerformance
  185. ? `
  186. // Performance Monitoring
  187. tracesSampleRate: 1.0, // Capture 100% of the transactions, reduce in production!`
  188. : ''
  189. }${
  190. hasSessionReplay
  191. ? `
  192. // Session Replay
  193. replaysSessionSampleRate: 0.1, // This sets the sample rate at 10%. You may want to change it to 100% while in development and then sample at a lower rate in production.
  194. replaysOnErrorSampleRate: 1.0, // If you're not already sampling the entire session, change the sample rate to 100% when sampling sessions where errors occur.`
  195. : ''
  196. }
  197. });
  198. });
  199. </script>`,
  200. {indent_size: 2}
  201. );
  202. const verifyCodeSnippet = beautify.html(
  203. `<script>
  204. myUndefinedFunction();
  205. </script>`,
  206. {indent_size: 2}
  207. );
  208. const toggleOptionalConfiguration = useCallback(() => {
  209. const show = !showOptionalConfig;
  210. setShowOptionalConfig(show);
  211. if (show) {
  212. trackAnalytics('onboarding.js_loader_optional_configuration_shown', {
  213. organization,
  214. platform: currentPlatform,
  215. project_id: project.id,
  216. });
  217. }
  218. }, [organization, project.id, currentPlatform, showOptionalConfig]);
  219. return (
  220. <DocsWrapper>
  221. <DocumentationWrapper>
  222. <h2>{t('Install')}</h2>
  223. <p>{t('Add this script tag to the top of the page:')}</p>
  224. <CodeSnippet dark language="html">
  225. {beautify.html(
  226. `<script src="${loaderLink}" crossorigin="anonymous"></script>`,
  227. {indent_size: 2, wrap_attributes: 'force-expand-multiline'}
  228. )}
  229. </CodeSnippet>
  230. <OptionalConfigWrapper>
  231. <ToggleButton
  232. priority="link"
  233. borderless
  234. size="zero"
  235. icon={<IconChevron direction={showOptionalConfig ? 'down' : 'right'} />}
  236. aria-label={t('Toggle optional configuration')}
  237. onClick={toggleOptionalConfiguration}
  238. />
  239. <h2 onClick={toggleOptionalConfiguration}>{t('Configuration (Optional)')}</h2>
  240. </OptionalConfigWrapper>
  241. {showOptionalConfig && (
  242. <div>
  243. <p>
  244. {t(
  245. "Initialise Sentry as early as possible in your application's lifecycle."
  246. )}
  247. </p>
  248. <CodeSnippet dark language="html">
  249. {configCodeSnippet}
  250. </CodeSnippet>
  251. </div>
  252. )}
  253. <h2>{t('Verify')}</h2>
  254. <p>
  255. {t(
  256. "This snippet contains an intentional error and can be used as a test to make sure that everything's working as expected."
  257. )}
  258. </p>
  259. <CodeSnippet dark language="html">
  260. {verifyCodeSnippet}
  261. </CodeSnippet>
  262. <hr />
  263. <h2>{t('Next Steps')}</h2>
  264. <ul>
  265. <li>
  266. <ExternalLink href="https://docs.sentry.io/platforms/javascript/sourcemaps/">
  267. {t('Source Maps')}
  268. </ExternalLink>
  269. {': '}
  270. {t('Learn how to enable readable stack traces in your Sentry errors.')}
  271. </li>
  272. <li>
  273. <ExternalLink href="https://docs.sentry.io/platforms/javascript/configuration/">
  274. {t('SDK Configuration')}
  275. </ExternalLink>
  276. {': '}
  277. {t('Learn how to configure your SDK using our Loader Script')}
  278. </li>
  279. {!products.includes(ProductSolution.PERFORMANCE_MONITORING) && (
  280. <li>
  281. <ExternalLink href="https://docs.sentry.io/platforms/javascript/performance/">
  282. {t('Performance Monitoring')}
  283. </ExternalLink>
  284. {': '}
  285. {t(
  286. 'Track down transactions to connect the dots between 10-second page loads and poor-performing API calls or slow database queries.'
  287. )}
  288. </li>
  289. )}
  290. {!products.includes(ProductSolution.SESSION_REPLAY) && (
  291. <li>
  292. <ExternalLink href="https://docs.sentry.io/platforms/javascript/session-replay/">
  293. {t('Session Replay')}
  294. </ExternalLink>
  295. {': '}
  296. {t(
  297. 'Get to the root cause of an error or latency issue faster by seeing all the technical details related to that issue in one visual replay on your web application.'
  298. )}
  299. </li>
  300. )}
  301. </ul>
  302. </DocumentationWrapper>
  303. </DocsWrapper>
  304. );
  305. }
  306. const DocsWrapper = styled(motion.div)``;
  307. const OptionalConfigWrapper = styled('div')`
  308. display: flex;
  309. cursor: pointer;
  310. `;
  311. const ToggleButton = styled(Button)`
  312. &,
  313. :hover {
  314. color: ${p => p.theme.gray500};
  315. }
  316. `;