The open and composable observability and data visualization platform. Visualize metrics, logs, and traces from multiple sources like Prometheus, Loki, Elasticsearch, InfluxDB, Postgres and many more.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
grafana/public/app/features/search/service/sql.ts

233 lines
6.8 KiB

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 (A–Z)",
// "meta": "",
// "name": "alpha-asc"
// },
// {
// "description": "Sort results in an alphabetically descending order",
// "displayName": "Alphabetically (Z–A)",
// "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 '';
};
}