2024-03-12 12:20:45 +05:30
|
|
|
import './LogsExplorer.styles.scss';
|
|
|
|
|
|
2024-06-18 19:04:06 +05:30
|
|
|
import * as Sentry from '@sentry/react';
|
2024-09-06 10:24:47 +05:30
|
|
|
import getLocalStorageKey from 'api/browser/localstorage/get';
|
|
|
|
|
import setLocalStorageApi from 'api/browser/localstorage/set';
|
|
|
|
|
import cx from 'classnames';
|
2023-08-30 20:24:16 +05:30
|
|
|
import ExplorerCard from 'components/ExplorerCard/ExplorerCard';
|
2024-09-06 10:24:47 +05:30
|
|
|
import QuickFilters from 'components/QuickFilters/QuickFilters';
|
2025-01-29 10:39:42 +04:30
|
|
|
import { QuickFiltersSource } from 'components/QuickFilters/types';
|
2024-09-06 10:24:47 +05:30
|
|
|
import { LOCALSTORAGE } from 'constants/localStorage';
|
2023-07-07 15:49:35 +03:00
|
|
|
import LogExplorerQuerySection from 'container/LogExplorerQuerySection';
|
2023-06-23 11:19:53 +03:00
|
|
|
import LogsExplorerViews from 'container/LogsExplorerViews';
|
2025-01-27 20:54:59 +04:30
|
|
|
import {
|
|
|
|
|
defaultLogsSelectedColumns,
|
|
|
|
|
defaultOptionsQuery,
|
|
|
|
|
URL_OPTIONS,
|
|
|
|
|
} from 'container/OptionsMenu/constants';
|
|
|
|
|
import { OptionsQuery } from 'container/OptionsMenu/types';
|
2024-02-12 00:23:19 +05:30
|
|
|
import LeftToolbarActions from 'container/QueryBuilder/components/ToolbarActions/LeftToolbarActions';
|
|
|
|
|
import RightToolbarActions from 'container/QueryBuilder/components/ToolbarActions/RightToolbarActions';
|
|
|
|
|
import Toolbar from 'container/Toolbar/Toolbar';
|
|
|
|
|
import { useQueryBuilder } from 'hooks/queryBuilder/useQueryBuilder';
|
2025-01-27 20:54:59 +04:30
|
|
|
import useUrlQueryData from 'hooks/useUrlQueryData';
|
|
|
|
|
import { isEqual, isNull } from 'lodash-es';
|
2023-11-15 16:46:20 +05:30
|
|
|
import ErrorBoundaryFallback from 'pages/ErrorBoundaryFallback/ErrorBoundaryFallback';
|
2025-01-31 10:25:47 +04:30
|
|
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
|
|
|
|
import { BaseAutocompleteData } from 'types/api/queryBuilder/queryAutocompleteResponse';
|
2023-08-30 20:24:16 +05:30
|
|
|
import { DataSource } from 'types/common/queryBuilder';
|
2023-06-16 13:38:39 +03:00
|
|
|
|
2023-07-07 15:49:35 +03:00
|
|
|
import { WrapperStyled } from './styles';
|
2024-09-06 10:24:47 +05:30
|
|
|
import { LogsQuickFiltersConfig, SELECTED_VIEWS } from './utils';
|
2023-06-16 13:38:39 +03:00
|
|
|
|
2023-06-23 21:39:59 +03:00
|
|
|
function LogsExplorer(): JSX.Element {
|
2024-07-08 20:02:10 +05:30
|
|
|
const [showFrequencyChart, setShowFrequencyChart] = useState(true);
|
2024-02-12 00:23:19 +05:30
|
|
|
const [selectedView, setSelectedView] = useState<SELECTED_VIEWS>(
|
|
|
|
|
SELECTED_VIEWS.SEARCH,
|
|
|
|
|
);
|
2024-09-06 10:24:47 +05:30
|
|
|
const [showFilters, setShowFilters] = useState<boolean>(() => {
|
|
|
|
|
const localStorageValue = getLocalStorageKey(
|
|
|
|
|
LOCALSTORAGE.SHOW_LOGS_QUICK_FILTERS,
|
|
|
|
|
);
|
|
|
|
|
if (!isNull(localStorageValue)) {
|
|
|
|
|
return localStorageValue === 'true';
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
});
|
2024-02-12 00:23:19 +05:30
|
|
|
|
|
|
|
|
const { handleRunQuery, currentQuery } = useQueryBuilder();
|
|
|
|
|
|
2024-08-16 13:11:39 +05:30
|
|
|
const listQueryKeyRef = useRef<any>();
|
|
|
|
|
|
|
|
|
|
const chartQueryKeyRef = useRef<any>();
|
|
|
|
|
|
|
|
|
|
const [isLoadingQueries, setIsLoadingQueries] = useState<boolean>(false);
|
|
|
|
|
|
2024-07-08 20:02:10 +05:30
|
|
|
const handleToggleShowFrequencyChart = (): void => {
|
|
|
|
|
setShowFrequencyChart(!showFrequencyChart);
|
2024-02-12 00:23:19 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const handleChangeSelectedView = (view: SELECTED_VIEWS): void => {
|
|
|
|
|
setSelectedView(view);
|
|
|
|
|
};
|
|
|
|
|
|
2024-09-06 10:24:47 +05:30
|
|
|
const handleFilterVisibilityChange = (): void => {
|
|
|
|
|
setLocalStorageApi(
|
|
|
|
|
LOCALSTORAGE.SHOW_LOGS_QUICK_FILTERS,
|
|
|
|
|
String(!showFilters),
|
|
|
|
|
);
|
|
|
|
|
setShowFilters((prev) => !prev);
|
|
|
|
|
};
|
|
|
|
|
|
2024-02-12 00:23:19 +05:30
|
|
|
// Switch to query builder view if there are more than 1 queries
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
if (currentQuery.builder.queryData.length > 1) {
|
|
|
|
|
handleChangeSelectedView(SELECTED_VIEWS.QUERY_BUILDER);
|
|
|
|
|
}
|
2024-08-22 23:59:22 +05:30
|
|
|
if (
|
|
|
|
|
currentQuery.builder.queryData.length === 1 &&
|
2024-09-18 18:02:17 +05:30
|
|
|
currentQuery.builder.queryData?.[0]?.groupBy?.length > 0
|
2024-08-22 23:59:22 +05:30
|
|
|
) {
|
|
|
|
|
handleChangeSelectedView(SELECTED_VIEWS.QUERY_BUILDER);
|
|
|
|
|
}
|
|
|
|
|
}, [currentQuery.builder.queryData, currentQuery.builder.queryData.length]);
|
2024-02-12 00:23:19 +05:30
|
|
|
|
2025-01-27 20:54:59 +04:30
|
|
|
const {
|
|
|
|
|
queryData: optionsQueryData,
|
|
|
|
|
redirectWithQuery: redirectWithOptionsData,
|
|
|
|
|
} = useUrlQueryData<OptionsQuery>(URL_OPTIONS, defaultOptionsQuery);
|
|
|
|
|
|
2025-01-31 10:25:47 +04:30
|
|
|
// Get and parse stored columns from localStorage
|
|
|
|
|
const logListOptionsFromLocalStorage = useMemo(() => {
|
|
|
|
|
const data = getLocalStorageKey(LOCALSTORAGE.LOGS_LIST_OPTIONS);
|
|
|
|
|
|
|
|
|
|
if (!data) return null;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const parsed = JSON.parse(data);
|
|
|
|
|
const hasValidColumns = Array.isArray(parsed?.selectColumns);
|
|
|
|
|
|
|
|
|
|
return hasValidColumns ? parsed.selectColumns : null;
|
|
|
|
|
} catch {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
|
|
// Check if the columns have the required columns (timestamp, body)
|
|
|
|
|
const hasRequiredColumns = useCallback(
|
|
|
|
|
(columns?: Array<{ key: string }> | null): boolean => {
|
|
|
|
|
if (!columns?.length) return false;
|
|
|
|
|
|
|
|
|
|
const hasTimestamp = columns.some((col) => col.key === 'timestamp');
|
|
|
|
|
const hasBody = columns.some((col) => col.key === 'body');
|
|
|
|
|
|
|
|
|
|
return hasTimestamp && hasBody;
|
|
|
|
|
},
|
|
|
|
|
[],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Merge the columns with the required columns (timestamp, body) if missing
|
|
|
|
|
const mergeWithRequiredColumns = useCallback(
|
|
|
|
|
(columns: BaseAutocompleteData[]): BaseAutocompleteData[] => [
|
|
|
|
|
// Add required columns (timestamp, body) if missing
|
|
|
|
|
...(!hasRequiredColumns(columns) ? defaultLogsSelectedColumns : []),
|
|
|
|
|
...columns,
|
|
|
|
|
],
|
|
|
|
|
[hasRequiredColumns],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Migrate the options query to the new format
|
|
|
|
|
const migrateOptionsQuery = useCallback(
|
|
|
|
|
(query: OptionsQuery): OptionsQuery => {
|
|
|
|
|
// Skip if already migrated
|
|
|
|
|
if (query.version) return query;
|
|
|
|
|
|
|
|
|
|
// Case 1: query has columns
|
|
|
|
|
if (query.selectColumns.length > 0) {
|
|
|
|
|
return {
|
|
|
|
|
...query,
|
|
|
|
|
version: 1,
|
|
|
|
|
selectColumns: mergeWithRequiredColumns(query.selectColumns),
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Case 2: No query columns in but we have localStorage columns
|
|
|
|
|
if (logListOptionsFromLocalStorage?.selectColumns?.length > 0) {
|
|
|
|
|
return {
|
|
|
|
|
...query,
|
|
|
|
|
version: 1,
|
|
|
|
|
selectColumns: mergeWithRequiredColumns(
|
|
|
|
|
logListOptionsFromLocalStorage.selectColumns,
|
|
|
|
|
),
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Case 3: No columns anywhere, use defaults
|
2025-01-27 20:54:59 +04:30
|
|
|
return {
|
|
|
|
|
...query,
|
|
|
|
|
version: 1,
|
2025-01-31 10:25:47 +04:30
|
|
|
selectColumns: defaultLogsSelectedColumns,
|
2025-01-27 20:54:59 +04:30
|
|
|
};
|
2025-01-31 10:25:47 +04:30
|
|
|
},
|
|
|
|
|
[mergeWithRequiredColumns, logListOptionsFromLocalStorage?.selectColumns],
|
|
|
|
|
);
|
2025-01-27 20:54:59 +04:30
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
const migratedQuery = migrateOptionsQuery(optionsQueryData);
|
|
|
|
|
// Only redirect if the query was actually modified
|
|
|
|
|
if (!isEqual(migratedQuery, optionsQueryData)) {
|
|
|
|
|
redirectWithOptionsData(migratedQuery);
|
|
|
|
|
}
|
2025-01-31 10:25:47 +04:30
|
|
|
}, [migrateOptionsQuery, optionsQueryData, redirectWithOptionsData]);
|
2025-01-27 20:54:59 +04:30
|
|
|
|
2024-02-12 00:23:19 +05:30
|
|
|
const isMultipleQueries = useMemo(
|
|
|
|
|
() =>
|
2024-08-23 22:29:07 +05:30
|
|
|
currentQuery.builder.queryData?.length > 1 ||
|
|
|
|
|
currentQuery.builder.queryFormulas?.length > 0,
|
2024-02-12 00:23:19 +05:30
|
|
|
[currentQuery],
|
|
|
|
|
);
|
|
|
|
|
|
2024-08-22 23:59:22 +05:30
|
|
|
const isGroupByPresent = useMemo(
|
|
|
|
|
() =>
|
2024-08-23 22:29:07 +05:30
|
|
|
currentQuery.builder.queryData?.length === 1 &&
|
|
|
|
|
currentQuery.builder.queryData?.[0]?.groupBy?.length > 0,
|
2024-08-22 23:59:22 +05:30
|
|
|
[currentQuery.builder.queryData],
|
|
|
|
|
);
|
|
|
|
|
|
2024-02-12 00:23:19 +05:30
|
|
|
const toolbarViews = useMemo(
|
|
|
|
|
() => ({
|
|
|
|
|
search: {
|
|
|
|
|
name: 'search',
|
|
|
|
|
label: 'Search',
|
2024-08-22 23:59:22 +05:30
|
|
|
disabled: isMultipleQueries || isGroupByPresent,
|
2024-02-12 00:23:19 +05:30
|
|
|
show: true,
|
|
|
|
|
},
|
|
|
|
|
queryBuilder: {
|
|
|
|
|
name: 'query-builder',
|
|
|
|
|
label: 'Query Builder',
|
|
|
|
|
disabled: false,
|
|
|
|
|
show: true,
|
|
|
|
|
},
|
|
|
|
|
clickhouse: {
|
|
|
|
|
name: 'clickhouse',
|
|
|
|
|
label: 'Clickhouse',
|
|
|
|
|
disabled: false,
|
|
|
|
|
show: false,
|
|
|
|
|
},
|
|
|
|
|
}),
|
2024-08-22 23:59:22 +05:30
|
|
|
[isGroupByPresent, isMultipleQueries],
|
2024-02-12 00:23:19 +05:30
|
|
|
);
|
|
|
|
|
|
2023-06-16 13:38:39 +03:00
|
|
|
return (
|
2024-06-18 19:04:06 +05:30
|
|
|
<Sentry.ErrorBoundary fallback={<ErrorBoundaryFallback />}>
|
2024-09-06 10:24:47 +05:30
|
|
|
<div className={cx('logs-module-page', showFilters ? 'filter-visible' : '')}>
|
|
|
|
|
{showFilters && (
|
|
|
|
|
<section className={cx('log-quick-filter-left-section')}>
|
|
|
|
|
<QuickFilters
|
2025-01-29 10:39:42 +04:30
|
|
|
source={QuickFiltersSource.LOGS_EXPLORER}
|
2024-09-06 10:24:47 +05:30
|
|
|
config={LogsQuickFiltersConfig}
|
|
|
|
|
handleFilterVisibilityChange={handleFilterVisibilityChange}
|
2024-02-12 00:23:19 +05:30
|
|
|
/>
|
2024-09-06 10:24:47 +05:30
|
|
|
</section>
|
|
|
|
|
)}
|
|
|
|
|
<section className={cx('log-module-right-section')}>
|
|
|
|
|
<Toolbar
|
|
|
|
|
showAutoRefresh={false}
|
|
|
|
|
leftActions={
|
|
|
|
|
<LeftToolbarActions
|
|
|
|
|
showFilter={showFilters}
|
|
|
|
|
handleFilterVisibilityChange={handleFilterVisibilityChange}
|
|
|
|
|
items={toolbarViews}
|
|
|
|
|
selectedView={selectedView}
|
|
|
|
|
onChangeSelectedView={handleChangeSelectedView}
|
|
|
|
|
onToggleHistrogramVisibility={handleToggleShowFrequencyChart}
|
|
|
|
|
showFrequencyChart={showFrequencyChart}
|
|
|
|
|
/>
|
|
|
|
|
}
|
|
|
|
|
rightActions={
|
|
|
|
|
<RightToolbarActions
|
|
|
|
|
onStageRunQuery={handleRunQuery}
|
|
|
|
|
listQueryKeyRef={listQueryKeyRef}
|
|
|
|
|
chartQueryKeyRef={chartQueryKeyRef}
|
|
|
|
|
isLoadingQueries={isLoadingQueries}
|
|
|
|
|
/>
|
|
|
|
|
}
|
|
|
|
|
showOldCTA
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
<WrapperStyled>
|
|
|
|
|
<div className="log-explorer-query-container">
|
|
|
|
|
<div>
|
|
|
|
|
<ExplorerCard sourcepage={DataSource.LOGS}>
|
|
|
|
|
<LogExplorerQuerySection selectedView={selectedView} />
|
|
|
|
|
</ExplorerCard>
|
|
|
|
|
</div>
|
|
|
|
|
<div className="logs-explorer-views">
|
|
|
|
|
<LogsExplorerViews
|
|
|
|
|
selectedView={selectedView}
|
|
|
|
|
showFrequencyChart={showFrequencyChart}
|
|
|
|
|
listQueryKeyRef={listQueryKeyRef}
|
|
|
|
|
chartQueryKeyRef={chartQueryKeyRef}
|
|
|
|
|
setIsLoadingQueries={setIsLoadingQueries}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</WrapperStyled>
|
|
|
|
|
</section>
|
|
|
|
|
</div>
|
2024-06-18 19:04:06 +05:30
|
|
|
</Sentry.ErrorBoundary>
|
2023-06-16 13:38:39 +03:00
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-23 21:39:59 +03:00
|
|
|
export default LogsExplorer;
|