From f5bad326b1e08cd725d18eff7b5d0236355be070 Mon Sep 17 00:00:00 2001 From: Kerry Gallagher Date: Mon, 15 Feb 2021 15:32:52 +0000 Subject: [PATCH] [Logs UI] Use useMlHref hook for ML links (#90935) (#91395) * Use useMlHref hook for ML related links --- .../analyze_in_ml_button.tsx | 110 ++++-------------- .../infra/public/hooks/use_link_props.tsx | 7 +- .../analyze_dataset_in_ml_action.tsx | 44 +++++-- .../top_categories/top_categories_section.tsx | 21 +++- .../sections/anomalies/log_entry_example.tsx | 49 ++++++-- 5 files changed, 120 insertions(+), 111 deletions(-) diff --git a/x-pack/plugins/infra/public/components/logging/log_analysis_results/analyze_in_ml_button.tsx b/x-pack/plugins/infra/public/components/logging/log_analysis_results/analyze_in_ml_button.tsx index c520243b5b24e..00c6b1f93ef88 100644 --- a/x-pack/plugins/infra/public/components/logging/log_analysis_results/analyze_in_ml_button.tsx +++ b/x-pack/plugins/infra/public/components/logging/log_analysis_results/analyze_in_ml_button.tsx @@ -7,97 +7,31 @@ import { EuiButton } from '@elastic/eui'; import { FormattedMessage } from '@kbn/i18n/react'; -import React from 'react'; -import { encode } from 'rison-node'; -import { TimeRange } from '../../../../common/http_api/shared/time_range'; -import { useLinkProps, LinkDescriptor } from '../../../hooks/use_link_props'; +import React, { useCallback } from 'react'; +import { useKibanaContextForPlugin } from '../../../hooks/use_kibana'; +import { shouldHandleLinkEvent } from '../../../hooks/use_link_props'; export const AnalyzeInMlButton: React.FunctionComponent<{ - jobId: string; - partition?: string; - timeRange: TimeRange; -}> = ({ jobId, partition, timeRange }) => { - const linkProps = useLinkProps( - typeof partition === 'string' - ? getEntitySpecificSingleMetricViewerLink(jobId, timeRange, { - 'event.dataset': partition, - }) - : getOverallAnomalyExplorerLinkDescriptor(jobId, timeRange) - ); - const buttonLabel = ( - + href?: string; +}> = ({ href }) => { + const { + services: { application }, + } = useKibanaContextForPlugin(); + + const handleClick = useCallback( + (e) => { + if (!href || !shouldHandleLinkEvent(e)) return; + application.navigateToUrl(href); + }, + [href, application] ); - return typeof partition === 'string' ? ( - - {buttonLabel} - - ) : ( - - {buttonLabel} + + return ( + + ); }; - -export const getOverallAnomalyExplorerLinkDescriptor = ( - jobId: string, - timeRange: TimeRange -): LinkDescriptor => { - const { from, to } = convertTimeRangeToParams(timeRange); - - const _g = encode({ - ml: { - jobIds: [jobId], - }, - time: { - from, - to, - }, - }); - - return { - app: 'ml', - pathname: '/explorer', - search: { _g }, - }; -}; - -export const getEntitySpecificSingleMetricViewerLink = ( - jobId: string, - timeRange: TimeRange, - entities: Record -): LinkDescriptor => { - const { from, to } = convertTimeRangeToParams(timeRange); - - const _g = encode({ - ml: { - jobIds: [jobId], - }, - time: { - from, - to, - mode: 'absolute', - }, - }); - - const _a = encode({ - mlTimeSeriesExplorer: { - entities, - }, - }); - - return { - app: 'ml', - pathname: '/timeseriesexplorer', - search: { _g, _a }, - }; -}; - -const convertTimeRangeToParams = (timeRange: TimeRange): { from: string; to: string } => { - return { - from: new Date(timeRange.startTime).toISOString(), - to: new Date(timeRange.endTime).toISOString(), - }; -}; diff --git a/x-pack/plugins/infra/public/hooks/use_link_props.tsx b/x-pack/plugins/infra/public/hooks/use_link_props.tsx index 225ed5ae4a191..72a538cd56281 100644 --- a/x-pack/plugins/infra/public/hooks/use_link_props.tsx +++ b/x-pack/plugins/infra/public/hooks/use_link_props.tsx @@ -69,9 +69,10 @@ export const useLinkProps = ( const onClick = useMemo(() => { return (e: React.MouseEvent | React.MouseEvent) => { - if (e.defaultPrevented || isModifiedEvent(e)) { + if (!shouldHandleLinkEvent(e)) { return; } + e.preventDefault(); const navigate = () => { @@ -119,3 +120,7 @@ const validateParams = ({ app, pathname, hash, search }: LinkDescriptor) => { const isModifiedEvent = (event: any) => !!(event.metaKey || event.altKey || event.ctrlKey || event.shiftKey); + +export const shouldHandleLinkEvent = ( + e: React.MouseEvent | React.MouseEvent +) => !e.defaultPrevented && !isModifiedEvent(e); diff --git a/x-pack/plugins/infra/public/pages/logs/log_entry_categories/sections/top_categories/analyze_dataset_in_ml_action.tsx b/x-pack/plugins/infra/public/pages/logs/log_entry_categories/sections/top_categories/analyze_dataset_in_ml_action.tsx index ba3553611c0e6..15e27705395bb 100644 --- a/x-pack/plugins/infra/public/pages/logs/log_entry_categories/sections/top_categories/analyze_dataset_in_ml_action.tsx +++ b/x-pack/plugins/infra/public/pages/logs/log_entry_categories/sections/top_categories/analyze_dataset_in_ml_action.tsx @@ -6,12 +6,14 @@ */ import { EuiButtonIcon, EuiToolTip } from '@elastic/eui'; +import moment from 'moment'; import { i18n } from '@kbn/i18n'; -import React from 'react'; - +import React, { useCallback } from 'react'; +import { useKibanaContextForPlugin } from '../../../../../hooks/use_kibana'; import { TimeRange } from '../../../../../../common/time/time_range'; -import { getEntitySpecificSingleMetricViewerLink } from '../../../../../components/logging/log_analysis_results'; -import { useLinkProps } from '../../../../../hooks/use_link_props'; +import { useMlHref, ML_PAGES } from '../../../../../../../ml/public'; +import { partitionField } from '../../../../../../common/log_analysis/job_parameters'; +import { shouldHandleLinkEvent } from '../../../../../hooks/use_link_props'; export const AnalyzeCategoryDatasetInMlAction: React.FunctionComponent<{ categorizationJobId: string; @@ -19,11 +21,32 @@ export const AnalyzeCategoryDatasetInMlAction: React.FunctionComponent<{ dataset: string; timeRange: TimeRange; }> = ({ categorizationJobId, categoryId, dataset, timeRange }) => { - const linkProps = useLinkProps( - getEntitySpecificSingleMetricViewerLink(categorizationJobId, timeRange, { - 'event.dataset': dataset, - mlcategory: `${categoryId}`, - }) + const { + services: { ml, http, application }, + } = useKibanaContextForPlugin(); + + const viewAnomalyInMachineLearningLink = useMlHref(ml, http.basePath.get(), { + page: ML_PAGES.SINGLE_METRIC_VIEWER, + pageState: { + jobIds: [categorizationJobId], + timeRange: { + from: moment(timeRange.startTime).format('YYYY-MM-DDTHH:mm:ss.SSSZ'), + to: moment(timeRange.endTime).format('YYYY-MM-DDTHH:mm:ss.SSSZ'), + mode: 'absolute', + }, + entities: { + [partitionField]: dataset, + mlcategory: `${categoryId}`, + }, + }, + }); + + const handleClick = useCallback( + (e) => { + if (!viewAnomalyInMachineLearningLink || !shouldHandleLinkEvent(e)) return; + application.navigateToUrl(viewAnomalyInMachineLearningLink); + }, + [application, viewAnomalyInMachineLearningLink] ); return ( @@ -32,7 +55,8 @@ export const AnalyzeCategoryDatasetInMlAction: React.FunctionComponent<{ aria-label={analyseCategoryDatasetInMlButtonLabel} iconType="machineLearningApp" data-test-subj="analyzeCategoryDatasetInMlButton" - {...linkProps} + href={viewAnomalyInMachineLearningLink} + onClick={handleClick} /> ); diff --git a/x-pack/plugins/infra/public/pages/logs/log_entry_categories/sections/top_categories/top_categories_section.tsx b/x-pack/plugins/infra/public/pages/logs/log_entry_categories/sections/top_categories/top_categories_section.tsx index 1aa6aabf864cc..f5b94bce74e67 100644 --- a/x-pack/plugins/infra/public/pages/logs/log_entry_categories/sections/top_categories/top_categories_section.tsx +++ b/x-pack/plugins/infra/public/pages/logs/log_entry_categories/sections/top_categories/top_categories_section.tsx @@ -6,6 +6,7 @@ */ import { EuiFlexGroup, EuiFlexItem, EuiLoadingSpinner, EuiSpacer, EuiTitle } from '@elastic/eui'; +import moment from 'moment'; import { i18n } from '@kbn/i18n'; import React from 'react'; @@ -18,6 +19,8 @@ import { AnalyzeInMlButton } from '../../../../../components/logging/log_analysi import { DatasetsSelector } from '../../../../../components/logging/log_analysis_results/datasets_selector'; import { TopCategoriesTable } from './top_categories_table'; import { SortOptions, ChangeSortOptions } from '../../use_log_entry_categories_results'; +import { useKibanaContextForPlugin } from '../../../../../hooks/use_kibana'; +import { useMlHref, ML_PAGES } from '../../../../../../../ml/public'; export const TopCategoriesSection: React.FunctionComponent<{ availableDatasets: string[]; @@ -48,6 +51,22 @@ export const TopCategoriesSection: React.FunctionComponent<{ sortOptions, changeSortOptions, }) => { + const { + services: { ml, http }, + } = useKibanaContextForPlugin(); + + const analyzeInMlLink = useMlHref(ml, http.basePath.get(), { + page: ML_PAGES.ANOMALY_EXPLORER, + pageState: { + jobIds: [jobId], + timeRange: { + from: moment(timeRange.startTime).format('YYYY-MM-DDTHH:mm:ss.SSSZ'), + to: moment(timeRange.endTime).format('YYYY-MM-DDTHH:mm:ss.SSSZ'), + mode: 'absolute', + }, + }, + }); + return ( <> @@ -66,7 +85,7 @@ export const TopCategoriesSection: React.FunctionComponent<{ /> - + diff --git a/x-pack/plugins/infra/public/pages/logs/log_entry_rate/sections/anomalies/log_entry_example.tsx b/x-pack/plugins/infra/public/pages/logs/log_entry_rate/sections/anomalies/log_entry_example.tsx index 1a794e6f78c39..4362f412d5a78 100644 --- a/x-pack/plugins/infra/public/pages/logs/log_entry_rate/sections/anomalies/log_entry_example.tsx +++ b/x-pack/plugins/infra/public/pages/logs/log_entry_rate/sections/anomalies/log_entry_example.tsx @@ -9,6 +9,8 @@ import React, { useMemo, useCallback, useState } from 'react'; import moment from 'moment'; import { encode } from 'rison-node'; import { i18n } from '@kbn/i18n'; +import { useMlHref, ML_PAGES } from '../../../../../../../ml/public'; +import { useKibanaContextForPlugin } from '../../../../../hooks/use_kibana'; import { euiStyled } from '../../../../../../../../../src/plugins/kibana_react/common'; import { getFriendlyNameForPartitionId } from '../../../../../../common/log_analysis'; import { @@ -25,10 +27,9 @@ import { LogColumnHeadersWrapper, LogColumnHeader, } from '../../../../../components/logging/log_text_stream/column_headers'; -import { useLinkProps } from '../../../../../hooks/use_link_props'; +import { useLinkProps, shouldHandleLinkEvent } from '../../../../../hooks/use_link_props'; import { TimeRange } from '../../../../../../common/time/time_range'; import { partitionField } from '../../../../../../common/log_analysis/job_parameters'; -import { getEntitySpecificSingleMetricViewerLink } from '../../../../../components/logging/log_analysis_results/analyze_in_ml_button'; import { LogEntryExample, isCategoryAnomaly } from '../../../../../../common/log_analysis'; import { LogColumnConfiguration, @@ -82,6 +83,9 @@ export const LogEntryExampleMessage: React.FunctionComponent = ({ timeRange, anomaly, }) => { + const { + services: { ml, http, application }, + } = useKibanaContextForPlugin(); const [isHovered, setIsHovered] = useState(false); const [isMenuOpen, setIsMenuOpen] = useState(false); const openMenu = useCallback(() => setIsMenuOpen(true), []); @@ -114,15 +118,32 @@ export const LogEntryExampleMessage: React.FunctionComponent = ({ }, }); - const viewAnomalyInMachineLearningLinkProps = useLinkProps( - getEntitySpecificSingleMetricViewerLink(anomaly.jobId, timeRange, { - [partitionField]: dataset, - ...(isCategoryAnomaly(anomaly) ? { mlcategory: anomaly.categoryId } : {}), - }) + const viewAnomalyInMachineLearningLink = useMlHref(ml, http.basePath.get(), { + page: ML_PAGES.SINGLE_METRIC_VIEWER, + pageState: { + jobIds: [anomaly.jobId], + timeRange: { + from: moment(timeRange.startTime).format('YYYY-MM-DDTHH:mm:ss.SSSZ'), + to: moment(timeRange.endTime).format('YYYY-MM-DDTHH:mm:ss.SSSZ'), + mode: 'absolute', + }, + entities: { + [partitionField]: dataset, + ...(isCategoryAnomaly(anomaly) ? { mlcategory: anomaly.categoryId } : {}), + }, + }, + }); + + const handleMlLinkClick = useCallback( + (e) => { + if (!viewAnomalyInMachineLearningLink || !shouldHandleLinkEvent(e)) return; + application.navigateToUrl(viewAnomalyInMachineLearningLink); + }, + [viewAnomalyInMachineLearningLink, application] ); const menuItems = useMemo(() => { - if (!viewInStreamLinkProps.onClick || !viewAnomalyInMachineLearningLinkProps.onClick) { + if (!viewInStreamLinkProps.onClick || !viewAnomalyInMachineLearningLink) { return undefined; } @@ -140,11 +161,17 @@ export const LogEntryExampleMessage: React.FunctionComponent = ({ }, { label: VIEW_ANOMALY_IN_ML_LABEL, - onClick: viewAnomalyInMachineLearningLinkProps.onClick, - href: viewAnomalyInMachineLearningLinkProps.href, + onClick: handleMlLinkClick, + href: viewAnomalyInMachineLearningLink, }, ]; - }, [id, openLogEntryFlyout, viewInStreamLinkProps, viewAnomalyInMachineLearningLinkProps]); + }, [ + id, + openLogEntryFlyout, + viewInStreamLinkProps, + viewAnomalyInMachineLearningLink, + handleMlLinkClick, + ]); return (