Skip to content

Commit aef64e3

Browse files
Fix export hosts csv UI and editing hidden columns UI (#17691)
relates to #17632 fixes a UI bug that prevented exporting hosts by csv. also updates the edit hidden columns modal to use column id instead of accessor. Id is safer to use as we are now using a function definition for the accessor for mdm enrollment status and mdm server URL columns. - [x] Manual QA for all new/changed functionality
1 parent d778328 commit aef64e3

File tree

3 files changed

+44
-24
lines changed

3 files changed

+44
-24
lines changed

frontend/pages/hosts/ManageHostsPage/HostTableConfig.tsx

+30-10
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
117117
<HeaderCell value="Hosts" isSortedDesc={cellProps.column.isSortedDesc} />
118118
),
119119
accessor: "display_name",
120+
id: "display_name",
120121
Cell: (cellProps: IHostTableStringCellProps) => {
121122
if (
122123
// if the host is pending, we want to disable the link to host details
@@ -175,6 +176,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
175176
/>
176177
),
177178
accessor: "hostname",
179+
id: "hostname",
178180
Cell: (cellProps: IHostTableStringCellProps) => (
179181
<TextCell value={cellProps.cell.value} />
180182
),
@@ -188,6 +190,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
188190
/>
189191
),
190192
accessor: "computer_name",
193+
id: "computer_name",
191194
Cell: (cellProps: IHostTableStringCellProps) => (
192195
<TextCell value={cellProps.cell.value} />
193196
),
@@ -198,6 +201,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
198201
<HeaderCell value="Team" isSortedDesc={cellProps.column.isSortedDesc} />
199202
),
200203
accessor: "team_name",
204+
id: "team_name",
201205
Cell: (cellProps) => (
202206
<TextCell value={cellProps.cell.value} formatter={hostTeamName} />
203207
),
@@ -228,6 +232,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
228232
},
229233
disableSortBy: true,
230234
accessor: "status",
235+
id: "status",
231236
Cell: (cellProps: IHostTableStringCellProps) => {
232237
const value = cellProps.cell.value;
233238
const tooltip = {
@@ -241,6 +246,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
241246
Header: "Issues",
242247
disableSortBy: true,
243248
accessor: "issues",
249+
id: "issues",
244250
Cell: (cellProps: IIssuesCellProps) => (
245251
<IssueCell
246252
issues={cellProps.row.original.issues}
@@ -257,6 +263,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
257263
/>
258264
),
259265
accessor: "gigs_disk_space_available",
266+
id: "gigs_disk_space_available",
260267
Cell: (cellProps: IHostTableNumberCellProps) => {
261268
const {
262269
id,
@@ -286,6 +293,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
286293
/>
287294
),
288295
accessor: "os_version",
296+
id: "os_version",
289297
Cell: (cellProps: IHostTableStringCellProps) => (
290298
<TextCell value={cellProps.cell.value} />
291299
),
@@ -299,6 +307,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
299307
/>
300308
),
301309
accessor: "osquery_version",
310+
id: "osquery_version",
302311
Cell: (cellProps: IHostTableStringCellProps) => (
303312
<TextCell value={cellProps.cell.value} />
304313
),
@@ -308,6 +317,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
308317
Header: "Used by",
309318
disableSortBy: true,
310319
accessor: "device_mapping",
320+
id: "device_mapping",
311321
Cell: (cellProps: IDeviceUserCellProps) => {
312322
const numUsers = cellProps.cell.value?.length || 0;
313323
const users = condenseDeviceUsers(cellProps.cell.value || []);
@@ -350,6 +360,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
350360
/>
351361
),
352362
accessor: "primary_ip",
363+
id: "primary_ip",
353364
Cell: (cellProps: IHostTableStringCellProps) => (
354365
<TextCell value={cellProps.cell.value} />
355366
),
@@ -374,7 +385,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
374385
},
375386
disableSortBy: true,
376387
accessor: (originalRow) => originalRow.mdm.enrollment_status,
377-
id: "mdm_enrollment_status",
388+
id: "mdm.enrollment_status",
378389
Cell: HostMdmStatusCell,
379390
},
380391
{
@@ -397,7 +408,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
397408
},
398409
disableSortBy: true,
399410
accessor: (originalRow) => originalRow.mdm.server_url,
400-
id: "mdm_server_url",
411+
id: "mdm.server_url",
401412
Cell: (cellProps: IHostTableStringCellProps) => {
402413
if (cellProps.row.original.platform === "chrome") {
403414
return NotSupported;
@@ -421,6 +432,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
421432
/>
422433
),
423434
accessor: "public_ip",
435+
id: "public_ip",
424436
Cell: (cellProps: IHostTableStringCellProps) => {
425437
return (
426438
<TextCell value={cellProps.cell.value ?? DEFAULT_EMPTY_CELL_VALUE} />
@@ -450,6 +462,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
450462
);
451463
},
452464
accessor: "detail_updated_at",
465+
id: "detail_updated_at",
453466
Cell: (cellProps: IHostTableStringCellProps) => (
454467
<TextCell
455468
value={{ timeString: cellProps.cell.value }}
@@ -480,6 +493,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
480493
);
481494
},
482495
accessor: "seen_time",
496+
id: "seen_time",
483497
Cell: (cellProps: IHostTableStringCellProps) => (
484498
<TextCell
485499
value={{ timeString: cellProps.cell.value }}
@@ -493,6 +507,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
493507
<HeaderCell value="UUID" isSortedDesc={cellProps.column.isSortedDesc} />
494508
),
495509
accessor: "uuid",
510+
id: "uuid",
496511
Cell: (cellProps: IHostTableStringCellProps) => (
497512
<TooltipTruncatedTextCell value={cellProps.cell.value} />
498513
),
@@ -506,6 +521,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
506521
/>
507522
),
508523
accessor: "last_restarted_at",
524+
id: "last_restarted_at",
509525
Cell: (cellProps: IHostTableStringCellProps) => {
510526
const { platform, last_restarted_at } = cellProps.row.original;
511527

@@ -527,6 +543,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
527543
Header: "CPU",
528544
disableSortBy: true,
529545
accessor: "cpu_type",
546+
id: "cpu_type",
530547
Cell: (cellProps: IHostTableStringCellProps) => (
531548
<TextCell value={cellProps.cell.value} />
532549
),
@@ -537,6 +554,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
537554
<HeaderCell value="RAM" isSortedDesc={cellProps.column.isSortedDesc} />
538555
),
539556
accessor: "memory",
557+
id: "memory",
540558
Cell: (cellProps: IHostTableNumberCellProps) => (
541559
<TextCell value={cellProps.cell.value} formatter={humanHostMemory} />
542560
),
@@ -550,6 +568,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
550568
/>
551569
),
552570
accessor: "primary_mac",
571+
id: "primary_mac",
553572
Cell: (cellProps: IHostTableStringCellProps) => (
554573
<TextCell value={cellProps.cell.value} />
555574
),
@@ -563,6 +582,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
563582
/>
564583
),
565584
accessor: "hardware_serial",
585+
id: "hardware_serial",
566586
Cell: (cellProps: IHostTableStringCellProps) => (
567587
<TextCell value={cellProps.cell.value} />
568588
),
@@ -576,6 +596,7 @@ const allHostTableHeaders: IHostTableColumnConfig[] = [
576596
/>
577597
),
578598
accessor: "hardware_model",
599+
id: "hardware_model",
579600
Cell: (cellProps: IHostTableStringCellProps) => (
580601
<TextCell value={cellProps.cell.value} />
581602
),
@@ -590,8 +611,8 @@ const defaultHiddenColumns = [
590611
"public_ip",
591612
"cpu_type",
592613
// TODO: should those be mdm.<blah>?
593-
"mdm_server_url",
594-
"mdm_enrollment_status",
614+
"mdm.server_url",
615+
"mdm.enrollment_status",
595616
"memory",
596617
"uptime",
597618
"uuid",
@@ -616,17 +637,17 @@ const generateAvailableTableHeaders = ({
616637
// skip over column headers that are not shown in free observer tier
617638
if (isFreeTier && isOnlyObserver) {
618639
if (
619-
currentColumn.accessor === "team_name" ||
640+
currentColumn.id === "team_name" ||
620641
currentColumn.id === "selection"
621642
) {
622643
return columns;
623644
}
624645
// skip over column headers that are not shown in free admin/maintainer
625646
} else if (isFreeTier) {
626647
if (
627-
currentColumn.accessor === "team_name" ||
628-
currentColumn.id === "mdm_server_url" ||
629-
currentColumn.id === "mdm_enrollment_status"
648+
currentColumn.id === "team_name" ||
649+
currentColumn.id === "mdm.server_url" ||
650+
currentColumn.id === "mdm.enrollment_status"
630651
) {
631652
return columns;
632653
}
@@ -646,7 +667,6 @@ const generateAvailableTableHeaders = ({
646667

647668
/**
648669
* Will generate a host table column configuration that a user currently sees.
649-
*
650670
*/
651671
const generateVisibleTableColumns = ({
652672
hiddenColumns,
@@ -660,7 +680,7 @@ const generateVisibleTableColumns = ({
660680
// remove columns set as hidden by the user.
661681
return generateAvailableTableHeaders({ isFreeTier, isOnlyObserver }).filter(
662682
(column) => {
663-
return !hiddenColumns.includes(column.accessor as string);
683+
return !hiddenColumns.includes(column.id as string);
664684
}
665685
);
666686
};

frontend/pages/hosts/ManageHostsPage/ManageHostsPage.tsx

+6-4
Original file line numberDiff line numberDiff line change
@@ -1295,10 +1295,12 @@ const ManageHostsPage = ({
12951295
isOnlyObserver,
12961296
});
12971297

1298-
const columnAccessors = tableColumns
1299-
.map((column) => (column.accessor ? column.accessor : ""))
1300-
.filter((element) => element);
1301-
visibleColumns = columnAccessors.join(",");
1298+
const columnIds = tableColumns
1299+
.map((column) => (column.id ? column.id : ""))
1300+
// "selection" colum does not include any relevent data for the CSV
1301+
// so we filter it out.
1302+
.filter((element) => element !== "" && element !== "selection");
1303+
visibleColumns = columnIds.join(",");
13021304
}
13031305

13041306
let options = {

frontend/pages/hosts/ManageHostsPage/components/EditColumnsModal/EditColumnsModal.jsx

+8-10
Original file line numberDiff line numberDiff line change
@@ -12,17 +12,17 @@ const useCheckboxListStateManagement = (allColumns, hiddenColumns) => {
1212
return allColumns.map((column) => {
1313
return {
1414
name: column.title,
15-
accessor: column.accessor,
16-
isChecked: !hiddenColumns.includes(column.accessor),
15+
id: column.id,
16+
isChecked: !hiddenColumns.includes(column.id),
1717
disableHidden: column.disableHidden,
1818
};
1919
});
2020
});
2121

22-
const updateColumnItems = (columnAccessor) => {
22+
const updateColumnItems = (columnId) => {
2323
setColumnItems((prevState) => {
2424
const selectedColumn = columnItems.find(
25-
(column) => column.accessor === columnAccessor
25+
(column) => column.id === columnId
2626
);
2727
const updatedColumn = {
2828
...selectedColumn,
@@ -31,9 +31,7 @@ const useCheckboxListStateManagement = (allColumns, hiddenColumns) => {
3131

3232
// this is replacing the column object with the updatedColumn we just created.
3333
const newState = prevState.map((currentColumn) => {
34-
return currentColumn.accessor === columnAccessor
35-
? updatedColumn
36-
: currentColumn;
34+
return currentColumn.id === columnId ? updatedColumn : currentColumn;
3735
});
3836
return newState;
3937
});
@@ -45,7 +43,7 @@ const useCheckboxListStateManagement = (allColumns, hiddenColumns) => {
4543
const getHiddenColumns = (columns) => {
4644
return columns
4745
.filter((column) => !column.isChecked)
48-
.map((column) => column.accessor);
46+
.map((column) => column.id);
4947
};
5048

5149
const EditColumnsModal = ({
@@ -67,11 +65,11 @@ const EditColumnsModal = ({
6765
{columnItems.map((column) => {
6866
if (column.disableHidden) return null;
6967
return (
70-
<div key={column.accessor}>
68+
<div key={column.id}>
7169
<Checkbox
7270
name={column.name}
7371
value={column.isChecked}
74-
onChange={() => updateColumnItems(column.accessor)}
72+
onChange={() => updateColumnItems(column.id)}
7573
>
7674
<span>{column.name}</span>
7775
</Checkbox>

0 commit comments

Comments
 (0)