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/plugins/datasource/loki/components/LokiContextUi.tsx

417 lines
14 KiB

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(4)};
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>
);
}