diff --git a/public/components/integrations/components/__tests__/__snapshots__/added_integration_table.test.tsx.snap b/public/components/integrations/components/__tests__/__snapshots__/added_integration_table.test.tsx.snap index be8c3a5e1..e4286b273 100644 --- a/public/components/integrations/components/__tests__/__snapshots__/added_integration_table.test.tsx.snap +++ b/public/components/integrations/components/__tests__/__snapshots__/added_integration_table.test.tsx.snap @@ -148,6 +148,7 @@ exports[`Added Integration Table View Test Renders added integration table view "name": "nginx", "templateName": "nginx", }, + "dataSourceMDSLabel": "Local cluster", "id": "ad7e6e30-0b99-11ee-b27c-c9863222e9bf", "name": "nginx", "templateName": "nginx", @@ -598,6 +599,7 @@ exports[`Added Integration Table View Test Renders added integration table view "name": "nginx", "templateName": "nginx", }, + "dataSourceMDSLabel": "Local cluster", "id": "ad7e6e30-0b99-11ee-b27c-c9863222e9bf", "name": "nginx", "templateName": "nginx", @@ -1503,3 +1505,1513 @@ exports[`Added Integration Table View Test Renders added integration table view `; + +exports[`Added Integration Table View Test Renders added integration table view using dummy data when MDS is disabled 1`] = ` + + + +
+ +
+ + +
+ +
+ + + +
+
+ + + + +
+ + + + + +
+
+
+
+
+
+
+
+
+ +
+ + +
+ + + Type + + } + closePopover={[Function]} + display="inlineBlock" + hasArrow={true} + id="field_value_selection_0" + isOpen={false} + ownFocus={true} + panelClassName="euiFilterGroup__popoverPanel" + panelPaddingSize="none" + > +
+
+ + + + + +
+
+
+
+
+
+
+
+
+
+
+
+ +
+ + +
+
+ +
+ +
+ +
+ + +
+ +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + Integration Name + + + + + + + + + + + + Source + + + + + + + + + + + + Date Added + + + + + + + + + + + + Actions + + + + + +
+
+ Integration Name +
+
+ + + nginx + + +
+
+
+ Source +
+
+ + + nginx + + +
+
+
+ Date Added +
+
+ +
+ 2023-06-15T16:28:36.370Z +
+
+
+
+
+ Actions +
+
+ + + + + + + +
+
+
+
+ +
+ +
+ + + +
+ +
+ + + : + 10 + + } + closePopover={[Function]} + display="inlineBlock" + hasArrow={true} + isOpen={false} + ownFocus={true} + panelPaddingSize="none" + > +
+
+ + + +
+
+
+
+
+ +
+ + + +
+
+
+
+
+
+ +
+ +
+ +
+ + + +`; diff --git a/public/components/integrations/components/__tests__/__snapshots__/setup_integration.test.tsx.snap b/public/components/integrations/components/__tests__/__snapshots__/setup_integration.test.tsx.snap index e7f9cd209..f25fbbee9 100644 --- a/public/components/integrations/components/__tests__/__snapshots__/setup_integration.test.tsx.snap +++ b/public/components/integrations/components/__tests__/__snapshots__/setup_integration.test.tsx.snap @@ -42,6 +42,7 @@ exports[`Integration Setup Page Renders integration setup page as expected 1`] = "enabledWorkflows": Array [], } } + handleSelectedDataSourceChange={[Function]} integration={ Object { "assets": Array [], @@ -255,6 +256,7 @@ exports[`Integration Setup Page Renders integration setup page as expected 1`] = "enabledWorkflows": Array [], } } + handleSelectedDataSourceChange={[Function]} integration={ Object { "assets": Array [], @@ -935,6 +937,8 @@ exports[`Integration Setup Page Renders integration setup page as expected 1`] = "enabledWorkflows": Array [], } } + dataSourceMDSId="" + dataSourceMDSLabel="" integration={ Object { "assets": Array [], diff --git a/public/components/integrations/components/__tests__/added_integration_table.test.tsx b/public/components/integrations/components/__tests__/added_integration_table.test.tsx index 95e9c2bd1..726e88192 100644 --- a/public/components/integrations/components/__tests__/added_integration_table.test.tsx +++ b/public/components/integrations/components/__tests__/added_integration_table.test.tsx @@ -3,12 +3,12 @@ * SPDX-License-Identifier: Apache-2.0 */ +import { waitFor } from '@testing-library/react'; import { configure, mount } from 'enzyme'; import Adapter from 'enzyme-adapter-react-16'; -import { waitFor } from '@testing-library/react'; -import { AddedIntegrationsTable } from '../added_integration_table'; -import { addedIntegrationData } from './testing_constants'; import React from 'react'; +import { AddedIntegrationsTable } from '../added_integration_table'; +import { addedIntegrationData, addedIntegrationDataWithoutMDS } from './testing_constants'; describe('Added Integration Table View Test', () => { configure({ adapter: new Adapter() }); @@ -20,4 +20,12 @@ describe('Added Integration Table View Test', () => { expect(wrapper).toMatchSnapshot(); }); }); + + it('Renders added integration table view using dummy data when MDS is disabled', async () => { + const wrapper = mount(); + + await waitFor(() => { + expect(wrapper).toMatchSnapshot(); + }); + }); }); diff --git a/public/components/integrations/components/__tests__/testing_constants.ts b/public/components/integrations/components/__tests__/testing_constants.ts index ae6220e74..465bc9ae9 100644 --- a/public/components/integrations/components/__tests__/testing_constants.ts +++ b/public/components/integrations/components/__tests__/testing_constants.ts @@ -175,6 +175,90 @@ export const addedIntegrationData: AddedIntegrationsTableProps = { loading: false, }; +export const addedIntegrationDataWithoutMDS: AddedIntegrationsTableProps = { + setData: () => {}, + http: httpClientMock, + data: { + hits: [ + { + name: 'nginx', + templateName: 'nginx', + dataSource: { sourceType: 'logs', dataset: 'nginx', namespace: 'prod' }, + creationDate: '2023-06-15T16:28:36.370Z', + status: 'active', + addedBy: 'admin', + assets: [ + { + assetType: 'index-pattern', + assetId: '3fc41705-8a23-49f4-926c-2819e0d7306d', + status: 'available', + isDefaultAsset: false, + description: 'ss4o_logs-nginx-prod', + }, + { + assetType: 'search', + assetId: 'a0415ddd-047d-4c02-8769-d14bfb70f525', + status: 'available', + isDefaultAsset: false, + description: '[NGINX Core Logs 1.0] Nginx Access Logs', + }, + { + assetType: 'visualization', + assetId: 'a17cd453-fb2f-4c24-81db-aedfc8682829', + status: 'available', + isDefaultAsset: false, + description: '[NGINX Core Logs 1.0] Response codes over time', + }, + { + assetType: 'search', + assetId: '3e47dfed-d9ff-4c1b-b425-04ffc8ed3fa9', + status: 'available', + isDefaultAsset: false, + description: '[NGINX Core Logs 1.0] Nginx Error Logs', + }, + { + assetType: 'visualization', + assetId: '641c2a03-eead-4900-94ee-e12d2fef8383', + status: 'available', + isDefaultAsset: false, + description: '[NGINX Core Logs 1.0] Errors over time', + }, + { + assetType: 'visualization', + assetId: 'ce61594d-8307-4358-9b7e-71101b3ed722', + status: 'available', + isDefaultAsset: false, + description: 'Data Volume', + }, + { + assetType: 'visualization', + assetId: '452bd6e3-3b50-407f-88f2-c35a29c56051', + status: 'available', + isDefaultAsset: false, + description: 'Top Paths', + }, + { + assetType: 'visualization', + assetId: '14a1ddab-08c1-4aba-ba3b-88bae36f7e50', + status: 'available', + isDefaultAsset: false, + description: 'Requests per Minute', + }, + { + assetType: 'dashboard', + assetId: '179bad58-c840-4c6c-9fd8-1667c14bd03a', + status: 'available', + isDefaultAsset: true, + description: '[NGINX Core Logs 1.0] Overview', + }, + ], + id: 'ad7e6e30-0b99-11ee-b27c-c9863222e9bf', + }, + ], + }, + loading: false, +}; + export const testIntegrationInstanceData = { data: { id: 'ad7e6e30-0b99-11ee-b27c-c9863222e9bf', diff --git a/public/components/integrations/components/added_integration.tsx b/public/components/integrations/components/added_integration.tsx index 99d3e3550..e445a30dd 100644 --- a/public/components/integrations/components/added_integration.tsx +++ b/public/components/integrations/components/added_integration.tsx @@ -22,14 +22,16 @@ import { EuiTableFieldDataColumnType, EuiText, } from '@elastic/eui'; -import React, { useEffect, useState } from 'react'; import truncate from 'lodash/truncate'; -import { PanelTitle } from '../../trace_analytics/components/common/helper_functions'; +import React, { useEffect, useState } from 'react'; +import { DataSourceViewConfig } from '../../../../../../src/plugins/data_source_management/public'; import { ASSET_FILTER_OPTIONS } from '../../../../common/constants/integrations'; import { INTEGRATIONS_BASE } from '../../../../common/constants/shared'; +import { dataSourceFilterFn } from '../../../../common/utils/shared'; +import { useToast } from '../../../../public/components/common/toast'; import { DeleteModal } from '../../common/helpers/delete_modal'; +import { PanelTitle } from '../../trace_analytics/components/common/helper_functions'; import { AddedIntegrationProps } from './integration_types'; -import { useToast } from '../../../../public/components/common/toast'; export const IntegrationHealthBadge = ({ status }: { status?: string }) => { switch (status) { @@ -45,7 +47,14 @@ export const IntegrationHealthBadge = ({ status }: { status?: string }) => { }; export function AddedIntegration(props: AddedIntegrationProps) { - const { http, integrationInstanceId, chrome } = props; + const { + http, + integrationInstanceId, + chrome, + dataSourceEnabled, + dataSourceManagement, + setActionMenu, + } = props; const { setToast } = useToast(); @@ -74,6 +83,8 @@ export function AddedIntegration(props: AddedIntegrationProps) { const [isModalVisible, setIsModalVisible] = useState(false); const [modalLayout, setModalLayout] = useState(); + const DataSourceMenu = dataSourceManagement?.ui?.getDataSourceMenu(); + const activateDeleteModal = (integrationName?: string) => { setModalLayout( + {dataSourceEnabled && ( + + )} diff --git a/public/components/integrations/components/added_integration_overview_page.tsx b/public/components/integrations/components/added_integration_overview_page.tsx index 83b0d000b..d406d570b 100644 --- a/public/components/integrations/components/added_integration_overview_page.tsx +++ b/public/components/integrations/components/added_integration_overview_page.tsx @@ -17,7 +17,7 @@ export interface AddedIntegrationsTableProps { data: AddedIntegrationsList; setData: React.Dispatch>; http: HttpStart; - dataSourceEnabled: string; + dataSourceEnabled: boolean; } export interface AddedIntegrationsList { @@ -33,6 +33,7 @@ export interface AddedIntegrationType { assets: any[]; addedBy: string; id: string; + references?: []; } export function AddedIntegrationOverviewPage(props: AddedIntegrationOverviewPageProps) { diff --git a/public/components/integrations/components/added_integration_table.tsx b/public/components/integrations/components/added_integration_table.tsx index 9702dd9d4..47abcbde6 100644 --- a/public/components/integrations/components/added_integration_table.tsx +++ b/public/components/integrations/components/added_integration_table.tsx @@ -93,7 +93,9 @@ export function AddedIntegrationsTable(props: AddedIntegrationsTableProps) { sortable: true, truncateText: true, render: (value, record) => ( - - - + + {truncate(record.dataSourceMDSLabel || 'Local cluster', { length: 100 })} + ), }); } @@ -132,8 +134,17 @@ export function AddedIntegrationsTable(props: AddedIntegrationsTableProps) { ); setIsModalVisible(true); }; - const integTemplateNames = [...new Set(props.data.hits.map((i) => i.templateName))].sort(); + let mdsLabels; + if (dataSourceEnabled) { + mdsLabels = [ + ...new Set( + props.data.hits.flatMap((hit) => + hit.references?.length > 0 ? hit.references.map((ref) => ref.name || 'Local cluster') : [] + ) + ), + ].sort(); + } const search = { box: { @@ -151,6 +162,21 @@ export function AddedIntegrationsTable(props: AddedIntegrationsTableProps) { view: name, })), }, + ...(dataSourceEnabled + ? [ + { + type: 'field_value_selection' as const, + field: 'dataSourceMDSLabel', + name: 'Data Source Name', + multiSelect: false, + options: mdsLabels?.map((name) => ({ + name, + value: name, + view: name, + })), + }, + ] + : []), ], }; @@ -159,7 +185,17 @@ export function AddedIntegrationsTable(props: AddedIntegrationsTableProps) { const templateName = integration.templateName; const creationDate = integration.creationDate; const name = integration.name; - return { id, templateName, creationDate, name, data: { templateName, name } }; + const dataSourceMDSLabel = integration.references + ? integration.references[0].name + : 'Local cluster'; + return { + id, + templateName, + creationDate, + name, + data: { templateName, name }, + dataSourceMDSLabel, + }; }); return ( diff --git a/public/components/integrations/components/create_integration_helpers.ts b/public/components/integrations/components/create_integration_helpers.ts index dcbc2da56..f6b8ed4c8 100644 --- a/public/components/integrations/components/create_integration_helpers.ts +++ b/public/components/integrations/components/create_integration_helpers.ts @@ -2,10 +2,10 @@ * Copyright OpenSearch Contributors * SPDX-License-Identifier: Apache-2.0 */ -import { Color, VALID_INDEX_NAME } from '../../../../common/constants/integrations'; import { HttpSetup } from '../../../../../../src/core/public'; +import { Color, VALID_INDEX_NAME } from '../../../../common/constants/integrations'; +import { CONSOLE_PROXY, DSL_BASE, INTEGRATIONS_BASE } from '../../../../common/constants/shared'; import { coreRefs } from '../../../framework/core_refs'; -import { CONSOLE_PROXY, INTEGRATIONS_BASE } from '../../../../common/constants/shared'; type ValidationResult = { ok: true } | { ok: false; errors: string[] }; @@ -18,6 +18,8 @@ interface AddIntegrationRequestParams { templateName: string; integration: IntegrationConfig; setToast: (title: string, color?: Color, text?: string | undefined) => void; + dataSourceMDSId?: string; + dataSourceMDSLabel?: string; name?: string; indexPattern?: string; workflows?: string[]; @@ -131,13 +133,13 @@ export const checkDataSourceName = ( export const fetchDataSourceMappings = async ( targetDataSource: string, - http: HttpSetup + http: HttpSetup, + dataSourceMDSId?: string ): Promise<{ [key: string]: { properties: Properties } } | null> => { return http - .post(CONSOLE_PROXY, { + .post(`${DSL_BASE}/integrations/mapping`, { query: { - path: `${targetDataSource}/_mapping`, - method: 'GET', + dataSourceMDSId, }, }) .then((response) => { @@ -204,7 +206,8 @@ export const doExistingDataSourceValidation = async ( const createComponentMapping = async ( componentName: string, - payload: ComponentMappingPayload + payload: ComponentMappingPayload, + dataSourceMDSId?: string ): Promise<{ [key: string]: { properties: Properties } } | null> => { const http = coreRefs.http!; const version = payload.template.mappings._meta.version; @@ -213,6 +216,7 @@ const createComponentMapping = async ( query: { path: `_component_template/ss4o_${componentName}-${version}-template`, method: 'POST', + dataSourceId: dataSourceMDSId, }, }); }; @@ -221,7 +225,8 @@ const createIndexMapping = async ( componentName: string, payload: ComponentMappingPayload, dataSourceName: string, - integration: IntegrationConfig + integration: IntegrationConfig, + dataSourceMDSId?: string ): Promise<{ [key: string]: { properties: Properties } } | null> => { const http = coreRefs.http!; const version = payload.template.mappings._meta.version; @@ -231,6 +236,7 @@ const createIndexMapping = async ( query: { path: `_index_template/ss4o_${componentName}-${integration.name}-${version}-sample`, method: 'POST', + dataSourceId: dataSourceMDSId, }, }); }; @@ -239,7 +245,8 @@ const createIndexPatternMappings = async ( targetDataSource: string, integrationTemplateId: string, integration: IntegrationConfig, - setToast: (title: string, color?: Color, text?: string | undefined) => void + setToast: (title: string, color?: Color, text?: string | undefined) => void, + dataSourceMDSId?: string ): Promise => { // TODO the nested methods still need the dataSource -> indexPattern rename applied, sub-methods // here still have old naming convention @@ -262,21 +269,21 @@ const createIndexPatternMappings = async ( if (key === integration.type) { return Promise.resolve(); } - return createComponentMapping(key, mapping as ComponentMappingPayload); + return createComponentMapping(key, mapping as ComponentMappingPayload, dataSourceMDSId); }) ); // In order to see our changes, we need to manually provoke a refresh - await http.post(CONSOLE_PROXY, { + await http.post(`${DSL_BASE}/integrations/refresh`, { query: { - path: '_refresh', - method: 'GET', + dataSourceMDSId, }, }); await createIndexMapping( integration.type, mappings[integration.type], targetDataSource, - integration + integration, + dataSourceMDSId ); } catch (err) { error = err.message; @@ -299,6 +306,8 @@ export async function addIntegrationRequest({ workflows, skipRedirect, dataSourceInfo, + dataSourceMDSId, + dataSourceMDSLabel, }: AddIntegrationRequestParams): Promise { const http = coreRefs.http!; if (addSample) { @@ -306,19 +315,24 @@ export async function addIntegrationRequest({ `ss4o_${integration.type}-${templateName}-*-sample`, templateName, integration, - setToast + setToast, + dataSourceMDSId ); name = `${templateName}-sample`; indexPattern = `ss4o_${integration.type}-${templateName}-sample-sample`; } const createReqBody: { + dataSourceMDSId?: string; + dataSourceMDSLabel?: string; name?: string; indexPattern?: string; workflows?: string[]; dataSource?: string; tableName?: string; } = { + dataSourceMDSId, + dataSourceMDSLabel, name, indexPattern, workflows, diff --git a/public/components/integrations/components/integration.tsx b/public/components/integrations/components/integration.tsx index d7b74dec0..9ae5f3fc5 100644 --- a/public/components/integrations/components/integration.tsx +++ b/public/components/integrations/components/integration.tsx @@ -5,7 +5,13 @@ /* eslint-disable react-hooks/exhaustive-deps */ import { + EuiButton, EuiLoadingSpinner, + EuiModal, + EuiModalBody, + EuiModalFooter, + EuiModalHeader, + EuiModalHeaderTitle, EuiOverlayMask, EuiPage, EuiPageBody, @@ -13,21 +19,30 @@ import { EuiTab, EuiTabs, } from '@elastic/eui'; -import React, { useEffect, useState } from 'react'; -import { IntegrationOverview } from './integration_overview_panel'; -import { IntegrationDetails } from './integration_details_panel'; -import { IntegrationFields } from './integration_fields_panel'; -import { IntegrationAssets } from './integration_assets_panel'; -import { AvailableIntegrationProps } from './integration_types'; +import React, { ComponentType, useEffect, useState } from 'react'; +import { DataSourceSelectorProps } from '../../../../../../src/plugins/data_source_management/public/components/data_source_selector/data_source_selector'; import { INTEGRATIONS_BASE } from '../../../../common/constants/shared'; -import { IntegrationScreenshots } from './integration_screenshots_panel'; +import { dataSourceFilterFn } from '../../../../common/utils/shared'; import { useToast } from '../../../../public/components/common/toast'; import { coreRefs } from '../../../framework/core_refs'; import { addIntegrationRequest } from './create_integration_helpers'; +import { IntegrationAssets } from './integration_assets_panel'; +import { IntegrationDetails } from './integration_details_panel'; +import { IntegrationFields } from './integration_fields_panel'; +import { IntegrationOverview } from './integration_overview_panel'; +import { IntegrationScreenshots } from './integration_screenshots_panel'; +import { AvailableIntegrationProps } from './integration_types'; export function Integration(props: AvailableIntegrationProps) { const http = coreRefs.http!; - const { integrationTemplateId, chrome } = props; + const { + integrationTemplateId, + chrome, + notifications, + dataSourceEnabled, + dataSourceManagement, + savedObjectsMDSClient, + } = props; const { setToast } = useToast(); const [integration, setIntegration] = useState({} as IntegrationConfig); @@ -35,6 +50,11 @@ export function Integration(props: AvailableIntegrationProps) { const [integrationMapping, setMapping] = useState(null); const [integrationAssets, setAssets] = useState([]); const [loading, setLoading] = useState(false); + const [isModalVisible, setIsModalVisible] = useState(false); + const closeModal = () => setIsModalVisible(false); + const showModal = () => setIsModalVisible(true); + const [dataSourceMDSId, setDataSourceMDSId] = useState(''); + const [dataSourceMDSLabel, setDataSourceMDSLabel] = useState(''); useEffect(() => { chrome.setBreadcrumbs([ @@ -117,6 +137,21 @@ export function Integration(props: AvailableIntegrationProps) { setSelectedTabId(id); }; + let DataSourceSelector: + | ComponentType + | React.JSX.IntrinsicAttributes + | null; + if (dataSourceEnabled) { + DataSourceSelector = dataSourceManagement.ui.DataSourceSelector; + } + + const onSelectedDataSource = (e) => { + const dataConnectionId = e[0] ? e[0].id : undefined; + setDataSourceMDSId(dataConnectionId); + const dataConnectionLabel = e[0] ? e[0].label : undefined; + setDataSourceMDSLabel(dataConnectionLabel); + }; + const renderTabs = () => { return tabs.map((tab, index) => ( ; + + if (isModalVisible) { + modal = ( + + + +

Select Data Source

+
+
+ + + + + + + + Close + + { + setLoading(true); + await addIntegrationRequest({ + addSample: true, + templateName: integration.name, + integration, + setToast, + dataSourceMDSId, + dataSourceMDSLabel, + }); + setLoading(false); + closeModal(); + }} + isLoading={loading} + fill + > + Add + + +
+ ); + } + if (Object.keys(integration).length === 0) { return ( @@ -148,14 +234,20 @@ export function Integration(props: AvailableIntegrationProps) { window.location.hash = `#/available/${integration.name}/setup`; }} setUpSample={async () => { - setLoading(true); - await addIntegrationRequest({ - addSample: true, - templateName: integration.name, - integration, - setToast, - }); - setLoading(false); + if (dataSourceEnabled) { + showModal(); + } else { + setLoading(true); + await addIntegrationRequest({ + addSample: true, + templateName: integration.name, + integration, + setToast, + dataSourceMDSId, + dataSourceMDSLabel, + }); + setLoading(false); + } }} loading={loading} /> @@ -173,6 +265,7 @@ export function Integration(props: AvailableIntegrationProps) { : IntegrationFields({ integration, integrationMapping })} + {modal} ); } diff --git a/public/components/integrations/components/integration_overview_panel.tsx b/public/components/integrations/components/integration_overview_panel.tsx index 1eb301f19..47698e7e7 100644 --- a/public/components/integrations/components/integration_overview_panel.tsx +++ b/public/components/integrations/components/integration_overview_panel.tsx @@ -6,11 +6,11 @@ import { EuiButton, EuiFlexGroup, + EuiFlexItem, + EuiPageContentHeaderSection, EuiPageHeader, EuiPageHeaderSection, EuiSpacer, - EuiFlexItem, - EuiPageContentHeaderSection, EuiText, } from '@elastic/eui'; import React from 'react'; diff --git a/public/components/integrations/components/integration_types.ts b/public/components/integrations/components/integration_types.ts index a42bd87d2..47c348edc 100644 --- a/public/components/integrations/components/integration_types.ts +++ b/public/components/integrations/components/integration_types.ts @@ -3,7 +3,14 @@ * SPDX-License-Identifier: Apache-2.0 */ -import { ChromeStart, HttpStart } from '../../../../../../src/core/public'; +import { + ChromeStart, + HttpStart, + MountPoint, + NotificationsStart, + SavedObjectsStart, +} from '../../../../../../src/core/public'; +import { DataSourceManagementPluginSetup } from '../../../../../../src/plugins/data_source_management/public'; export interface AvailableIntegrationOverviewPageProps { http: HttpStart; @@ -13,7 +20,7 @@ export interface AvailableIntegrationOverviewPageProps { export interface AddedIntegrationOverviewPageProps { http: HttpStart; chrome: ChromeStart; - dataSourceEnabled: string; + dataSourceEnabled: boolean; } export interface AvailableIntegrationProps { @@ -25,10 +32,19 @@ export interface AddedIntegrationProps { http: HttpStart; chrome: ChromeStart; integrationInstanceId: string; + notifications: NotificationsStart; + dataSourceEnabled: boolean; + dataSourceManagement: DataSourceManagementPluginSetup; + savedObjectsMDSClient: SavedObjectsStart; + setActionMenu: (menuMount: MountPoint | undefined) => void; } export interface AvailableIntegrationProps { http: HttpStart; chrome: ChromeStart; integrationTemplateId: string; + notifications: NotificationsStart; + dataSourceEnabled: boolean; + dataSourceManagement: DataSourceManagementPluginSetup; + savedObjectsMDSClient: SavedObjectsStart; } diff --git a/public/components/integrations/components/setup_integration.tsx b/public/components/integrations/components/setup_integration.tsx index 1d8df5abb..b3f2f69b5 100644 --- a/public/components/integrations/components/setup_integration.tsx +++ b/public/components/integrations/components/setup_integration.tsx @@ -22,8 +22,9 @@ import React, { useEffect, useState } from 'react'; import { NotificationsStart, SavedObjectsStart } from '../../../../../../src/core/public'; import { DataSourceManagementPluginSetup } from '../../../../../../src/plugins/data_source_management/public'; import { Color } from '../../../../common/constants/integrations'; -import { CONSOLE_PROXY, INTEGRATIONS_BASE } from '../../../../common/constants/shared'; +import { INTEGRATIONS_BASE } from '../../../../common/constants/shared'; import { IntegrationConnectionType } from '../../../../common/types/integrations'; +import { SQLService } from '../../../../public/services/requests/sql'; import { coreRefs } from '../../../framework/core_refs'; import { addIntegrationRequest } from './create_integration_helpers'; import { SetupIntegrationFormInputs } from './setup_integration_inputs'; @@ -50,14 +51,18 @@ export interface IntegrationConfigProps { dataSourceEnabled: boolean; dataSourceManagement: DataSourceManagementPluginSetup; savedObjectsMDSClient: SavedObjectsStart; + handleSelectedDataSourceChange: (dataSourceMDSId?: string, dataSourceMDSLabel?: string) => void; } type SetupCallout = { show: true; title: string; color?: Color; text?: string } | { show: false }; +const sqlService = new SQLService(coreRefs.http!); + const runQuery = async ( query: string, datasource: string, - sessionId: string | null + sessionId: string | undefined, + dataSourceMDSId?: string ): Promise> => { // Used for polling const sleep = (ms: number) => { @@ -65,24 +70,22 @@ const runQuery = async ( }; try { - const http = coreRefs.http!; - const queryResponse: { queryId: string; sessionId: string } = await http.post(CONSOLE_PROXY, { - body: JSON.stringify({ query, datasource, lang: 'sql', sessionId }), - query: { - path: '_plugins/_async_query', - method: 'POST', + const queryResponse: { queryId: string; sessionId: string } = await sqlService.fetch( + { + query, + datasource, + lang: 'sql', + sessionId, }, - }); + dataSourceMDSId + ); + let poll: { status: string; error?: string } = { status: 'undefined' }; - const [queryId, newSessionId] = [queryResponse.queryId, queryResponse.sessionId]; + const { queryId, sessionId: newSessionId } = queryResponse; + while (!poll.error) { - poll = await http.post(CONSOLE_PROXY, { - body: '{}', - query: { - path: '_plugins/_async_query/' + queryId, - method: 'GET', - }, - }); + poll = await sqlService.fetchWithJobId({ queryId }, dataSourceMDSId); + if (poll.status.toLowerCase() === 'success') { return { ok: true, @@ -100,6 +103,7 @@ const runQuery = async ( } await sleep(3000); } + return { ok: false, error: new Error(poll.error) }; } catch (err) { console.error(err); @@ -138,16 +142,20 @@ const addIntegration = async ({ integration, setLoading, setCalloutLikeToast, + dataSourceMDSId, + dataSourceMDSLabel, setIsInstalling, }: { config: IntegrationSetupInputs; integration: IntegrationConfig; setLoading: (loading: boolean) => void; setCalloutLikeToast: (title: string, color?: Color, text?: string) => void; + dataSourceMDSId?: string; + dataSourceMDSLabel?: string; setIsInstalling?: (isInstalling: boolean, success?: boolean) => void; }) => { setLoading(true); - let sessionId: string | null = null; + let sessionId: string | undefined; if (config.connectionType === 'index') { let enabledWorkflows: string[] | undefined; @@ -163,6 +171,8 @@ const addIntegration = async ({ templateName: integration.name, integration, setToast: setCalloutLikeToast, + dataSourceMDSId, + dataSourceMDSLabel, name: config.displayName, indexPattern: config.connectionDataSource, skipRedirect: setIsInstalling ? true : false, @@ -180,7 +190,6 @@ const addIntegration = async ({ const assets: { data: ParsedIntegrationAsset[] } = await http.get( `${INTEGRATIONS_BASE}/repository/${integration.name}/assets` ); - for (const query of assets.data.filter( (a: ParsedIntegrationAsset): a is ParsedIntegrationAsset & { type: 'query' } => a.type === 'query' @@ -191,7 +200,12 @@ const addIntegration = async ({ } const queryStr = prepareQuery(query.query, config); - const result = await runQuery(queryStr, config.connectionDataSource, sessionId); + const result = await runQuery( + queryStr, + config.connectionDataSource, + sessionId, + dataSourceMDSId + ); if (!result.ok) { setLoading(false); setCalloutLikeToast('Failed to add integration', 'danger', result.error.message); @@ -205,6 +219,8 @@ const addIntegration = async ({ templateName: integration.name, integration, setToast: setCalloutLikeToast, + dataSourceMDSId, + dataSourceMDSLabel, name: config.displayName, indexPattern: `flint_${config.connectionDataSource}_${ config.connectionDatabaseName ?? 'default' @@ -245,6 +261,8 @@ export function SetupBottomBar({ loading, setLoading, setSetupCallout, + dataSourceMDSId, + dataSourceMDSLabel, unsetIntegration, setIsInstalling, }: { @@ -253,6 +271,8 @@ export function SetupBottomBar({ loading: boolean; setLoading: (loading: boolean) => void; setSetupCallout: (setupCallout: SetupCallout) => void; + dataSourceMDSId?: string; + dataSourceMDSLabel?: string; unsetIntegration?: () => void; setIsInstalling?: (isInstalling: boolean, success?: boolean) => void; }) { @@ -306,6 +326,8 @@ export function SetupBottomBar({ setIsInstalling(newLoading); }, setCalloutLikeToast, + dataSourceMDSId, + dataSourceMDSLabel, setIsInstalling, }); } else { @@ -314,6 +336,8 @@ export function SetupBottomBar({ config, setLoading, setCalloutLikeToast, + dataSourceMDSId, + dataSourceMDSLabel, setIsInstalling, }); } @@ -384,6 +408,8 @@ export function SetupIntegrationForm({ const [setupCallout, setSetupCallout] = useState({ show: false } as SetupCallout); const [showLoading, setShowLoading] = useState(false); + const [dataSourceMDSId, setDataSourceMDSId] = useState(''); + const [dataSourceMDSLabel, setDataSourceMDSLabel] = useState(''); useEffect(() => { const getTemplate = async () => { @@ -397,6 +423,11 @@ export function SetupIntegrationForm({ const updateConfig = (updates: Partial) => setConfig(Object.assign({}, integConfig, updates)); + const handleSelectedDataSourceChange = (id?: string, label?: string) => { + setDataSourceMDSId(id); + setDataSourceMDSLabel(label); + }; + const IntegrationInputFormComponent = forceConnection?.type === 'securityLake' || integConfig.connectionType === 'securityLake' ? SetupIntegrationInputsForSecurityLake @@ -416,6 +447,7 @@ export function SetupIntegrationForm({ notifications={notifications} dataSourceEnabled={dataSourceEnabled} savedObjectsMDSClient={savedObjectsMDSClient} + handleSelectedDataSourceChange={handleSelectedDataSourceChange} /> )} @@ -428,6 +460,8 @@ export function SetupIntegrationForm({ loading={showLoading} setLoading={setShowLoading} setSetupCallout={setSetupCallout} + dataSourceMDSId={dataSourceMDSId} + dataSourceMDSLabel={dataSourceMDSLabel} unsetIntegration={unsetIntegration} setIsInstalling={setIsInstalling} /> diff --git a/public/components/integrations/components/setup_integration_inputs.tsx b/public/components/integrations/components/setup_integration_inputs.tsx index a9b9437f3..a7c5cab15 100644 --- a/public/components/integrations/components/setup_integration_inputs.tsx +++ b/public/components/integrations/components/setup_integration_inputs.tsx @@ -19,6 +19,7 @@ import { NotificationsStart, SavedObjectsStart } from '../../../../../../src/cor import { DataSourceManagementPluginSetup } from '../../../../../../src/plugins/data_source_management/public'; import { CONSOLE_PROXY, DATACONNECTIONS_BASE } from '../../../../common/constants/shared'; import { IntegrationConnectionType } from '../../../../common/types/integrations'; +import { dataSourceFilterFn } from '../../../../common/utils/shared'; import { coreRefs } from '../../../framework/core_refs'; import { IntegrationConfigProps, IntegrationSetupInputs } from './setup_integration'; @@ -76,7 +77,8 @@ const integrationConnectionSelectorItems: Array<{ ]; const suggestDataSources = async ( - type: IntegrationConnectionType + type: IntegrationConnectionType, + dataSourceMDSId?: string ): Promise> => { const http = coreRefs.http!; try { @@ -86,6 +88,7 @@ const suggestDataSources = async ( query: { path: '_data_stream/ss4o_*', method: 'GET', + dataSourceId: dataSourceMDSId ?? '', }, }); return ( @@ -94,7 +97,9 @@ const suggestDataSources = async ( }) ?? [] ); } else if (type === 's3' || type === 'securityLake') { - const result = (await http.get(DATACONNECTIONS_BASE)) as Array<{ + const result = (await http.get( + `${DATACONNECTIONS_BASE}/dataSourceMDSId=${dataSourceMDSId ?? ''}` + )) as Array<{ name: string; connector: string; }>; @@ -196,6 +201,7 @@ export function IntegrationConnectionInputs({ notifications, savedObjectsMDSClient, lockConnectionType, + handleSelectedDataSourceChange, }: { config: IntegrationSetupInputs; updateConfig: (updates: Partial) => void; @@ -204,9 +210,17 @@ export function IntegrationConnectionInputs({ dataSourceEnabled: boolean; dataSourceManagement: DataSourceManagementPluginSetup; savedObjectsMDSClient: SavedObjectsStart; + handleSelectedDataSourceChange: (dataSourceMDSId?: string, dataSourceMDSLabel?: string) => void; lockConnectionType?: boolean; }) { + const [dataSourceMDSId, setDataSourceMDSId] = useState(''); const connectionType = INTEGRATION_CONNECTION_DATA_SOURCE_TYPES.get(config.connectionType)!; + const onSelectedDataSource = (e) => { + const dataConnectionId = e[0] ? e[0].id : undefined; + setDataSourceMDSId(dataConnectionId); + const dataConnectionLabel = e[0] ? e[0].label : undefined; + handleSelectedDataSourceChange(dataConnectionId, dataConnectionLabel); + }; let DataSourceSelector; if (dataSourceEnabled) { @@ -219,14 +233,14 @@ export function IntegrationConnectionInputs({ useEffect(() => { const updateDataSources = async () => { - const data = await suggestDataSources(config.connectionType); + const data = await suggestDataSources(config.connectionType, dataSourceMDSId); setDataSourceSuggestions(data); setIsSuggestionsLoading(false); }; setIsSuggestionsLoading(true); updateDataSources(); - }, [config.connectionType]); + }, [config.connectionType, dataSourceMDSId]); return ( <> @@ -242,6 +256,8 @@ export function IntegrationConnectionInputs({ disabled={false} fullWidth={false} removePrepend={true} + onSelectedDataSource={onSelectedDataSource} + dataSourceFilter={dataSourceFilterFn} /> @@ -438,6 +454,7 @@ export function SetupIntegrationFormInputs(props: IntegrationConfigProps) { dataSourceManagement, notifications, savedObjectsMDSClient, + handleSelectedDataSourceChange, } = props; return ( @@ -475,6 +492,7 @@ export function SetupIntegrationFormInputs(props: IntegrationConfigProps) { notifications={notifications} dataSourceEnabled={dataSourceEnabled} savedObjectsMDSClient={savedObjectsMDSClient} + handleSelectedDataSourceChange={handleSelectedDataSourceChange} /> {config.connectionType === 's3' ? ( <> diff --git a/public/components/integrations/home.tsx b/public/components/integrations/home.tsx index e30215a63..0b99dfeba 100644 --- a/public/components/integrations/home.tsx +++ b/public/components/integrations/home.tsx @@ -7,6 +7,7 @@ import React from 'react'; import { HashRouter, Route, RouteComponentProps, Switch } from 'react-router-dom'; import { ChromeBreadcrumb, + MountPoint, NotificationsStart, SavedObjectsStart, } from '../../../../../src/core/public'; @@ -26,6 +27,7 @@ interface HomeProps extends RouteComponentProps, AppAnalyticsCoreDeps { dataSourceEnabled: boolean; dataSourceManagement: DataSourceManagementPluginSetup; savedObjectsMDSClient: SavedObjectsStart; + setActionMenu: (menuMount: MountPoint | undefined) => void; } export const Home = (props: HomeProps) => { @@ -36,6 +38,7 @@ export const Home = (props: HomeProps) => { dataSourceManagement, savedObjectsMDSClient, notifications, + setActionMenu, } = props; const commonProps = { @@ -69,6 +72,11 @@ export const Home = (props: HomeProps) => { )} /> @@ -79,6 +87,10 @@ export const Home = (props: HomeProps) => { )} /> diff --git a/public/components/metrics/top_menu/metrics_export_panel.tsx b/public/components/metrics/top_menu/metrics_export_panel.tsx index ce8cb3e51..7ac52a7a1 100644 --- a/public/components/metrics/top_menu/metrics_export_panel.tsx +++ b/public/components/metrics/top_menu/metrics_export_panel.tsx @@ -6,8 +6,8 @@ import { EuiCompressedComboBox, EuiCompressedFieldText, - EuiFlexGroup, EuiCompressedFormRow, + EuiFlexGroup, EuiFlexItem, EuiForm, EuiHorizontalRule, diff --git a/public/components/metrics/top_menu/top_menu.tsx b/public/components/metrics/top_menu/top_menu.tsx index 92c7adf89..c1da76368 100644 --- a/public/components/metrics/top_menu/top_menu.tsx +++ b/public/components/metrics/top_menu/top_menu.tsx @@ -5,10 +5,10 @@ import { EuiCompressedFieldText, - EuiFlexGroup, - EuiFlexItem, EuiCompressedSelect, EuiCompressedSuperDatePicker, + EuiFlexGroup, + EuiFlexItem, } from '@elastic/eui'; import React from 'react'; import { useDispatch, useSelector } from 'react-redux'; diff --git a/public/components/trace_analytics/components/common/__tests__/__snapshots__/search_bar.test.tsx.snap b/public/components/trace_analytics/components/common/__tests__/__snapshots__/search_bar.test.tsx.snap index e6e545312..a95f3e81e 100644 --- a/public/components/trace_analytics/components/common/__tests__/__snapshots__/search_bar.test.tsx.snap +++ b/public/components/trace_analytics/components/common/__tests__/__snapshots__/search_bar.test.tsx.snap @@ -1064,4 +1064,4 @@ exports[`Search bar components renders search bar 1`] = `
-`; \ No newline at end of file +`; diff --git a/public/components/trace_analytics/components/services/__tests__/__snapshots__/services.test.tsx.snap b/public/components/trace_analytics/components/services/__tests__/__snapshots__/services.test.tsx.snap index 315ab6f04..cf0a8c08d 100644 --- a/public/components/trace_analytics/components/services/__tests__/__snapshots__/services.test.tsx.snap +++ b/public/components/trace_analytics/components/services/__tests__/__snapshots__/services.test.tsx.snap @@ -6296,4 +6296,4 @@ exports[`Services component renders services page 1`] = ` -`; \ No newline at end of file +`; diff --git a/public/components/trace_analytics/components/traces/__tests__/__snapshots__/traces.test.tsx.snap b/public/components/trace_analytics/components/traces/__tests__/__snapshots__/traces.test.tsx.snap index 36ea53e2f..3bc1aafbc 100644 --- a/public/components/trace_analytics/components/traces/__tests__/__snapshots__/traces.test.tsx.snap +++ b/public/components/trace_analytics/components/traces/__tests__/__snapshots__/traces.test.tsx.snap @@ -5266,4 +5266,4 @@ exports[`Traces component renders traces page 1`] = ` -`; \ No newline at end of file +`; diff --git a/server/adaptors/integrations/__test__/builder.test.ts b/server/adaptors/integrations/__test__/builder.test.ts index ca596e213..4ebec0ad3 100644 --- a/server/adaptors/integrations/__test__/builder.test.ts +++ b/server/adaptors/integrations/__test__/builder.test.ts @@ -3,8 +3,8 @@ * SPDX-License-Identifier: Apache-2.0 */ -import { TEST_INTEGRATION_CONFIG } from '../../../../test/constants'; import { SavedObjectsClientContract } from '../../../../../../src/core/server'; +import { TEST_INTEGRATION_CONFIG } from '../../../../test/constants'; import { IntegrationInstanceBuilder } from '../integrations_builder'; import { IntegrationReader } from '../repository/integration_reader'; import * as mockUtils from '../repository/utils'; diff --git a/server/adaptors/integrations/integrations_builder.ts b/server/adaptors/integrations/integrations_builder.ts index 41b0dc53f..033acf580 100644 --- a/server/adaptors/integrations/integrations_builder.ts +++ b/server/adaptors/integrations/integrations_builder.ts @@ -4,14 +4,16 @@ */ import { v4 as uuidv4 } from 'uuid'; +import { SavedObjectsBulkCreateObject } from '../../../../../src/core/public'; import { SavedObjectsClientContract } from '../../../../../src/core/server'; import { IntegrationReader } from './repository/integration_reader'; -import { SavedObjectsBulkCreateObject } from '../../../../../src/core/public'; import { deepCheck } from './repository/utils'; interface BuilderOptions { name: string; indexPattern: string; + dataSourceMDSId?: string; + dataSourceMDSLabel?: string; workflows?: string[]; dataSource?: string; tableName?: string; @@ -186,15 +188,23 @@ export class IntegrationInstanceBuilder { new Error('Attempted to create instance with invalid template', config.error) ); } - return Promise.resolve({ + const instance: IntegrationInstance = { name: options.name, templateName: config.value.name, - // Before data sources existed we called the index pattern a data source. Now we need the old - // name for BWC but still use the new data sources in building, so we map the variable only - // for returned output here dataSource: options.indexPattern, creationDate: new Date().toISOString(), assets: refs, - }); + }; + if (options.dataSourceMDSId) { + instance.references = [ + { + id: options.dataSourceMDSId, + name: options.dataSourceMDSLabel, + type: 'data-source', + }, + ]; + } + + return Promise.resolve(instance); } } diff --git a/server/adaptors/integrations/integrations_manager.ts b/server/adaptors/integrations/integrations_manager.ts index e24ccfae1..5ac26edb8 100644 --- a/server/adaptors/integrations/integrations_manager.ts +++ b/server/adaptors/integrations/integrations_manager.ts @@ -4,12 +4,12 @@ */ import path from 'path'; -import { addRequestToMetric } from '../../common/metrics/metrics_helper'; import { SavedObject, SavedObjectsClientContract } from '../../../../../src/core/server/types'; +import { addRequestToMetric } from '../../common/metrics/metrics_helper'; import { IntegrationInstanceBuilder } from './integrations_builder'; -import { TemplateManager } from './repository/repository'; import { FileSystemDataAdaptor } from './repository/fs_data_adaptor'; import { IndexDataAdaptor } from './repository/index_data_adaptor'; +import { TemplateManager } from './repository/repository'; export class IntegrationsManager { client: SavedObjectsClientContract; @@ -157,6 +157,8 @@ export class IntegrationsManager { templateName: string, name: string, indexPattern: string, + dataSourceMDSId?: string, + dataSourceMDSLabel?: string, workflows?: string[], dataSource?: string, tableName?: string @@ -173,6 +175,8 @@ export class IntegrationsManager { const result = await this.instanceBuilder.build(template, { name, indexPattern, + dataSourceMDSId, + dataSourceMDSLabel, workflows, dataSource, tableName, diff --git a/server/adaptors/integrations/types.ts b/server/adaptors/integrations/types.ts index 95a789504..0f229526f 100644 --- a/server/adaptors/integrations/types.ts +++ b/server/adaptors/integrations/types.ts @@ -113,6 +113,7 @@ interface IntegrationInstance { dataSource: string; creationDate: string; assets: AssetReference[]; + references?: []; } interface IntegrationInstanceResult extends IntegrationInstance { diff --git a/server/routes/dsl.ts b/server/routes/dsl.ts index 0560ae049..43bec95f1 100644 --- a/server/routes/dsl.ts +++ b/server/routes/dsl.ts @@ -3,17 +3,17 @@ * SPDX-License-Identifier: Apache-2.0 */ -import { schema } from '@osd/config-schema'; import { RequestParams } from '@elastic/elasticsearch'; +import { schema } from '@osd/config-schema'; import { IRouter } from '../../../../src/core/server'; -import { DSLFacet } from '../services/facets/dsl_facet'; import { DSL_BASE, - DSL_SEARCH, DSL_CAT, DSL_MAPPING, + DSL_SEARCH, DSL_SETTINGS, } from '../../common/constants/shared'; +import { DSLFacet } from '../services/facets/dsl_facet'; export function registerDslRoute( { router }: { router: IRouter; facet: DSLFacet }, @@ -236,4 +236,72 @@ export function registerDslRoute( } } ); + + router.post( + { + path: `${DSL_BASE}/integrations/refresh`, + validate: { + body: schema.any(), + query: schema.object({ + dataSourceMDSId: schema.maybe(schema.string({ defaultValue: '' })), + }), + }, + }, + async (context, request, response) => { + const dataSourceMDSId = request.query.dataSourceMDSId; + try { + let resp; + if (dataSourceEnabled && dataSourceMDSId) { + const client = await context.dataSource.opensearch.legacy.getClient(dataSourceMDSId); + resp = await client.callAPI('indices.refresh'); + } else { + resp = await context.core.opensearch.legacy.client.callAsCurrentUser('indices.refresh'); + } + return response.ok({ + body: resp, + }); + } catch (error) { + if (error.statusCode !== 404) console.error(error); + return response.custom({ + statusCode: error.statusCode || 500, + body: error.message, + }); + } + } + ); + + router.post( + { + path: `${DSL_BASE}/integrations/mapping`, + validate: { + body: schema.any(), + query: schema.object({ + dataSourceMDSId: schema.maybe(schema.string({ defaultValue: '' })), + }), + }, + }, + async (context, request, response) => { + const dataSourceMDSId = request.query.dataSourceMDSId; + try { + let resp; + if (dataSourceEnabled && dataSourceMDSId) { + const client = await context.dataSource.opensearch.legacy.getClient(dataSourceMDSId); + resp = await client.callAPI('indices.getMapping'); + } else { + resp = await context.core.opensearch.legacy.client.callAsCurrentUser( + 'indices.getMapping' + ); + } + return response.ok({ + body: resp, + }); + } catch (error) { + if (error.statusCode !== 404) console.error(error); + return response.custom({ + statusCode: error.statusCode || 500, + body: error.message, + }); + } + } + ); } diff --git a/server/routes/integrations/integrations_router.ts b/server/routes/integrations/integrations_router.ts index 68eed60d9..d48077af3 100644 --- a/server/routes/integrations/integrations_router.ts +++ b/server/routes/integrations/integrations_router.ts @@ -7,11 +7,11 @@ import { schema } from '@osd/config-schema'; import * as mime from 'mime'; import sanitize from 'sanitize-filename'; import { IRouter } from '../../../../../src/core/server'; -import { INTEGRATIONS_BASE } from '../../../common/constants/shared'; import { OpenSearchDashboardsResponse, OpenSearchDashboardsResponseFactory, } from '../../../../../src/core/server/http/router'; +import { INTEGRATIONS_BASE } from '../../../common/constants/shared'; import { IntegrationsManager } from '../../adaptors/integrations/integrations_manager'; /** @@ -71,6 +71,8 @@ export function registerIntegrationsRoute(router: IRouter) { templateName: schema.string(), }), body: schema.object({ + dataSourceMDSId: schema.maybe(schema.string({ defaultValue: '' })), + dataSourceMDSLabel: schema.maybe(schema.string({ defaultValue: '' })), name: schema.string(), indexPattern: schema.string(), workflows: schema.maybe(schema.arrayOf(schema.string())), @@ -86,6 +88,8 @@ export function registerIntegrationsRoute(router: IRouter) { request.params.templateName, request.body.name, request.body.indexPattern, + request.body.dataSourceMDSId, + request.body.dataSourceMDSLabel, request.body.workflows, request.body.dataSource, request.body.tableName