D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
usr
/
share
/
grafana
/
public
/
app
/
plugins
/
datasource
/
loki
/
components
/
Filename :
LokiContextUi.tsx
back
Copy
import { css } from '@emotion/css'; import React, { useCallback, useEffect, useMemo, useState } from 'react'; import { useAsync } from 'react-use'; import { GrafanaTheme2, LogRowModel, renderMarkdown, SelectableValue } from '@grafana/data'; import { reportInteraction } from '@grafana/runtime'; import { Button, Collapse, Icon, InlineField, InlineFieldRow, InlineSwitch, Label, MultiSelect, RenderUserContentAsHTML, Spinner, Tooltip, useStyles2, } from '@grafana/ui'; import store from 'app/core/store'; import { RawQuery } from '../../prometheus/querybuilder/shared/RawQuery'; import { LogContextProvider, LOKI_LOG_CONTEXT_PRESERVED_LABELS, PreservedLabels, SHOULD_INCLUDE_PIPELINE_OPERATIONS, } from '../LogContextProvider'; import { escapeLabelValueInSelector } from '../languageUtils'; import { isQueryWithParser } from '../queryUtils'; import { lokiGrammar } from '../syntax'; import { ContextFilter, LokiQuery } from '../types'; export interface LokiContextUiProps { logContextProvider: LogContextProvider; row: LogRowModel; updateFilter: (value: ContextFilter[]) => void; onClose: () => void; origQuery?: LokiQuery; runContextQuery?: () => void; } function getStyles(theme: GrafanaTheme2) { return { labels: css` display: flex; gap: ${theme.spacing(0.5)}; `, wrapper: css` display: flex; flex-direction: column; flex: 1; gap: ${theme.spacing(0.5)}; position: relative; `, textWrapper: css` display: flex; align-items: center; `, hidden: css` visibility: hidden; `, label: css` max-width: 100%; &:first-of-type { margin-bottom: ${theme.spacing(2)}; } &:not(:first-of-type) { margin: ${theme.spacing(2)} 0; } `, rawQueryContainer: css` text-align: start; line-break: anywhere; margin-top: -${theme.spacing(0.25)}; margin-right: ${theme.spacing(6)}; min-height: ${theme.spacing(4)}; `, ui: css` background-color: ${theme.colors.background.secondary}; padding: ${theme.spacing(2)}; `, rawQuery: css` display: inline; `, queryDescription: css` margin-left: ${theme.spacing(0.5)}; `, iconButton: css` position: absolute; top: ${theme.spacing(1)}; right: ${theme.spacing(1)}; z-index: ${theme.zIndex.navbarFixed}; `, operationsToggle: css` margin: ${theme.spacing(1)} 0 ${theme.spacing(-1)} 0; & > div { margin: 0; & > label { padding: 0; } } `, }; } export const IS_LOKI_LOG_CONTEXT_UI_OPEN = 'isLogContextQueryUiOpen'; export function LokiContextUi(props: LokiContextUiProps) { const { row, logContextProvider, updateFilter, onClose, origQuery, runContextQuery } = props; const styles = useStyles2(getStyles); const [contextFilters, setContextFilters] = useState<ContextFilter[]>([]); const [initialized, setInitialized] = useState(false); const [loading, setLoading] = useState(false); const [isOpen, setIsOpen] = useState(store.getBool(IS_LOKI_LOG_CONTEXT_UI_OPEN, false)); const [includePipelineOperations, setIncludePipelineOperations] = useState( store.getBool(SHOULD_INCLUDE_PIPELINE_OPERATIONS, false) ); const timerHandle = React.useRef<number>(); const previousInitialized = React.useRef<boolean>(false); const previousContextFilters = React.useRef<ContextFilter[]>([]); const isInitialState = useMemo(() => { // Initial query has all regular labels enabled and all parsed labels disabled if (initialized && contextFilters.some((filter) => filter.fromParser === filter.enabled)) { return false; } // if we include pipeline operations, we also want to enable the revert button if (includePipelineOperations && logContextProvider.queryContainsValidPipelineStages(origQuery)) { return false; } return true; }, [contextFilters, includePipelineOperations, initialized, logContextProvider, origQuery]); useEffect(() => { if (!initialized) { return; } // don't trigger if we initialized, this will be the same query anyways. if (!previousInitialized.current) { previousInitialized.current = initialized; return; } if (contextFilters.filter(({ enabled, fromParser }) => enabled && !fromParser).length === 0) { setContextFilters(previousContextFilters.current); return; } previousContextFilters.current = structuredClone(contextFilters); if (timerHandle.current) { clearTimeout(timerHandle.current); } setLoading(true); timerHandle.current = window.setTimeout(() => { updateFilter(contextFilters.filter(({ enabled }) => enabled)); // We are storing the removed labels and selected extracted labels in local storage so we can // preselect the labels in the UI in the next log context view. const preservedLabels: PreservedLabels = { removedLabels: [], selectedExtractedLabels: [], }; contextFilters.forEach(({ enabled, fromParser, label }) => { // We only want to store real labels that were removed from the initial query if (!enabled && !fromParser) { preservedLabels.removedLabels.push(label); } // Or extracted labels that were added to the initial query if (enabled && fromParser) { preservedLabels.selectedExtractedLabels.push(label); } }); store.set(LOKI_LOG_CONTEXT_PRESERVED_LABELS, JSON.stringify(preservedLabels)); setLoading(false); }, 1500); return () => { clearTimeout(timerHandle.current); }; // eslint-disable-next-line react-hooks/exhaustive-deps }, [contextFilters, initialized]); useEffect(() => { return () => { clearTimeout(timerHandle.current); onClose(); }; }, [onClose]); useAsync(async () => { setLoading(true); const initContextFilters = await logContextProvider.getInitContextFilters(row.labels, origQuery); setContextFilters(initContextFilters); setInitialized(true); setLoading(false); }); useEffect(() => { reportInteraction('grafana_explore_logs_loki_log_context_loaded', { logRowUid: row.uid, type: 'load', }); return () => { reportInteraction('grafana_explore_logs_loki_log_context_loaded', { logRowUid: row.uid, type: 'unload', }); }; }, [row.uid]); const realLabels = contextFilters.filter(({ fromParser }) => !fromParser); const realLabelsEnabled = realLabels.filter(({ enabled }) => enabled); const parsedLabels = contextFilters.filter(({ fromParser }) => fromParser); const parsedLabelsEnabled = parsedLabels.filter(({ enabled }) => enabled); const contextFilterToSelectFilter = useCallback((contextFilter: ContextFilter): SelectableValue<string> => { return { label: `${contextFilter.label}="${escapeLabelValueInSelector(contextFilter.value)}"`, value: contextFilter.label, }; }, []); // Currently we support adding of parser and showing parsed labels only if there is 1 parser const showParsedLabels = origQuery && isQueryWithParser(origQuery.expr).parserCount === 1 && parsedLabels.length > 0; let queryExpr = logContextProvider.prepareExpression( contextFilters.filter(({ enabled }) => enabled), origQuery ); return ( <div className={styles.wrapper}> <Tooltip content={'Revert to initial log context query.'}> <div className={styles.iconButton}> <Button data-testid="revert-button" icon="history-alt" variant="secondary" disabled={isInitialState} onClick={(e) => { reportInteraction('grafana_explore_logs_loki_log_context_reverted', { logRowUid: row.uid, }); setContextFilters((contextFilters) => { return contextFilters.map((contextFilter) => ({ ...contextFilter, // For revert to initial query we need to enable all labels and disable all parsed labels enabled: !contextFilter.fromParser, })); }); // We are removing the preserved labels from local storage so we can preselect the labels in the UI store.delete(LOKI_LOG_CONTEXT_PRESERVED_LABELS); store.delete(SHOULD_INCLUDE_PIPELINE_OPERATIONS); setIncludePipelineOperations(false); }} /> </div> </Tooltip> <Collapse collapsible={true} isOpen={isOpen} onToggle={() => { store.set(IS_LOKI_LOG_CONTEXT_UI_OPEN, !isOpen); setIsOpen((isOpen) => !isOpen); reportInteraction('grafana_explore_logs_loki_log_context_toggled', { logRowUid: row.uid, action: !isOpen ? 'open' : 'close', }); }} label={ <div className={styles.rawQueryContainer}> {initialized ? ( <> <RawQuery lang={{ grammar: lokiGrammar, name: 'loki' }} query={queryExpr} className={styles.rawQuery} /> <Tooltip content="The initial log context query is created from all labels defining the stream for the selected log line. Use the editor below to customize the log context query."> <Icon name="info-circle" size="sm" className={styles.queryDescription} /> </Tooltip> </> ) : ( <Spinner /> )} </div> } > <div className={styles.ui}> <Label className={styles.label} description="The initial log context query is created from all labels defining the stream for the selected log line. You can broaden your search by removing one or more of the label filters." > Widen the search </Label> <MultiSelect isLoading={loading} options={realLabels.map(contextFilterToSelectFilter)} value={realLabelsEnabled.map(contextFilterToSelectFilter)} closeMenuOnSelect={true} maxMenuHeight={200} noOptionsMessage="No further labels available" onChange={(keys, actionMeta) => { if (actionMeta.action === 'select-option') { reportInteraction('grafana_explore_logs_loki_log_context_filtered', { logRowUid: row.uid, type: 'label', action: 'select', }); } if (actionMeta.action === 'remove-value') { reportInteraction('grafana_explore_logs_loki_log_context_filtered', { logRowUid: row.uid, type: 'label', action: 'remove', }); } return setContextFilters( contextFilters.map((filter) => { if (filter.fromParser) { return filter; } filter.enabled = keys.some((key) => key.value === filter.label); return filter; }) ); }} /> {showParsedLabels && ( <> <Label className={styles.label} description={`By using a parser in your original query, you can use filters for extracted labels. Refine your search by applying extracted labels created from the selected log line.`} > Refine the search </Label> <MultiSelect isLoading={loading} options={parsedLabels.map(contextFilterToSelectFilter)} value={parsedLabelsEnabled.map(contextFilterToSelectFilter)} closeMenuOnSelect={true} maxMenuHeight={200} noOptionsMessage="No further labels available" isClearable={true} onChange={(keys, actionMeta) => { if (actionMeta.action === 'select-option') { reportInteraction('grafana_explore_logs_loki_log_context_filtered', { logRowUid: row.uid, type: 'parsed_label', action: 'select', }); } if (actionMeta.action === 'remove-value') { reportInteraction('grafana_explore_logs_loki_log_context_filtered', { logRowUid: row.uid, type: 'parsed_label', action: 'remove', }); } setContextFilters( contextFilters.map((filter) => { if (!filter.fromParser) { return filter; } filter.enabled = keys.some((key) => key.value === filter.label); return filter; }) ); }} /> </> )} {logContextProvider.queryContainsValidPipelineStages(origQuery) && ( <InlineFieldRow className={styles.operationsToggle}> <InlineField label="Include LogQL pipeline operations" tooltip={ <RenderUserContentAsHTML content={renderMarkdown( "This will include LogQL operations such as `line_format` or `label_format`. It won't include line or label filter operations." )} /> } > <InlineSwitch value={includePipelineOperations} showLabel={true} transparent={true} onChange={(e) => { reportInteraction('grafana_explore_logs_loki_log_context_pipeline_toggled', { logRowUid: row.uid, action: e.currentTarget.checked ? 'enable' : 'disable', }); store.set(SHOULD_INCLUDE_PIPELINE_OPERATIONS, e.currentTarget.checked); setIncludePipelineOperations(e.currentTarget.checked); if (runContextQuery) { runContextQuery(); } }} /> </InlineField> </InlineFieldRow> )} </div> </Collapse> </div> ); }