grafana/public/app/features/explore/LiveLogs.tsx
Dominik Prokop a9c94ec93b
Explore: Update the way Loki retrieve log context (#17204)
* Move log's typings into grafana/ui
* Update the way context is retrieved for Loki

Major changes:

1. getLogRowContext expects row to be of LogRowModel type
2. getLogRowContext accepts generic options object, specific to a datasource of interest. limit option has been removed, and now it's a part of Loki's context query options (see below)
3. LogRowContextProvider performs two queries now. Before, it was Loki ds that performed queries in both directions when getLogRowContext.
4. Loki's getLogRowContext accepts options object of a type:

interface LokiContextQueryOptions {
    direction?: 'BACKWARD' | 'FORWARD';
    limit?: number;
}

This will enable querying in either direction independently and also slightly simplifies the way query errors are handled.

LogRowContextProvider maps the results to a Loki specific context types, basically string[][], as raw log lines are displayed in first version.
2019-05-22 23:10:05 +02:00

126 lines
3.3 KiB
TypeScript

import React, { PureComponent } from 'react';
import { css, cx } from 'emotion';
import {
Themeable,
withTheme,
GrafanaTheme,
selectThemeVariant,
LinkButton,
LogsModel,
LogRowModel,
} from '@grafana/ui';
import ElapsedTime from './ElapsedTime';
import { ButtonSize, ButtonVariant } from '@grafana/ui/src/components/Button/AbstractButton';
const getStyles = (theme: GrafanaTheme) => ({
logsRowsLive: css`
label: logs-rows-live;
display: flex;
flex-flow: column nowrap;
height: 65vh;
overflow-y: auto;
:first-child {
margin-top: auto !important;
}
`,
logsRowFresh: css`
label: logs-row-fresh;
color: ${theme.colors.text};
background-color: ${selectThemeVariant({ light: theme.colors.gray6, dark: theme.colors.gray1 }, theme.type)};
`,
logsRowOld: css`
label: logs-row-old;
opacity: 0.8;
`,
logsRowsIndicator: css`
font-size: ${theme.typography.size.md};
padding: ${theme.spacing.sm} 0;
display: flex;
align-items: center;
`,
});
export interface Props extends Themeable {
logsResult?: LogsModel;
stopLive: () => void;
}
export interface State {
renderCount: number;
}
class LiveLogs extends PureComponent<Props, State> {
private liveEndDiv: HTMLDivElement = null;
constructor(props: Props) {
super(props);
this.state = { renderCount: 0 };
}
componentDidUpdate(prevProps: Props) {
const prevRows: LogRowModel[] = prevProps.logsResult ? prevProps.logsResult.rows : [];
const rows: LogRowModel[] = this.props.logsResult ? this.props.logsResult.rows : [];
if (prevRows !== rows) {
this.setState({
renderCount: this.state.renderCount + 1,
});
}
if (this.liveEndDiv) {
this.liveEndDiv.scrollIntoView(false);
}
}
render() {
const { theme } = this.props;
const { renderCount } = this.state;
const styles = getStyles(theme);
const rowsToRender: LogRowModel[] = this.props.logsResult ? this.props.logsResult.rows : [];
return (
<>
<div className={cx(['logs-rows', styles.logsRowsLive])}>
{rowsToRender.map((row: any, index) => {
return (
<div
className={row.fresh ? cx(['logs-row', styles.logsRowFresh]) : cx(['logs-row', styles.logsRowOld])}
key={`${row.timeEpochMs}-${index}`}
>
<div className="logs-row__localtime" title={`${row.timestamp} (${row.timeFromNow})`}>
{row.timeLocal}
</div>
<div className="logs-row__message">{row.entry}</div>
</div>
);
})}
<div
ref={element => {
this.liveEndDiv = element;
if (this.liveEndDiv) {
this.liveEndDiv.scrollIntoView(false);
}
}}
/>
</div>
<div className={cx([styles.logsRowsIndicator])}>
<span>
Last line received: <ElapsedTime renderCount={renderCount} humanize={true} /> ago
</span>
<LinkButton
onClick={this.props.stopLive}
size={ButtonSize.Medium}
variant={ButtonVariant.Transparent}
style={{ color: theme.colors.orange }}
>
Stop Live
</LinkButton>
</div>
</>
);
}
}
export const LiveLogsWithTheme = withTheme(LiveLogs);