mirror of
https://github.com/fleetdm/fleet
synced 2026-05-06 14:58:33 +00:00
276 lines
7.4 KiB
TypeScript
276 lines
7.4 KiB
TypeScript
/** software/os OS tab > Table */
|
|
|
|
import React, { useCallback, useMemo } from "react";
|
|
import { InjectedRouter } from "react-router";
|
|
import { Row } from "react-table";
|
|
|
|
import PATHS from "router/paths";
|
|
|
|
import { GITHUB_NEW_ISSUE_LINK } from "utilities/constants";
|
|
|
|
import CustomLink from "components/CustomLink";
|
|
import TableContainer from "components/TableContainer";
|
|
import LastUpdatedText from "components/LastUpdatedText";
|
|
import { ITableQueryData } from "components/TableContainer/TableContainer";
|
|
import TableCount from "components/TableContainer/TableCount";
|
|
import { SingleValue } from "react-select-5";
|
|
import DropdownWrapper from "components/forms/fields/DropdownWrapper";
|
|
import { CustomOptionType } from "components/forms/fields/DropdownWrapper/DropdownWrapper";
|
|
|
|
import EmptySoftwareTable from "pages/SoftwarePage/components/tables/EmptySoftwareTable";
|
|
import { IOSVersionsResponse } from "services/entities/operating_systems";
|
|
|
|
import generateTableConfig from "pages/DashboardPage/cards/OperatingSystems/OSTableConfig";
|
|
import { getPathWithQueryParams } from "utilities/url";
|
|
import { getNextLocationPath } from "utilities/helpers";
|
|
import { SelectedPlatform } from "interfaces/platform";
|
|
|
|
const baseClass = "software-os-table";
|
|
|
|
interface IRowProps extends Row {
|
|
original: {
|
|
os_version_id?: string;
|
|
};
|
|
}
|
|
|
|
interface ISoftwareOSTableProps {
|
|
router: InjectedRouter;
|
|
isSoftwareEnabled: boolean;
|
|
data?: IOSVersionsResponse;
|
|
perPage: number;
|
|
orderDirection: "asc" | "desc";
|
|
orderKey: string;
|
|
currentPage: number;
|
|
teamId?: number;
|
|
isLoading: boolean;
|
|
platform?: SelectedPlatform;
|
|
}
|
|
|
|
const PLATFORM_FILTER_OPTIONS = [
|
|
{
|
|
disabled: false,
|
|
label: "All platforms",
|
|
value: "all",
|
|
},
|
|
{
|
|
disabled: false,
|
|
label: "macOS",
|
|
value: "darwin",
|
|
},
|
|
{
|
|
disabled: false,
|
|
label: "Windows",
|
|
value: "windows",
|
|
},
|
|
{
|
|
disabled: false,
|
|
label: "Linux",
|
|
value: "linux",
|
|
},
|
|
{
|
|
disabled: false,
|
|
label: "ChromeOS",
|
|
value: "chrome",
|
|
},
|
|
{
|
|
disabled: false,
|
|
label: "iOS",
|
|
value: "ios",
|
|
},
|
|
{
|
|
disabled: false,
|
|
label: "iPadOS",
|
|
value: "ipados",
|
|
},
|
|
];
|
|
|
|
const SoftwareOSTable = ({
|
|
router,
|
|
isSoftwareEnabled,
|
|
data,
|
|
perPage,
|
|
orderDirection,
|
|
orderKey,
|
|
currentPage,
|
|
teamId,
|
|
isLoading,
|
|
platform,
|
|
}: ISoftwareOSTableProps) => {
|
|
const determineQueryParamChange = useCallback(
|
|
(newTableQuery: ITableQueryData) => {
|
|
const changedEntry = Object.entries(newTableQuery).find(([key, val]) => {
|
|
switch (key) {
|
|
case "sortDirection":
|
|
return val !== orderDirection;
|
|
case "sortHeader":
|
|
return val !== orderKey;
|
|
case "pageIndex":
|
|
return val !== currentPage;
|
|
case "platform":
|
|
return val !== platform;
|
|
default:
|
|
return false;
|
|
}
|
|
});
|
|
return changedEntry?.[0] ?? "";
|
|
},
|
|
[platform, currentPage, orderDirection, orderKey]
|
|
);
|
|
|
|
const generateNewQueryParams = useCallback(
|
|
(newTableQuery: ITableQueryData, changedParam: string) => {
|
|
return {
|
|
team_id: teamId,
|
|
order_direction: newTableQuery.sortDirection,
|
|
order_key: newTableQuery.sortHeader,
|
|
page: changedParam === "pageIndex" ? newTableQuery.pageIndex : 0,
|
|
platform,
|
|
};
|
|
},
|
|
[teamId, platform]
|
|
);
|
|
|
|
const onQueryChange = useCallback(
|
|
(newTableQuery: ITableQueryData) => {
|
|
// we want to determine which query param has changed in order to
|
|
// reset the page index to 0 if any other param has changed.
|
|
const changedParam = determineQueryParamChange(newTableQuery);
|
|
// if nothing has changed, don't update the route. this can happen when
|
|
// this handler is called on the initial render.
|
|
if (changedParam === "") return;
|
|
|
|
const newRoute = getNextLocationPath({
|
|
pathPrefix: PATHS.SOFTWARE_OS,
|
|
routeTemplate: "",
|
|
queryParams: generateNewQueryParams(newTableQuery, changedParam),
|
|
});
|
|
|
|
router.replace(newRoute);
|
|
},
|
|
[determineQueryParamChange, generateNewQueryParams, router]
|
|
);
|
|
|
|
const softwareTableHeaders = useMemo(() => {
|
|
if (!data) return [];
|
|
return generateTableConfig(teamId, router, {
|
|
includeName: true,
|
|
includeVulnerabilities: true,
|
|
includeIcon: true,
|
|
});
|
|
}, [data, router, teamId]);
|
|
|
|
const handleRowSelect = (row: IRowProps) => {
|
|
const path = getPathWithQueryParams(
|
|
PATHS.SOFTWARE_OS_DETAILS(Number(row.original.os_version_id)),
|
|
{ team_id: teamId }
|
|
);
|
|
|
|
router.push(path);
|
|
};
|
|
|
|
// Determines if a user should be able to filter the table
|
|
const hasData = data?.os_versions && data?.os_versions.length > 0;
|
|
const hasPlatformFilter = platform !== "all";
|
|
|
|
const showFilterHeaders = isSoftwareEnabled && (hasData || hasPlatformFilter);
|
|
|
|
const renderSoftwareCount = () => {
|
|
if (!data) return null;
|
|
|
|
return (
|
|
<>
|
|
<TableCount name="items" count={data?.count} />
|
|
{showFilterHeaders && data?.counts_updated_at && (
|
|
<LastUpdatedText
|
|
lastUpdatedAt={data.counts_updated_at}
|
|
customTooltipText={
|
|
<>
|
|
The last time software data was <br />
|
|
updated, including vulnerabilities <br />
|
|
and host counts.
|
|
</>
|
|
}
|
|
/>
|
|
)}
|
|
</>
|
|
);
|
|
};
|
|
|
|
const renderTableHelpText = () => {
|
|
return (
|
|
<div>
|
|
Seeing unexpected software or vulnerabilities?{" "}
|
|
<CustomLink
|
|
url={GITHUB_NEW_ISSUE_LINK}
|
|
text="File an issue on GitHub"
|
|
newTab
|
|
/>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
const handlePlatformFilterDropdownChange = (
|
|
platformSelected: SingleValue<CustomOptionType>
|
|
) => {
|
|
router?.replace(
|
|
getNextLocationPath({
|
|
pathPrefix: PATHS.SOFTWARE_OS,
|
|
queryParams: {
|
|
team_id: teamId,
|
|
order_direction: orderDirection,
|
|
order_key: orderKey,
|
|
page: 0,
|
|
platform: platformSelected?.value,
|
|
},
|
|
})
|
|
);
|
|
};
|
|
|
|
const renderPlatformDropdown = () => {
|
|
return (
|
|
<DropdownWrapper
|
|
name="os-platform-dropdown"
|
|
value={platform || "all"}
|
|
className={`${baseClass}__platform-dropdown`}
|
|
options={PLATFORM_FILTER_OPTIONS}
|
|
onChange={handlePlatformFilterDropdownChange}
|
|
variant="table-filter"
|
|
/>
|
|
);
|
|
};
|
|
|
|
return (
|
|
<div className={baseClass}>
|
|
<TableContainer
|
|
columnConfigs={softwareTableHeaders}
|
|
data={data?.os_versions ?? []}
|
|
isLoading={isLoading}
|
|
resultsTitle="items"
|
|
emptyComponent={() => (
|
|
<EmptySoftwareTable
|
|
tableName="operating systems"
|
|
isSoftwareDisabled={!isSoftwareEnabled}
|
|
noSearchQuery // non-searchable table renders not detecting by default
|
|
/>
|
|
)}
|
|
defaultSortHeader={orderKey}
|
|
defaultSortDirection={orderDirection}
|
|
pageIndex={currentPage}
|
|
manualSortBy
|
|
pageSize={perPage}
|
|
showMarkAllPages={false}
|
|
isAllPagesSelected={false}
|
|
customControl={showFilterHeaders ? renderPlatformDropdown : undefined}
|
|
disableNextPage={!data?.meta.has_next_results}
|
|
searchable={false}
|
|
onQueryChange={onQueryChange}
|
|
renderCount={renderSoftwareCount}
|
|
renderTableHelpText={renderTableHelpText}
|
|
disableMultiRowSelect
|
|
onSelectSingleRow={handleRowSelect}
|
|
/>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default SoftwareOSTable;
|