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

[Saved Queries] Improve saved query management #170599

Merged
merged 79 commits into from
Feb 12, 2024
Merged
Show file tree
Hide file tree
Changes from 34 commits
Commits
Show all changes
79 commits
Select commit Hold shift + click to select a range
744a72e
Optimize saved queries exist check
davismcphee Nov 4, 2023
ef1a293
Check for saved query title conflicts at save instead of preloading a…
davismcphee Nov 4, 2023
88026dd
Add pagination to saved query list
davismcphee Nov 4, 2023
ab567ca
Get rid of saved query _all endpoint
davismcphee Nov 4, 2023
4e37a21
Improve saved query filtering and pagination
davismcphee Nov 4, 2023
f215a1d
Move delete button out of list to improve UX and improve a11y
davismcphee Nov 4, 2023
5d7082f
Add support for titleKeyword field and sorting on title
davismcphee Nov 4, 2023
f67c59a
Add active badge
davismcphee Nov 6, 2023
4e61c8a
[CI] Auto-commit changed files from 'node scripts/check_mappings_upda…
kibanamachine Nov 6, 2023
72ef1e7
[CI] Auto-commit changed files from 'node scripts/jest_integration -u…
kibanamachine Nov 6, 2023
b18ce80
UI cleanup -- remove and replace it with EUI prop, change props to…
davismcphee Nov 7, 2023
1dc1b0d
Remove unused useEuiTheme
davismcphee Nov 8, 2023
4a38b0a
Fix issue where load and delete query buttons are still enabled after…
davismcphee Nov 8, 2023
14a0d90
[CI] Auto-commit changed files from 'node scripts/check_mappings_upda…
kibanamachine Dec 6, 2023
03c2d38
Update saved query popover design to match new mockups
davismcphee Dec 7, 2023
9ea296a
Add query count display
davismcphee Dec 7, 2023
9fa2fff
Clear selected query after search if it no longer exists on the page
davismcphee Dec 7, 2023
8a99931
Improve the logic that pulls the loaded query to the first page
davismcphee Dec 7, 2023
cf0b432
Fix Jest tests
davismcphee Dec 8, 2023
5b85dbf
Fix issue deleting saved queries, and fix failing FTR tests
davismcphee Dec 8, 2023
27df4a4
Fix broken tests
davismcphee Dec 8, 2023
e8598a2
Attempting to fix failing test
davismcphee Dec 8, 2023
17c60c3
Drop saved query management list style override
davismcphee Jan 16, 2024
1b0408b
Small type cleanup and removing modified translation entries.
davismcphee Jan 16, 2024
c7ea202
Added comment about Axe failure
davismcphee Jan 16, 2024
81676a1
Fix bug in Lens where loading the same saved query which is already l…
davismcphee Jan 17, 2024
543d44a
Add Jest tests for saved_query_management_list.test.tsx
davismcphee Jan 17, 2024
fcb45ea
Finish adding Jest tests
davismcphee Jan 17, 2024
6663eee
Cleanup in saved_query_mangement_list
davismcphee Jan 17, 2024
008b853
Jest test cleanup
davismcphee Jan 17, 2024
18dd2b3
Update route_handler_context Jest tests
davismcphee Jan 17, 2024
30e66ec
Add model version tests
davismcphee Jan 18, 2024
35f83cf
[CI] Auto-commit changed files from 'node scripts/lint_ts_projects --…
kibanamachine Jan 18, 2024
24f2911
[CI] Auto-commit changed files from 'node scripts/jest_integration -u…
kibanamachine Jan 18, 2024
61e509a
Merge branch 'main' into saved-query-impovements
davismcphee Jan 26, 2024
c629b8c
Fix update button underline on hover and the focus outline being cut off
davismcphee Jan 27, 2024
88989da
Clean up query menu panels
davismcphee Jan 27, 2024
cafc480
Simplify query name help text
davismcphee Jan 27, 2024
9beba60
Improve query save and delete messages
davismcphee Jan 27, 2024
e48acf9
Remove /_all endpoint
davismcphee Jan 27, 2024
e99675a
Implement server side duplicate title check, and improve error handling
davismcphee Jan 27, 2024
e3938b5
Make saved query update button aware of time filter
davismcphee Jan 27, 2024
70c3c4d
Refine saved query search functionality
davismcphee Jan 29, 2024
daabd0b
Drop saved query switchToModelVersionAt
davismcphee Jan 29, 2024
7e87c5f
Merge branch 'main' into saved-query-impovements
davismcphee Jan 29, 2024
fda58ed
Fix duplicate query check
davismcphee Jan 29, 2024
303e1b8
Remove saved query title help text
davismcphee Jan 29, 2024
7d677de
Reset currentPanelId on popover open, fixes minor styling bug
davismcphee Jan 29, 2024
acf4cf0
Remove selectable list item background override
davismcphee Jan 29, 2024
02d7d60
No longer clear selected saved query on page change
davismcphee Jan 29, 2024
6da59d9
Add disabled button text
davismcphee Feb 1, 2024
711d620
Migrate saved_query endpoint error handling to Boom
davismcphee Feb 1, 2024
70fbfd1
Update Jest route_handler_context Jest tests
davismcphee Feb 1, 2024
572c7a0
Add saved query duplicate title check
davismcphee Feb 1, 2024
99ba526
Update saved query search functionality
davismcphee Feb 2, 2024
367d4f5
Merge branch 'main' into saved-query-impovements
davismcphee Feb 2, 2024
8287ae3
Clean up saved query search logic
davismcphee Feb 2, 2024
8f1385f
Fix failing tests by waiting for saved query form to close after saving
davismcphee Feb 3, 2024
c765fed
Fix broken saved query Cypress tests
davismcphee Feb 3, 2024
257dffb
Convert saved query API tests to TypeScript, and add tests for search…
davismcphee Feb 6, 2024
fd0e331
Add utilities to saved query API integration tests to simplify tests
davismcphee Feb 6, 2024
f62e5ce
Reorganize saved query API integration tests
davismcphee Feb 6, 2024
fea1606
Ensure all saved query filter changes are detected
davismcphee Feb 7, 2024
1da5917
Fix issue where saved query 'Include filters' and 'Include time filte…
davismcphee Feb 7, 2024
5ea22f5
Add loading indicator to query list
davismcphee Feb 7, 2024
0d06849
Fix issue where query count could show a negative value in some situa…
davismcphee Feb 7, 2024
eee8645
Allow searching for queries with titles containing special characters
davismcphee Feb 7, 2024
c6cff43
Merge branch 'main' into saved-query-impovements
davismcphee Feb 7, 2024
a505553
Merge branch 'main' into saved-query-impovements
kibanamachine Feb 7, 2024
94e9361
Fix React errors caused by calling usePrettyDuration within itemTitle…
davismcphee Feb 8, 2024
032a918
Switch from lodash max to Math.max
davismcphee Feb 8, 2024
8736091
[CI] Auto-commit changed files from 'node scripts/lint_ts_projects --…
kibanamachine Feb 8, 2024
9a09649
Merge branch 'main' into saved-query-impovements
stratoula Feb 8, 2024
d98d2c2
Merge branch 'main' into saved-query-impovements
davismcphee Feb 9, 2024
6dd865a
Fix current_mappings.json
davismcphee Feb 9, 2024
9888240
Merge branch 'main' into saved-query-impovements
davismcphee Feb 9, 2024
5eff403
Merge branch 'main' into saved-query-impovements
davismcphee Feb 9, 2024
62281ff
Merge branch 'main' into saved-query-impovements
davismcphee Feb 9, 2024
41605ec
Merge branch 'main' into saved-query-impovements
davismcphee Feb 12, 2024
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 @@ -172,7 +172,8 @@
],
"query": [
"description",
"title"
"title",
"titleKeyword"
],
"kql-telemetry": [],
"search-session": [
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -572,6 +572,9 @@
"title": {
"type": "text"
},
"titleKeyword": {
"type": "keyword"
},
"description": {
"type": "text"
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -130,7 +130,7 @@ describe('checking migration metadata changes on all registered SO types', () =>
"osquery-pack-asset": "cd140bc2e4b092e93692b587bf6e38051ef94c75",
"osquery-saved-query": "6095e288750aa3164dfe186c74bc5195c2bf2bd4",
"policy-settings-protection-updates-note": "33924bb246f9e5bcb876109cc83e3c7a28308352",
"query": "21cbbaa09abb679078145ce90087b1e88b7eae95",
"query": "501bece68f26fe561286a488eabb1a8ab12f1137",
"risk-engine-configuration": "b105d4a3c6adce40708d729d12e5ef3c8fbd9508",
"rules-settings": "892a2918ebaeba809a612b8d97cec0b07c800b5f",
"sample-data-telemetry": "37441b12f5b0159c2d6d5138a494c9f440e950b5",
Expand Down
2 changes: 1 addition & 1 deletion src/plugins/data/public/query/mocks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ const createStartContractMock = () => {
addToQueryLog: jest.fn(),
filterManager: createFilterManagerMock(),
queryString: queryStringManagerMock.createStartContract(),
savedQueries: { getSavedQuery: jest.fn() } as any,
savedQueries: { getSavedQuery: jest.fn(), getSavedQueryCount: jest.fn() } as any,
state$: new Observable(),
getState: jest.fn(),
timefilter: timefilterServiceMock.createStartContract(),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,6 @@ const {
findSavedQueries,
createQuery,
updateQuery,
getAllSavedQueries,
getSavedQueryCount,
} = createSavedQueryService(http);

Expand Down Expand Up @@ -64,19 +63,6 @@ describe('saved query service', () => {
});
});

describe('getAllSavedQueries', function () {
it('should post and extract the saved queries from the response', async () => {
http.post.mockResolvedValue({
total: 0,
savedQueries: [{ attributes: savedQueryAttributes }],
});
const result = await getAllSavedQueries();
expect(http.post).toBeCalled();
expect(http.post).toHaveBeenCalledWith(`${SAVED_QUERY_BASE_URL}/_all`, { version });
expect(result).toEqual([{ attributes: savedQueryAttributes }]);
});
});

describe('findSavedQueries', function () {
it('should post and return the total & saved queries', async () => {
http.post.mockResolvedValue({
Expand Down
10 changes: 0 additions & 10 deletions src/plugins/data/public/query/saved_query/saved_query_service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -30,15 +30,6 @@ export const createSavedQueryService = (http: HttpStart) => {
return savedQuery;
};

// we have to tell the saved objects client how many to fetch, otherwise it defaults to fetching 20 per page
const getAllSavedQueries = async (): Promise<SavedQuery[]> => {
const { savedQueries } = await http.post<{ savedQueries: SavedQuery[] }>(
`${SAVED_QUERY_BASE_URL}/_all`,
{ version }
);
return savedQueries;
};

// findSavedQueries will do a 'match_all' if no search string is passed in
const findSavedQueries = async (
search: string = '',
Expand Down Expand Up @@ -71,7 +62,6 @@ export const createSavedQueryService = (http: HttpStart) => {
return {
createQuery,
updateQuery,
getAllSavedQueries,
findSavedQueries,
getSavedQuery,
deleteSavedQuery,
Expand Down
1 change: 0 additions & 1 deletion src/plugins/data/public/query/saved_query/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,6 @@ export type { SavedQuery, SavedQueryAttributes };
export interface SavedQueryService {
createQuery: (attributes: SavedQueryAttributes) => Promise<SavedQuery>;
updateQuery: (id: string, attributes: SavedQueryAttributes) => Promise<SavedQuery>;
getAllSavedQueries: () => Promise<SavedQuery[]>;
findSavedQueries: (
searchText?: string,
perPage?: number,
Expand Down
62 changes: 48 additions & 14 deletions src/plugins/data/server/query/route_handler_context.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,10 @@ import { coreMock } from '@kbn/core/server/mocks';
import { FilterStateStore, Query } from '@kbn/es-query';
import { DATA_VIEW_SAVED_OBJECT_TYPE } from '../../common';
import type { SavedObject, SavedQueryAttributes } from '../../common';
import { registerSavedQueryRouteHandlerContext } from './route_handler_context';
import {
InternalSavedQueryAttributes,
registerSavedQueryRouteHandlerContext,
} from './route_handler_context';
import { SavedObjectsFindResponse, SavedObjectsUpdateResponse } from '@kbn/core/server';

const mockContext = {
Expand All @@ -31,6 +34,10 @@ const savedQueryAttributes: SavedQueryAttributes = {
},
filters: [],
};
const internalSavedQueryAttributes: InternalSavedQueryAttributes = {
...savedQueryAttributes,
titleKeyword: 'foo',
};
const savedQueryAttributesBar: SavedQueryAttributes = {
title: 'bar',
description: 'baz',
Expand Down Expand Up @@ -90,19 +97,23 @@ describe('saved query route handler context', () => {

describe('create', function () {
it('should create a saved object for the given attributes', async () => {
const mockResponse: SavedObject<SavedQueryAttributes> = {
const mockResponse: SavedObject<InternalSavedQueryAttributes> = {
id: 'foo',
type: 'query',
attributes: savedQueryAttributes,
attributes: internalSavedQueryAttributes,
references: [],
};
mockSavedObjectsClient.create.mockResolvedValue(mockResponse);

const response = await context.create(savedQueryAttributes);

expect(mockSavedObjectsClient.create).toHaveBeenCalledWith('query', savedQueryAttributes, {
references: [],
});
expect(mockSavedObjectsClient.create).toHaveBeenCalledWith(
'query',
internalSavedQueryAttributes,
{
references: [],
}
);
expect(response).toEqual({
id: 'foo',
attributes: savedQueryAttributes,
Expand All @@ -117,10 +128,13 @@ describe('saved query route handler context', () => {
query: { match_all: {} },
},
};
const mockResponse: SavedObject<SavedQueryAttributes> = {
const mockResponse: SavedObject<InternalSavedQueryAttributes> = {
id: 'foo',
type: 'query',
attributes: savedQueryAttributesWithQueryObject,
attributes: {
...savedQueryAttributesWithQueryObject,
titleKeyword: 'foo',
},
references: [],
};
mockSavedObjectsClient.create.mockResolvedValue(mockResponse);
Expand All @@ -136,10 +150,13 @@ describe('saved query route handler context', () => {
filters: savedQueryAttributesWithFilters.filters,
timefilter: savedQueryAttributesWithFilters.timefilter,
};
const mockResponse: SavedObject<SavedQueryAttributes> = {
const mockResponse: SavedObject<InternalSavedQueryAttributes> = {
id: 'foo',
type: 'query',
attributes: serializedSavedQueryAttributesWithFilters,
attributes: {
...serializedSavedQueryAttributesWithFilters,
titleKeyword: 'foo',
},
references: [],
};
mockSavedObjectsClient.create.mockResolvedValue(mockResponse);
Expand Down Expand Up @@ -176,10 +193,10 @@ describe('saved query route handler context', () => {

describe('update', function () {
it('should update a saved object for the given attributes', async () => {
const mockResponse: SavedObject<SavedQueryAttributes> = {
const mockResponse: SavedObject<InternalSavedQueryAttributes> = {
id: 'foo',
type: 'query',
attributes: savedQueryAttributes,
attributes: internalSavedQueryAttributes,
references: [],
};
mockSavedObjectsClient.update.mockResolvedValue(mockResponse);
Expand All @@ -189,7 +206,7 @@ describe('saved query route handler context', () => {
expect(mockSavedObjectsClient.update).toHaveBeenCalledWith(
'query',
'foo',
savedQueryAttributes,
internalSavedQueryAttributes,
{
references: [],
}
Expand Down Expand Up @@ -241,6 +258,14 @@ describe('saved query route handler context', () => {

const response = await context.find();

expect(mockSavedObjectsClient.find).toHaveBeenCalledWith({
type: 'query',
page: 1,
perPage: 50,
search: '',
sortField: 'titleKeyword',
sortOrder: 'asc',
});
expect(response.savedQueries).toEqual([{ id: 'foo', attributes: savedQueryAttributes }]);
});

Expand Down Expand Up @@ -274,10 +299,12 @@ describe('saved query route handler context', () => {
const response = await context.find({ search: 'foo' });

expect(mockSavedObjectsClient.find).toHaveBeenCalledWith({
type: 'query',
page: 1,
perPage: 50,
search: 'foo',
type: 'query',
sortField: undefined,
sortOrder: undefined,
});
expect(response.savedQueries).toEqual([{ id: 'foo', attributes: savedQueryAttributes }]);
});
Expand Down Expand Up @@ -361,6 +388,8 @@ describe('saved query route handler context', () => {
page: 1,
perPage: 2,
search: '',
sortField: 'titleKeyword',
sortOrder: 'asc',
type: 'query',
});
expect(response.savedQueries).toEqual(
Expand Down Expand Up @@ -568,6 +597,11 @@ describe('saved query route handler context', () => {

const response = await context.count();

expect(mockSavedObjectsClient.find).toHaveBeenCalledWith({
type: 'query',
page: 0,
perPage: 0,
});
expect(response).toEqual(1);
});
});
Expand Down
71 changes: 36 additions & 35 deletions src/plugins/data/server/query/route_handler_context.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,15 +8,24 @@

import { CustomRequestHandlerContext, RequestHandlerContext, SavedObject } from '@kbn/core/server';
import { isFilters, isOfQueryType } from '@kbn/es-query';
import { omit } from 'lodash';
import { isQuery, SavedQueryAttributes } from '../../common';
import { extract, inject } from '../../common/query/filters/persistable_state';

export interface InternalSavedQueryAttributes extends SavedQueryAttributes {
titleKeyword: string;
}

function injectReferences({
id,
attributes,
attributes: internalAttributes,
namespaces,
references,
}: Pick<SavedObject<SavedQueryAttributes>, 'id' | 'attributes' | 'namespaces' | 'references'>) {
}: Pick<
SavedObject<InternalSavedQueryAttributes>,
'id' | 'attributes' | 'namespaces' | 'references'
>) {
const attributes: SavedQueryAttributes = omit(internalAttributes, 'titleKeyword');
const { query } = attributes;
if (isOfQueryType(query) && typeof query.query === 'string') {
try {
Expand Down Expand Up @@ -48,8 +57,9 @@ function extractReferences({
}
}

const attributes: SavedQueryAttributes = {
const attributes: InternalSavedQueryAttributes = {
title: title.trim(),
titleKeyword: title.trim(),
description: description.trim(),
query: {
...query,
Expand Down Expand Up @@ -83,7 +93,7 @@ export async function registerSavedQueryRouteHandlerContext(context: RequestHand
verifySavedQuery(attrs);
const { attributes, references } = extractReferences(attrs);

const savedObject = await soClient.create<SavedQueryAttributes>('query', attributes, {
const savedObject = await soClient.create<InternalSavedQueryAttributes>('query', attributes, {
references,
});

Expand All @@ -97,9 +107,14 @@ export async function registerSavedQueryRouteHandlerContext(context: RequestHand
verifySavedQuery(attrs);
const { attributes, references } = extractReferences(attrs);

const savedObject = await soClient.update<SavedQueryAttributes>('query', id, attributes, {
references,
});
const savedObject = await soClient.update<InternalSavedQueryAttributes>(
'query',
id,
attributes,
{
references,
}
);

// TODO: Handle properly
if (savedObject.error) throw new Error(savedObject.error.message);
Expand All @@ -108,10 +123,8 @@ export async function registerSavedQueryRouteHandlerContext(context: RequestHand
};

const getSavedQuery = async (id: string) => {
const { saved_object: savedObject, outcome } = await soClient.resolve<SavedQueryAttributes>(
'query',
id
);
const { saved_object: savedObject, outcome } =
await soClient.resolve<InternalSavedQueryAttributes>('query', id);
if (outcome === 'conflict') {
throw new Error(`Multiple saved queries found with ID: ${id} (legacy URL alias conflict)`);
} else if (savedObject.error) {
Expand All @@ -121,41 +134,30 @@ export async function registerSavedQueryRouteHandlerContext(context: RequestHand
};

const getSavedQueriesCount = async () => {
const { total } = await soClient.find<SavedQueryAttributes>({
const { total } = await soClient.find<InternalSavedQueryAttributes>({
type: 'query',
page: 0,
perPage: 0,
Comment on lines +198 to +199
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ℹ️ Minor performance optimization to no longer request documents when we only need the total.

});
return total;
};

const findSavedQueries = async ({ page = 1, perPage = 50, search = '' } = {}) => {
const { total, saved_objects: savedObjects } = await soClient.find<SavedQueryAttributes>({
type: 'query',
page,
perPage,
search,
});
const { total, saved_objects: savedObjects } =
await soClient.find<InternalSavedQueryAttributes>({
type: 'query',
page,
perPage,
search,
sortField: search ? undefined : 'titleKeyword',
sortOrder: search ? undefined : 'asc',
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ℹ️ If no search term is passed, we sort by title alphabetically. If a search term is passed, we use the default score based sorting.

});

const savedQueries = savedObjects.map(injectReferences);

return { total, savedQueries };
};

const getAllSavedQueries = async () => {
const finder = soClient.createPointInTimeFinder<SavedQueryAttributes>({
type: 'query',
perPage: 100,
});

const savedObjects: Array<SavedObject<SavedQueryAttributes>> = [];
for await (const response of finder.find()) {
savedObjects.push(...(response.saved_objects ?? []));
}
await finder.close();

const savedQueries = savedObjects.map(injectReferences);
return { total: savedQueries.length, savedQueries };
};

const deleteSavedQuery = async (id: string) => {
return await soClient.delete('query', id, { force: true });
};
Expand All @@ -166,7 +168,6 @@ export async function registerSavedQueryRouteHandlerContext(context: RequestHand
get: getSavedQuery,
count: getSavedQueriesCount,
find: findSavedQueries,
getAll: getAllSavedQueries,
delete: deleteSavedQuery,
};
}
Expand Down
Loading