From 0428b06b7c02734e8beac4657a09b41983c42a21 Mon Sep 17 00:00:00 2001 From: Amardeepsingh Siglani Date: Wed, 11 Jan 2023 04:31:48 +0530 Subject: [PATCH 1/3] fixed issues; improved UX Signed-off-by: Amardeepsingh Siglani --- .../FieldMappingsTable.tsx | 25 +++++- .../FieldNameSelector.tsx | 4 +- .../containers/ConfigureFieldMapping.tsx | 84 ++++++++++++------- 3 files changed, 75 insertions(+), 38 deletions(-) diff --git a/public/pages/CreateDetector/components/ConfigureFieldMapping/components/RequiredFieldMapping/FieldMappingsTable.tsx b/public/pages/CreateDetector/components/ConfigureFieldMapping/components/RequiredFieldMapping/FieldMappingsTable.tsx index 9c445c5d7..1a255ffbd 100644 --- a/public/pages/CreateDetector/components/ConfigureFieldMapping/components/RequiredFieldMapping/FieldMappingsTable.tsx +++ b/public/pages/CreateDetector/components/ConfigureFieldMapping/components/RequiredFieldMapping/FieldMappingsTable.tsx @@ -6,6 +6,7 @@ import React, { Component } from 'react'; import { RouteComponentProps } from 'react-router-dom'; import { + CriteriaWithPagination, EuiBasicTableColumn, EuiEmptyPrompt, EuiIcon, @@ -42,12 +43,27 @@ interface FieldMappingsTableProps extends RouteCompon mappingProps: MappingProps[T]; } -interface FieldMappingsTableState {} +interface FieldMappingsTableState { + pageIndex: number; +} export default class FieldMappingsTable extends Component< FieldMappingsTableProps, FieldMappingsTableState > { + constructor(props: FieldMappingsTableProps) { + super(props); + this.state = { + pageIndex: 0, + }; + } + + private onTableChange = (nextValues: CriteriaWithPagination) => { + this.setState({ + pageIndex: nextValues.page.index, + }); + }; + render() { const { loading, indexFields, ruleFields } = this.props; let items: FieldMappingsTableItem[]; @@ -70,7 +86,6 @@ export default class FieldMappingsTable extends Compo { field: 'ruleFieldName', name: 'Rule field name', - sortable: true, dataType: 'string', width: '25%', render: (ruleFieldName: string) => ruleFieldName || DEFAULT_EMPTY_DATA, @@ -85,7 +100,6 @@ export default class FieldMappingsTable extends Compo { field: 'logFieldName', name: 'Log field name', - sortable: true, dataType: 'string', width: '45%', render: (logFieldName: string, entry: FieldMappingsTableItem) => { @@ -150,9 +164,12 @@ export default class FieldMappingsTable extends Compo loading={loading} items={items} columns={columns} - pagination={true} + pagination={{ + pageIndex: this.state.pageIndex, + }} sorting={sorting} isSelectable={false} + onTableChange={this.onTableChange} message={ []; + selectedOptions: EuiComboBoxOptionOption[] | undefined; errorMessage?: string; } @@ -32,7 +32,7 @@ export default class FieldNameSelector extends Component[]) => { - this.setState({ selectedOptions }); + this.setState({ selectedOptions: selectedOptions.length ? selectedOptions : undefined }); this.props.onChange(selectedOptions[0]?.label); }; diff --git a/public/pages/CreateDetector/components/ConfigureFieldMapping/containers/ConfigureFieldMapping.tsx b/public/pages/CreateDetector/components/ConfigureFieldMapping/containers/ConfigureFieldMapping.tsx index 8d5df4ec8..de0d142bd 100644 --- a/public/pages/CreateDetector/components/ConfigureFieldMapping/containers/ConfigureFieldMapping.tsx +++ b/public/pages/CreateDetector/components/ConfigureFieldMapping/containers/ConfigureFieldMapping.tsx @@ -80,13 +80,7 @@ export default class ConfigureFieldMapping extends Component< }); this.setState({ createdMappings: existingMappings, - mappingsData: { - ...mappingsView.response, - unmapped_field_aliases: [ - 'timestamp', - ...(mappingsView.response.unmapped_field_aliases || []), - ], - }, + mappingsData: mappingsView.response, }); this.updateMappingSharedState(existingMappings); } @@ -111,11 +105,16 @@ export default class ConfigureFieldMapping extends Component< return invalidFields; } - onMappingCreation = (ruleFieldName: string, indxFieldName: string): void => { + onMappingCreation = (ruleFieldName: string, indexFieldName: string): void => { const newMappings: ruleFieldToIndexFieldMap = { ...this.state.createdMappings, - [ruleFieldName]: indxFieldName, + [ruleFieldName]: indexFieldName, }; + + if (!indexFieldName) { + delete newMappings[ruleFieldName]; + } + const invalidMappingFieldNames = this.getInvalidMappingFieldNames(newMappings); this.setState({ createdMappings: newMappings, @@ -142,17 +141,28 @@ export default class ConfigureFieldMapping extends Component< ...createdMappings, }; - // read only data const mappedRuleFields: string[] = []; - const mappedLogFields: string[] = []; + const logFields: Set = new Set(mappingsData.unmapped_index_fields || []); + let pendingCount = mappingsData.unmapped_field_aliases?.length || 0; + const unmappedRuleFields = [...(mappingsData.unmapped_field_aliases || [])]; + Object.keys(mappingsData.properties).forEach((ruleFieldName) => { mappedRuleFields.unshift(ruleFieldName); - mappedLogFields.unshift(mappingsData.properties[ruleFieldName].path); + + // Need this check to avoid adding undefined value + // When user removes existing mapping for default mapped values, the mapping will be undefined + if (existingMappings[ruleFieldName]) { + logFields.add(existingMappings[ruleFieldName]); + } }); - // edit data - const ruleFields = [...(mappingsData.unmapped_field_aliases || [])]; - const indexFields = [...(mappingsData.unmapped_index_fields || [])]; + Object.keys(existingMappings).forEach((mappedRuleField) => { + if (unmappedRuleFields.includes(mappedRuleField)) { + pendingCount--; + } + }); + + const indexFieldOptions = Array.from(logFields); return (
@@ -168,24 +178,31 @@ export default class ConfigureFieldMapping extends Component< - {ruleFields.length > 0 ? ( + {unmappedRuleFields.length > 0 ? ( <> - -

- To generate accurate findings, we recommend mapping the following security rules - fields with the log field from your data source. -

-
+ {pendingCount > 0 ? ( + +

+ To generate accurate findings, we recommend mapping the following security rules + fields with the log fields in your data source. +

+
+ ) : ( + +

Your data source have been mapped with all security rule fields.

+
+ )} + - + {...this.props} loading={loading} - ruleFields={ruleFields} - indexFields={indexFields} + ruleFields={unmappedRuleFields} + indexFields={indexFieldOptions} mappingProps={{ type: MappingViewType.Edit, existingMappings, @@ -213,7 +230,7 @@ export default class ConfigureFieldMapping extends Component< buttonContent={
-

{`View mapped fields (${mappedRuleFields.length})`}

+

{`Default mapped fields (${mappedRuleFields.length})`}

} @@ -222,13 +239,16 @@ export default class ConfigureFieldMapping extends Component< initialIsOpen={false} > - + {...this.props} loading={loading} ruleFields={mappedRuleFields} - indexFields={mappedLogFields} + indexFields={indexFieldOptions} mappingProps={{ - type: MappingViewType.Readonly, + type: MappingViewType.Edit, + existingMappings, + invalidMappingFieldNames, + onMappingCreation: this.onMappingCreation, }} /> From 88e32ee7fd879a4ffb82a147ed2a2081bfce02e5 Mon Sep 17 00:00:00 2001 From: Amardeepsingh Siglani Date: Wed, 11 Jan 2023 04:35:38 +0530 Subject: [PATCH 2/3] removed unwanted change Signed-off-by: Amardeepsingh Siglani --- .../components/RequiredFieldMapping/FieldNameSelector.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/public/pages/CreateDetector/components/ConfigureFieldMapping/components/RequiredFieldMapping/FieldNameSelector.tsx b/public/pages/CreateDetector/components/ConfigureFieldMapping/components/RequiredFieldMapping/FieldNameSelector.tsx index e268edee3..2399714e2 100644 --- a/public/pages/CreateDetector/components/ConfigureFieldMapping/components/RequiredFieldMapping/FieldNameSelector.tsx +++ b/public/pages/CreateDetector/components/ConfigureFieldMapping/components/RequiredFieldMapping/FieldNameSelector.tsx @@ -14,7 +14,7 @@ interface SIEMFieldNameProps { } interface SIEMFieldNameState { - selectedOptions: EuiComboBoxOptionOption[] | undefined; + selectedOptions: EuiComboBoxOptionOption[]; errorMessage?: string; } @@ -32,7 +32,7 @@ export default class FieldNameSelector extends Component[]) => { - this.setState({ selectedOptions: selectedOptions.length ? selectedOptions : undefined }); + this.setState({ selectedOptions }); this.props.onChange(selectedOptions[0]?.label); }; From e6c9f2978eee27c2df78971262e6295702a48d57 Mon Sep 17 00:00:00 2001 From: Amardeepsingh Siglani Date: Wed, 11 Jan 2023 05:07:43 +0530 Subject: [PATCH 3/3] override default mapping with created Signed-off-by: Amardeepsingh Siglani --- .../containers/ConfigureFieldMapping.tsx | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/public/pages/CreateDetector/components/ConfigureFieldMapping/containers/ConfigureFieldMapping.tsx b/public/pages/CreateDetector/components/ConfigureFieldMapping/containers/ConfigureFieldMapping.tsx index de0d142bd..281225cd1 100644 --- a/public/pages/CreateDetector/components/ConfigureFieldMapping/containers/ConfigureFieldMapping.tsx +++ b/public/pages/CreateDetector/components/ConfigureFieldMapping/containers/ConfigureFieldMapping.tsx @@ -76,7 +76,9 @@ export default class ConfigureFieldMapping extends Component< if (mappingsView.ok) { const existingMappings = { ...this.state.createdMappings }; Object.keys(mappingsView.response.properties).forEach((ruleFieldName) => { - existingMappings[ruleFieldName] = mappingsView.response.properties[ruleFieldName].path; + existingMappings[ruleFieldName] = + this.state.createdMappings[ruleFieldName] || + mappingsView.response.properties[ruleFieldName].path; }); this.setState({ createdMappings: existingMappings,