mirror of
https://github.com/grafana/grafana.git
synced 2025-02-13 00:55:47 -06:00
* Refactor: Uses nanosecond string to retreive LogRow contexts * Reafactor: Small changes to comments after PR comments
147 lines
4.9 KiB
TypeScript
147 lines
4.9 KiB
TypeScript
import { DataQueryResponse, DataQueryError, LogRowModel } from '@grafana/ui';
|
|
import { useState, useEffect } from 'react';
|
|
import flatten from 'lodash/flatten';
|
|
import useAsync from 'react-use/lib/useAsync';
|
|
|
|
export interface LogRowContextRows {
|
|
before?: string[];
|
|
after?: string[];
|
|
}
|
|
export interface LogRowContextQueryErrors {
|
|
before?: string;
|
|
after?: string;
|
|
}
|
|
|
|
export interface HasMoreContextRows {
|
|
before: boolean;
|
|
after: boolean;
|
|
}
|
|
|
|
interface LogRowContextProviderProps {
|
|
row: LogRowModel;
|
|
getRowContext: (row: LogRowModel, options?: any) => Promise<DataQueryResponse>;
|
|
children: (props: {
|
|
result: LogRowContextRows;
|
|
errors: LogRowContextQueryErrors;
|
|
hasMoreContextRows: HasMoreContextRows;
|
|
updateLimit: () => void;
|
|
}) => JSX.Element;
|
|
}
|
|
|
|
export const getRowContexts = async (
|
|
getRowContext: (row: LogRowModel, options?: any) => Promise<DataQueryResponse>,
|
|
row: LogRowModel,
|
|
limit: number
|
|
) => {
|
|
const promises = [
|
|
getRowContext(row, {
|
|
limit,
|
|
}),
|
|
getRowContext(row, {
|
|
limit: limit + 1, // Lets add one more to the limit as we're filtering out one row see comment below
|
|
direction: 'FORWARD',
|
|
}),
|
|
];
|
|
|
|
const results: Array<DataQueryResponse | DataQueryError> = await Promise.all(promises.map(p => p.catch(e => e)));
|
|
|
|
return {
|
|
data: results.map((result, index) => {
|
|
const dataResult: DataQueryResponse = result as DataQueryResponse;
|
|
if (!dataResult.data) {
|
|
return [];
|
|
}
|
|
|
|
// We need to filter out the row we're basing our search from because of how start/end params work in Loki API
|
|
// see https://github.com/grafana/loki/issues/597#issuecomment-506408980
|
|
// the alternative to create our own add 1 nanosecond method to the a timestamp string would be quite complex
|
|
return dataResult.data.map(series => {
|
|
const filteredRows = series.rows.filter((r: any) => r[0] !== row.timestamp);
|
|
return filteredRows.map((row: any) => row[1]);
|
|
});
|
|
}),
|
|
errors: results.map(result => {
|
|
const errorResult: DataQueryError = result as DataQueryError;
|
|
if (!errorResult.message) {
|
|
return null;
|
|
}
|
|
|
|
return errorResult.message;
|
|
}),
|
|
};
|
|
};
|
|
|
|
export const LogRowContextProvider: React.FunctionComponent<LogRowContextProviderProps> = ({
|
|
getRowContext,
|
|
row,
|
|
children,
|
|
}) => {
|
|
// React Hook that creates a number state value called limit to component state and a setter function called setLimit
|
|
// The intial value for limit is 10
|
|
// Used for the number of rows to retrieve from backend from a specific point in time
|
|
const [limit, setLimit] = useState(10);
|
|
|
|
// React Hook that creates an object state value called result to component state and a setter function called setResult
|
|
// The intial value for result is null
|
|
// Used for sorting the response from backend
|
|
const [result, setResult] = useState<{
|
|
data: string[][];
|
|
errors: string[];
|
|
}>(null);
|
|
|
|
// React Hook that creates an object state value called hasMoreContextRows to component state and a setter function called setHasMoreContextRows
|
|
// The intial value for hasMoreContextRows is {before: true, after: true}
|
|
// Used for indicating in UI if there are more rows to load in a given direction
|
|
const [hasMoreContextRows, setHasMoreContextRows] = useState({
|
|
before: true,
|
|
after: true,
|
|
});
|
|
|
|
// React Hook that resolves two promises every time the limit prop changes
|
|
// First promise fetches limit number of rows backwards in time from a specific point in time
|
|
// Second promise fetches limit number of rows forwards in time from a specific point in time
|
|
const { value } = useAsync(async () => {
|
|
return await getRowContexts(getRowContext, row, limit); // Moved it to a separate function for debugging purposes
|
|
}, [limit]);
|
|
|
|
// React Hook that performs a side effect every time the value (from useAsync hook) prop changes
|
|
// The side effect changes the result state with the response from the useAsync hook
|
|
// The side effect changes the hasMoreContextRows state if there are more context rows before or after the current result
|
|
useEffect(() => {
|
|
if (value) {
|
|
setResult(currentResult => {
|
|
let hasMoreLogsBefore = true,
|
|
hasMoreLogsAfter = true;
|
|
|
|
if (currentResult && currentResult.data[0].length === value.data[0].length) {
|
|
hasMoreLogsBefore = false;
|
|
}
|
|
|
|
if (currentResult && currentResult.data[1].length === value.data[1].length) {
|
|
hasMoreLogsAfter = false;
|
|
}
|
|
|
|
setHasMoreContextRows({
|
|
before: hasMoreLogsBefore,
|
|
after: hasMoreLogsAfter,
|
|
});
|
|
|
|
return value;
|
|
});
|
|
}
|
|
}, [value]);
|
|
|
|
return children({
|
|
result: {
|
|
before: result ? flatten(result.data[0]) : [],
|
|
after: result ? flatten(result.data[1]) : [],
|
|
},
|
|
errors: {
|
|
before: result ? result.errors[0] : null,
|
|
after: result ? result.errors[1] : null,
|
|
},
|
|
hasMoreContextRows,
|
|
updateLimit: () => setLimit(limit + 10),
|
|
});
|
|
};
|