Skip to content

Commit

Permalink
[8.x] [Security Solution] Event Renderer Virtualization (#193316) (#1…
Browse files Browse the repository at this point in the history
…96587)

# Backport

This will backport the following commits from `main` to `8.x`:
- [[Security Solution] Event Renderer Virtualization
(#193316)](#193316)

<!--- Backport version: 9.4.3 -->

### Questions ?
Please refer to the [Backport tool
documentation](https://github.com/sqren/backport)

<!--BACKPORT [{"author":{"name":"Jatin
Kathuria","email":"[email protected]"},"sourceCommit":{"committedDate":"2024-10-16T17:53:49Z","message":"[Security
Solution] Event Renderer Virtualization (#193316)\n\n##
Summary\r\n\r\nThis PR implements virtualization when Event Renderers
are enabled.\r\nIdeally from UX pespective nothing should change but
from performance\r\nperspective, the event renderers should be
scalable.\r\n\r\n### Testing checklist\r\n\r\n1. UX is working same as
before when Event Renderers are enabled.\r\n2. Operations such as
increasing page size from `10` to `100` are not\r\ntaking as much time
as before. Below operations can be used to test.\r\n a. Closing /
Opening Timeline\r\n b. Changes `Rows per page`\r\n c. Changes tabs from
query to any other and back.\r\n\r\n### Before\r\nIn below video, you
will notice how long it took to change `pageSize` to\r\n100 and all 100
rows are rendered at
once.\r\n\r\n\r\nhttps://github.com/user-attachments/assets/106669c9-bda8-4b7d-af3f-b64824bde397\r\n\r\n\r\n###
After\r\n\r\n\r\nhttps://github.com/user-attachments/assets/356d9e1f-caf1-4f88-9223-0e563939bf6b\r\n\r\n\r\n\r\n>
[!Note]\r\n> 1. Also test in small screen. The table should be
scrollable but\r\nnothing out of ordinary.\r\n> 2. Additionally, try to
load data which has `network_flow` process so\r\nas to create bigger and
varied Event Renderers.\r\n\r\n---------\r\n\r\nCo-authored-by: Cee Chen
<[email protected]>","sha":"fa92a8ede7bce32456e3d6a6307761b4209248f9","branchLabelMapping":{"^v9.0.0$":"main","^v8.16.0$":"8.x","^v(\\d+).(\\d+).\\d+$":"$1.$2"}},"sourcePullRequest":{"labels":["backport","release_note:skip","v9.0.0","Team:Threat
Hunting:Investigations","ci:project-deploy-security","v8.16.0"],"title":"[Security
Solution] Event Renderer
Virtualization","number":193316,"url":"https://github.com/elastic/kibana/pull/193316","mergeCommit":{"message":"[Security
Solution] Event Renderer Virtualization (#193316)\n\n##
Summary\r\n\r\nThis PR implements virtualization when Event Renderers
are enabled.\r\nIdeally from UX pespective nothing should change but
from performance\r\nperspective, the event renderers should be
scalable.\r\n\r\n### Testing checklist\r\n\r\n1. UX is working same as
before when Event Renderers are enabled.\r\n2. Operations such as
increasing page size from `10` to `100` are not\r\ntaking as much time
as before. Below operations can be used to test.\r\n a. Closing /
Opening Timeline\r\n b. Changes `Rows per page`\r\n c. Changes tabs from
query to any other and back.\r\n\r\n### Before\r\nIn below video, you
will notice how long it took to change `pageSize` to\r\n100 and all 100
rows are rendered at
once.\r\n\r\n\r\nhttps://github.com/user-attachments/assets/106669c9-bda8-4b7d-af3f-b64824bde397\r\n\r\n\r\n###
After\r\n\r\n\r\nhttps://github.com/user-attachments/assets/356d9e1f-caf1-4f88-9223-0e563939bf6b\r\n\r\n\r\n\r\n>
[!Note]\r\n> 1. Also test in small screen. The table should be
scrollable but\r\nnothing out of ordinary.\r\n> 2. Additionally, try to
load data which has `network_flow` process so\r\nas to create bigger and
varied Event Renderers.\r\n\r\n---------\r\n\r\nCo-authored-by: Cee Chen
<[email protected]>","sha":"fa92a8ede7bce32456e3d6a6307761b4209248f9"}},"sourceBranch":"main","suggestedTargetBranches":["8.x"],"targetPullRequestStates":[{"branch":"main","label":"v9.0.0","branchLabelMappingKey":"^v9.0.0$","isSourceBranch":true,"state":"MERGED","url":"https://github.com/elastic/kibana/pull/193316","number":193316,"mergeCommit":{"message":"[Security
Solution] Event Renderer Virtualization (#193316)\n\n##
Summary\r\n\r\nThis PR implements virtualization when Event Renderers
are enabled.\r\nIdeally from UX pespective nothing should change but
from performance\r\nperspective, the event renderers should be
scalable.\r\n\r\n### Testing checklist\r\n\r\n1. UX is working same as
before when Event Renderers are enabled.\r\n2. Operations such as
increasing page size from `10` to `100` are not\r\ntaking as much time
as before. Below operations can be used to test.\r\n a. Closing /
Opening Timeline\r\n b. Changes `Rows per page`\r\n c. Changes tabs from
query to any other and back.\r\n\r\n### Before\r\nIn below video, you
will notice how long it took to change `pageSize` to\r\n100 and all 100
rows are rendered at
once.\r\n\r\n\r\nhttps://github.com/user-attachments/assets/106669c9-bda8-4b7d-af3f-b64824bde397\r\n\r\n\r\n###
After\r\n\r\n\r\nhttps://github.com/user-attachments/assets/356d9e1f-caf1-4f88-9223-0e563939bf6b\r\n\r\n\r\n\r\n>
[!Note]\r\n> 1. Also test in small screen. The table should be
scrollable but\r\nnothing out of ordinary.\r\n> 2. Additionally, try to
load data which has `network_flow` process so\r\nas to create bigger and
varied Event Renderers.\r\n\r\n---------\r\n\r\nCo-authored-by: Cee Chen
<[email protected]>","sha":"fa92a8ede7bce32456e3d6a6307761b4209248f9"}},{"branch":"8.x","label":"v8.16.0","branchLabelMappingKey":"^v8.16.0$","isSourceBranch":false,"state":"NOT_CREATED"}]}]
BACKPORT-->

Co-authored-by: Jatin Kathuria <[email protected]>
  • Loading branch information
kibanamachine and logeekal authored Oct 16, 2024
1 parent 6909ffe commit fd658fb
Show file tree
Hide file tree
Showing 14 changed files with 491 additions and 265 deletions.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -1633,7 +1633,7 @@
"@types/react-router-dom": "^5.3.3",
"@types/react-syntax-highlighter": "^15.4.0",
"@types/react-test-renderer": "^17.0.2",
"@types/react-virtualized": "^9.21.22",
"@types/react-virtualized": "^9.21.30",
"@types/react-window": "^1.8.8",
"@types/react-window-infinite-loader": "^1.0.9",
"@types/redux-actions": "^2.6.1",
Expand Down
5 changes: 5 additions & 0 deletions x-pack/plugins/security_solution/common/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -513,3 +513,8 @@ export const CASE_ATTACHMENT_ENDPOINT_TYPE_ID = 'endpoint' as const;
*/
export const MAX_MANUAL_RULE_RUN_LOOKBACK_WINDOW_DAYS = 90;
export const MAX_MANUAL_RULE_RUN_BULK_SIZE = 100;

/*
* Whether it is a Jest environment
*/
export const JEST_ENVIRONMENT = typeof jest !== 'undefined';
Original file line number Diff line number Diff line change
Expand Up @@ -40,69 +40,67 @@ import { useStatefulRowRenderer } from './use_stateful_row_renderer';
* which focuses the current or next row, respectively.
* - A screen-reader-only message provides additional context and instruction
*/
export const StatefulRowRenderer = ({
ariaRowindex,
containerRef,
event,
lastFocusedAriaColindex,
rowRenderers,
timelineId,
}: {
ariaRowindex: number;
containerRef: React.MutableRefObject<HTMLDivElement | null>;
event: TimelineItem;
lastFocusedAriaColindex: number;
rowRenderers: RowRenderer[];
timelineId: string;
}) => {
const { focusOwnership, onFocus, onKeyDown, onOutsideClick } = useStatefulEventFocus({
export const StatefulRowRenderer = React.memo(
({
ariaRowindex,
colindexAttribute: ARIA_COLINDEX_ATTRIBUTE,
containerRef,
event,
lastFocusedAriaColindex,
onColumnFocused: noop,
rowindexAttribute: ARIA_ROWINDEX_ATTRIBUTE,
});

const { rowRenderer } = useStatefulRowRenderer({
data: event.ecs,
rowRenderers,
});

const content = useMemo(
() =>
rowRenderer && (
// eslint-disable-next-line jsx-a11y/no-noninteractive-element-interactions
<div className={getRowRendererClassName(ariaRowindex)} role="dialog" onFocus={onFocus}>
<EuiOutsideClickDetector onOutsideClick={onOutsideClick}>
<EuiFocusTrap clickOutsideDisables={true} disabled={focusOwnership !== 'owned'}>
<EuiScreenReaderOnly data-test-subj="eventRendererScreenReaderOnly">
<p>{i18n.YOU_ARE_IN_AN_EVENT_RENDERER(ariaRowindex)}</p>
</EuiScreenReaderOnly>
<EuiFlexGroup direction="column" onKeyDown={onKeyDown}>
<EuiFlexItem grow={true}>
{rowRenderer.renderRow({
data: event.ecs,
isDraggable: true,
scopeId: timelineId,
})}
</EuiFlexItem>
</EuiFlexGroup>
</EuiFocusTrap>
</EuiOutsideClickDetector>
</div>
),
[
timelineId,
}: {
ariaRowindex: number;
containerRef: React.MutableRefObject<HTMLDivElement | null>;
event: TimelineItem;
lastFocusedAriaColindex: number;
rowRenderers: RowRenderer[];
timelineId: string;
}) => {
const { focusOwnership, onFocus, onKeyDown, onOutsideClick } = useStatefulEventFocus({
ariaRowindex,
event.ecs,
focusOwnership,
onFocus,
onKeyDown,
onOutsideClick,
rowRenderer,
timelineId,
]
);
colindexAttribute: ARIA_COLINDEX_ATTRIBUTE,
containerRef,
lastFocusedAriaColindex,
onColumnFocused: noop,
rowindexAttribute: ARIA_ROWINDEX_ATTRIBUTE,
});

const { rowRenderer } = useStatefulRowRenderer({
data: event.ecs,
rowRenderers,
});

const row = useMemo(() => {
const result = rowRenderer?.renderRow({
data: event.ecs,
isDraggable: false,
scopeId: timelineId,
});
return result;
}, [rowRenderer, event.ecs, timelineId]);

const content = useMemo(
() =>
rowRenderer && (
// eslint-disable-next-line jsx-a11y/no-noninteractive-element-interactions
<div className={getRowRendererClassName(ariaRowindex)} role="dialog" onFocus={onFocus}>
<EuiOutsideClickDetector onOutsideClick={onOutsideClick}>
<EuiFocusTrap clickOutsideDisables={true} disabled={focusOwnership !== 'owned'}>
<EuiScreenReaderOnly data-test-subj="eventRendererScreenReaderOnly">
<p>{i18n.YOU_ARE_IN_AN_EVENT_RENDERER(ariaRowindex)}</p>
</EuiScreenReaderOnly>
<EuiFlexGroup direction="column" onKeyDown={onKeyDown}>
<EuiFlexItem grow={true}>{row}</EuiFlexItem>
</EuiFlexGroup>
</EuiFocusTrap>
</EuiOutsideClickDetector>
</div>
),
[ariaRowindex, focusOwnership, onFocus, onKeyDown, onOutsideClick, rowRenderer, row]
);

return content;
}
);

return content;
};
StatefulRowRenderer.displayName = 'StatefulRowRenderer';
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ interface UseStatefulRowRendererArgs {

export function useStatefulRowRenderer(args: UseStatefulRowRendererArgs) {
const { data, rowRenderers } = args;

const rowRenderer = useMemo(() => getRowRenderer({ data, rowRenderers }), [data, rowRenderers]);

const result = useMemo(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,10 +5,11 @@
* 2.0.
*/

import React, { useMemo } from 'react';
import React, { useMemo, useEffect } from 'react';
import type { EuiDataGridCellValueElementProps } from '@elastic/eui';
import type { SortColumnTable } from '@kbn/securitysolution-data-table';
import type { TimelineItem } from '@kbn/timelines-plugin/common';
import { JEST_ENVIRONMENT } from '../../../../../../common/constants';
import { useLicense } from '../../../../../common/hooks/use_license';
import { SourcererScopeName } from '../../../../../sourcerer/store/model';
import { useSourcererDataView } from '../../../../../sourcerer/containers';
Expand All @@ -21,6 +22,7 @@ import { TimelineControlColumnCellRender } from '../../unified_components/data_t
import type { ColumnHeaderOptions } from '../../../../../../common/types';
import { useTimelineColumns } from './use_timeline_columns';
import type { UnifiedTimelineDataGridCellContext } from '../../types';
import { useTimelineUnifiedDataTableContext } from '../../unified_components/data_table/use_timeline_unified_data_table_context';

interface UseTimelineControlColumnArgs {
columns: ColumnHeaderOptions[];
Expand Down Expand Up @@ -59,6 +61,58 @@ export const useTimelineControlColumn = ({
const ACTION_BUTTON_COUNT = isEnterprisePlus ? 6 : 5;
const { localColumns } = useTimelineColumns(columns);

const RowCellRender = useMemo(
() =>
function TimelineControlColumnCellRenderer(
props: EuiDataGridCellValueElementProps & UnifiedTimelineDataGridCellContext
) {
const ctx = useTimelineUnifiedDataTableContext();

useEffect(() => {
props.setCellProps({
className:
ctx.expanded?.id === events[props.rowIndex]?._id
? 'unifiedDataTable__cell--expanded'
: '',
});
});

/*
* In some cases, when number of events is updated
* but new table is not yet rendered it can result
* in the mismatch between the number of events v/s
* the number of rows in the table currently rendered.
*
* */
if ('rowIndex' in props && props.rowIndex >= events.length) return <></>;
return (
<TimelineControlColumnCellRender
rowIndex={props.rowIndex}
columnId={props.columnId}
timelineId={timelineId}
ariaRowindex={props.rowIndex}
checked={false}
columnValues=""
data={events[props.rowIndex].data}
ecsData={events[props.rowIndex].ecs}
loadingEventIds={EMPTY_STRING_ARRAY}
eventId={events[props.rowIndex]?._id}
index={props.rowIndex}
onEventDetailsPanelOpened={noOp}
onRowSelected={noOp}
refetch={refetch}
showCheckboxes={false}
setEventsLoading={noOp}
setEventsDeleted={noOp}
pinnedEventIds={pinnedEventIds}
eventIdToNoteIds={eventIdToNoteIds}
toggleShowNotes={onToggleShowNotes}
/>
);
},
[events, timelineId, refetch, pinnedEventIds, eventIdToNoteIds, onToggleShowNotes]
);

// We need one less when the unified components are enabled because the document expand is provided by the unified data table
const UNIFIED_COMPONENTS_ACTION_BUTTON_COUNT = ACTION_BUTTON_COUNT - 1;
return useMemo(() => {
Expand All @@ -84,49 +138,7 @@ export const useTimelineControlColumn = ({
/>
);
},
rowCellRender: (
props: EuiDataGridCellValueElementProps & UnifiedTimelineDataGridCellContext
) => {
/*
* In some cases, when number of events is updated
* but new table is not yet rendered it can result
* in the mismatch between the number of events v/s
* the number of rows in the table currently rendered.
*
* */
if ('rowIndex' in props && props.rowIndex >= events.length) return <></>;
props.setCellProps({
className:
props.expandedEventId === events[props.rowIndex]?._id
? 'unifiedDataTable__cell--expanded'
: '',
});

return (
<TimelineControlColumnCellRender
rowIndex={props.rowIndex}
columnId={props.columnId}
timelineId={timelineId}
ariaRowindex={props.rowIndex}
checked={false}
columnValues=""
data={events[props.rowIndex].data}
ecsData={events[props.rowIndex].ecs}
loadingEventIds={EMPTY_STRING_ARRAY}
eventId={events[props.rowIndex]?._id}
index={props.rowIndex}
onEventDetailsPanelOpened={noOp}
onRowSelected={noOp}
refetch={refetch}
showCheckboxes={false}
setEventsLoading={noOp}
setEventsDeleted={noOp}
pinnedEventIds={pinnedEventIds}
eventIdToNoteIds={eventIdToNoteIds}
toggleShowNotes={onToggleShowNotes}
/>
);
},
rowCellRender: JEST_ENVIRONMENT ? RowCellRender : React.memo(RowCellRender),
}));
} else {
return getDefaultControlColumn(ACTION_BUTTON_COUNT).map((x) => ({
Expand All @@ -142,11 +154,7 @@ export const useTimelineControlColumn = ({
sort,
activeTab,
timelineId,
refetch,
events,
pinnedEventIds,
eventIdToNoteIds,
onToggleShowNotes,
ACTION_BUTTON_COUNT,
RowCellRender,
]);
};
Loading

0 comments on commit fd658fb

Please sign in to comment.