grafana/public/app/plugins/panel/logs/LogsPanel.tsx

873 lines
32 KiB
TypeScript

import { css, cx } from '@emotion/css';
import { groupBy } from 'lodash';
import { useCallback, useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react';
import * as React from 'react';
import { isObservable, lastValueFrom } from 'rxjs';
import {
AbsoluteTimeRange,
CoreApp,
DataFrame,
DataHoverClearEvent,
DataHoverEvent,
DataQueryResponse,
DataSourceApi,
dateTimeForTimeZone,
GrafanaTheme2,
hasLogsContextSupport,
hasLogsContextUiSupport,
Labels,
LogRowContextOptions,
LogRowModel,
LogsSortOrder,
PanelData,
PanelProps,
TimeRange,
TimeZone,
toUtc,
urlUtil,
LogSortOrderChangeEvent,
LoadingState,
rangeUtil,
} from '@grafana/data';
import { Trans } from '@grafana/i18n';
import { config, getAppEvents } from '@grafana/runtime';
import { ScrollContainer, usePanelContext, useStyles2 } from '@grafana/ui';
import { getFieldLinksForExplore } from 'app/features/explore/utils/links';
import { ControlledLogRows } from 'app/features/logs/components/ControlledLogRows';
import { InfiniteScroll } from 'app/features/logs/components/InfiniteScroll';
import { LogRowContextModal } from 'app/features/logs/components/log-context/LogRowContextModal';
import { LogLineContext } from 'app/features/logs/components/panel/LogLineContext';
import { LogList } from 'app/features/logs/components/panel/LogList';
import { PanelDataErrorView } from 'app/features/panel/components/PanelDataErrorView';
import { combineResponses } from 'app/plugins/datasource/loki/mergeResponses';
import { createAndCopyShortLink, getLogsPermalinkRange } from '../../../core/utils/shortLinks';
import { LogLabels } from '../../../features/logs/components/LogLabels';
import { LogRows } from '../../../features/logs/components/LogRows';
import { COMMON_LABELS, dataFrameToLogsModel, dedupLogRows } from '../../../features/logs/logsModel';
import {
GetFieldLinksFn,
isCoreApp,
isIsFilterLabelActive,
isLogLineMenuCustomItems,
isOnClickFilterLabel,
isOnClickFilterOutLabel,
isOnClickFilterOutString,
isOnClickFilterString,
isOnClickHideField,
isOnClickShowField,
isOnLogOptionsChange,
isOnNewLogsReceivedType,
isReactNodeArray,
isSetDisplayedFields,
onNewLogsReceivedType,
Options,
} from './types';
import { useDatasourcesFromTargets } from './useDatasourcesFromTargets';
interface LogsPanelProps extends PanelProps<Options> {
/**
* Adds a key => value filter to the query referenced by the provided DataFrame refId. Used by Log details and Logs table.
* onClickFilterLabel?: (key: string, value: string, frame?: DataFrame) => void;
*
* Adds a negative key => value filter to the query referenced by the provided DataFrame refId. Used by Log details and Logs table.
* onClickFilterOutLabel?: (key: string, value: string, frame?: DataFrame) => void;
*
* Adds a string filter to the query referenced by the provided DataFrame refId. Used by the Logs popover menu.
* onClickFilterOutString?: (value: string, refId?: string) => void;
*
* Removes a string filter to the query referenced by the provided DataFrame refId. Used by the Logs popover menu.
* onClickFilterString?: (value: string, refId?: string) => void;
*
* Determines if a given key => value filter is active in a given query. Used by Log details.
* isFilterLabelActive?: (key: string, value: string, refId?: string) => Promise<boolean>;
*
* Array of field names to display instead of the log line. Pass a list of fields or an empty array to enable hide/show fields in Log Details.
* displayedFields?: string[]
*
* Called from the "eye" icon in Log Details to request showing the displayed field. If ommited, a default implementation is used.
* onClickShowField?: (key: string) => void;
*
* Called from the "eye" icon in Log Details to request hiding the displayed field. If ommited, a default implementation is used.
* onClickHideField?: (key: string) => void;
*
* Called from the new Log Details Panel when fields are reordered. If ommited, a default implementation is used.
* setDisplayedFields?: (key: string) => void;
*
* Passed to the LogRowMenuCell component to be rendered before the default actions in the menu.
* logRowMenuIconsBefore?: ReactNode[];
*
* Passed to the LogRowMenuCell component to be rendered after the default actions in the menu.
* logRowMenuIconsAfter?: ReactNode[];
*
* Callback to be invoked when enableInfiniteScrolling and new logs have been received after an scroll event.
* onNewLogsReceived?: (allLogs: DataFrame[], newLogs: DataFrame[]) => void;
*
* Log Controls props:
*
* Enables a sidebar with controls for scrolling, sort order, deduplication, filtering, timestamps, wrapping, etc.
* showControls?: boolean
*
* If controls are enabled, the component will use this key to store changes to the aforementioned options.
* controlsStorageKey?: string
*
* If controls are enabled, this function is called when a change is made in one of the options from the controls.
* onLogOptionsChange?: (option: LogListControlOptions, value: string | boolean | string[]) => void;
*
* When the feature toggle newLogsPanel is enabled, you can pass extra options to the LogLineMenu component.
* These options are an array of items with { label, onClick } or { divider: true } for dividers.
* logLineMenuCustomItems?: LogLineMenuCustomItem[];
*
* Use the default, bigger, font size, or a smaller one. Defaults to "default".
* fontSize?: 'default' | 'small'
*
* Set the mode used by the Log Details panel. Displayed as a sidebar, or inline below the log line. Defaults to "inline".
* detailsMode?: 'inline' | 'sidebar'
*
* When showing timestamps, toggle between showing nanoseconds or milliseconds.
* timestampResolution?: 'ms' | 'ns'
*/
}
interface LogsPermalinkUrlState {
logs?: {
id?: string;
};
}
const noCommonLabels: Labels = {};
export const LogsPanel = ({
data,
timeZone,
fieldConfig,
options: {
showControls,
controlsStorageKey,
showLabels,
showTime,
wrapLogMessage,
showCommonLabels,
prettifyLogMessage,
sortOrder,
dedupStrategy,
enableLogDetails,
showLogContextToggle,
onClickFilterLabel,
onClickFilterOutLabel,
onClickFilterOutString,
onClickFilterString,
onLogOptionsChange,
isFilterLabelActive,
logRowMenuIconsBefore,
logRowMenuIconsAfter,
logLineMenuCustomItems,
enableInfiniteScrolling,
onNewLogsReceived,
fontSize,
syntaxHighlighting,
detailsMode: detailsModeProp,
noInteractions,
timestampResolution,
...options
},
height,
id,
}: LogsPanelProps) => {
const isAscending = sortOrder === LogsSortOrder.Ascending;
const style = useStyles2(getStyles);
const logsContainerRef = useRef<HTMLDivElement>(null);
const [contextRow, setContextRow] = useState<LogRowModel | null>(null);
const [scrollElement, setScrollElement] = useState<HTMLDivElement | null>(null);
const [displayedFields, setDisplayedFields] = useState<string[]>(options.displayedFields ?? []);
// Loading state to be passed as a prop to the <InfiniteScroll> component
const [infiniteScrolling, setInfiniteScrolling] = useState(false);
// Loading ref to prevent firing multiple requests
const loadingRef = useRef(false);
const [panelData, setPanelData] = useState(data);
const dataSourcesMap = useDatasourcesFromTargets(panelData.request?.targets);
// Prevents the scroll position to change when new data from infinite scrolling is received
const keepScrollPositionRef = useRef<null | 'infinite-scroll' | 'user'>(null);
let closeCallback = useRef<() => void>();
const { app, eventBus, onAddAdHocFilter } = usePanelContext();
useEffect(() => {
getAppEvents().publish(
new LogSortOrderChangeEvent({
order: sortOrder,
})
);
}, [sortOrder]);
const onLogRowHover = useCallback(
(row?: LogRowModel) => {
if (row) {
eventBus.publish(
new DataHoverEvent({
point: {
time: row.timeEpochMs,
},
})
);
}
},
[eventBus]
);
const onLogContainerMouseLeave = useCallback(() => {
eventBus.publish(new DataHoverClearEvent());
}, [eventBus]);
const onCloseContext = useCallback(() => {
setContextRow(null);
if (closeCallback.current) {
closeCallback.current();
}
}, [closeCallback]);
const onOpenContext = useCallback(
(row: LogRowModel, onClose: () => void) => {
setContextRow(row);
closeCallback.current = onClose;
},
[closeCallback]
);
const showContextToggle = useCallback(
(row: LogRowModel): boolean => {
if (
!row.dataFrame.refId ||
!dataSourcesMap ||
(!showLogContextToggle &&
panelData.request?.app !== CoreApp.Dashboard &&
panelData.request?.app !== CoreApp.PanelEditor &&
panelData.request?.app !== CoreApp.PanelViewer)
) {
return false;
}
const dataSource = dataSourcesMap.get(row.dataFrame.refId);
return hasLogsContextSupport(dataSource);
},
[dataSourcesMap, showLogContextToggle, panelData.request?.app]
);
const showPermaLink = useCallback(() => {
return !(
panelData.request?.app !== CoreApp.Dashboard &&
panelData.request?.app !== CoreApp.PanelEditor &&
panelData.request?.app !== CoreApp.PanelViewer
);
}, [panelData.request?.app]);
const getLogRowContext = useCallback(
async (row: LogRowModel, origRow: LogRowModel, options: LogRowContextOptions): Promise<DataQueryResponse> => {
if (!origRow.dataFrame.refId || !dataSourcesMap) {
return Promise.resolve({ data: [] });
}
const query = panelData.request?.targets[0];
if (!query) {
return Promise.resolve({ data: [] });
}
const dataSource = dataSourcesMap.get(origRow.dataFrame.refId);
if (!hasLogsContextSupport(dataSource)) {
return Promise.resolve({ data: [] });
}
options.scopedVars = panelData.request?.scopedVars;
return dataSource.getLogRowContext(row, options, query);
},
[panelData.request?.targets, panelData.request?.scopedVars, dataSourcesMap]
);
const getLogRowContextUi = useCallback(
(origRow: LogRowModel, runContextQuery?: () => void): React.ReactNode => {
if (!origRow.dataFrame.refId || !dataSourcesMap) {
return <></>;
}
const query = panelData.request?.targets[0];
if (!query) {
return <></>;
}
const dataSource = dataSourcesMap.get(origRow.dataFrame.refId);
if (!hasLogsContextUiSupport(dataSource)) {
return <></>;
}
if (!dataSource.getLogRowContextUi) {
return <></>;
}
return dataSource.getLogRowContextUi(origRow, runContextQuery, query, panelData.request?.scopedVars);
},
[panelData.request?.targets, panelData.request?.scopedVars, dataSourcesMap]
);
// Important to memoize stuff here, as panel rerenders a lot for example when resizing.
const [logRows, deduplicatedRows, commonLabels] = useMemo(() => {
const logs = panelData
? dataFrameToLogsModel(
panelData.series,
panelData.request?.intervalMs,
undefined,
panelData.request?.targets,
Boolean(enableInfiniteScrolling)
)
: null;
const logRows = logs?.rows || [];
const commonLabels = logs?.meta?.find((m) => m.label === COMMON_LABELS);
const deduplicatedRows = dedupLogRows(logRows, dedupStrategy);
return [logRows, deduplicatedRows, commonLabels];
}, [dedupStrategy, enableInfiniteScrolling, panelData]);
const onPermalinkClick = useCallback(
async (row: LogRowModel) => {
return await copyDashboardUrl(row, logRows, panelData.timeRange);
},
[panelData.timeRange, logRows]
);
useEffect(() => {
if (data.state !== LoadingState.Loading) {
setPanelData(data);
}
}, [data]);
useLayoutEffect(() => {
if (config.featureToggles.newLogsPanel) {
return;
}
if (!logsContainerRef.current || !scrollElement || keepScrollPositionRef.current) {
keepScrollPositionRef.current =
keepScrollPositionRef.current === 'infinite-scroll' ? null : keepScrollPositionRef.current;
return;
}
/**
* In dashboards, users with newest logs at the bottom have the expectation of keeping the scroll at the bottom
* when new data is received. See https://github.com/grafana/grafana/pull/37634
*/
if (panelData.request?.app === CoreApp.Dashboard || panelData.request?.app === CoreApp.PanelEditor) {
scrollElement.scrollTo(0, isAscending ? logsContainerRef.current.scrollHeight : 0);
}
}, [panelData.request?.app, isAscending, scrollElement, logRows]);
const getFieldLinks: GetFieldLinksFn = useCallback(
(field, rowIndex, dataFrame, vars) => {
return getFieldLinksForExplore({
field,
rowIndex,
range: panelData.timeRange,
dataFrame: dataFrame,
vars,
});
},
[panelData]
);
/**
* Scrolls the given row into view.
*/
const scrollIntoView = useCallback(
(row: HTMLElement) => {
scrollElement?.scrollTo({
top: row.offsetTop,
behavior: 'smooth',
});
},
[scrollElement]
);
const handleOnClickFilterLabel = useCallback(
(key: string, value: string) => {
onAddAdHocFilter?.({
key,
value,
operator: '=',
});
},
[onAddAdHocFilter]
);
const handleOnClickFilterOutLabel = useCallback(
(key: string, value: string) => {
onAddAdHocFilter?.({
key,
value,
operator: '!=',
});
},
[onAddAdHocFilter]
);
const showField = useCallback(
(key: string) => {
const index = displayedFields?.indexOf(key);
if (index === -1) {
setDisplayedFields(displayedFields?.concat(key));
}
},
[displayedFields]
);
const hideField = useCallback(
(key: string) => {
const index = displayedFields?.indexOf(key);
if (index !== undefined && index > -1) {
setDisplayedFields(displayedFields?.filter((k) => key !== k));
}
},
[displayedFields]
);
useEffect(() => {
if (options.displayedFields) {
setDisplayedFields(options.displayedFields);
}
}, [options.displayedFields]);
// Respect the scroll position when refreshing the panel
useEffect(() => {
function handleScroll() {
if (!scrollElement) {
return;
}
// Signal to keep the user scroll position
keepScrollPositionRef.current = 'user';
const atTheBottom = scrollElement.scrollHeight - scrollElement.scrollTop - scrollElement.clientHeight === 0;
// Except when the user resets the scroll to the original position depending on the sort direction
if (scrollElement.scrollTop === 0 && !isAscending) {
keepScrollPositionRef.current = null;
} else if (atTheBottom && isAscending) {
keepScrollPositionRef.current = null;
}
}
scrollElement?.addEventListener('scroll', handleScroll);
scrollElement?.addEventListener('wheel', handleScroll);
return () => {
scrollElement?.removeEventListener('scroll', handleScroll);
scrollElement?.removeEventListener('wheel', handleScroll);
};
}, [isAscending, scrollElement]);
const loadMoreLogs = useCallback(
async (scrollRange: AbsoluteTimeRange) => {
if (!data.request || !config.featureToggles.logsInfiniteScrolling || loadingRef.current) {
return;
}
loadingRef.current = true;
setInfiniteScrolling(true);
const onNewLogsReceivedCallback = isOnNewLogsReceivedType(onNewLogsReceived) ? onNewLogsReceived : undefined;
let newSeries: DataFrame[] = [];
try {
newSeries = await requestMoreLogs(dataSourcesMap, panelData, scrollRange, timeZone, onNewLogsReceivedCallback);
} catch (e) {
console.error(e);
} finally {
setInfiniteScrolling(false);
loadingRef.current = false;
}
keepScrollPositionRef.current = 'infinite-scroll';
setPanelData({
...panelData,
series: newSeries,
});
},
[data.request, dataSourcesMap, onNewLogsReceived, panelData, timeZone]
);
const renderCommonLabels = () => (
<div className={cx(style.labelContainer, isAscending && style.labelContainerAscending)}>
<span className={style.label}>
<Trans i18nKey="logs.logs-panel.render-common-labels.common-labels">Common labels:</Trans>
</span>
<LogLabels
labels={typeof commonLabels?.value === 'object' ? commonLabels?.value : noCommonLabels}
emptyMessage="(no common labels)"
/>
</div>
);
const initialScrollPosition = useMemo(() => {
/**
* In dashboards, users with newest logs at the bottom have the expectation of keeping the scroll at the bottom
* when new data is received. See https://github.com/grafana/grafana/pull/37634
*/
if (app === CoreApp.Dashboard || app === CoreApp.PanelEditor) {
return sortOrder === LogsSortOrder.Ascending ? 'bottom' : 'top';
}
return 'top';
}, [app, sortOrder]);
const storageKey = useMemo(() => {
if (controlsStorageKey) {
return controlsStorageKey;
}
if (!data.request) {
return undefined;
}
return `${data.request?.dashboardUID}.${id}`;
}, [controlsStorageKey, data.request, id]);
if (!data || logRows.length === 0) {
return <PanelDataErrorView fieldConfig={fieldConfig} panelId={id} data={data} needsStringField />;
}
// Passing callbacks control the display of the filtering buttons. We want to pass it only if onAddAdHocFilter is defined.
const defaultOnClickFilterLabel = onAddAdHocFilter ? handleOnClickFilterLabel : undefined;
const defaultOnClickFilterOutLabel = onAddAdHocFilter ? handleOnClickFilterOutLabel : undefined;
const onClickShowField = isOnClickShowField(options.onClickShowField) ? options.onClickShowField : showField;
const onClickHideField = isOnClickHideField(options.onClickHideField) ? options.onClickHideField : hideField;
const setDisplayedFieldsFn = isSetDisplayedFields(options.setDisplayedFields)
? options.setDisplayedFields
: setDisplayedFields;
// In Dashboards, default to inline. Otherwise, let apps control or have automatic behavior.
const detailsMode = detailsModeProp ? detailsModeProp : app === CoreApp.Dashboard ? 'inline' : undefined;
return (
<>
{(!config.featureToggles.newLogsPanel || !config.featureToggles.newLogContext) && contextRow && (
<LogRowContextModal
open={contextRow !== null}
row={contextRow}
onClose={onCloseContext}
getRowContext={(row, options) => getLogRowContext(row, contextRow, options)}
logsSortOrder={sortOrder}
timeZone={timeZone}
getLogRowContextUi={getLogRowContextUi}
/>
)}
{config.featureToggles.newLogsPanel && config.featureToggles.newLogContext && getLogRowContext && contextRow && (
<LogLineContext
open={contextRow !== null}
log={contextRow}
onClose={onCloseContext}
getRowContext={(row, options) => getLogRowContext(row, contextRow, options)}
getLogRowContextUi={getLogRowContextUi}
logOptionsStorageKey={controlsStorageKey}
timeZone={timeZone}
displayedFields={displayedFields}
onClickShowField={showField}
onClickHideField={hideField}
/>
)}
{config.featureToggles.newLogsPanel && (
<div
onMouseLeave={onLogContainerMouseLeave}
className={style.logListContainer}
style={height ? { minHeight: height } : undefined}
ref={(element: HTMLDivElement) => setScrollElement(element)}
>
{deduplicatedRows.length > 0 && scrollElement && (
<LogList
app={isCoreApp(app) ? app : CoreApp.Dashboard}
containerElement={scrollElement}
dedupStrategy={dedupStrategy}
detailsMode={detailsMode}
displayedFields={displayedFields}
enableLogDetails={enableLogDetails}
fontSize={fontSize}
getFieldLinks={getFieldLinks}
isLabelFilterActive={isIsFilterLabelActive(isFilterLabelActive) ? isFilterLabelActive : undefined}
initialScrollPosition={initialScrollPosition}
loading={infiniteScrolling}
logLineMenuCustomItems={
isLogLineMenuCustomItems(logLineMenuCustomItems) ? logLineMenuCustomItems : undefined
}
logs={deduplicatedRows}
logSupportsContext={showContextToggle}
loadMore={enableInfiniteScrolling ? loadMoreLogs : undefined}
noInteractions={noInteractions}
onClickFilterLabel={
isOnClickFilterLabel(onClickFilterLabel) ? onClickFilterLabel : defaultOnClickFilterLabel
}
onClickFilterOutLabel={
isOnClickFilterOutLabel(onClickFilterOutLabel) ? onClickFilterOutLabel : defaultOnClickFilterOutLabel
}
onClickFilterString={isOnClickFilterString(onClickFilterString) ? onClickFilterString : undefined}
onClickFilterOutString={
isOnClickFilterOutString(onClickFilterOutString) ? onClickFilterOutString : undefined
}
onClickShowField={displayedFields !== undefined ? onClickShowField : undefined}
onClickHideField={displayedFields !== undefined ? onClickHideField : undefined}
onLogLineHover={onLogRowHover}
onLogOptionsChange={isOnLogOptionsChange(onLogOptionsChange) ? onLogOptionsChange : undefined}
onOpenContext={onOpenContext}
onPermalinkClick={showPermaLink() ? onPermalinkClick : undefined}
permalinkedLogId={getLogsPanelState()?.logs?.id ?? undefined}
prettifyJSON={prettifyLogMessage}
setDisplayedFields={setDisplayedFieldsFn}
showControls={Boolean(showControls)}
showTime={showTime}
showUniqueLabels={showLabels}
sortOrder={sortOrder}
logOptionsStorageKey={storageKey}
syntaxHighlighting={syntaxHighlighting}
timeRange={data.timeRange}
timestampResolution={timestampResolution}
timeZone={timeZone}
wrapLogMessage={wrapLogMessage}
/>
)}
</div>
)}
{!config.featureToggles.newLogsPanel && !showControls && (
<ScrollContainer ref={(scrollElement) => setScrollElement(scrollElement)}>
<div onMouseLeave={onLogContainerMouseLeave} className={style.container} ref={logsContainerRef}>
{showCommonLabels && !isAscending && renderCommonLabels()}
<InfiniteScroll
loading={infiniteScrolling}
loadMoreLogs={enableInfiniteScrolling ? loadMoreLogs : undefined}
range={data.timeRange}
timeZone={timeZone}
rows={logRows}
scrollElement={scrollElement}
sortOrder={sortOrder}
>
<LogRows
scrollElement={scrollElement}
scrollIntoView={scrollIntoView}
permalinkedRowId={getLogsPanelState()?.logs?.id ?? undefined}
onPermalinkClick={showPermaLink() ? onPermalinkClick : undefined}
logRows={logRows}
showContextToggle={showContextToggle}
deduplicatedRows={deduplicatedRows}
dedupStrategy={dedupStrategy}
showLabels={showLabels}
showTime={showTime}
wrapLogMessage={wrapLogMessage}
prettifyLogMessage={prettifyLogMessage}
timeZone={timeZone}
getFieldLinks={getFieldLinks}
logsSortOrder={sortOrder}
enableLogDetails={enableLogDetails}
previewLimit={isAscending ? logRows.length : undefined}
onLogRowHover={onLogRowHover}
app={isCoreApp(app) ? app : CoreApp.Dashboard}
onOpenContext={onOpenContext}
onClickFilterLabel={
isOnClickFilterLabel(onClickFilterLabel) ? onClickFilterLabel : defaultOnClickFilterLabel
}
onClickFilterOutLabel={
isOnClickFilterOutLabel(onClickFilterOutLabel) ? onClickFilterOutLabel : defaultOnClickFilterOutLabel
}
onClickFilterString={isOnClickFilterString(onClickFilterString) ? onClickFilterString : undefined}
onClickFilterOutString={
isOnClickFilterOutString(onClickFilterOutString) ? onClickFilterOutString : undefined
}
isFilterLabelActive={isIsFilterLabelActive(isFilterLabelActive) ? isFilterLabelActive : undefined}
displayedFields={displayedFields}
onClickShowField={displayedFields !== undefined ? onClickShowField : undefined}
onClickHideField={displayedFields !== undefined ? onClickHideField : undefined}
logRowMenuIconsBefore={isReactNodeArray(logRowMenuIconsBefore) ? logRowMenuIconsBefore : undefined}
logRowMenuIconsAfter={isReactNodeArray(logRowMenuIconsAfter) ? logRowMenuIconsAfter : undefined}
// Ascending order causes scroll to stick to the bottom, so previewing is futile
renderPreview={isAscending ? false : true}
timeRange={data.timeRange}
/>
</InfiniteScroll>
{showCommonLabels && isAscending && renderCommonLabels()}
</div>
</ScrollContainer>
)}
{!config.featureToggles.newLogsPanel && showControls && (
<div onMouseLeave={onLogContainerMouseLeave} className={style.controlledLogsContainer}>
{showCommonLabels && !isAscending && renderCommonLabels()}
<ControlledLogRows
ref={(scrollElement: HTMLDivElement | null) => setScrollElement(scrollElement)}
visualisationType="logs"
loading={infiniteScrolling}
loadMoreLogs={enableInfiniteScrolling ? loadMoreLogs : undefined}
range={data.timeRange}
logRows={logRows}
deduplicatedRows={deduplicatedRows}
dedupStrategy={dedupStrategy}
onClickFilterLabel={
isOnClickFilterLabel(onClickFilterLabel) ? onClickFilterLabel : defaultOnClickFilterLabel
}
onClickFilterOutLabel={
isOnClickFilterOutLabel(onClickFilterOutLabel) ? onClickFilterOutLabel : defaultOnClickFilterOutLabel
}
showContextToggle={showContextToggle}
showLabels={showLabels}
showTime={showTime}
enableLogDetails={enableLogDetails}
wrapLogMessage={wrapLogMessage}
prettifyLogMessage={prettifyLogMessage}
timeZone={timeZone}
getFieldLinks={getFieldLinks}
logsSortOrder={sortOrder}
displayedFields={displayedFields}
onClickShowField={displayedFields !== undefined ? onClickShowField : undefined}
onClickHideField={displayedFields !== undefined ? onClickHideField : undefined}
app={isCoreApp(app) ? app : CoreApp.Dashboard}
onLogRowHover={onLogRowHover}
onOpenContext={onOpenContext}
onPermalinkClick={onPermalinkClick}
permalinkedRowId={getLogsPanelState()?.logs?.id ?? undefined}
scrollIntoView={scrollIntoView}
isFilterLabelActive={isIsFilterLabelActive(isFilterLabelActive) ? isFilterLabelActive : undefined}
onClickFilterString={isOnClickFilterString(onClickFilterString) ? onClickFilterString : undefined}
onClickFilterOutString={
isOnClickFilterOutString(onClickFilterOutString) ? onClickFilterOutString : undefined
}
logRowMenuIconsBefore={isReactNodeArray(logRowMenuIconsBefore) ? logRowMenuIconsBefore : undefined}
logRowMenuIconsAfter={isReactNodeArray(logRowMenuIconsAfter) ? logRowMenuIconsAfter : undefined}
onLogOptionsChange={isOnLogOptionsChange(onLogOptionsChange) ? onLogOptionsChange : undefined}
logOptionsStorageKey={controlsStorageKey}
// Ascending order causes scroll to stick to the bottom, so previewing is futile
renderPreview={isAscending ? false : true}
timeRange={data.timeRange}
/>
{showCommonLabels && isAscending && renderCommonLabels()}
</div>
)}
</>
);
};
const getStyles = (theme: GrafanaTheme2) => ({
container: css({
marginBottom: theme.spacing(1.5),
}),
logListContainer: css({
minHeight: '100%',
maxHeight: '100%',
display: 'flex',
flex: 1,
flexDirection: 'column',
overflow: 'hidden',
}),
controlledLogsContainer: css({
height: '100%',
}),
labelContainer: css({
margin: theme.spacing(0, 0, 0.5, 0.5),
display: 'flex',
alignItems: 'center',
}),
labelContainerAscending: css({
margin: theme.spacing(0.5, 0, 0.5, 0),
}),
label: css({
marginRight: theme.spacing(0.5),
fontSize: theme.typography.bodySmall.fontSize,
fontWeight: theme.typography.fontWeightMedium,
}),
});
function getLogsPanelState(): LogsPermalinkUrlState | undefined {
const urlParams = urlUtil.getUrlSearchParams();
const panelStateEncoded = urlParams?.panelState;
if (
panelStateEncoded &&
Array.isArray(panelStateEncoded) &&
panelStateEncoded?.length > 0 &&
typeof panelStateEncoded[0] === 'string'
) {
try {
return JSON.parse(panelStateEncoded[0]);
} catch (e) {
console.error('error parsing logsPanelState', e);
}
}
return undefined;
}
async function copyDashboardUrl(row: LogRowModel, rows: LogRowModel[], timeRange: TimeRange) {
// this is an extra check, to be sure that we are not
// creating permalinks for logs without an id-field.
// normally it should never happen, because we do not
// display the permalink button in such cases.
if (row.rowId === undefined || !row.dataFrame.refId) {
return;
}
// get panel state, add log-row-id
const panelState = {
logs: { id: row.uid },
};
// Grab the current dashboard URL
const currentURL = new URL(window.location.href);
// Add panel state containing the rowId, and absolute time range from the current query, but leave everything else the same, if the user is in edit mode when grabbing the link, that's what will be linked to, etc.
currentURL.searchParams.set('panelState', JSON.stringify(panelState));
const range = getLogsPermalinkRange(row, rows, {
from: toUtc(timeRange.from).valueOf(),
to: toUtc(timeRange.to).valueOf(),
});
currentURL.searchParams.set('from', range.from.toString());
currentURL.searchParams.set('to', range.to.toString());
await createAndCopyShortLink(currentURL.toString());
return Promise.resolve();
}
export async function requestMoreLogs(
dataSourcesMap: Map<string, DataSourceApi>,
panelData: PanelData,
timeRange: AbsoluteTimeRange,
timeZone: TimeZone,
onNewLogsReceived?: onNewLogsReceivedType
) {
if (!panelData.request) {
return [];
}
const range: TimeRange = rangeUtil.convertRawToRange({
from: dateTimeForTimeZone(timeZone, timeRange.from),
to: dateTimeForTimeZone(timeZone, timeRange.to),
});
const targetGroups = groupBy(panelData.request.targets, 'datasource.uid');
const dataRequests = [];
for (const uid in targetGroups) {
const dataSource = dataSourcesMap.get(panelData.request.targets[0].refId);
if (!dataSource) {
console.warn(`Could not resolve data source for target ${panelData.request.targets[0].refId}`);
continue;
}
dataRequests.push(
dataSource.query({
...panelData.request,
range,
targets: targetGroups[uid],
})
);
}
const responses = await Promise.all(dataRequests);
let updatedSeries = panelData.series;
for (const response of responses) {
const newData = isObservable(response) ? await lastValueFrom(response) : response;
updatedSeries = combineResponses(
{
data: updatedSeries,
},
{ data: newData.data }
).data;
if (onNewLogsReceived) {
onNewLogsReceived(updatedSeries, newData.data);
}
}
return updatedSeries;
}