diff --git a/opensearch_dashboards.json b/opensearch_dashboards.json index 551bd678..5aa5d906 100644 --- a/opensearch_dashboards.json +++ b/opensearch_dashboards.json @@ -8,7 +8,7 @@ "optionalPlugins": [ "dataSource", "dataSourceManagement", - "dataExplorer" + "assistantDashboards" ], "requiredPlugins": [ "opensearchDashboardsUtils", diff --git a/public/components/DiscoverAction/SuggestAnomalyDetector.test.tsx b/public/components/DiscoverAction/SuggestAnomalyDetector.test.tsx index cca54108..63e753c7 100644 --- a/public/components/DiscoverAction/SuggestAnomalyDetector.test.tsx +++ b/public/components/DiscoverAction/SuggestAnomalyDetector.test.tsx @@ -16,50 +16,13 @@ import { import { Provider } from 'react-redux'; import configureStore from '../../redux/configureStore'; -import GenerateAnomalyDetector from './SuggestAnomalyDetector'; -import { DiscoverActionContext } from '../../../../../src/plugins/data_explorer/public'; +import SuggestAnomalyDetector from './SuggestAnomalyDetector'; import { fieldFormatsMock } from '../../../../../src/plugins/data/common/field_formats/mocks'; import { IndexPattern } from '../../../../../src/plugins/data/common'; import userEvent from '@testing-library/user-event'; import { HttpFetchOptionsWithPath } from '../../../../../src/core/public'; import { BASE_NODE_API_PATH } from '../../../utils/constants'; - -const notifications = { - toasts: { - addDanger: jest.fn().mockName('addDanger'), - addSuccess: jest.fn().mockName('addSuccess'), - } -}; - -const getNotifications = () => { - return notifications; -} - -jest.mock('../../services', () => ({ - ...jest.requireActual('../../services'), - getNotifications: getNotifications, -})); - -const renderWithRouter = (context: DiscoverActionContext) => ({ - ...render( - - - - ( - - - - )} - /> - - - - ), -}); +import { getQueryService } from '../../services'; export function shouldReadFieldFromDocValues(aggregatable: boolean, opensearchType: string) { return ( @@ -69,7 +32,6 @@ export function shouldReadFieldFromDocValues(aggregatable: boolean, opensearchTy ); } - function stubbedSampleFields() { return [ ['bytes', 'long', true, true, { count: 10 }], @@ -141,108 +103,123 @@ function createIndexPattern(id: string): IndexPattern { }; } -const expectedAnomalyDetector = { - name: "test-pattern_anomaly_detector", - description: "Created based on the OpenSearch Assistant", - indices: ["test-pattern"], - filterQuery: { - match_all: {} - }, - uiMetadata: { - features: { - feature_responseLatency: { - featureType: "simple_aggs", - aggregationBy: "avg", - aggregationOf: "responseLatency" - }, - feature_response: { - featureType: "simple_aggs", - aggregationBy: "sum", - aggregationOf: "response" - } - }, - filters: [] - }, - featureAttributes: [ - { - featureName: "feature_responseLatency", - featureEnabled: true, - importance: 1, - aggregationQuery: { - feature_response_latency: { - avg: { - field: "responseLatency" - } - } - } - }, - { - featureName: "feature_response", - featureEnabled: true, - importance: 1, - aggregationQuery: { - feature_response: { - sum: { - field: "response" - } - } - } - } - ], - timeField: "timestamp", - detectionInterval: { - period: { - interval: 10, - unit: "Minutes" - } - }, - windowDelay: { - period: { - interval: 1, - unit: "Minutes" - } - }, - shingleSize: 8, - categoryField: ["ip"] +const mockedIndexPattern = createIndexPattern('test-pattern'); + +const notifications = { + toasts: { + addDanger: jest.fn().mockName('addDanger'), + addSuccess: jest.fn().mockName('addSuccess'), + } }; +const getNotifications = () => { + return notifications; +} + +jest.mock('../../services', () => ({ + ...jest.requireActual('../../services'), + getNotifications: getNotifications, + getQueryService: jest.fn().mockReturnValue({ + queryString: { + getQuery: jest.fn(), + }, + }), + getIndexPatternService: () => ({ + get: () => (mockedIndexPattern) + }) +})); + +const renderWithRouter = () => ({ + ...render( + + + + ( + + + + )} + /> + + + + ), +}); + describe('GenerateAnomalyDetector spec', () => { - describe('Renders loading component', () => { + describe('Renders failed', () => { + beforeEach(() => { + jest.clearAllMocks(); + }); + + it('renders with invalid dataset type', async () => { + const queryService = getQueryService(); + queryService.queryString.getQuery.mockReturnValue({ + dataset: { + id: undefined, + title: undefined, + type: 'INDEX' + }, + }); + + + const { queryByText } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).toBeNull(); + + await waitFor(() => { + expect(getNotifications().toasts.addDanger).toHaveBeenCalledTimes(1); + expect(getNotifications().toasts.addDanger).toHaveBeenCalledWith( + 'Unsupported dataset type' + ); + }); + }); + it('renders empty component', async () => { - httpClientMock.post = jest.fn().mockResolvedValue({ - ok: true, - generatedParameters: { - categoryField: '', - aggregationField: '', - aggregationMethod: '', - dateFields: '', + const queryService = getQueryService(); + queryService.queryString.getQuery.mockReturnValue({ + dataset: { + id: undefined, + title: undefined, + type: 'INDEX_PATTERN' }, }); - const context = { - indexPattern: createIndexPattern(''), - }; - const { queryByText } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).toBeNull(); + const { queryByText } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).toBeNull(); await waitFor(() => { expect(getNotifications().toasts.addDanger).toHaveBeenCalledTimes(1); expect(getNotifications().toasts.addDanger).toHaveBeenCalledWith( - 'Cannot extract index pattern from the context' + 'Cannot extract complete index info from the context' ); }); }); + }); + + describe('Renders loading component', () => { + beforeEach(() => { + jest.clearAllMocks(); + const queryService = getQueryService(); + queryService.queryString.getQuery.mockReturnValue({ + dataset: { + id: 'test-pattern', + title: 'test-pattern', + type: 'INDEX_PATTERN', + timeFieldName: '@timestamp', + }, + }); + }); it('renders with empty generated parameters', async () => { httpClientMock.post = jest.fn().mockResolvedValue({ ok: true, }); - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(getNotifications().toasts.addDanger).toHaveBeenCalledTimes(1); @@ -263,11 +240,8 @@ describe('GenerateAnomalyDetector spec', () => { }, }); - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(getNotifications().toasts.addDanger).toHaveBeenCalledTimes(1); @@ -288,11 +262,8 @@ describe('GenerateAnomalyDetector spec', () => { }, }); - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(getNotifications().toasts.addDanger).toHaveBeenCalledTimes(1); @@ -313,11 +284,8 @@ describe('GenerateAnomalyDetector spec', () => { }, }); - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(getNotifications().toasts.addDanger).toHaveBeenCalledTimes(1); @@ -338,11 +306,8 @@ describe('GenerateAnomalyDetector spec', () => { }, }); - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(queryByText('Create detector')).not.toBeNull(); @@ -354,8 +319,20 @@ describe('GenerateAnomalyDetector spec', () => { }); - describe('Test API calls', () => { + beforeEach(() => { + jest.clearAllMocks(); + const queryService = getQueryService(); + queryService.queryString.getQuery.mockReturnValue({ + dataset: { + id: 'test-pattern', + title: 'test-pattern', + type: 'INDEX_PATTERN', + timeFieldName: '@timestamp', + }, + }); + }); + it('All API calls execute successfully', async () => { httpClientMock.post = jest.fn((pathOrOptions: string | HttpFetchOptionsWithPath) => { const url = typeof pathOrOptions === 'string' ? pathOrOptions : pathOrOptions.path; @@ -384,11 +361,8 @@ describe('GenerateAnomalyDetector spec', () => { } }); - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText, getByTestId } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText, getByTestId } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(queryByText('Generating parameters...')).toBeNull(); expect(queryByText('Create detector')).not.toBeNull(); @@ -407,15 +381,6 @@ describe('GenerateAnomalyDetector spec', () => { body: JSON.stringify({ index: 'test-pattern' }), } ); - expect(httpClientMock.post).toHaveBeenCalledWith( - `${BASE_NODE_API_PATH}/detectors`, - { - body: JSON.stringify(expectedAnomalyDetector), - } - ); - expect(httpClientMock.post).toHaveBeenCalledWith( - `${BASE_NODE_API_PATH}/detectors/test/start` - ); expect(getNotifications().toasts.addSuccess).toHaveBeenCalledTimes(1); }); }); @@ -426,11 +391,8 @@ describe('GenerateAnomalyDetector spec', () => { error: 'Generate parameters failed' }); - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(getNotifications().toasts.addDanger).toHaveBeenCalledTimes(1); expect(getNotifications().toasts.addDanger).toHaveBeenCalledWith( @@ -472,12 +434,8 @@ describe('GenerateAnomalyDetector spec', () => { }, }); - - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText, getByTestId } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText, getByTestId } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(queryByText('Generating parameters...')).toBeNull(); @@ -539,11 +497,8 @@ describe('GenerateAnomalyDetector spec', () => { }); - const context = { - indexPattern: createIndexPattern('test-pattern'), - } - const { queryByText, getByTestId } = renderWithRouter(context); - expect(queryByText('Suggest anomaly detector')).not.toBeNull(); + const { queryByText, getByTestId } = renderWithRouter(); + expect(queryByText('Suggested anomaly detector')).not.toBeNull(); await waitFor(() => { expect(queryByText('Generating parameters...')).toBeNull(); diff --git a/public/components/DiscoverAction/SuggestAnomalyDetector.tsx b/public/components/DiscoverAction/SuggestAnomalyDetector.tsx index a5ec1a59..7831c18c 100644 --- a/public/components/DiscoverAction/SuggestAnomalyDetector.tsx +++ b/public/components/DiscoverAction/SuggestAnomalyDetector.tsx @@ -24,7 +24,7 @@ import { EuiComboBox, } from '@elastic/eui'; import '../FeatureAnywhereContextMenu/CreateAnomalyDetector/styles.scss'; -import { useDispatch } from 'react-redux'; +import { useDispatch, useSelector } from 'react-redux'; import { isEmpty, get } from 'lodash'; import { Field, @@ -54,6 +54,7 @@ import { } from '../../../server/utils/constants'; import { focusOnFirstWrongFeature, + getCategoryFields, initialFeatureValue, validateFeatures, } from '../../pages/ConfigureModel/utils/helpers'; @@ -61,7 +62,7 @@ import { formikToDetector } from '../../pages/ReviewAndCreate/utils/helpers'; import { FormattedFormRow } from '../FormattedFormRow/FormattedFormRow'; import { FeatureAccordion } from '../../pages/ConfigureModel/components/FeatureAccordion'; import { AD_DOCS_LINK, DEFAULT_SHINGLE_SIZE, MAX_FEATURE_NUM, PLUGIN_NAME } from '../../utils/constants'; -import { getNotifications } from '../../services'; +import { getNotifications, getQueryService } from '../../services'; import { prettifyErrorMessage } from '../../../server/utils/helpers'; import EnhancedAccordion from '../FeatureAnywhereContextMenu/EnhancedAccordion'; import MinimalAccordion from '../FeatureAnywhereContextMenu/MinimalAccordion'; @@ -69,10 +70,11 @@ import { DataFilterList } from '../../pages/DefineDetector/components/DataFilter import { generateParameters } from '../../redux/reducers/assistant'; import { FEATURE_TYPE } from '../../models/interfaces'; import { FeaturesFormikValues } from '../../pages/ConfigureModel/models/interfaces'; -import { DiscoverActionContext } from '../../../../../src/plugins/data_explorer/public/types'; import { getMappings } from '../../redux/reducers/opensearch'; import { mountReactNode } from '../../../../../src/core/public/utils'; import { formikToDetectorName } from '../FeatureAnywhereContextMenu/CreateAnomalyDetector/helpers'; +import { DEFAULT_DATA } from '../../../../../src/plugins/data/common'; +import { AppState } from '../../redux/reducers'; export interface GeneratedParameters { categoryField: string; @@ -80,41 +82,35 @@ export interface GeneratedParameters { dateFields: string[]; } -function GenerateAnomalyDetector({ +function SuggestAnomalyDetector({ closeFlyout, - context, }: { closeFlyout: any; - context: DiscoverActionContext; }) { const dispatch = useDispatch(); const notifications = getNotifications(); - const indexPatternId = context.indexPattern?.id; - const indexPatternName = context.indexPattern?.title; - if (!indexPatternId || !indexPatternName) { + const queryString = getQueryService().queryString; + const dataset = queryString.getQuery().dataset || queryString.getDefaultQuery().dataset; + const datasetType = dataset.type; + if (datasetType != DEFAULT_DATA.SET_TYPES.INDEX_PATTERN && datasetType != DEFAULT_DATA.SET_TYPES.INDEX) { notifications.toasts.addDanger( - 'Cannot extract index pattern from the context' + 'Unsupported dataset type' ); return <>; } - const dataSourceId = context.indexPattern?.dataSourceRef?.id; - const timeFieldFromIndexPattern = context.indexPattern?.timeFieldName; - const fieldsFromContext = context.indexPattern?.fields || []; - const [categoricalFields, dateFields] = fieldsFromContext.reduce( - ([cFields, dFields], indexPatternField) => { - const esType = indexPatternField.spec.esTypes?.[0]; - const name = indexPatternField.spec.name; - if (esType === 'keyword' || esType === 'ip') { - cFields.push(name); - } else if (esType === 'date') { - dFields.push(name); - } - return [cFields, dFields]; - }, - [[], []] as [string[], string[]] - ) || [[], []]; + const indexPatternId = dataset.id; + // indexName could be a index pattern or a concrete index + const indexName = dataset.title; + const timeFieldName = dataset.timeFieldName; + if (!indexPatternId || !indexName || !timeFieldName) { + notifications.toasts.addDanger( + 'Cannot extract complete index info from the context' + ); + return <>; + } + const dataSourceId = dataset.dataSource?.id; const [isLoading, setIsLoading] = useState(true); const [buttonName, setButtonName] = useState( 'Generating parameters...' @@ -127,14 +123,22 @@ function GenerateAnomalyDetector({ const [delayValue, setDelayValue] = useState(1); const [enabled, setEnabled] = useState(false); const [detectorName, setDetectorName] = useState( - formikToDetectorName(indexPatternName.substring(0, 40)) + formikToDetectorName(indexName.substring(0, 40)) + ); + const indexDataTypes = useSelector( + (state: AppState) => state.opensearch.dataTypes ); + const categoricalFields = getCategoryFields(indexDataTypes); + + const dateFields = get(indexDataTypes, 'date', []) as string[]; + const dateNanoFields = get(indexDataTypes, 'date_nanos', []) as string[]; + const allDateFields = dateFields.concat(dateNanoFields); // let LLM to generate parameters for creating anomaly detector async function getParameters() { try { const result = await dispatch( - generateParameters(indexPatternName!, dataSourceId) + generateParameters(indexName!, dataSourceId) ); const rawGeneratedParameters = get(result, 'generatedParameters'); if (!rawGeneratedParameters) { @@ -207,11 +211,8 @@ function GenerateAnomalyDetector({ useEffect(() => { async function fetchData() { + await dispatch(getMappings(indexName, dataSourceId)); await getParameters(); - const getMappingDispatchCall = dispatch( - getMappings(indexPatternName, dataSourceId) - ); - await Promise.all([getMappingDispatchCall]); } fetchData(); }, []); @@ -353,8 +354,8 @@ function GenerateAnomalyDetector({ let initialDetectorValue = { name: detectorName, - index: [{ label: indexPatternName }], - timeField: timeFieldFromIndexPattern, + index: [{ label: indexName }], + timeField: timeFieldName, interval: intervalValue, windowDelay: delayValue, shingleSize: DEFAULT_SHINGLE_SIZE, @@ -382,7 +383,7 @@ function GenerateAnomalyDetector({

- Suggest anomaly detector + Suggested anomaly detector

@@ -729,7 +730,7 @@ function GenerateAnomalyDetector({ data-test-subj="timestampFilter" id="timeField" placeholder="Find timestamp" - options={dateFields.map((field) => { + options={allDateFields.map((field) => { return { label: field, }; @@ -750,7 +751,7 @@ function GenerateAnomalyDetector({ label: field.value, }, ] - : [{ label: timeFieldFromIndexPattern }] + : [{ label: timeFieldName }] } singleSelection={{ asPlainText: true }} isClearable={false} @@ -857,4 +858,4 @@ function GenerateAnomalyDetector({ ); } -export default GenerateAnomalyDetector; +export default SuggestAnomalyDetector; diff --git a/public/plugin.ts b/public/plugin.ts index 68eedfcf..a340e6be 100644 --- a/public/plugin.ts +++ b/public/plugin.ts @@ -27,7 +27,7 @@ import { } from '../../../src/plugins/embeddable/public'; import { ACTION_AD } from './action/ad_dashboard_action'; import { APP_PATH, DASHBOARD_PAGE_NAV_ID, DETECTORS_PAGE_NAV_ID, OVERVIEW_PAGE_NAV_ID, PLUGIN_NAME } from './utils/constants'; -import { getActions } from './utils/contextMenu/getActions'; +import { ACTION_SUGGEST_AD, getActions, getSuggestAnomalyDetectorAction } from './utils/contextMenu/getActions'; import { overlayAnomaliesFunction } from './expressions/overlay_anomalies'; import { setClient, @@ -42,7 +42,8 @@ import { setDataSourceManagementPlugin, setDataSourceEnabled, setNavigationUI, - setApplication + setApplication, + setIndexPatternService } from './services'; import { AnomalyDetectionOpenSearchDashboardsPluginStart } from 'public'; import { @@ -50,16 +51,16 @@ import { VisAugmenterStart, } from '../../../src/plugins/vis_augmenter/public'; import { UiActionsStart } from '../../../src/plugins/ui_actions/public'; -import { DataPublicPluginStart } from '../../../src/plugins/data/public'; +import { DataPublicPluginSetup, DataPublicPluginStart } from '../../../src/plugins/data/public'; import { DataSourceManagementPluginSetup } from '../../../src/plugins/data_source_management/public'; import { DataSourcePluginSetup } from '../../../src/plugins/data_source/public'; import { NavigationPublicPluginStart } from '../../../src/plugins/navigation/public'; -import { getDiscoverAction } from './utils/discoverAction'; -import { DataExplorerPluginSetup } from '../../../src/plugins/data_explorer/public'; +import { AssistantSetup } from '../../../plugins/dashboards-assistant/public'; declare module '../../../src/plugins/ui_actions/public' { export interface ActionContextMapping { [ACTION_AD]: {}; + [ACTION_SUGGEST_AD]: {} } } @@ -70,7 +71,7 @@ export interface AnomalyDetectionSetupDeps { visAugmenter: VisAugmenterSetup; dataSourceManagement: DataSourceManagementPluginSetup; dataSource: DataSourcePluginSetup; - dataExplorer: DataExplorerPluginSetup; + data: DataPublicPluginSetup; } export interface AnomalyDetectionStartDeps { @@ -192,9 +193,9 @@ export class AnomalyDetectionOpenSearchDashboardsPlugin plugins.uiActions.addTriggerAction(CONTEXT_MENU_TRIGGER, action); }); - // Add action to Discover - const discoverAction = getDiscoverAction(); - plugins.dataExplorer.registerDiscoverAction(discoverAction); + // Add suggest anomaly detector action to the uiActions in Discover + const suggestAnomalyDetectorAction = getSuggestAnomalyDetectorAction(); + plugins.uiActions.addTriggerAction(plugins.assistantDashboards.assistantTriggers.AI_ASSISTANT_TRIGGER, suggestAnomalyDetectorAction); // registers the expression function used to render anomalies on an Augmented Visualization plugins.expressions.registerFunction(overlayAnomaliesFunction); diff --git a/public/utils/contextMenu/getActions.tsx b/public/utils/contextMenu/getActions.tsx index f58a7a9e..30ae041e 100644 --- a/public/utils/contextMenu/getActions.tsx +++ b/public/utils/contextMenu/getActions.tsx @@ -7,7 +7,7 @@ import React from 'react'; import { i18n } from '@osd/i18n'; import { EuiIconType } from '@elastic/eui'; import { toMountPoint } from '../../../../../src/plugins/opensearch_dashboards_react/public'; -import { Action } from '../../../../../src/plugins/ui_actions/public'; +import { Action, createAction } from '../../../../../src/plugins/ui_actions/public'; import { createADAction } from '../../action/ad_dashboard_action'; import AnywhereParentFlyout from '../../components/FeatureAnywhereContextMenu/AnywhereParentFlyout'; import { Provider } from 'react-redux'; @@ -16,6 +16,9 @@ import DocumentationTitle from '../../components/FeatureAnywhereContextMenu/Docu import { AD_FEATURE_ANYWHERE_LINK, ANOMALY_DETECTION_ICON } from '../constants'; import { getClient, getOverlays } from '../../../public/services'; import { FLYOUT_MODES } from '../../../public/components/FeatureAnywhereContextMenu/AnywhereParentFlyout/constants'; +import SuggestAnomalyDetector from '../../../public/components/DiscoverAction/SuggestAnomalyDetector'; + +export const ACTION_SUGGEST_AD = 'suggestAnomalyDetector'; // This is used to create all actions in the same context menu const grouping: Action['grouping'] = [ @@ -31,23 +34,23 @@ const grouping: Action['grouping'] = [ export const getActions = () => { const getOnClick = (startingFlyout) => - async ({ embeddable }) => { - const overlayService = getOverlays(); - const openFlyout = overlayService.openFlyout; - const store = configureStore(getClient()); - const overlay = openFlyout( - toMountPoint( - - overlay.close()} - /> - - ), - { size: 'm', className: 'context-menu__flyout' } - ); - }; + async ({ embeddable }) => { + const overlayService = getOverlays(); + const openFlyout = overlayService.openFlyout; + const store = configureStore(getClient()); + const overlay = openFlyout( + toMountPoint( + + overlay.close()} + /> + + ), + { size: 'm', className: 'context-menu__flyout' } + ); + }; return [ { @@ -87,3 +90,31 @@ export const getActions = () => { }, ].map((options) => createADAction({ ...options, grouping })); }; + +export const getSuggestAnomalyDetectorAction = () => { + const onClick = async function () { + const overlayService = getOverlays(); + const openFlyout = overlayService.openFlyout; + const store = configureStore(getClient()); + const overlay = openFlyout( + toMountPoint( + + overlay.close()} + /> + + ) + ); + } + + return createAction({ + id: 'suggestAnomalyDetector', + order: 100, + type: ACTION_SUGGEST_AD, + getDisplayName: () => 'Suggest anomaly detector', + getIconType: () => ANOMALY_DETECTION_ICON, + execute: async () => { + onClick(); + }, + }); +} diff --git a/public/utils/discoverAction.tsx b/public/utils/discoverAction.tsx deleted file mode 100644 index 9b345ee8..00000000 --- a/public/utils/discoverAction.tsx +++ /dev/null @@ -1,38 +0,0 @@ -/* - * Copyright OpenSearch Contributors - * SPDX-License-Identifier: Apache-2.0 - */ - -import React from 'react' -import { ANOMALY_DETECTION_ICON } from "./constants"; -import GenerateAnomalyDetector from "../components/DiscoverAction/SuggestAnomalyDetector"; -import { getClient, getOverlays } from '../../public/services'; -import { toMountPoint } from "../../../../src/plugins/opensearch_dashboards_react/public"; -import { Provider } from "react-redux"; -import configureStore from '../redux/configureStore'; -import { DiscoverAction, DiscoverActionContext } from "../../../../src/plugins/data_explorer/public/types"; - -export const getDiscoverAction = (): DiscoverAction => { - const onClick = function (context: DiscoverActionContext) { - const overlayService = getOverlays(); - const openFlyout = overlayService.openFlyout; - const store = configureStore(getClient()); - const overlay = openFlyout( - toMountPoint( - - overlay.close()} - context={context} - /> - - ) - ); - } - - return { - order: 0, - name: 'Suggest anomaly detector', - iconType: ANOMALY_DETECTION_ICON, - onClick: onClick, - } -}; diff --git a/server/routes/assistant.ts b/server/routes/assistant.ts index 6669b9c3..5c1a0cd0 100644 --- a/server/routes/assistant.ts +++ b/server/routes/assistant.ts @@ -56,16 +56,14 @@ export default class AssistantService { }); if ( - !getAgentResponse || - !getAgentResponse['configuration'] || - !getAgentResponse['configuration']['agent_id'] + !getAgentResponse || !(getAgentResponse.ml_configuration?.agent_id || getAgentResponse.configuration?.agent_id) ) { throw new Error( 'Cannot get flow agent id for generating anomaly detector' ); } - const agentId = getAgentResponse['configuration']['agent_id']; + const agentId = getAgentResponse.ml_configuration?.agent_id || getAgentResponse.configuration?.agent_id; const executeAgentResponse = await callWithRequest('ml.executeAgent', { agentId: agentId, diff --git a/server/utils/constants.ts b/server/utils/constants.ts index 8c071556..1a756187 100644 --- a/server/utils/constants.ts +++ b/server/utils/constants.ts @@ -133,4 +133,4 @@ export const HISTORICAL_TASK_TYPES = [ export const CUSTOM_AD_RESULT_INDEX_PREFIX = 'opensearch-ad-plugin-result-'; -export const SUGGEST_ANOMALY_DETECTOR_CONFIG_ID = 'suggest_anomaly_detector'; +export const SUGGEST_ANOMALY_DETECTOR_CONFIG_ID = 'os_suggest_ad';