2019-05-22 16:10:05 -05:00
|
|
|
import { DataQueryResponse, DataQueryError, LogRowModel } from '@grafana/ui';
|
2019-05-20 01:44:37 -05:00
|
|
|
import { useState, useEffect } from 'react';
|
2019-05-22 16:10:05 -05:00
|
|
|
import flatten from 'lodash/flatten';
|
2019-05-20 01:44:37 -05:00
|
|
|
import useAsync from 'react-use/lib/useAsync';
|
|
|
|
|
|
|
|
export interface LogRowContextRows {
|
2019-05-22 16:10:05 -05:00
|
|
|
before?: string[];
|
|
|
|
after?: string[];
|
2019-05-20 01:44:37 -05:00
|
|
|
}
|
|
|
|
export interface LogRowContextQueryErrors {
|
|
|
|
before?: string;
|
|
|
|
after?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface HasMoreContextRows {
|
|
|
|
before: boolean;
|
|
|
|
after: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface LogRowContextProviderProps {
|
|
|
|
row: LogRowModel;
|
2019-05-22 16:10:05 -05:00
|
|
|
getRowContext: (row: LogRowModel, options?: any) => Promise<DataQueryResponse>;
|
2019-05-20 01:44:37 -05:00
|
|
|
children: (props: {
|
|
|
|
result: LogRowContextRows;
|
|
|
|
errors: LogRowContextQueryErrors;
|
|
|
|
hasMoreContextRows: HasMoreContextRows;
|
|
|
|
updateLimit: () => void;
|
|
|
|
}) => JSX.Element;
|
|
|
|
}
|
|
|
|
|
|
|
|
export const LogRowContextProvider: React.FunctionComponent<LogRowContextProviderProps> = ({
|
|
|
|
getRowContext,
|
|
|
|
row,
|
|
|
|
children,
|
|
|
|
}) => {
|
|
|
|
const [limit, setLimit] = useState(10);
|
2019-05-22 16:10:05 -05:00
|
|
|
const [result, setResult] = useState<{
|
|
|
|
data: string[][];
|
|
|
|
errors: string[];
|
|
|
|
}>(null);
|
2019-05-20 01:44:37 -05:00
|
|
|
const [hasMoreContextRows, setHasMoreContextRows] = useState({
|
|
|
|
before: true,
|
|
|
|
after: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
const { value } = useAsync(async () => {
|
2019-05-22 16:10:05 -05:00
|
|
|
const promises = [
|
|
|
|
getRowContext(row, {
|
|
|
|
limit,
|
|
|
|
}),
|
|
|
|
getRowContext(row, {
|
|
|
|
limit,
|
|
|
|
direction: 'FORWARD',
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
|
|
|
|
const results: Array<DataQueryResponse | DataQueryError> = await Promise.all(promises.map(p => p.catch(e => e)));
|
|
|
|
|
2019-05-20 01:44:37 -05:00
|
|
|
return {
|
2019-05-22 16:10:05 -05:00
|
|
|
data: results.map(result => {
|
|
|
|
if ((result as DataQueryResponse).data) {
|
|
|
|
return (result as DataQueryResponse).data.map(series => {
|
|
|
|
return series.rows.map(row => row[1]);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
errors: results.map(result => {
|
|
|
|
if ((result as DataQueryError).message) {
|
|
|
|
return (result as DataQueryError).message;
|
2019-05-20 01:44:37 -05:00
|
|
|
} else {
|
2019-05-22 16:10:05 -05:00
|
|
|
return null;
|
2019-05-20 01:44:37 -05:00
|
|
|
}
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
}, [limit]);
|
|
|
|
|
|
|
|
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: {
|
2019-05-22 16:10:05 -05:00
|
|
|
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,
|
2019-05-20 01:44:37 -05:00
|
|
|
},
|
|
|
|
hasMoreContextRows,
|
|
|
|
updateLimit: () => setLimit(limit + 10),
|
|
|
|
});
|
|
|
|
};
|