grafana/public/app/features/search/service/sql.ts
Josh Hunt 0c36b247af
Chore: Refactor Search/Folder view types into DashboardViewItem (#63162)
* Chore: Refactor Search/Folder view types into DashboardViewItem

* uid is not optional in api

* rename queryResultToNestedFolderItem function

* Fix error from locationInfo being empty

* change queryResultToViewItem to take view instead

* Fix sortMeta fields not showing on search cards

* Show correct parent for panel search results
2023-02-23 15:52:54 +00:00

233 lines
6.8 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import { ArrayVector, DataFrame, DataFrameView, FieldType, getDisplayProcessor, SelectableValue } from '@grafana/data';
import { config } from '@grafana/runtime';
import { TermCount } from 'app/core/components/TagFilter/TagFilter';
import { backendSrv } from 'app/core/services/backend_srv';
import { DEFAULT_MAX_VALUES, TYPE_KIND_MAP } from '../constants';
import { DashboardSearchHit, DashboardSearchItemType } from '../types';
import { LocationInfo } from './types';
import { replaceCurrentFolderQuery } from './utils';
import { DashboardQueryResult, GrafanaSearcher, QueryResponse, SearchQuery } from '.';
interface APIQuery {
query?: string;
tag?: string[];
limit?: number;
page?: number;
type?: DashboardSearchItemType;
// DashboardIds []int64
dashboardUID?: string[];
folderIds?: number[];
sort?: string;
starred?: boolean;
}
// Internal object to hold folderId
interface LocationInfoEXT extends LocationInfo {
folderId?: number;
}
export class SQLSearcher implements GrafanaSearcher {
locationInfo: Record<string, LocationInfoEXT> = {
general: {
kind: 'folder',
name: 'General',
url: '/dashboards',
folderId: 0,
},
}; // share location info with everyone
private async composeQuery(apiQuery: APIQuery, searchOptions: SearchQuery): Promise<APIQuery> {
const query = await replaceCurrentFolderQuery(searchOptions);
if (query.query === '*') {
if (query.kind?.length === 1 && TYPE_KIND_MAP[query.kind[0]]) {
apiQuery.type = TYPE_KIND_MAP[query.kind[0]];
}
} else if (query.query?.length) {
apiQuery.query = query.query;
}
if (query.uid) {
apiQuery.dashboardUID = query.uid;
} else if (query.location?.length) {
let info = this.locationInfo[query.location];
if (!info) {
// This will load all folder folders
await this.doAPIQuery({ type: DashboardSearchItemType.DashFolder, limit: 999 });
info = this.locationInfo[query.location];
}
apiQuery.folderIds = [info?.folderId ?? 0];
}
return apiQuery;
}
async search(query: SearchQuery): Promise<QueryResponse> {
if (query.facet?.length) {
throw new Error('facets not supported!');
}
const q = await this.composeQuery(
{
limit: query.limit ?? DEFAULT_MAX_VALUES, // default 1k max values
tag: query.tags,
sort: query.sort,
},
query
);
return this.doAPIQuery(q);
}
async starred(query: SearchQuery): Promise<QueryResponse> {
if (query.facet?.length) {
throw new Error('facets not supported!');
}
const q = await this.composeQuery(
{
limit: query.limit ?? DEFAULT_MAX_VALUES, // default 1k max values
tag: query.tags,
sort: query.sort,
starred: query.starred,
},
query
);
return this.doAPIQuery(q);
}
// returns the appropriate sorting options
async getSortOptions(): Promise<SelectableValue[]> {
// {
// "sortOptions": [
// {
// "description": "Sort results in an alphabetically ascending order",
// "displayName": "Alphabetically (AZ)",
// "meta": "",
// "name": "alpha-asc"
// },
// {
// "description": "Sort results in an alphabetically descending order",
// "displayName": "Alphabetically (ZA)",
// "meta": "",
// "name": "alpha-desc"
// }
// ]
// }
const opts = await backendSrv.get('/api/search/sorting');
return opts.sortOptions.map((v: any) => ({
value: v.name,
label: v.displayName,
}));
}
// NOTE: the bluge query will find tags within the current results, the SQL based one does not
async tags(query: SearchQuery): Promise<TermCount[]> {
const terms = await backendSrv.get<TermCount[]>('/api/dashboards/tags');
return terms.sort((a, b) => b.count - a.count);
}
async doAPIQuery(query: APIQuery): Promise<QueryResponse> {
const rsp = await backendSrv.get<DashboardSearchHit[]>('/api/search', query);
// Field values (columnar)
const kind: string[] = [];
const name: string[] = [];
const uid: string[] = [];
const url: string[] = [];
const tags: string[][] = [];
const location: string[] = [];
const sortBy: number[] = [];
let sortMetaName: string | undefined;
for (let hit of rsp) {
const k = hit.type === 'dash-folder' ? 'folder' : 'dashboard';
kind.push(k);
name.push(hit.title);
uid.push(hit.uid);
url.push(hit.url);
tags.push(hit.tags);
sortBy.push(hit.sortMeta!);
let v = hit.folderUid;
if (!v && k === 'dashboard') {
v = 'general';
}
location.push(v!);
if (hit.sortMetaName?.length) {
sortMetaName = hit.sortMetaName;
}
if (hit.folderUid && hit.folderTitle) {
this.locationInfo[hit.folderUid] = {
kind: 'folder',
name: hit.folderTitle,
url: hit.folderUrl!,
folderId: hit.folderId,
};
} else if (k === 'folder') {
this.locationInfo[hit.uid] = {
kind: k,
name: hit.title!,
url: hit.url,
folderId: hit.id,
};
}
}
const data: DataFrame = {
fields: [
{ name: 'kind', type: FieldType.string, config: {}, values: new ArrayVector(kind) },
{ name: 'name', type: FieldType.string, config: {}, values: new ArrayVector(name) },
{ name: 'uid', type: FieldType.string, config: {}, values: new ArrayVector(uid) },
{ name: 'url', type: FieldType.string, config: {}, values: new ArrayVector(url) },
{ name: 'tags', type: FieldType.other, config: {}, values: new ArrayVector(tags) },
{ name: 'location', type: FieldType.string, config: {}, values: new ArrayVector(location) },
],
length: name.length,
meta: {
custom: {
count: name.length,
max_score: 1,
locationInfo: this.locationInfo,
},
},
};
// Add enterprise sort fields as a field in the frame
if (sortMetaName?.length && sortBy.length) {
data.meta!.custom!.sortBy = sortMetaName;
data.fields.push({
name: sortMetaName, // Used in display
type: FieldType.number,
config: {},
values: new ArrayVector(sortBy),
});
}
for (const field of data.fields) {
field.display = getDisplayProcessor({ field, theme: config.theme2 });
}
const view = new DataFrameView<DashboardQueryResult>(data);
return {
totalRows: data.length,
view,
// Paging not supported with this version
loadMoreItems: async (startIndex: number, stopIndex: number): Promise<void> => {},
isItemLoaded: (index: number): boolean => true,
};
}
getFolderViewSort = () => {
// sorts alphabetically in memory after retrieving the folders from the database
return '';
};
}