Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Logs UI] Support the Unified Search Bar for Query input #143222

Merged
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
00e77ef
Use Unified Search Bar for query input
Kerry350 Oct 12, 2022
d116eb7
Fix initial state quirks
Kerry350 Oct 12, 2022
8c1506a
Improve type checking
Kerry350 Oct 12, 2022
d723bf5
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Oct 13, 2022
ac3c90c
Add new showQueryMenu prop with tests / story
Kerry350 Oct 13, 2022
32d6f67
Add showQueryMenu option to Logs and the create_search_bar wrapper
Kerry350 Oct 13, 2022
fe33caa
Remove unused variables
Kerry350 Oct 13, 2022
33e494b
Update x-pack/plugins/infra/public/containers/logs/log_filter/log_fil…
Kerry350 Oct 14, 2022
09f8330
PR tweaks
Kerry350 Oct 14, 2022
140e99c
Merge branch '142719-logs-unified-search-bar-query-input' of github.c…
Kerry350 Oct 14, 2022
e0e70e8
Fix loading from URL
Kerry350 Oct 14, 2022
a7abf8f
Adjust state handling to account for invalid manager queries
Kerry350 Oct 18, 2022
ffe3e45
Merge remote-tracking branch 'upstream' into 142719-logs-unified-sear…
Kerry350 Oct 18, 2022
3147810
Show filter error within UI and block requests
Kerry350 Oct 18, 2022
a2c5024
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Oct 19, 2022
e322bbc
PR amendments
Kerry350 Oct 19, 2022
246786c
Merge branch 'main' into 142719-logs-unified-search-bar-query-input
kibanamachine Oct 19, 2022
96d17b0
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Oct 20, 2022
a395d27
Use a real DataView reference with derivedDataView
Kerry350 Oct 20, 2022
929e9df
Update x-pack/plugins/infra/public/containers/logs/log_filter/with_lo…
Kerry350 Oct 20, 2022
94b86c9
Merge branch '142719-logs-unified-search-bar-query-input' of github.c…
Kerry350 Oct 20, 2022
fd193dd
Move location of default query setting
Kerry350 Oct 20, 2022
1007c0b
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Oct 20, 2022
6cdbe6b
Respond to dataView changes
Kerry350 Oct 20, 2022
d0805d5
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Oct 24, 2022
f2ccde9
Revert "Use a real DataView reference with derivedDataView"
Kerry350 Oct 24, 2022
b484f3e
Add "type" property rather than use a "real" data view
Kerry350 Oct 24, 2022
809238b
Update x-pack/plugins/infra/public/containers/logs/log_filter/log_fil…
Kerry350 Oct 25, 2022
3d13548
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Oct 25, 2022
6256bc8
Revert "Revert "Use a real DataView reference with derivedDataView""
Kerry350 Oct 25, 2022
3595f0f
Use real dataView in resolved logView
Kerry350 Oct 25, 2022
dd161d7
Merge branch '142719-logs-unified-search-bar-query-input' of github.c…
Kerry350 Oct 25, 2022
04e7404
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Oct 31, 2022
827621d
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Oct 31, 2022
f948012
Small tweaks
Kerry350 Oct 31, 2022
8867aef
Fix tests in link_to_logs.test.tsx
Kerry350 Oct 31, 2022
3f5f8d8
Update snapshot for log_views_client.test.ts
Kerry350 Oct 31, 2022
274cec2
Merge remote-tracking branch 'upstream/main' into 142719-logs-unified…
Kerry350 Nov 1, 2022
7c99c5e
Use createStubDataView over custom object
Kerry350 Nov 1, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -292,6 +292,15 @@ storiesOf('SearchBar', module)
query: { query: 'Test: miaou', language: 'kuery' },
} as unknown as SearchBarProps)
)
.add('with query menu off', () =>
wrapSearchBarInContext({
showDatePicker: false,
showFilterBar: false,
showQueryInput: true,
showQueryMenu: false,
query: { query: 'Test: miaou', language: 'kuery' },
} as unknown as SearchBarProps)
)
.add('with only the filter bar and the date picker on', () =>
wrapSearchBarInContext({
showDatePicker: true,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -191,6 +191,7 @@ export function createSearchBar({
showDatePicker={props.showDatePicker}
showFilterBar={props.showFilterBar}
showQueryBar={props.showQueryBar}
showQueryMenu={props.showQueryMenu}
showQueryInput={props.showQueryInput}
showSaveQuery={props.showSaveQuery}
showSubmitButton={props.showSubmitButton}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -138,6 +138,7 @@ describe('SearchBar', () => {
const FILTER_BAR = '[data-test-subj="unifiedFilterBar"]';
const QUERY_BAR = '.kbnQueryBar';
const QUERY_INPUT = '[data-test-subj="unifiedQueryInput"]';
const QUERY_MENU_BUTTON = '[data-test-subj="showQueryBarMenu"]';
const EDITOR = '[data-test-subj="unifiedTextLangEditor"]';

beforeEach(() => {
Expand Down Expand Up @@ -221,6 +222,20 @@ describe('SearchBar', () => {
expect(component.find(QUERY_INPUT).length).toBeFalsy();
});

it('Should NOT render the query menu button, if disabled', () => {
const component = mount(
wrapSearchBarInContext({
indexPatterns: [mockIndexPattern],
screenTitle: 'test screen',
onQuerySubmit: noop,
query: kqlQuery,
showQueryMenu: false,
})
);

expect(component.find(QUERY_MENU_BUTTON).length).toBeFalsy();
});

it('Should render query bar and filter bar', () => {
const component = mount(
wrapSearchBarInContext({
Expand Down
6 changes: 4 additions & 2 deletions src/plugins/unified_search/public/search_bar/search_bar.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ export interface SearchBarOwnProps<QT extends AggregateQuery | Query = Query> {
dataTestSubj?: string;
// Togglers
showQueryBar?: boolean;
showQueryMenu?: boolean;
showQueryInput?: boolean;
showFilterBar?: boolean;
showDatePicker?: boolean;
Expand Down Expand Up @@ -123,6 +124,7 @@ class SearchBarUI<QT extends (Query | AggregateQuery) | Query = Query> extends C
> {
public static defaultProps = {
showQueryBar: true,
showQueryMenu: true,
showFilterBar: true,
showDatePicker: true,
showSubmitButton: true,
Expand Down Expand Up @@ -450,7 +452,7 @@ class SearchBarUI<QT extends (Query | AggregateQuery) | Query = Query> extends C
/>
);

const queryBarMenu = (
const queryBarMenu = this.props.showQueryMenu ? (
<QueryBarMenu
nonKqlMode={this.props.nonKqlMode}
language={
Expand Down Expand Up @@ -490,7 +492,7 @@ class SearchBarUI<QT extends (Query | AggregateQuery) | Query = Query> extends C
: undefined
}
/>
);
) : undefined;

let filterBar;
if (this.shouldRenderFilterBar()) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,102 +5,93 @@
* 2.0.
*/

import { buildEsQuery, DataViewBase, Query } from '@kbn/es-query';
import { useMemo } from 'react';
import { buildEsQuery, DataViewBase, Query, AggregateQuery } from '@kbn/es-query';
import createContainer from 'constate';
import { useCallback, useState } from 'react';
import useDebounce from 'react-use/lib/useDebounce';
import { useKibanaQuerySettings } from '../../../utils/use_kibana_query_settings';
import { BuiltEsQuery } from '../log_stream';
import { useKibanaContextForPlugin } from '../../../hooks/use_kibana';
import { useSubscription } from '../../../utils/use_observable';

interface ILogFilterState {
filterQuery: {
parsedQuery: BuiltEsQuery;
serializedQuery: string;
originalQuery: Query;
} | null;
filterQueryDraft: Query;
validationErrors: string[];
}

const initialLogFilterState: ILogFilterState = {
filterQuery: null,
filterQueryDraft: {
language: 'kuery',
query: '',
},
validationErrors: [],
};

const validationDebounceTimeout = 1000; // milliseconds

export const useLogFilterState = ({ indexPattern }: { indexPattern: DataViewBase }) => {
const [logFilterState, setLogFilterState] = useState<ILogFilterState>(initialLogFilterState);
const {
data: {
query: { queryString },
},
} = useKibanaContextForPlugin().services;
const kibanaQuerySettings = useKibanaQuerySettings();

const parseQuery = useCallback(
(filterQuery: Query) => buildEsQuery(indexPattern, filterQuery, [], kibanaQuerySettings),
[indexPattern, kibanaQuerySettings]
);

const setLogFilterQueryDraft = useCallback((filterQueryDraft: Query) => {
setLogFilterState((previousLogFilterState) => ({
...previousLogFilterState,
filterQueryDraft,
validationErrors: [],
}));
}, []);
function isQuery(value: Query | AggregateQuery): value is Query {
return value.hasOwnProperty('query');
}
Kerry350 marked this conversation as resolved.
Show resolved Hide resolved

const [, cancelPendingValidation] = useDebounce(
() => {
setLogFilterState((previousLogFilterState) => {
try {
parseQuery(logFilterState.filterQueryDraft);
return {
...previousLogFilterState,
validationErrors: [],
};
} catch (error) {
return {
...previousLogFilterState,
validationErrors: [`${error}`],
};
const getLogFilterQuery = useCallback(
(filterQuery: Query | AggregateQuery) => {
try {
// NOTE: We sync with the QueryString manager - and therefore other solutions - but we don't support SQL syntax.
if (!isQuery(filterQuery)) {
throw new Error('Only Query types are supported');
}
});
const parsedQuery = parseQuery(filterQuery);
return {
parsedQuery,
serializedQuery: JSON.stringify(parsedQuery),
originalQuery: filterQuery,
};
} catch (error) {
// NOTE: If parsing fails or an AggregateQuery is in use
queryString.setQuery({
language: 'kuery',
query: '',
});
Kerry350 marked this conversation as resolved.
Show resolved Hide resolved
}
},
validationDebounceTimeout,
[logFilterState.filterQueryDraft, parseQuery]
[parseQuery, queryString]
);

const [logFilterState, setLogFilterState] = useState<ILogFilterState>({
filterQuery: getLogFilterQuery(queryString.getQuery()) || null,
});
Kerry350 marked this conversation as resolved.
Show resolved Hide resolved

const applyLogFilterQuery = useCallback(
(filterQuery: Query) => {
cancelPendingValidation();
try {
const parsedQuery = parseQuery(filterQuery);
setLogFilterState((previousLogFilterState) => ({
...previousLogFilterState,
filterQuery: {
parsedQuery,
serializedQuery: JSON.stringify(parsedQuery),
originalQuery: filterQuery,
},
filterQueryDraft: filterQuery,
validationErrors: [],
}));
} catch (error) {
(filterQuery: Query | AggregateQuery) => {
const logFilterQuery = getLogFilterQuery(filterQuery);
if (logFilterQuery) {
setLogFilterState((previousLogFilterState) => ({
...previousLogFilterState,
validationErrors: [`${error}`],
filterQuery: logFilterQuery,
}));
}
},
[cancelPendingValidation, parseQuery]
[getLogFilterQuery]
);

useSubscription(
useMemo(() => queryString.getUpdates$(), [queryString]),
{
next: useCallback(() => {
const esQuery = queryString.getQuery();
applyLogFilterQuery(esQuery);
}, [applyLogFilterQuery, queryString]),
}
Kerry350 marked this conversation as resolved.
Show resolved Hide resolved
);

return {
filterQuery: logFilterState.filterQuery,
filterQueryDraft: logFilterState.filterQueryDraft,
isFilterQueryDraftValid: logFilterState.validationErrors.length === 0,
setLogFilterQueryDraft,
applyLogFilterQuery,
};
};
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,23 +10,30 @@ import React from 'react';
import { Query } from '@kbn/es-query';
import { replaceStateKeyInQueryString, UrlStateContainer } from '../../../utils/url_state';
import { useLogFilterStateContext } from './log_filter_state';
import { useKibanaContextForPlugin } from '../../../hooks/use_kibana';

export const WithLogFilterUrlState: React.FC = () => {
const { filterQuery, applyLogFilterQuery } = useLogFilterStateContext();
const { filterQuery } = useLogFilterStateContext();
const {
data: {
query: { queryString },
},
} = useKibanaContextForPlugin().services;

return (
<UrlStateContainer
populateWithInitialState={true} // NOTE: Since the filterQuery is ultimately determined by the QueryString manager populate with the initial state
urlState={filterQuery?.originalQuery}
urlStateKey="logFilter"
mapToUrlState={mapToFilterQuery}
onChange={(urlState) => {
if (urlState) {
applyLogFilterQuery(urlState);
queryString.setQuery(urlState);
}
}}
onInitialize={(urlState) => {
if (urlState) {
applyLogFilterQuery(urlState);
queryString.setQuery(urlState);
Kerry350 marked this conversation as resolved.
Show resolved Hide resolved
}
}}
/>
Expand Down
45 changes: 21 additions & 24 deletions x-pack/plugins/infra/public/pages/logs/stream/page_toolbar.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,8 @@
*/

import { EuiFlexGroup, EuiFlexItem } from '@elastic/eui';
import { Query } from '@kbn/es-query';
import { i18n } from '@kbn/i18n';
import React from 'react';
import { QueryStringInput } from '@kbn/unified-search-plugin/public';
import { DataView } from '@kbn/data-views-plugin/public';
import { euiStyled } from '@kbn/kibana-react-plugin/common';
import { useKibanaContextForPlugin } from '../../../hooks/use_kibana';
Expand All @@ -18,7 +16,6 @@ import { LogDatepicker } from '../../../components/logging/log_datepicker';
import { LogHighlightsMenu } from '../../../components/logging/log_highlights_menu';
import { LogTextScaleControls } from '../../../components/logging/log_text_scale_controls';
import { LogTextWrapControls } from '../../../components/logging/log_text_wrap_controls';
import { useLogFilterStateContext } from '../../../containers/logs/log_filter';
import { useLogEntryFlyoutContext } from '../../../containers/logs/log_flyout';
import { useLogHighlightsStateContext } from '../../../containers/logs/log_highlights/log_highlights';
import { useLogPositionStateContext } from '../../../containers/logs/log_position';
Expand All @@ -29,11 +26,18 @@ export const LogsToolbar = () => {
const { derivedDataView } = useLogViewContext();
const { availableTextScales, setTextScale, setTextWrap, textScale, textWrap } =
useLogViewConfigurationContext();
const { filterQueryDraft, isFilterQueryDraftValid, applyLogFilterQuery, setLogFilterQueryDraft } =
useLogFilterStateContext();
const { setSurroundingLogsId } = useLogEntryFlyoutContext();
const { http, notifications, docLinks, uiSettings, data, storage, unifiedSearch } =
useKibanaContextForPlugin().services;
const {
http,
notifications,
docLinks,
uiSettings,
data,
storage,
unifiedSearch: {
ui: { SearchBar },
},
} = useKibanaContextForPlugin().services;

const {
setHighlightTerms,
Expand All @@ -57,27 +61,20 @@ export const LogsToolbar = () => {
<div>
<EuiFlexGroup alignItems="center" justifyContent="spaceBetween" gutterSize="l" wrap>
<QueryBarFlexItem>
<QueryStringInput
disableLanguageSwitcher={true}
<SearchBar
appName={i18n.translate('xpack.infra.appName', {
defaultMessage: 'Infra logs',
})}
iconType="search"
indexPatterns={[derivedDataView as DataView]}
isInvalid={!isFilterQueryDraftValid}
onChange={(query: Query) => {
setSurroundingLogsId(null);
setLogFilterQueryDraft(query);
}}
onSubmit={(query: Query) => {
setSurroundingLogsId(null);
applyLogFilterQuery(query);
}}
placeholder={i18n.translate('xpack.infra.logsPage.toolbar.kqlSearchFieldPlaceholder', {
defaultMessage: 'Search for log entries… (e.g. host.name:host-1)',
})}
query={filterQueryDraft}
appName={i18n.translate('xpack.infra.appName', {
defaultMessage: 'Infra logs',
})}
deps={{ unifiedSearch, notifications, http, docLinks, uiSettings, data, storage }}
useDefaultBehaviors={true}
indexPatterns={[derivedDataView as DataView]}
showQueryInput={true}
showQueryMenu={false}
showFilterBar={false}
showDatePicker={false}
/>
</QueryBarFlexItem>
<EuiFlexItem grow={false}>
Expand Down
2 changes: 1 addition & 1 deletion x-pack/plugins/infra/public/utils/url_state.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,11 @@

import { parse, stringify } from 'query-string';
import { History, Location } from 'history';
import { throttle } from 'lodash';
import React from 'react';
import { Route, RouteProps } from 'react-router-dom';
import { decode, encode, RisonValue } from 'rison-node';
import { url } from '@kbn/kibana-utils-plugin/public';
import { throttle } from 'lodash';

interface UrlStateContainerProps<UrlState> {
urlState: UrlState | undefined;
Expand Down