diff --git a/src/renderer/api/cadt/v1/audit/audit.api.ts b/src/renderer/api/cadt/v1/audit/audit.api.ts new file mode 100644 index 00000000..7f405e01 --- /dev/null +++ b/src/renderer/api/cadt/v1/audit/audit.api.ts @@ -0,0 +1,46 @@ +import {cadtApi, auditTag} from "../"; + +const host: string = 'http://localhost:31310' + +interface GetAuditParams { + page: number; + orgUid: string; + search?: string; + order?: string; +} + +interface GetAuditResponse { + page: number, + pageCount: number, + data: any[] +} + +const auditApi = cadtApi.injectEndpoints({ + endpoints: (builder) => ({ + getAudit: builder.query({ + query: ({ page, orgUid, search, order }: GetAuditParams) => { + // Initialize the params object with page and limit + const params: GetAuditParams & {limit: number} = { orgUid, page, limit: 10 }; + + if (search) { + params.search = search.replace(/[^a-zA-Z0-9 _.-]+/, ''); + } + + if (order) { + params.order = order; + } + + return { + url: `${host}/v1/audit`, + params, // Use the constructed params object + method: 'GET', + }; + }, + providesTags: (_response, _error, {orgUid}) => [{type: auditTag, id: orgUid}], + }) + }) +}); + +export const { + useGetAuditQuery +} = auditApi; \ No newline at end of file diff --git a/src/renderer/api/cadt/v1/audit/index.ts b/src/renderer/api/cadt/v1/audit/index.ts new file mode 100644 index 00000000..0a3a334d --- /dev/null +++ b/src/renderer/api/cadt/v1/audit/index.ts @@ -0,0 +1 @@ +export * from './audit.api'; \ No newline at end of file diff --git a/src/renderer/api/cadt/v1/index.ts b/src/renderer/api/cadt/v1/index.ts index 3f949ea3..f4b62c4e 100644 --- a/src/renderer/api/cadt/v1/index.ts +++ b/src/renderer/api/cadt/v1/index.ts @@ -3,16 +3,17 @@ import {createApi, fetchBaseQuery} from '@reduxjs/toolkit/query/react'; const projectsTag: string = 'projects'; const organizationsTag: string = 'organizations'; const unitsTag: string = 'projects'; +const auditTag: string = 'audit'; const baseQuery = fetchBaseQuery({ baseUrl: '/', }); -export { projectsTag, organizationsTag, unitsTag}; +export { projectsTag, organizationsTag, unitsTag, auditTag}; export const cadtApi = createApi({ baseQuery, reducerPath: 'cadtApi', - tagTypes: [projectsTag, organizationsTag, unitsTag], + tagTypes: [projectsTag, organizationsTag, unitsTag, auditTag], endpoints: () => ({}), }); diff --git a/src/renderer/api/index.ts b/src/renderer/api/index.ts index acbe8d0e..84e492e1 100644 --- a/src/renderer/api/index.ts +++ b/src/renderer/api/index.ts @@ -1,3 +1,4 @@ export * from './cadt/v1/organizations'; export * from './cadt/v1/units'; -export * from './cadt/v1/projects'; \ No newline at end of file +export * from './cadt/v1/projects'; +export * from './cadt/v1/audit'; \ No newline at end of file diff --git a/src/renderer/components/blocks/layout/LeftNav.tsx b/src/renderer/components/blocks/layout/LeftNav.tsx index e17d341d..f2f4433b 100644 --- a/src/renderer/components/blocks/layout/LeftNav.tsx +++ b/src/renderer/components/blocks/layout/LeftNav.tsx @@ -64,6 +64,13 @@ const LeftNav = () => { > + navigate(ROUTES.AUDIT)} + > + + void>; + setOrder?: (sort: string) => void; + order?: string; + totalPages: number; + totalCount: number; +} + +const AuditsTable: React.FC = ({ + data, + isLoading, + currentPage, + onPageChange, + setOrder, + order, + totalPages, + totalCount, +}) => { + + + const columns = useMemo( + () => [ + { + title: , + key: 'table', + }, + { + title: , + key: 'onchainConfirmationTimeStamp', + render: (row: any) => { + return <>{formatToDataTimeFromSeconds(row.onchainConfirmationTimeStamp)}; + } + }, + { + title: , + key: 'type', + }, + { + title: , + key: 'rootHash', + }, + { + title: , + key: 'author', + }, + { + title: , + key: 'comment', + }, + ], + [], + ); + + return ( +
+ + + + + } + /> +
+ ); +}; + +export { AuditsTable }; diff --git a/src/renderer/components/blocks/tables/index.ts b/src/renderer/components/blocks/tables/index.ts index 0161d472..572cefaa 100644 --- a/src/renderer/components/blocks/tables/index.ts +++ b/src/renderer/components/blocks/tables/index.ts @@ -1,4 +1,5 @@ export * from './ProjectsListTable'; export * from './SkeletonTable'; export * from './UnitsListTable'; +export * from './AuditsTable'; export * from './GlossaryTable'; \ No newline at end of file diff --git a/src/renderer/pages/Audit/AuditPage.tsx b/src/renderer/pages/Audit/AuditPage.tsx new file mode 100644 index 00000000..563de953 --- /dev/null +++ b/src/renderer/pages/Audit/AuditPage.tsx @@ -0,0 +1,116 @@ +import React, { useCallback } from 'react'; +import { useGetAuditQuery } from '@/api'; +import { useQueryParamState } from '@/hooks'; +import {debounce, DebouncedFunc} from 'lodash'; +import { + OrganizationSelector, + IndeterminateProgressOverlay, + SkeletonTable, + AuditsTable, + SearchBox, +} from '@/components'; +import {FormattedMessage} from "react-intl"; + +const AuditPage: React.FC = () => { + const [currentPage, setCurrentPage] = useQueryParamState('page', '1'); + const [orgUid, setOrgUid] = useQueryParamState('orgUid', undefined); + const [search, setSearch] = useQueryParamState('search', undefined); + const [order, setOrder] = useQueryParamState('order', undefined); + + const { + data: auditData, + isLoading: auditLoading, + isFetching: auditFetching, + error: auditError, + } = useGetAuditQuery({ page: Number(currentPage), orgUid, search, order }, {skip: !orgUid}); + + const handlePageChange: DebouncedFunc = useCallback( + debounce((page) => setCurrentPage(page), 800), + [setCurrentPage], + ); + + const handleOrganizationSelected = useCallback( + (organization: any) => { + setOrgUid(organization?.orgUid); + }, + [setOrgUid], + ); + + const handleSearchChange: DebouncedFunc = useCallback( + debounce((event: any) => { + setSearch(event.target.value); + }, 800), + [setSearch, debounce], + ); + + const handleSetOrder = useCallback(() => { + const currentDirection = order; + + // Cycle through 'ASC', 'DESC', and no order ('') + switch (currentDirection) { + case 'ASC': + setOrder(`DESC`); + break; + case 'DESC': + setOrder(''); + break; + default: + setOrder(`ASC`); + break; + } + + }, [order, setOrder]); + + if (!orgUid){ + return ( + <> +
+ +
+
+ +
+ + ); + } + + if (auditLoading) { + return ; + } + + if (auditError) { + return ; + } + + if (!auditData) { + return ; + } + + return ( + <> + {auditFetching && } +
+ + +
+ <> + {auditLoading ? ( + + ) : ( + + )} + + + ); +}; + +export {AuditPage}; diff --git a/src/renderer/pages/Audit/index.ts b/src/renderer/pages/Audit/index.ts new file mode 100644 index 00000000..25f5689f --- /dev/null +++ b/src/renderer/pages/Audit/index.ts @@ -0,0 +1 @@ +export * from './AuditPage'; \ No newline at end of file diff --git a/src/renderer/pages/ProjectsList/ProjectsList.tsx b/src/renderer/pages/ProjectsList/ProjectsListPage.tsx similarity index 90% rename from src/renderer/pages/ProjectsList/ProjectsList.tsx rename to src/renderer/pages/ProjectsList/ProjectsListPage.tsx index a9584f35..015c1cea 100644 --- a/src/renderer/pages/ProjectsList/ProjectsList.tsx +++ b/src/renderer/pages/ProjectsList/ProjectsListPage.tsx @@ -1,7 +1,7 @@ import React, { useCallback } from 'react'; import { useGetProjectsQuery } from '@/api'; import { useQueryParamState, useColumnOrderHandler } from '@/hooks'; -import { debounce } from 'lodash'; +import {debounce, DebouncedFunc} from 'lodash'; import { OrganizationSelector, IndeterminateProgressOverlay, @@ -12,7 +12,7 @@ import { } from '@/components'; import {FormattedMessage} from "react-intl"; -const ProjectsList: React.FC = () => { +const ProjectsListPage: React.FC = () => { const [currentPage, setCurrentPage] = useQueryParamState('page', '1'); const [orgUid, setOrgUid] = useQueryParamState('orgUid', undefined); const [search, setSearch] = useQueryParamState('search', undefined); @@ -26,7 +26,7 @@ const ProjectsList: React.FC = () => { error: projectsError, } = useGetProjectsQuery({ page: Number(currentPage), orgUid, search, order }); - const handlePageChange = useCallback( + const handlePageChange: DebouncedFunc = useCallback( debounce((page) => setCurrentPage(page), 800), [setCurrentPage], ); @@ -38,7 +38,7 @@ const ProjectsList: React.FC = () => { [setOrgUid], ); - const handleSearchChange = useCallback( + const handleSearchChange: DebouncedFunc = useCallback( debounce((event: any) => { setSearch(event.target.value); }, 800), @@ -84,4 +84,4 @@ const ProjectsList: React.FC = () => { ); }; -export { ProjectsList }; +export { ProjectsListPage }; diff --git a/src/renderer/pages/ProjectsList/index.ts b/src/renderer/pages/ProjectsList/index.ts index 5353d519..85e99f26 100644 --- a/src/renderer/pages/ProjectsList/index.ts +++ b/src/renderer/pages/ProjectsList/index.ts @@ -1 +1 @@ -export * from './ProjectsList'; \ No newline at end of file +export * from './ProjectsListPage'; \ No newline at end of file diff --git a/src/renderer/pages/UnitsList/UnitsList.tsx b/src/renderer/pages/UnitsList/UnitsListPage.tsx similarity index 90% rename from src/renderer/pages/UnitsList/UnitsList.tsx rename to src/renderer/pages/UnitsList/UnitsListPage.tsx index 5ff36686..a06a8125 100644 --- a/src/renderer/pages/UnitsList/UnitsList.tsx +++ b/src/renderer/pages/UnitsList/UnitsListPage.tsx @@ -1,7 +1,7 @@ import React, { useCallback } from 'react'; import { useGetUnitsQuery } from '@/api'; import { useQueryParamState, useColumnOrderHandler } from '@/hooks'; -import { debounce } from 'lodash'; +import {debounce, DebouncedFunc} from 'lodash'; import { OrganizationSelector, IndeterminateProgressOverlay, @@ -12,7 +12,7 @@ import { } from '@/components'; import {FormattedMessage} from "react-intl"; -const UnitsList: React.FC = () => { +const UnitsListPage: React.FC = () => { const [currentPage, setCurrentPage] = useQueryParamState('page', '1'); const [orgUid, setOrgUid] = useQueryParamState('orgUid', undefined); const [search, setSearch] = useQueryParamState('search', undefined); @@ -26,7 +26,7 @@ const UnitsList: React.FC = () => { error: unitsError, } = useGetUnitsQuery({ page: Number(currentPage), orgUid, search, order }); - const handlePageChange = useCallback( + const handlePageChange: DebouncedFunc = useCallback( debounce((page) => setCurrentPage(page), 800), [setCurrentPage], ); @@ -38,7 +38,7 @@ const UnitsList: React.FC = () => { [setOrgUid], ); - const handleSearchChange = useCallback( + const handleSearchChange: DebouncedFunc = useCallback( debounce((event: any) => { setSearch(event.target.value); }, 800), @@ -84,4 +84,4 @@ const UnitsList: React.FC = () => { ); }; -export { UnitsList }; +export { UnitsListPage }; diff --git a/src/renderer/pages/UnitsList/index.ts b/src/renderer/pages/UnitsList/index.ts index ca0a28cb..3d284f52 100644 --- a/src/renderer/pages/UnitsList/index.ts +++ b/src/renderer/pages/UnitsList/index.ts @@ -1 +1 @@ -export * from './UnitsList'; \ No newline at end of file +export * from './UnitsListPage'; \ No newline at end of file diff --git a/src/renderer/pages/index.ts b/src/renderer/pages/index.ts index 26df64e5..5d6e41de 100644 --- a/src/renderer/pages/index.ts +++ b/src/renderer/pages/index.ts @@ -1,4 +1,5 @@ export * from './Error'; export * from './ProjectsList'; export * from './UnitsList'; -export * from './Glossary'; \ No newline at end of file +export * from './Audit'; +export * from './Glossary'; diff --git a/src/renderer/routes/AppNavigator.tsx b/src/renderer/routes/AppNavigator.tsx index 42e17eb6..edb56a54 100644 --- a/src/renderer/routes/AppNavigator.tsx +++ b/src/renderer/routes/AppNavigator.tsx @@ -1,5 +1,11 @@ import React from 'react'; -import { BrowserRouter as Router, Routes, Route, Navigate, redirect } from 'react-router-dom'; +import { + BrowserRouter as Router, + Routes, + Route, + Navigate, + redirect, +} from 'react-router-dom'; import ROUTES from '@/routes/route-constants'; import * as Pages from '@/pages'; import { Template } from '@/components'; @@ -15,9 +21,8 @@ const AppNavigator: React.FC = () => { /> }> } /> - } /> - } /> - + } /> + } /> } /> } /> diff --git a/src/renderer/routes/route-constants.ts b/src/renderer/routes/route-constants.ts index fed68231..e8877e5c 100644 --- a/src/renderer/routes/route-constants.ts +++ b/src/renderer/routes/route-constants.ts @@ -1,5 +1,6 @@ export default { PROJECTS_LIST: '/projects', UNITS_LIST: '/units', + AUDIT: '/audit', GLOSSARY: '/glossary', }; \ No newline at end of file diff --git a/src/renderer/translations/tokens/en-US.json b/src/renderer/translations/tokens/en-US.json index a64d42a2..acb84668 100644 --- a/src/renderer/translations/tokens/en-US.json +++ b/src/renderer/translations/tokens/en-US.json @@ -29,7 +29,16 @@ "project-status": "Project Status", "unit-metric": "Unit Metric", "validation-body": "Validation Body", + "table": "Table", + "timestamp": "Timestamp", + "type": "Type", + "root-hash": "Root Hash", + "author": "Author", + "comment": "Comment", + "please-select-an-organization-to-view-audit-data": "Please select an organization to view audit data", + "audits": "Audits", + "validation-body": "Validation Body", "glossary": "Glossary", "term": "Term", "description": "Description" -} \ No newline at end of file +} diff --git a/src/renderer/utils/transforms.ts b/src/renderer/utils/transforms.ts new file mode 100644 index 00000000..264950ad --- /dev/null +++ b/src/renderer/utils/transforms.ts @@ -0,0 +1,10 @@ +export const formatToDataTimeFromSeconds = (seconds: number): string => { + return new Intl.DateTimeFormat('en-US', { + year: 'numeric', + month: '2-digit', + day: '2-digit', + hour: '2-digit', + minute: '2-digit', + second: '2-digit' + }).format(new Date(seconds * 1000)) +} \ No newline at end of file