From c73b5d99835f6ca456e28ebb76f61d39e8961960 Mon Sep 17 00:00:00 2001 From: debsmita1 Date: Mon, 30 Sep 2024 22:33:32 +0530 Subject: [PATCH] [Bulk import]: fix added repositories count --- plugins/bulk-import/dev/index.tsx | 5 +- plugins/bulk-import/package.json | 1 + .../src/api/BulkImportBackendClient.test.ts | 10 +- .../src/api/BulkImportBackendClient.ts | 6 +- .../AddRepositoriesForm.test.tsx | 2 +- .../AddRepositoriesTable.test.tsx | 2 +- .../AddRepositories/AddRepositoriesTable.tsx | 7 +- .../AddRepositories/OrganizationTableRow.tsx | 2 +- .../OrganizationsColumnHeader.ts | 4 +- .../ReposSelectDrawerColumnHeader.ts | 2 +- .../RepositoriesColumnHeader.ts | 2 +- .../AddRepositories/RepositoriesHeader.tsx | 27 ++- .../AddRepositories/RepositoriesTable.tsx | 3 +- .../AddRepositories/RepositoryTableRow.tsx | 18 +- .../src/components/BulkImportPage.test.tsx | 6 +- .../src/components/BulkImportPage.tsx | 22 ++- .../PreviewFile/PreviewFile.test.tsx | 2 +- .../PreviewPullRequestForm.test.tsx | 2 +- .../PreviewFile/PreviewPullRequests.test.tsx | 2 +- .../AddedRepositoriesTableBody.tsx | 70 +++++++ .../Repositories/AddedRepositoryTableRow.tsx | 80 ++++++++ .../Repositories/CatalogInfoAction.test.tsx | 11 +- .../Repositories/CatalogInfoAction.tsx | 6 +- .../DeleteRepositoryDialog.test.tsx | 98 +++++++--- .../Repositories/DeleteRepositoryDialog.tsx | 51 +++-- .../Repositories/RepositoriesList.test.tsx | 31 +-- .../Repositories/RepositoriesList.tsx | 184 +++++++++++++----- .../Repositories/RepositoriesListColumns.ts | 43 ++++ .../Repositories/RepositoriesListColumns.tsx | 98 ---------- .../Repositories/RepositoriesListToolbar.tsx | 54 +++-- .../src/hooks/useAddedRepositories.test.ts | 12 +- .../src/hooks/useAddedRepositories.ts | 153 ++++++--------- plugins/bulk-import/src/mocks/mockData.ts | 175 +++++++++-------- .../bulk-import/src/types/response-types.ts | 7 + .../src/utils/repository-utils.tsx | 46 +++++ plugins/bulk-import/tests/bulkImport.spec.ts | 2 +- yarn.lock | 2 +- 37 files changed, 792 insertions(+), 456 deletions(-) create mode 100644 plugins/bulk-import/src/components/Repositories/AddedRepositoriesTableBody.tsx create mode 100644 plugins/bulk-import/src/components/Repositories/AddedRepositoryTableRow.tsx create mode 100644 plugins/bulk-import/src/components/Repositories/RepositoriesListColumns.ts delete mode 100644 plugins/bulk-import/src/components/Repositories/RepositoriesListColumns.tsx diff --git a/plugins/bulk-import/dev/index.tsx b/plugins/bulk-import/dev/index.tsx index e2c20a7bd54..002bb20e26c 100644 --- a/plugins/bulk-import/dev/index.tsx +++ b/plugins/bulk-import/dev/index.tsx @@ -27,6 +27,7 @@ import { BulkImportPage, bulkImportPlugin } from '../src/plugin'; import { APITypes, ImportJobResponse, + ImportJobs, ImportJobStatus, OrgAndRepoResponse, RepositoryStatus, @@ -78,7 +79,7 @@ class MockBulkImportApi implements BulkImportAPI { _page: number, _size: number, _seachString: string, - ): Promise { + ): Promise { return mockGetImportJobs; } @@ -112,7 +113,7 @@ class MockBulkImportApi implements BulkImportAPI { repo: string, _defaultBranch: string, ): Promise { - return mockGetImportJobs.find( + return mockGetImportJobs.imports.find( i => i.repository.url === repo, ) as ImportJobStatus; } diff --git a/plugins/bulk-import/package.json b/plugins/bulk-import/package.json index d9116d5db41..8c0ce0b1362 100644 --- a/plugins/bulk-import/package.json +++ b/plugins/bulk-import/package.json @@ -48,6 +48,7 @@ "@material-ui/lab": "^4.0.0-alpha.61", "@mui/icons-material": "5.16.4", "@mui/material": "^5.12.2", + "@tanstack/react-query": "^4.29.21", "formik": "^2.4.5", "js-yaml": "^4.1.0", "lodash": "^4.17.21", diff --git a/plugins/bulk-import/src/api/BulkImportBackendClient.test.ts b/plugins/bulk-import/src/api/BulkImportBackendClient.test.ts index 4eb62066770..e2dba59c00c 100644 --- a/plugins/bulk-import/src/api/BulkImportBackendClient.test.ts +++ b/plugins/bulk-import/src/api/BulkImportBackendClient.test.ts @@ -90,7 +90,7 @@ const handlers = [ (req, res, ctx) => { const test = req.headers.get('Content-Type'); if (test === 'application/json') { - return res(ctx.status(200), ctx.json(mockGetImportJobs[1])); + return res(ctx.status(200), ctx.json(mockGetImportJobs.imports[1])); } return res(ctx.status(404)); }, @@ -103,7 +103,7 @@ const handlers = [ return res( ctx.status(200), ctx.json( - mockGetImportJobs.filter(r => + mockGetImportJobs.imports.filter(r => r.repository.name?.includes(searchParam), ), ), @@ -288,7 +288,9 @@ describe('BulkImportBackendClient', () => { it('getImportJobs should retrieve the import jobs based on search string', async () => { const jobs = await bulkImportApi.getImportJobs(1, 2, 'cup'); expect(jobs).toEqual( - mockGetImportJobs.filter(r => r.repository.name?.includes('cup')), + mockGetImportJobs.imports.filter(r => + r.repository.name?.includes('cup'), + ), ); }); @@ -316,7 +318,7 @@ describe('BulkImportBackendClient', () => { ); expect(response.status).toBe(RepositoryStatus.WAIT_PR_APPROVAL); - expect(response).toEqual(mockGetImportJobs[1]); + expect(response).toEqual(mockGetImportJobs.imports[1]); }); }); diff --git a/plugins/bulk-import/src/api/BulkImportBackendClient.ts b/plugins/bulk-import/src/api/BulkImportBackendClient.ts index 7c588cb082c..e6924bf6f39 100644 --- a/plugins/bulk-import/src/api/BulkImportBackendClient.ts +++ b/plugins/bulk-import/src/api/BulkImportBackendClient.ts @@ -8,6 +8,7 @@ import { APITypes, CreateImportJobRepository, ImportJobResponse, + ImportJobs, ImportJobStatus, OrgAndRepoResponse, } from '../types'; @@ -25,7 +26,7 @@ export type BulkImportAPI = { page: number, size: number, searchString: string, - ) => Promise; + ) => Promise; createImportJobs: ( importRepositories: CreateImportJobRepository[], dryRun?: boolean, @@ -87,11 +88,12 @@ export class BulkImportBackendClient implements BulkImportAPI { const { token: idToken } = await this.identityApi.getCredentials(); const backendUrl = this.configApi.getString('backend.baseUrl'); const jsonResponse = await fetch( - `${backendUrl}/api/bulk-import/imports?pagePerIntegration=${page}&sizePerIntegration=${size}&search=${searchString}`, + `${backendUrl}/api/bulk-import/imports?page=${page}&size=${size}&search=${searchString}`, { headers: { 'Content-Type': 'application/json', ...(idToken && { Authorization: `Bearer ${idToken}` }), + 'api-version': 'v2', }, }, ); diff --git a/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesForm.test.tsx b/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesForm.test.tsx index 56867cf6cb4..ca25db78100 100644 --- a/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesForm.test.tsx +++ b/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesForm.test.tsx @@ -49,7 +49,7 @@ class MockBulkImportApi { repo: string, _defaultBranch: string, ): Promise { - return mockGetImportJobs.find( + return mockGetImportJobs.imports.find( i => i.repository.url === repo, ) as ImportJobStatus; } diff --git a/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesTable.test.tsx b/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesTable.test.tsx index bc7c52cf8ee..a84cd1021d0 100644 --- a/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesTable.test.tsx +++ b/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesTable.test.tsx @@ -44,7 +44,7 @@ class MockBulkImportApi { repo: string, _defaultBranch: string, ): Promise { - return mockGetImportJobs.find( + return mockGetImportJobs.imports.find( i => i.repository.url === repo, ) as ImportJobStatus; } diff --git a/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesTable.tsx b/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesTable.tsx index 22e007f66b0..8f1eb6f8817 100644 --- a/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesTable.tsx +++ b/plugins/bulk-import/src/components/AddRepositories/AddRepositoriesTable.tsx @@ -11,13 +11,16 @@ export const AddRepositoriesTable = ({ title }: { title: string }) => { const { values } = useFormikContext(); const [searchString, setSearchString] = React.useState(''); const [page, setPage] = React.useState(0); - + const handleSearch = (str: string) => { + setSearchString(str); + setPage(0); + }; return ( {values.repositoryType === RepositorySelection.Repository ? ( diff --git a/plugins/bulk-import/src/components/AddRepositories/OrganizationTableRow.tsx b/plugins/bulk-import/src/components/AddRepositories/OrganizationTableRow.tsx index 98a4c6bddf3..e0fd32c59b5 100644 --- a/plugins/bulk-import/src/components/AddRepositories/OrganizationTableRow.tsx +++ b/plugins/bulk-import/src/components/AddRepositories/OrganizationTableRow.tsx @@ -31,7 +31,7 @@ export const OrganizationTableRow = ({ ).length; return ( - + {data.orgName} diff --git a/plugins/bulk-import/src/components/AddRepositories/OrganizationsColumnHeader.ts b/plugins/bulk-import/src/components/AddRepositories/OrganizationsColumnHeader.ts index 3ed3db84e0d..901c2c02291 100644 --- a/plugins/bulk-import/src/components/AddRepositories/OrganizationsColumnHeader.ts +++ b/plugins/bulk-import/src/components/AddRepositories/OrganizationsColumnHeader.ts @@ -8,12 +8,12 @@ export const OrganizationsColumnHeader: TableColumn[] = [ }, { id: 'url', title: 'URL', field: 'organizationUrl' }, { - id: 'selectedRepositories', + id: 'selected-repositories', title: 'Selected repositories', field: 'selectedRepositories', }, { - id: 'catalogInfoYaml', + id: 'cataloginfoyaml', title: 'catalog-info.yaml', field: 'catalogInfoYaml.status', }, diff --git a/plugins/bulk-import/src/components/AddRepositories/ReposSelectDrawerColumnHeader.ts b/plugins/bulk-import/src/components/AddRepositories/ReposSelectDrawerColumnHeader.ts index 640f2cee2d3..98e7923db15 100644 --- a/plugins/bulk-import/src/components/AddRepositories/ReposSelectDrawerColumnHeader.ts +++ b/plugins/bulk-import/src/components/AddRepositories/ReposSelectDrawerColumnHeader.ts @@ -12,7 +12,7 @@ export const ReposSelectDrawerColumnHeader: TableColumn[] = [ field: 'repoUrl', }, { - id: 'catalogInfoYaml', + id: 'cataloginfoyaml', title: '', field: 'catalogInfoYaml.status', }, diff --git a/plugins/bulk-import/src/components/AddRepositories/RepositoriesColumnHeader.ts b/plugins/bulk-import/src/components/AddRepositories/RepositoriesColumnHeader.ts index e3b3757363e..89b38bfabe3 100644 --- a/plugins/bulk-import/src/components/AddRepositories/RepositoriesColumnHeader.ts +++ b/plugins/bulk-import/src/components/AddRepositories/RepositoriesColumnHeader.ts @@ -17,7 +17,7 @@ export const RepositoriesColumnHeader: TableColumn[] = [ field: 'organizationUrl', }, { - id: 'catalogInfoYaml', + id: 'cataloginfoyaml', title: 'catalog-info.yaml', field: 'catalogInfoYaml.status', }, diff --git a/plugins/bulk-import/src/components/AddRepositories/RepositoriesHeader.tsx b/plugins/bulk-import/src/components/AddRepositories/RepositoriesHeader.tsx index ceba8f88e2d..df96fcfea20 100644 --- a/plugins/bulk-import/src/components/AddRepositories/RepositoriesHeader.tsx +++ b/plugins/bulk-import/src/components/AddRepositories/RepositoriesHeader.tsx @@ -9,6 +9,7 @@ import { } from '@material-ui/core'; import { Order } from '../../types'; +import { RepositoriesListColumns } from '../Repositories/RepositoriesListColumns'; import { OrganizationsColumnHeader } from './OrganizationsColumnHeader'; import { RepositoriesColumnHeader } from './RepositoriesColumnHeader'; import { ReposSelectDrawerColumnHeader } from './ReposSelectDrawerColumnHeader'; @@ -22,15 +23,17 @@ export const RepositoriesHeader = ({ onRequestSort, isDataLoading, showOrganizations, + showImportJobs, isRepoSelectDrawer = false, }: { - numSelected: number; + numSelected?: number; onRequestSort: (event: React.MouseEvent, property: any) => void; order: Order; orderBy: string | undefined; - rowCount: number; + rowCount?: number; isDataLoading?: boolean; showOrganizations?: boolean; + showImportJobs?: boolean; isRepoSelectDrawer?: boolean; onSelectAllClick?: (event: React.ChangeEvent) => void; }) => { @@ -43,6 +46,9 @@ export const RepositoriesHeader = ({ if (showOrganizations) { return OrganizationsColumnHeader; } + if (showImportJobs) { + return RepositoriesListColumns; + } if (isRepoSelectDrawer) { return ReposSelectDrawerColumnHeader; } @@ -52,7 +58,7 @@ export const RepositoriesHeader = ({ return ( - {getColumnHeader().map((headCell, index) => ( + {getColumnHeader()?.map((headCell, index) => ( - {index === 0 && !showOrganizations && ( + {index === 0 && !showOrganizations && !showImportJobs && ( 0 && numSelected < rowCount} - checked={rowCount > 0 && numSelected === rowCount} + indeterminate={ + (numSelected && + rowCount && + numSelected > 0 && + numSelected < rowCount) || + false + } + checked={ + ((rowCount ?? 0) > 0 && numSelected === rowCount) || false + } onChange={onSelectAllClick} inputProps={{ 'aria-label': 'select all repositories', @@ -85,6 +99,7 @@ export const RepositoriesHeader = ({ active={orderBy === headCell.field} direction={orderBy === headCell.field ? order : 'asc'} onClick={createSortHandler(headCell.field)} + disabled={headCell.sorting === false} > {headCell.title} diff --git a/plugins/bulk-import/src/components/AddRepositories/RepositoriesTable.tsx b/plugins/bulk-import/src/components/AddRepositories/RepositoriesTable.tsx index 51433b06aa2..83e241f0fc6 100644 --- a/plugins/bulk-import/src/components/AddRepositories/RepositoriesTable.tsx +++ b/plugins/bulk-import/src/components/AddRepositories/RepositoriesTable.tsx @@ -39,7 +39,7 @@ export const RepositoriesTable = ({ drawerOrganization?: string; updateSelectedReposInDrawer?: (repos: AddedRepositories) => void; }) => { - const { setFieldValue, values } = + const { setFieldValue, values, setStatus } = useFormikContext(); const [order, setOrder] = React.useState('asc'); const [orderBy, setOrderBy] = React.useState(); @@ -200,6 +200,7 @@ export const RepositoriesTable = ({ const updateSelection = (newSelected: AddedRepositories) => { setSelected(newSelected); + setStatus(null); if (drawerOrganization && updateSelectedReposInDrawer) { // Update in the context of the drawer diff --git a/plugins/bulk-import/src/components/AddRepositories/RepositoryTableRow.tsx b/plugins/bulk-import/src/components/AddRepositories/RepositoryTableRow.tsx index a8259ff947d..583e9815f34 100644 --- a/plugins/bulk-import/src/components/AddRepositories/RepositoryTableRow.tsx +++ b/plugins/bulk-import/src/components/AddRepositories/RepositoryTableRow.tsx @@ -72,23 +72,17 @@ export const RepositoryTableRow = ({ - <> - {urlHelper(data?.repoUrl || '')} - - + {urlHelper(data?.repoUrl || '')} + {!isDrawer && ( - <> - {urlHelper(data?.organizationUrl || '')} - - + {urlHelper(data?.organizationUrl || '')} + )} diff --git a/plugins/bulk-import/src/components/BulkImportPage.test.tsx b/plugins/bulk-import/src/components/BulkImportPage.test.tsx index fff39ebc078..c66a66ff2b3 100644 --- a/plugins/bulk-import/src/components/BulkImportPage.test.tsx +++ b/plugins/bulk-import/src/components/BulkImportPage.test.tsx @@ -38,12 +38,12 @@ describe('BulkImport Page', () => { mockUsePermission.mockReturnValue({ loading: false, allowed: true }); mockUseAddedRepositories.mockReturnValue({ loaded: true, - data: [], - retry: jest.fn(), + data: { addedRepositories: [], totalJobs: 0 }, + refetch: jest.fn(), error: undefined, }); await renderInTestApp(); - expect(screen.getByText('Added repositories (0)')).toBeInTheDocument(); + expect(screen.getByText('Added repositories')).toBeInTheDocument(); }); it('should not render if user is not authorized to access the bulk import plugin', async () => { diff --git a/plugins/bulk-import/src/components/BulkImportPage.tsx b/plugins/bulk-import/src/components/BulkImportPage.tsx index 18970457e4b..1f74fa4fb9e 100644 --- a/plugins/bulk-import/src/components/BulkImportPage.tsx +++ b/plugins/bulk-import/src/components/BulkImportPage.tsx @@ -5,6 +5,7 @@ import { usePermission } from '@backstage/plugin-permission-react'; import { Alert, AlertTitle } from '@material-ui/lab'; import FormControl from '@mui/material/FormControl'; +import { QueryClient, QueryClientProvider } from '@tanstack/react-query'; import { Formik } from 'formik'; import { bulkImportPermission } from '@janus-idp/backstage-plugin-bulk-import-common'; @@ -27,6 +28,7 @@ export const BulkImportPage = () => { excludedRepositories: {}, approvalTool: ApprovalTool.Git, }; + const queryClient = new QueryClient(); const bulkImportViewPermissionResult = usePermission({ permission: bulkImportPermission, @@ -39,15 +41,17 @@ export const BulkImportPage = () => { } if (bulkImportViewPermissionResult.allowed) { return ( - {}} - > - - - - + + {}} + > + + + + + ); } return ( diff --git a/plugins/bulk-import/src/components/PreviewFile/PreviewFile.test.tsx b/plugins/bulk-import/src/components/PreviewFile/PreviewFile.test.tsx index b236ac89d98..86032c9c09f 100644 --- a/plugins/bulk-import/src/components/PreviewFile/PreviewFile.test.tsx +++ b/plugins/bulk-import/src/components/PreviewFile/PreviewFile.test.tsx @@ -45,7 +45,7 @@ class MockBulkImportApi { repo: string, _defaultBranch: string, ): Promise { - return mockGetImportJobs.find( + return mockGetImportJobs.imports.find( i => i.repository.url === repo, ) as ImportJobStatus; } diff --git a/plugins/bulk-import/src/components/PreviewFile/PreviewPullRequestForm.test.tsx b/plugins/bulk-import/src/components/PreviewFile/PreviewPullRequestForm.test.tsx index be133a4a5cd..75c4420de19 100644 --- a/plugins/bulk-import/src/components/PreviewFile/PreviewPullRequestForm.test.tsx +++ b/plugins/bulk-import/src/components/PreviewFile/PreviewPullRequestForm.test.tsx @@ -44,7 +44,7 @@ class MockBulkImportApi { repo: string, _defaultBranch: string, ): Promise { - return mockGetImportJobs.find( + return mockGetImportJobs.imports.find( i => i.repository.url === repo, ) as ImportJobStatus; } diff --git a/plugins/bulk-import/src/components/PreviewFile/PreviewPullRequests.test.tsx b/plugins/bulk-import/src/components/PreviewFile/PreviewPullRequests.test.tsx index 7ca7835857b..4cae6df03ed 100644 --- a/plugins/bulk-import/src/components/PreviewFile/PreviewPullRequests.test.tsx +++ b/plugins/bulk-import/src/components/PreviewFile/PreviewPullRequests.test.tsx @@ -44,7 +44,7 @@ class MockBulkImportApi { repo: string, _defaultBranch: string, ): Promise { - return mockGetImportJobs.find( + return mockGetImportJobs.imports.find( i => i.repository.url === repo, ) as ImportJobStatus; } diff --git a/plugins/bulk-import/src/components/Repositories/AddedRepositoriesTableBody.tsx b/plugins/bulk-import/src/components/Repositories/AddedRepositoriesTableBody.tsx new file mode 100644 index 00000000000..1d382a08e01 --- /dev/null +++ b/plugins/bulk-import/src/components/Repositories/AddedRepositoriesTableBody.tsx @@ -0,0 +1,70 @@ +import * as React from 'react'; + +import { makeStyles, TableBody, TableCell, TableRow } from '@material-ui/core'; +import CircularProgress from '@mui/material/CircularProgress'; + +import { AddRepositoryData } from '../../types'; +import { AddedRepositoryTableRow } from './AddedRepositoryTableRow'; +import { RepositoriesListColumns } from './RepositoriesListColumns'; + +const useStyles = makeStyles(theme => ({ + empty: { + padding: theme.spacing(2), + display: 'flex', + justifyContent: 'center', + }, +})); + +export const AddedRepositoriesTableBody = ({ + loading, + rows, + emptyRows, +}: { + loading: boolean; + emptyRows: number; + rows: AddRepositoryData[]; +}) => { + const classes = useStyles(); + + if (loading) { + return ( + + + +
+ +
+ + + + ); + } else if (rows?.length > 0) { + return ( + + {rows.map(row => { + return ; + })} + {emptyRows > 0 && ( + + + + )} + + ); + } + return ( + + + +
+ No records found +
+ + + + ); +}; diff --git a/plugins/bulk-import/src/components/Repositories/AddedRepositoryTableRow.tsx b/plugins/bulk-import/src/components/Repositories/AddedRepositoryTableRow.tsx new file mode 100644 index 00000000000..e2e1caf0171 --- /dev/null +++ b/plugins/bulk-import/src/components/Repositories/AddedRepositoryTableRow.tsx @@ -0,0 +1,80 @@ +import * as React from 'react'; + +import { Link } from '@backstage/core-components'; + +import { makeStyles, TableCell, TableRow } from '@material-ui/core'; +import OpenInNewIcon from '@mui/icons-material/OpenInNew'; +import { useFormikContext } from 'formik'; + +import { AddRepositoriesFormValues, AddRepositoryData } from '../../types'; +import { + calculateLastUpdated, + getImportStatus, + urlHelper, +} from '../../utils/repository-utils'; +import CatalogInfoAction from './CatalogInfoAction'; +import DeleteRepository from './DeleteRepository'; +import SyncRepository from './SyncRepository'; + +const useStyles = makeStyles(() => ({ + tableCellStyle: { + lineHeight: '1.5rem', + fontSize: '0.875rem', + }, +})); + +const ImportStatus = ({ data }: { data: AddRepositoryData }) => { + const { values } = useFormikContext(); + return getImportStatus( + values.repositories?.[data.id]?.catalogInfoYaml?.status as string, + true, + ); +}; + +const LastUpdated = ({ data }: { data: AddRepositoryData }) => { + const { values } = useFormikContext(); + return calculateLastUpdated( + values.repositories?.[data.id]?.catalogInfoYaml?.lastUpdated || '', + ); +}; + +export const AddedRepositoryTableRow = ({ + data, +}: { + data: AddRepositoryData; +}) => { + const classes = useStyles(); + + return ( + + + {data.repoName} + + + + {urlHelper(data?.repoUrl || '')} + + + + + + {urlHelper(data?.organizationUrl || '')} + + + + + + + + + + + + + + + + + + ); +}; diff --git a/plugins/bulk-import/src/components/Repositories/CatalogInfoAction.test.tsx b/plugins/bulk-import/src/components/Repositories/CatalogInfoAction.test.tsx index 91fddeec0ec..1bc0b7758a1 100644 --- a/plugins/bulk-import/src/components/Repositories/CatalogInfoAction.test.tsx +++ b/plugins/bulk-import/src/components/Repositories/CatalogInfoAction.test.tsx @@ -51,7 +51,7 @@ describe('CatalogInfoAction', () => { it('should allow users to edit the catalog-info.yaml PR if the PR is waiting to be approved', async () => { mockUseAsync.mockReturnValue({ loading: false, - value: mockGetImportJobs[0], + value: mockGetImportJobs.imports[0], }); mockUsePermission.mockReturnValue({ loading: false, allowed: true }); @@ -62,7 +62,7 @@ describe('CatalogInfoAction', () => { values: { repositories: { ['org/dessert/cupcake']: { - ...mockGetImportJobs[0], + ...mockGetImportJobs.imports[0], catalogInfoYaml: { status: RepositoryStatus.WAIT_PR_APPROVAL, }, @@ -90,7 +90,10 @@ describe('CatalogInfoAction', () => { it('should allow users to view the catalog-info.yaml if the entity is registered', async () => { mockUseAsync.mockReturnValue({ loading: false, - value: { ...mockGetImportJobs[0], status: RepositoryStatus.ADDED }, + value: { + ...mockGetImportJobs.imports[0], + status: RepositoryStatus.ADDED, + }, }); mockUsePermission.mockReturnValue({ loading: false, allowed: true }); @@ -101,7 +104,7 @@ describe('CatalogInfoAction', () => { values: { repositories: { ['org/dessert/cupcake']: { - ...mockGetImportJobs[0], + ...mockGetImportJobs.imports[0], catalogInfoYaml: { status: RepositoryStatus.ADDED, }, diff --git a/plugins/bulk-import/src/components/Repositories/CatalogInfoAction.tsx b/plugins/bulk-import/src/components/Repositories/CatalogInfoAction.tsx index ea699939c7a..2bef9b2627d 100644 --- a/plugins/bulk-import/src/components/Repositories/CatalogInfoAction.tsx +++ b/plugins/bulk-import/src/components/Repositories/CatalogInfoAction.tsx @@ -90,8 +90,6 @@ const CatalogInfoAction = ({ data }: { data: AddRepositoryData }) => { if (Object.keys(drawerData || {}).length === 0) { setDrawerData(value as ImportJobStatus); } - } else { - removeQueryParams(); } } // eslint-disable-next-line react-hooks/exhaustive-deps @@ -123,9 +121,9 @@ const CatalogInfoAction = ({ data }: { data: AddRepositoryData }) => { ({ useApi: jest.fn(), })); +const createTestQueryClient = () => + new QueryClient({ + defaultOptions: { + queries: { + retry: false, // Disable retries for testing + }, + }, + }); +let queryClient: QueryClient; +beforeEach(() => { + queryClient = createTestQueryClient(); +}); + describe('DeleteRepositoryDialog', () => { it('renders delete repository dialog correctly', () => { render( - , + + + , ); expect( screen.queryByText(/Remove cupcake repository?/i), @@ -30,40 +45,75 @@ describe('DeleteRepositoryDialog', () => { }); it('does not render when not open', () => { - const { queryByText } = render( - , + render( + + + , + ); + expect( + screen.queryByText(/Remove cupcake repository?/i), + ).not.toBeInTheDocument(); + }); + + it('should show an error if repository url is missing', async () => { + const repo = { + ...mockGetRepositories.repositories[0], + repoUrl: '', + url: '', + }; + + render( + + + , ); - expect(queryByText(/Remove cupcake repository?/i)).not.toBeInTheDocument(); + expect( + screen.queryByText(/Remove cupcake repository?/i), + ).toBeInTheDocument(); + const deleteButton = screen.getByText('Remove'); + fireEvent.click(deleteButton); + await waitFor(() => { + expect( + screen.queryByText( + /Unable to remove repository. Repository URL missing./i, + ), + ).toBeInTheDocument(); + }); }); it('shows an error when the deletion fails', async () => { const mockDeleteRepository = jest .fn() - .mockResolvedValue({ error: { message: 'Error occured' } }); + .mockReturnValue({ err: 'Error occured' }); const useApiMock = useApi as jest.Mock; useApiMock.mockReturnValue({ deleteImportAction: mockDeleteRepository, }); - - const user = userEvent.setup(); render( - , + + + , ); const deleteButton = screen.getByText('Remove'); - await user.click(deleteButton); + fireEvent.click(deleteButton); await waitFor(() => { expect( - screen.queryByText(/Unable to remove repository. Error occured/i), + screen.queryByText('Unable to remove repository. Error occured'), ).toBeInTheDocument(); + expect(deleteButton).toBeDisabled(); }); }); }); diff --git a/plugins/bulk-import/src/components/Repositories/DeleteRepositoryDialog.tsx b/plugins/bulk-import/src/components/Repositories/DeleteRepositoryDialog.tsx index fb454ceaf5c..afbf71d5aae 100644 --- a/plugins/bulk-import/src/components/Repositories/DeleteRepositoryDialog.tsx +++ b/plugins/bulk-import/src/components/Repositories/DeleteRepositoryDialog.tsx @@ -15,10 +15,11 @@ import DialogContent from '@mui/material/DialogContent'; import DialogTitle from '@mui/material/DialogTitle'; import IconButton from '@mui/material/IconButton'; import Typography from '@mui/material/Typography'; +import { useMutation } from '@tanstack/react-query'; import { get } from 'lodash'; import { bulkImportApiRef } from '../../api/BulkImportBackendClient'; -import { AddRepositoryData } from '../../types'; +import { AddRepositoryData, ImportJobStatus } from '../../types'; const useStyles = makeStyles((theme: Theme) => createStyles({ @@ -69,27 +70,33 @@ const DeleteRepositoryDialog = ({ closeDialog: () => void; }) => { const classes = useStyles(); - const [error, setError] = React.useState(''); - const [isSubmitting, setIsSubmitting] = React.useState(false); + const [deleteError, setDeleteError] = React.useState(''); const bulkImportApi = useApi(bulkImportApiRef); + const deleteRepository = async (deleteRepo: AddRepositoryData) => { + return await bulkImportApi.deleteImportAction( + deleteRepo.repoUrl || '', + deleteRepo.defaultBranch || 'main', + ); + }; + const mutationDelete = useMutation(deleteRepository, { + onSuccess: (data: ImportJobStatus | Response) => { + if (get(data, 'err')) { + setDeleteError(`Unable to remove repository. ${get(data, 'err')}`); + } else { + closeDialog(); + } + }, + onError: (error: Error) => { + setDeleteError(`Unable to remove repository. ${error.message}`); + }, + }); const handleClickRemove = async () => { - setIsSubmitting(true); if (!repository.repoUrl || !repository?.defaultBranch) { - setIsSubmitting(false); - setError( - `Unable to remove repository. ${!repository?.repoUrl ? 'Repository URL missing' : 'Repository default branch is missing'}`, + setDeleteError( + `Unable to remove repository. ${!repository?.repoUrl ? 'Repository URL missing.' : 'Repository default branch is missing.'}`, ); } else { - const value = await bulkImportApi.deleteImportAction( - repository.repoUrl, - repository.defaultBranch, - ); - setIsSubmitting(false); - if (get(value, 'error')) { - setError(`Unable to remove repository. ${get(value, 'error.message')}`); - } else { - closeDialog(); - } + mutationDelete.mutate(repository); } }; @@ -127,9 +134,9 @@ const DeleteRepositoryDialog = ({ Catalog page. - {error && ( + {deleteError && ( - {error} + {deleteError} )} @@ -137,9 +144,11 @@ const DeleteRepositoryDialog = ({ variant="contained" className={`${classes.deleteButton} ${classes.button}`} onClick={() => handleClickRemove()} - disabled={isSubmitting} + disabled={mutationDelete.isLoading || !!deleteError} startIcon={ - isSubmitting && + mutationDelete.isLoading && ( + + ) } > Remove diff --git a/plugins/bulk-import/src/components/Repositories/RepositoriesList.test.tsx b/plugins/bulk-import/src/components/Repositories/RepositoriesList.test.tsx index 203d5598a2a..2f780457110 100644 --- a/plugins/bulk-import/src/components/Repositories/RepositoriesList.test.tsx +++ b/plugins/bulk-import/src/components/Repositories/RepositoriesList.test.tsx @@ -8,7 +8,7 @@ import { render, screen } from '@testing-library/react'; import { useFormikContext } from 'formik'; import { useAddedRepositories } from '../../hooks'; -import { mockGetImportJobs } from '../../mocks/mockData'; +import { mockGetImportJobs, mockGetRepositories } from '../../mocks/mockData'; import { RepositoriesList } from './RepositoriesList'; jest.mock('react', () => ({ @@ -86,10 +86,13 @@ jest.mock('./RepositoriesListColumns', () => ({ const mockAsyncData = { loaded: true, - data: mockGetImportJobs, + data: { + addedRepositories: mockGetImportJobs.imports, + totalJobs: mockGetImportJobs.imports.length, + }, totalCount: 1, error: undefined, - retry: jest.fn(), + refetch: jest.fn(), }; const mockUseAddedRepositories = useAddedRepositories as jest.MockedFunction< @@ -105,6 +108,7 @@ describe('RepositoriesList', () => { (useFormikContext as jest.Mock).mockReturnValue({ status: null, setFieldValue: jest.fn(), + values: mockGetRepositories.repositories, }); mockUseAddedRepositories.mockReturnValue(mockAsyncData); render( @@ -119,12 +123,7 @@ describe('RepositoriesList', () => { expect( screen.getByText('Added repositories (4)', { exact: false }), ).toBeInTheDocument(); - expect(screen.getByText('Name')).toBeInTheDocument(); - expect(screen.getByText('Repo URL')).toBeInTheDocument(); - expect(screen.getByText('Organization')).toBeInTheDocument(); - expect(screen.getByText('Status')).toBeInTheDocument(); - expect(screen.getByText('Last updated')).toBeInTheDocument(); - expect(screen.getByText('Actions')).toBeInTheDocument(); + expect(screen.getByTestId('import-jobs')).toBeInTheDocument(); }); it('should render the component and display empty content when no data', async () => { @@ -132,7 +131,10 @@ describe('RepositoriesList', () => { status: null, setFieldValue: jest.fn(), }); - mockUseAddedRepositories.mockReturnValue({ ...mockAsyncData, data: [] }); + mockUseAddedRepositories.mockReturnValue({ + ...mockAsyncData, + data: { addedRepositories: [], totalJobs: 0 }, + }); render( @@ -142,9 +144,9 @@ describe('RepositoriesList', () => { ); expect( - screen.getByText('Added repositories (0)', { exact: false }), + screen.getByText('Added repositories', { exact: false }), ).toBeInTheDocument(); - const emptyMessage = screen.getByTestId('added-repositories-table-empty'); + const emptyMessage = screen.getByTestId('no-import-jobs-found'); expect(emptyMessage).toBeInTheDocument(); expect(emptyMessage).toHaveTextContent('No records found'); }); @@ -157,7 +159,10 @@ describe('RepositoriesList', () => { }, setFieldValue: jest.fn(), }); - mockUseAddedRepositories.mockReturnValue({ ...mockAsyncData, data: [] }); + mockUseAddedRepositories.mockReturnValue({ + ...mockAsyncData, + data: { addedRepositories: [], totalJobs: 0 }, + }); render( diff --git a/plugins/bulk-import/src/components/Repositories/RepositoriesList.tsx b/plugins/bulk-import/src/components/Repositories/RepositoriesList.tsx index dd290593c4f..27cb657ba4e 100644 --- a/plugins/bulk-import/src/components/Repositories/RepositoriesList.tsx +++ b/plugins/bulk-import/src/components/Repositories/RepositoriesList.tsx @@ -1,48 +1,49 @@ import React from 'react'; import { useLocation, useNavigate } from 'react-router-dom'; -import { ErrorPage, Table } from '@backstage/core-components'; +import { ErrorPage } from '@backstage/core-components'; -import { makeStyles } from '@material-ui/core'; +import { + Box, + Paper, + Table, + TableContainer, + TablePagination, +} from '@material-ui/core'; import { useDeleteDialog, useDrawer } from '@janus-idp/shared-react'; import { useAddedRepositories } from '../../hooks/useAddedRepositories'; -import { AddRepositoryData } from '../../types'; +import { AddRepositoryData, Order } from '../../types'; +import { getComparator } from '../../utils/repository-utils'; +import { RepositoriesHeader } from '../AddRepositories/RepositoriesHeader'; +import { AddedRepositoriesTableBody } from './AddedRepositoriesTableBody'; import DeleteRepositoryDialog from './DeleteRepositoryDialog'; import EditCatalogInfo from './EditCatalogInfo'; -import { columns } from './RepositoriesListColumns'; import { RepositoriesListToolbar } from './RepositoriesListToolbar'; -const useStyles = makeStyles(theme => ({ - empty: { - padding: theme.spacing(2), - display: 'flex', - justifyContent: 'center', - }, -})); - export const RepositoriesList = () => { const navigate = useNavigate(); const location = useLocation(); const searchParams = new URLSearchParams(location.search); + const [order, setOrder] = React.useState('asc'); + const [orderBy, setOrderBy] = React.useState(); const { openDialog, setOpenDialog, deleteComponent } = useDeleteDialog(); const { openDrawer, setOpenDrawer, drawerData } = useDrawer(); const [pageNumber, setPageNumber] = React.useState(0); const [rowsPerPage, setRowsPerPage] = React.useState(5); - const [searchString, setSearchString] = React.useState(''); - const classes = useStyles(); + const [debouncedSearch, setDebouncedSearch] = React.useState(''); const { data: importJobs, error: errJobs, loaded: jobsLoaded, - retry, - } = useAddedRepositories(pageNumber + 1, rowsPerPage, searchString); + refetch, + } = useAddedRepositories(pageNumber + 1, rowsPerPage, debouncedSearch); const closeDialog = () => { setOpenDialog(false); - retry(); + refetch(); }; const closeDrawer = () => { @@ -55,33 +56,138 @@ export const RepositoriesList = () => { setOpenDrawer(false); }; + const handleRequestSort = ( + _event: React.MouseEvent, + property: string, + ) => { + const isAsc = orderBy === property && order === 'asc'; + setOrder(isAsc ? 'desc' : 'asc'); + setOrderBy(property); + }; + + // Avoid a layout jump when reaching the last page with empty rows. + const emptyRows = + pageNumber > 0 ? Math.max(0, rowsPerPage - importJobs.totalJobs) : 0; + + const sortedData = React.useMemo(() => { + return [...(importJobs.addedRepositories || [])]?.sort( + getComparator(order, orderBy as string), + ); + }, [importJobs.addedRepositories, order, orderBy]); + + const handleSearch = (str: string) => { + setDebouncedSearch(str); + setPageNumber(0); + }; + if (Object.keys(errJobs || {}).length > 0) { return ; } return ( - <> - - + + + +
+ + +
+ + { + setPageNumber(page); + }} + onRowsPerPageChange={event => { + setRowsPerPage(parseInt(event.target.value, 10)); + }} + labelRowsPerPage={null} + /> + {openDrawer && ( + + )} + {openDialog && ( + + )} +
+
+ ); +}; + +/* + + { setSearchString(search); }} - onPageChange={(page: number, pageSize: number) => { - setPageNumber(page); - setRowsPerPage(pageSize); - }} - onRowsPerPageChange={(pageSize: number) => { - setRowsPerPage(pageSize); - }} title={ !jobsLoaded || !importJobs ? 'Added repositories' - : `Added repositories (${importJobs.length})` + : `Added repositories (${importJobs.totalJobs})` } options={{ padding: 'default', search: true, paging: true }} - data={importJobs ?? []} + data={importJobs.addedRepositories ?? []} isLoading={!jobsLoaded} - columns={columns} + columns={RepositoriesListColumns} + components={{ + Pagination: () => ( + { + setPageNumber(page); + }} + onRowsPerPageChange={event => { + setRowsPerPage(event.target.value as unknown as number); + }} + labelRowsPerPage={null} + /> + ), + }} emptyContent={
{
} /> - {openDrawer && ( - - )} - {openDialog && ( - - )} - - ); -}; +*/ diff --git a/plugins/bulk-import/src/components/Repositories/RepositoriesListColumns.ts b/plugins/bulk-import/src/components/Repositories/RepositoriesListColumns.ts new file mode 100644 index 00000000000..1e7ef940e0b --- /dev/null +++ b/plugins/bulk-import/src/components/Repositories/RepositoriesListColumns.ts @@ -0,0 +1,43 @@ +import { TableColumn } from '@backstage/core-components'; + +import { AddRepositoryData } from '../../types'; + +export const RepositoriesListColumns: TableColumn[] = [ + { + id: 'name', + title: 'Name', + field: 'repoName', + type: 'string', + }, + { + id: 'repo-url', + title: 'Repo URL', + field: 'repoUrl', + type: 'string', + }, + { + id: 'organization', + title: 'Organization', + field: 'organizationUrl', + type: 'string', + }, + { + id: 'status', + title: 'Status', + field: 'catalogInfoYaml.status', + type: 'string', + }, + { + id: 'last-updated', + title: 'Last updated', + field: 'catalogInfoYaml.lastUpdated', + type: 'datetime', + }, + { + id: 'actions', + title: 'Actions', + field: 'actions', + sorting: false, + type: 'string', + }, +]; diff --git a/plugins/bulk-import/src/components/Repositories/RepositoriesListColumns.tsx b/plugins/bulk-import/src/components/Repositories/RepositoriesListColumns.tsx deleted file mode 100644 index a1c1500cda6..00000000000 --- a/plugins/bulk-import/src/components/Repositories/RepositoriesListColumns.tsx +++ /dev/null @@ -1,98 +0,0 @@ -import React from 'react'; - -import { Link, TableColumn } from '@backstage/core-components'; - -import OpenInNewIcon from '@mui/icons-material/OpenInNew'; -import { useFormikContext } from 'formik'; - -import { AddRepositoriesFormValues, AddRepositoryData } from '../../types'; -import { - calculateLastUpdated, - descendingComparator, - getImportStatus, - urlHelper, -} from '../../utils/repository-utils'; -import CatalogInfoAction from './CatalogInfoAction'; -import DeleteRepository from './DeleteRepository'; -import SyncRepository from './SyncRepository'; - -const ImportStatus = ({ data }: { data: AddRepositoryData }) => { - const { values } = useFormikContext(); - return getImportStatus( - values.repositories[data.id]?.catalogInfoYaml?.status as string, - true, - ); -}; - -const LastUpdated = ({ data }: { data: AddRepositoryData }) => { - const { values } = useFormikContext(); - return calculateLastUpdated( - values.repositories[data.id]?.catalogInfoYaml?.lastUpdated || '', - ); -}; - -export const columns: TableColumn[] = [ - { - title: 'Name', - field: 'repoName', - type: 'string', - }, - { - title: 'Repo URL', - field: 'repoUrl', - type: 'string', - align: 'left', - render: (props: AddRepositoryData) => { - return ( - - {urlHelper(props.repoUrl || '')} - - - ); - }, - }, - { - title: 'Organization', - field: 'organizationUrl', - type: 'string', - align: 'left', - render: (props: AddRepositoryData) => { - return ( - - {urlHelper(props.organizationUrl || '')} - - - ); - }, - }, - { - title: 'Status', - field: 'catalogInfoYaml.status', - type: 'string', - align: 'left', - customSort: (a: AddRepositoryData, b: AddRepositoryData) => - descendingComparator(a, b, 'catalogInfoYaml.status'), - render: (data: AddRepositoryData) => , - }, - { - title: 'Last updated', - field: 'catalogInfoYaml.lastUpdated', - type: 'datetime', - align: 'left', - render: (data: AddRepositoryData) => , - }, - { - title: 'Actions', - field: 'actions', - sorting: false, - type: 'string', - align: 'left', - render: (data: AddRepositoryData) => ( - <> - - - - - ), - }, -]; diff --git a/plugins/bulk-import/src/components/Repositories/RepositoriesListToolbar.tsx b/plugins/bulk-import/src/components/Repositories/RepositoriesListToolbar.tsx index c5f95ac42bb..f42ecb0bde0 100644 --- a/plugins/bulk-import/src/components/Repositories/RepositoriesListToolbar.tsx +++ b/plugins/bulk-import/src/components/Repositories/RepositoriesListToolbar.tsx @@ -2,17 +2,19 @@ import React from 'react'; import { LinkButton } from '@backstage/core-components'; -import { makeStyles } from '@material-ui/core'; +import { makeStyles, Toolbar } from '@material-ui/core'; import { Alert, AlertTitle } from '@material-ui/lab'; +import Typography from '@mui/material/Typography'; import { useFormikContext } from 'formik'; import { AddRepositoriesFormValues } from '../../types'; +import { RepositoriesSearchBar } from '../AddRepositories/AddRepositoriesSearchBar'; const useStyles = makeStyles(theme => ({ toolbar: { display: 'flex', justifyContent: 'end', - marginBottom: '24px', + padding: '14px', }, rbacPreReqLink: { color: theme.palette.link, @@ -22,15 +24,28 @@ const useStyles = makeStyles(theme => ({ }, })); -export const RepositoriesListToolbar = () => { +export const RepositoriesListToolbar = ({ + jobs, + setSearchString, +}: { + jobs: number; + setSearchString: (str: string) => void; +}) => { const { status, setStatus } = useFormikContext(); + const [search, setSearch] = React.useState(''); const classes = useStyles(); const handleCloseAlert = () => { setStatus(null); }; + + const handleSearch = (filter: string) => { + setSearchString(filter); + setSearch(filter); + }; + return ( -
+ <> {(status?.title || status?.url) && ( <> handleCloseAlert()}> @@ -42,16 +57,27 @@ export const RepositoriesListToolbar = () => {
)} - - + - Add - - -
+ {jobs ? `Added repositories (${jobs})` : 'Added repositories'} + + + + + Add + + + + + ); }; diff --git a/plugins/bulk-import/src/hooks/useAddedRepositories.test.ts b/plugins/bulk-import/src/hooks/useAddedRepositories.test.ts index cdd5d7d82e9..5fb06c7efe7 100644 --- a/plugins/bulk-import/src/hooks/useAddedRepositories.test.ts +++ b/plugins/bulk-import/src/hooks/useAddedRepositories.test.ts @@ -10,6 +10,16 @@ jest.mock('@backstage/core-plugin-api', () => ({ }), })); +jest.mock('@tanstack/react-query', () => ({ + ...jest.requireActual('@tanstack/react-query'), + useQuery: jest.fn().mockReturnValue({ + data: mockGetImportJobs, + isLoading: false, + error: null, + refetch: jest.fn(), + }), +})); + jest.mock('formik', () => ({ ...jest.requireActual('formik'), useFormikContext: jest.fn().mockReturnValue({ @@ -22,7 +32,7 @@ describe('useAddedRepositories', () => { const { result } = renderHook(() => useAddedRepositories(1, 5, '')); await waitFor(() => { expect(result.current.loaded).toBeTruthy(); - expect(result.current.data).toHaveLength(4); + expect(result.current.data.totalJobs).toBe(4); }); }); }); diff --git a/plugins/bulk-import/src/hooks/useAddedRepositories.ts b/plugins/bulk-import/src/hooks/useAddedRepositories.ts index 213c80abe45..e9a654f1207 100644 --- a/plugins/bulk-import/src/hooks/useAddedRepositories.ts +++ b/plugins/bulk-import/src/hooks/useAddedRepositories.ts @@ -1,5 +1,5 @@ import React from 'react'; -import { useAsync, useAsyncRetry, useDebounce, useInterval } from 'react-use'; +import { useAsync, useDebounce } from 'react-use'; import { configApiRef, @@ -7,6 +7,7 @@ import { useApi, } from '@backstage/core-plugin-api'; +import { useQuery } from '@tanstack/react-query'; import { useFormikContext } from 'formik'; import { @@ -18,30 +19,30 @@ import { bulkImportApiRef } from '../api/BulkImportBackendClient'; import { AddRepositoriesFormValues, AddRepositoryData, - ImportJobStatus, - RepositoryStatus, + ImportJobs, } from '../types'; -import { getPRTemplate } from '../utils/repository-utils'; +import { prepareDataForAddedRepositories } from '../utils/repository-utils'; export const useAddedRepositories = ( - page: number, + pageNumber: number, rowsPerPage: number, searchString: string, pollInterval?: number, ): { loaded: boolean; - data: AddRepositoryData[]; + data: { + addedRepositories: AddRepositoryData[]; + totalJobs: number; + }; error: any; - retry: () => void; + refetch: () => void; } => { - const [addedRepositoriesData, setAddedRepositoriesData] = React.useState< - AddRepositoryData[] - >([]); + const [addedRepositoriesData, setAddedRepositoriesData] = React.useState<{ + [id: string]: AddRepositoryData; + }>({}); + const [totalImportJobs, setTotalImportJobs] = React.useState(0); const [loaded, setLoaded] = React.useState(false); const [debouncedSearch, setDebouncedSearch] = React.useState(searchString); - const [errorState, setErrorState] = React.useState< - { [key: string]: string | undefined } | undefined - >(); const identityApi = useApi(identityApiRef); const configApi = useApi(configApiRef); const mounted = React.useRef(false); @@ -65,10 +66,24 @@ export const useAddedRepositories = ( const bulkImportApi = useApi(bulkImportApiRef); const { setFieldValue } = useFormikContext(); - const { value, loading, error, retry } = useAsyncRetry( - async () => - await bulkImportApi.getImportJobs(page, rowsPerPage, searchString), - [page, rowsPerPage, debouncedSearch], + const fetchAddedRepositories = async ( + page: number, + size: number, + searchStr: string, + ) => { + const response = await bulkImportApi.getImportJobs(page, size, searchStr); + return response; + }; + + const { + data: value, + error, + isLoading: loading, + refetch, + } = useQuery( + ['importJobs', pageNumber, rowsPerPage, debouncedSearch], + () => fetchAddedRepositories(pageNumber, rowsPerPage, debouncedSearch), + { keepPreviousData: true, refetchInterval: pollInterval || 60000 }, ); React.useEffect(() => { @@ -78,64 +93,18 @@ export const useAddedRepositories = ( }; }, []); - const prepareDataForAddedRepositories = React.useCallback( - ( - addedRepositories: ImportJobStatus[] | Response, - isLoading: boolean, - errorData: { [key: string]: string | undefined } | undefined, - ) => { - if (!isLoading && !errorData && mounted.current) { - if (!Array.isArray(addedRepositories)) { - setAddedRepositoriesData([]); - } else { - const repoData: { [id: string]: AddRepositoryData } = - addedRepositories?.reduce((acc, val: ImportJobStatus) => { - const id = `${val.repository.organization}/${val.repository.name}`; - return { - ...acc, - [id]: { - id, - repoName: val.repository.name, - defaultBranch: val.repository.defaultBranch, - orgName: val.repository.organization, - repoUrl: val.repository.url, - organizationUrl: val?.repository?.url?.substring( - 0, - val.repository.url.indexOf(val?.repository?.name || '') - 1, - ), - catalogInfoYaml: { - status: val.status - ? RepositoryStatus[val.status as RepositoryStatus] - : RepositoryStatus.NotGenerated, - prTemplate: getPRTemplate( - val.repository.name || '', - val.repository.organization || '', - user as string, - baseUrl as string, - val.repository.url || '', - val.repository.defaultBranch || 'main', - ), - pullRequest: val?.github?.pullRequest?.url || '', - lastUpdated: val.lastUpdate, - }, - }, - }; - }, {}); - setFieldValue(`repositories`, repoData); - setAddedRepositoriesData(Object.values(repoData)); - } - setLoaded(true); - } else if (errorData && mounted.current) { + const prepareData = React.useCallback( + (addedRepositories: ImportJobs | Response, isLoading: boolean) => { + if (!isLoading) { + const repoData = prepareDataForAddedRepositories( + addedRepositories, + user as string, + baseUrl as string, + ); + setAddedRepositoriesData(repoData); + setFieldValue(`repositories`, repoData); + setTotalImportJobs((addedRepositories as ImportJobs)?.totalCount || 0); setLoaded(true); - setErrorState({ - ...(errorData ?? {}), - ...((addedRepositories as Response)?.statusText - ? { - name: 'Error', - message: (addedRepositories as Response)?.statusText, - } - : {}), - }); } }, [ @@ -143,31 +112,31 @@ export const useAddedRepositories = ( baseUrl, setFieldValue, setAddedRepositoriesData, - setErrorState, - setLoaded, + setTotalImportJobs, ], ); - const debouncedUpdateResources = useDebounceCallback( - prepareDataForAddedRepositories, - 250, - ); + const debouncedUpdateResources = useDebounceCallback(prepareData, 250); React.useEffect(() => { - debouncedUpdateResources?.(value, loading, error); - }, [debouncedUpdateResources, value, loading, error]); - - useInterval( - () => { - retry(); - }, - loading ? null : pollInterval || 60000, - ); + debouncedUpdateResources?.(value, loading); + }, [debouncedUpdateResources, value, loading]); return useDeepCompareMemoize({ - data: addedRepositoriesData, + data: { + addedRepositories: Object.values(addedRepositoriesData), + totalJobs: totalImportJobs, + }, loaded, - error: errorState, - retry, + error: { + ...(error ?? {}), + ...((value as Response)?.statusText + ? { + name: 'Error', + message: (value as Response)?.statusText, + } + : {}), + }, + refetch, }); }; diff --git a/plugins/bulk-import/src/mocks/mockData.ts b/plugins/bulk-import/src/mocks/mockData.ts index df7a1481bad..9e4c0350236 100644 --- a/plugins/bulk-import/src/mocks/mockData.ts +++ b/plugins/bulk-import/src/mocks/mockData.ts @@ -1,4 +1,4 @@ -import { AddedRepositories, ApprovalTool, ImportJobStatus } from '../types'; +import { AddedRepositories, ApprovalTool, ImportJobs } from '../types'; export const mockGetOrganizations = { errors: [], @@ -140,100 +140,105 @@ export const mockGetRepositories = { sizePerIntegration: 5, }; -export const mockGetImportJobs: ImportJobStatus[] = [ - { - approvalTool: ApprovalTool.Git, - github: { - pullRequest: { - number: 90, - url: 'https://github.com/org/dessert/cupcake/pull/90', - body: 'PR body', - catalogInfoContent: - 'apiVersion: backstage.io/v1alpha1\nkind: Component\nmetadata:\n name: che1\n annotations:\n github.com/project-slug: debsmita1/che\nspec:\n type: other\n lifecycle: unknown\n owner: user:default/debsmita1\n', - title: 'PR title', +export const mockGetImportJobs: ImportJobs = { + imports: [ + { + approvalTool: ApprovalTool.Git, + github: { + pullRequest: { + number: 90, + url: 'https://github.com/org/dessert/cupcake/pull/90', + body: 'PR body', + catalogInfoContent: + 'apiVersion: backstage.io/v1alpha1\nkind: Component\nmetadata:\n name: che1\n annotations:\n github.com/project-slug: debsmita1/che\nspec:\n type: other\n lifecycle: unknown\n owner: user:default/debsmita1\n', + title: 'PR title', + }, }, - }, - lastUpdate: '2024-07-17T13:46:37Z', - repository: { - id: 'org/dessert/cupcake', - name: 'cupcake', - url: 'https://github.com/org/dessert/cupcake', - defaultBranch: 'master', - organization: 'org/dessert', - }, - id: 'org/dessert/cupcake', - status: 'WAIT_PR_APPROVAL', - }, - { - approvalTool: ApprovalTool.Git, - github: { - pullRequest: { - body: 'PR body', - catalogInfoContent: - '\nkind: Component\nmetadata:\n name: che1\n annotations:\n github.com/project-slug: debsmita1/che\nspec:\n type: other\n lifecycle: unknown\n owner: user:default/debsmita1\n', - title: 'PR title', - number: 91, - url: 'https://github.com/org/dessert/donut/pull/91', + lastUpdate: '2024-07-17T13:46:37Z', + repository: { + id: 'org/dessert/cupcake', + name: 'cupcake', + url: 'https://github.com/org/dessert/cupcake', + defaultBranch: 'master', + organization: 'org/dessert', }, + id: 'org/dessert/cupcake', + status: 'WAIT_PR_APPROVAL', }, - lastUpdate: '2024-07-18T13:46:37Z', - repository: { - id: 'org/dessert/donut', - name: 'donut', - url: 'https://github.com/org/dessert/donut', - defaultBranch: 'master', - organization: 'org/dessert', - }, - id: 'org/dessert/donut', - status: 'WAIT_PR_APPROVAL', - }, - { - approvalTool: ApprovalTool.Git, - github: { - pullRequest: { - number: 94, - url: 'https://github.com/org/food/food-app/pull/94', - body: 'PR body', - catalogInfoContent: - 'apiVersion: backstage.io/v1alpha1\nkind: Component\nmetadata:\n name: che1\n annotations:\n github.com/project-slug: debsmita1/che\nspec:\n type: other\n lifecycle: unknown\n owner: user:default/debsmita1\n', - title: 'PR title', + { + approvalTool: ApprovalTool.Git, + github: { + pullRequest: { + body: 'PR body', + catalogInfoContent: + '\nkind: Component\nmetadata:\n name: che1\n annotations:\n github.com/project-slug: debsmita1/che\nspec:\n type: other\n lifecycle: unknown\n owner: user:default/debsmita1\n', + title: 'PR title', + number: 91, + url: 'https://github.com/org/dessert/donut/pull/91', + }, }, + lastUpdate: '2024-07-18T13:46:37Z', + repository: { + id: 'org/dessert/donut', + name: 'donut', + url: 'https://github.com/org/dessert/donut', + defaultBranch: 'master', + organization: 'org/dessert', + }, + id: 'org/dessert/donut', + status: 'WAIT_PR_APPROVAL', }, - lastUpdate: '2024-07-21T13:46:37Z', - repository: { + { + approvalTool: ApprovalTool.Git, + github: { + pullRequest: { + number: 94, + url: 'https://github.com/org/food/food-app/pull/94', + body: 'PR body', + catalogInfoContent: + 'apiVersion: backstage.io/v1alpha1\nkind: Component\nmetadata:\n name: che1\n annotations:\n github.com/project-slug: debsmita1/che\nspec:\n type: other\n lifecycle: unknown\n owner: user:default/debsmita1\n', + title: 'PR title', + }, + }, + lastUpdate: '2024-07-21T13:46:37Z', + repository: { + id: 'org/food/food-app', + name: 'food-app', + url: 'https://github.com/org/food/food-app', + defaultBranch: 'master', + organization: 'org/food', + }, id: 'org/food/food-app', - name: 'food-app', - url: 'https://github.com/org/food/food-app', - defaultBranch: 'master', - organization: 'org/food', + status: 'WAIT_PR_APPROVAL', }, - id: 'org/food/food-app', - status: 'WAIT_PR_APPROVAL', - }, - { - approvalTool: ApprovalTool.Git, - github: { - pullRequest: { - number: 95, - url: 'https://github.com/org/pet-store-boston/pet-app/pull/95', - body: 'PR body', - catalogInfoContent: - 'apiVersion: backstage.io/v1alpha1\nkind: Component\nmetadata:\n name: che1\n annotations:\n github.com/project-slug: debsmita1/che\nspec:\n type: other\n lifecycle: unknown\n owner: user:default/debsmita1\n', - title: 'PR title', + { + approvalTool: ApprovalTool.Git, + github: { + pullRequest: { + number: 95, + url: 'https://github.com/org/pet-store-boston/pet-app/pull/95', + body: 'PR body', + catalogInfoContent: + 'apiVersion: backstage.io/v1alpha1\nkind: Component\nmetadata:\n name: che1\n annotations:\n github.com/project-slug: debsmita1/che\nspec:\n type: other\n lifecycle: unknown\n owner: user:default/debsmita1\n', + title: 'PR title', + }, + }, + lastUpdate: '2024-07-22T13:46:37Z', + repository: { + id: 'org/pet-store-boston/pet-app', + name: 'pet-app', + url: 'https://github.com/org/pet-store-boston/pet-app', + defaultBranch: 'master', + organization: 'org/pet-store-boston', }, - }, - lastUpdate: '2024-07-22T13:46:37Z', - repository: { id: 'org/pet-store-boston/pet-app', - name: 'pet-app', - url: 'https://github.com/org/pet-store-boston/pet-app', - defaultBranch: 'master', - organization: 'org/pet-store-boston', + status: 'ADDED', }, - id: 'org/pet-store-boston/pet-app', - status: 'ADDED', - }, -]; + ], + page: 1, + size: 5, + totalCount: 4, +}; export const mockSelectedRepositories: AddedRepositories = { ['org/dessert/cupcake']: mockGetRepositories.repositories[0], diff --git a/plugins/bulk-import/src/types/response-types.ts b/plugins/bulk-import/src/types/response-types.ts index 3fe2ba08f9f..0568a978f52 100644 --- a/plugins/bulk-import/src/types/response-types.ts +++ b/plugins/bulk-import/src/types/response-types.ts @@ -35,6 +35,13 @@ export type ImportJobStatus = { repository: Repository; }; +export type ImportJobs = { + imports: ImportJobStatus[]; + page: number; + size: number; + totalCount: number; +}; + export type OrgAndRepoResponse = { errors?: string[]; repositories?: Repository[]; diff --git a/plugins/bulk-import/src/utils/repository-utils.tsx b/plugins/bulk-import/src/utils/repository-utils.tsx index 02582a01bad..b2b34146cf4 100644 --- a/plugins/bulk-import/src/utils/repository-utils.tsx +++ b/plugins/bulk-import/src/utils/repository-utils.tsx @@ -16,6 +16,7 @@ import { CreateImportJobRepository, ErrorType, ImportJobResponse, + ImportJobs, ImportJobStatus, ImportStatus, JobErrors, @@ -552,3 +553,48 @@ export const evaluatePRTemplate = ( }; } }; + +export const prepareDataForAddedRepositories = ( + addedRepositories: ImportJobs | Response | undefined, + user: string, + baseUrl: string, +) => { + if (!Array.isArray((addedRepositories as ImportJobs)?.imports)) { + return {}; + } + const importJobs = addedRepositories as ImportJobs; + const repoData: { [id: string]: AddRepositoryData } = + importJobs.imports?.reduce((acc, val: ImportJobStatus) => { + const id = `${val.repository.organization}/${val.repository.name}`; + return { + ...acc, + [id]: { + id, + repoName: val.repository.name, + defaultBranch: val.repository.defaultBranch, + orgName: val.repository.organization, + repoUrl: val.repository.url, + organizationUrl: val?.repository?.url?.substring( + 0, + val.repository.url.indexOf(val?.repository?.name || '') - 1, + ), + catalogInfoYaml: { + status: val.status + ? RepositoryStatus[val.status as RepositoryStatus] + : RepositoryStatus.NotGenerated, + prTemplate: getPRTemplate( + val.repository.name || '', + val.repository.organization || '', + user, + baseUrl, + val.repository.url || '', + val.repository.defaultBranch || 'main', + ), + pullRequest: val?.github?.pullRequest?.url || '', + lastUpdated: val.lastUpdate, + }, + }, + }; + }, {}); + return repoData; +}; diff --git a/plugins/bulk-import/tests/bulkImport.spec.ts b/plugins/bulk-import/tests/bulkImport.spec.ts index 8ed2c0db4da..3fec4d5c1da 100644 --- a/plugins/bulk-import/tests/bulkImport.spec.ts +++ b/plugins/bulk-import/tests/bulkImport.spec.ts @@ -53,7 +53,7 @@ test.describe('Bulk import plugin', () => { }); test('Remove repository alert window is shown', async () => { - await page.getByPlaceholder('Filter').fill('cupcake'); + await page.getByPlaceholder('Search').fill('cupcake'); await page.locator('span[data-testid="delete-repository"]').first().click(); await expect(page.getByText('Remove cupcake repository?')).toBeVisible(); await expect( diff --git a/yarn.lock b/yarn.lock index 056e71608f3..073b2cb0a99 100644 --- a/yarn.lock +++ b/yarn.lock @@ -13307,7 +13307,7 @@ resolved "https://registry.yarnpkg.com/@tanstack/query-core/-/query-core-4.36.1.tgz#79f8c1a539d47c83104210be2388813a7af2e524" integrity sha512-DJSilV5+ytBP1FbFcEJovv4rnnm/CokuVvrBEtW/Va9DvuJ3HksbXUJEpI0aV1KtuL4ZoO9AVE6PyNLzF7tLeA== -"@tanstack/react-query@^4.36.1": +"@tanstack/react-query@^4.29.21", "@tanstack/react-query@^4.36.1": version "4.36.1" resolved "https://registry.yarnpkg.com/@tanstack/react-query/-/react-query-4.36.1.tgz#acb589fab4085060e2e78013164868c9c785e5d2" integrity sha512-y7ySVHFyyQblPl3J3eQBWpXZkliroki3ARnBKsdJchlgt7yJLRDUcf4B8soufgiYt3pEQIkBWBx1N9/ZPIeUWw==