index.tsx 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. import {Fragment} from 'react';
  2. import {RouteComponentProps} from 'react-router';
  3. import styled from '@emotion/styled';
  4. import {
  5. addErrorMessage,
  6. addLoadingMessage,
  7. addSuccessMessage,
  8. } from 'sentry/actionCreators/indicator';
  9. import ExternalLink from 'sentry/components/links/externalLink';
  10. import Pagination from 'sentry/components/pagination';
  11. import {PanelTable} from 'sentry/components/panels';
  12. import SearchBar from 'sentry/components/searchBar';
  13. import {t, tct} from 'sentry/locale';
  14. import {space} from 'sentry/styles/space';
  15. import {Organization, Project, SourceMapsArchive} from 'sentry/types';
  16. import {decodeScalar} from 'sentry/utils/queryString';
  17. import routeTitleGen from 'sentry/utils/routeTitle';
  18. import AsyncView from 'sentry/views/asyncView';
  19. import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader';
  20. import TextBlock from 'sentry/views/settings/components/text/textBlock';
  21. import SourceMapsArchiveRow from './sourceMapsArchiveRow';
  22. type Props = RouteComponentProps<{projectId: string}, {}> & {
  23. organization: Organization;
  24. project: Project;
  25. };
  26. type State = AsyncView['state'] & {
  27. archives: SourceMapsArchive[];
  28. };
  29. class ProjectSourceMaps extends AsyncView<Props, State> {
  30. getTitle() {
  31. const {projectId} = this.props.params;
  32. return routeTitleGen(t('Source Maps'), projectId, false);
  33. }
  34. getDefaultState(): State {
  35. return {
  36. ...super.getDefaultState(),
  37. archives: [],
  38. };
  39. }
  40. getEndpoints(): ReturnType<AsyncView['getEndpoints']> {
  41. return [['archives', this.getArchivesUrl(), {query: {query: this.getQuery()}}]];
  42. }
  43. getArchivesUrl() {
  44. const {organization, project} = this.props;
  45. return `/projects/${organization.slug}/${project.slug}/files/source-maps/`;
  46. }
  47. handleSearch = (query: string) => {
  48. const {location, router} = this.props;
  49. router.push({
  50. ...location,
  51. query: {...location.query, cursor: undefined, query},
  52. });
  53. };
  54. handleDelete = async (name: string) => {
  55. addLoadingMessage(t('Removing artifacts\u2026'));
  56. try {
  57. await this.api.requestPromise(this.getArchivesUrl(), {
  58. method: 'DELETE',
  59. query: {name},
  60. });
  61. this.fetchData();
  62. addSuccessMessage(t('Artifacts removed.'));
  63. } catch {
  64. addErrorMessage(t('Unable to remove artifacts. Please try again.'));
  65. }
  66. };
  67. getQuery() {
  68. const {query} = this.props.location.query;
  69. return decodeScalar(query);
  70. }
  71. getEmptyMessage() {
  72. if (this.getQuery()) {
  73. return t('There are no archives that match your search.');
  74. }
  75. return t('There are no archives for this project.');
  76. }
  77. renderLoading() {
  78. return this.renderBody();
  79. }
  80. renderArchives() {
  81. const {archives} = this.state;
  82. if (!archives.length) {
  83. return null;
  84. }
  85. const {organization, project} = this.props;
  86. return archives.map(a => {
  87. return (
  88. <SourceMapsArchiveRow
  89. key={a.name}
  90. archive={a}
  91. orgId={organization.slug}
  92. projectId={project.slug}
  93. onDelete={this.handleDelete}
  94. />
  95. );
  96. });
  97. }
  98. renderBody() {
  99. const {loading, archives, archivesPageLinks} = this.state;
  100. return (
  101. <Fragment>
  102. <SettingsPageHeader
  103. title={t('Source Maps')}
  104. action={
  105. <SearchBar
  106. placeholder={t('Filter Archives')}
  107. onSearch={this.handleSearch}
  108. query={this.getQuery()}
  109. width="280px"
  110. />
  111. }
  112. />
  113. <TextBlock>
  114. {tct(
  115. `These source map archives help Sentry identify where to look when Javascript is minified. By providing this information, you can get better context for your stack traces when debugging. To learn more about source maps, [link: read the docs].`,
  116. {
  117. link: (
  118. <ExternalLink href="https://docs.sentry.io/platforms/javascript/sourcemaps/" />
  119. ),
  120. }
  121. )}
  122. </TextBlock>
  123. <StyledPanelTable
  124. headers={[
  125. t('Archive'),
  126. <ArtifactsColumn key="artifacts">{t('Artifacts')}</ArtifactsColumn>,
  127. t('Type'),
  128. t('Date Created'),
  129. '',
  130. ]}
  131. emptyMessage={this.getEmptyMessage()}
  132. isEmpty={archives.length === 0}
  133. isLoading={loading}
  134. >
  135. {this.renderArchives()}
  136. </StyledPanelTable>
  137. <Pagination pageLinks={archivesPageLinks} />
  138. </Fragment>
  139. );
  140. }
  141. }
  142. const StyledPanelTable = styled(PanelTable)`
  143. grid-template-columns:
  144. minmax(120px, 1fr) max-content minmax(85px, max-content) minmax(265px, max-content)
  145. 75px;
  146. `;
  147. const ArtifactsColumn = styled('div')`
  148. text-align: right;
  149. padding-right: ${space(1.5)};
  150. margin-right: ${space(0.25)};
  151. `;
  152. export default ProjectSourceMaps;