diff --git a/src/containers/AppWithClusters/AppWithClusters.tsx b/src/containers/AppWithClusters/AppWithClusters.tsx index 3275d8d57..b8cdc6026 100644 --- a/src/containers/AppWithClusters/AppWithClusters.tsx +++ b/src/containers/AppWithClusters/AppWithClusters.tsx @@ -3,6 +3,7 @@ import React from 'react'; import type {Store} from '@reduxjs/toolkit'; import type {History} from 'history'; +import {getLogsLink as getLogsLinkDefault} from '../../utils/logs'; import type { GetLogsLink, GetMonitoringClusterLink, @@ -31,7 +32,7 @@ export interface AppWithClustersProps { export function AppWithClusters({ store, history, - getLogsLink, + getLogsLink = getLogsLinkDefault, getMonitoringLink = getMonitoringLinkDefault, getMonitoringClusterLink = getMonitoringClusterLinkDefault, userSettings, diff --git a/src/containers/AppWithClusters/ExtendedCluster/ExtendedCluster.tsx b/src/containers/AppWithClusters/ExtendedCluster/ExtendedCluster.tsx index c7fdfa617..595556be1 100644 --- a/src/containers/AppWithClusters/ExtendedCluster/ExtendedCluster.tsx +++ b/src/containers/AppWithClusters/ExtendedCluster/ExtendedCluster.tsx @@ -38,12 +38,19 @@ const getAdditionalBalancerInfo = (balancer: string) => { }; }; -const getAdditionalClusterProps = ( - clusterName: string | undefined, - monitoring: string | undefined, - balancer: string | undefined, - getMonitoringClusterLink?: GetMonitoringClusterLink, -) => { +interface GetAdditionalClusterProps { + clusterName: string | undefined; + monitoring: string | undefined; + balancer: string | undefined; + getMonitoringClusterLink?: GetMonitoringClusterLink; +} + +const getAdditionalClusterProps = ({ + clusterName, + monitoring, + balancer, + getMonitoringClusterLink, +}: GetAdditionalClusterProps) => { const additionalClusterProps: AdditionalClusterProps = {}; if (monitoring && getMonitoringClusterLink) { @@ -61,14 +68,25 @@ const getAdditionalClusterProps = ( return additionalClusterProps; }; -const getAdditionalTenantsProps = ( - clusterName: string | undefined, - monitoring: string | undefined, - balancer: string | undefined, - useClusterBalancerAsBackend: boolean | undefined, - getMonitoringLink?: GetMonitoringLink, - getLogsLink?: GetLogsLink, -) => { +interface GetAdditionalTenantsProps { + clusterName: string | undefined; + monitoring: string | undefined; + balancer: string | undefined; + logging: string | undefined; + useClusterBalancerAsBackend: boolean | undefined; + getMonitoringLink?: GetMonitoringLink; + getLogsLink?: GetLogsLink; +} + +const getAdditionalTenantsProps = ({ + clusterName, + monitoring, + balancer, + logging, + useClusterBalancerAsBackend, + getMonitoringLink, + getLogsLink, +}: GetAdditionalTenantsProps) => { const additionalTenantsProps: AdditionalTenantsProps = {}; additionalTenantsProps.prepareTenantBackend = ( @@ -104,12 +122,12 @@ const getAdditionalTenantsProps = ( }; } - if (clusterName && getLogsLink) { + if (logging && getLogsLink) { additionalTenantsProps.getLogsLink = (dbName?: string) => { if (dbName) { return getLogsLink({ dbName, - clusterName, + logging, }); } @@ -133,27 +151,28 @@ export function ExtendedCluster({ getLogsLink, }: ExtendedClusterProps) { const additionalNodesProps = useAdditionalNodesProps(); - const {name, balancer, monitoring} = useClusterBaseInfo(); + const {name, balancer, monitoring, logging} = useClusterBaseInfo(); const [useClusterBalancerAsBackend] = useSetting(USE_CLUSTER_BALANCER_AS_BACKEND_KEY); return (
diff --git a/src/containers/AppWithClusters/ExtendedTenant/ExtendedTenant.tsx b/src/containers/AppWithClusters/ExtendedTenant/ExtendedTenant.tsx index 0b8205ffe..379fd75ca 100644 --- a/src/containers/AppWithClusters/ExtendedTenant/ExtendedTenant.tsx +++ b/src/containers/AppWithClusters/ExtendedTenant/ExtendedTenant.tsx @@ -15,7 +15,7 @@ export function ExtendedTenant({ getMonitoringLink, getLogsLink, }: ExtendedTenantProps) { - const {monitoring, name: clusterName} = useClusterBaseInfo(); + const {monitoring, logging} = useClusterBaseInfo(); const additionalNodesProps = useAdditionalNodesProps(); const additionalTenantProps = { @@ -31,10 +31,10 @@ export function ExtendedTenant({ return null; }, getLogsLink: (dbName?: string) => { - if (clusterName && dbName && getLogsLink) { + if (logging && dbName && getLogsLink) { return getLogsLink({ dbName, - clusterName, + logging, }); } diff --git a/src/utils/__test__/logs.test.ts b/src/utils/__test__/logs.test.ts new file mode 100644 index 000000000..e9b43e233 --- /dev/null +++ b/src/utils/__test__/logs.test.ts @@ -0,0 +1,51 @@ +import {getLogsLink} from '../logs'; + +describe('getLogsLink', () => { + test('should insert dbName into logs URL query', () => { + const loggingData = { + url: 'https://logging.url/projects/kikimr/logs?from=now-1h&to=now&query=%7Bproject+%3D+%22kikimr%22%2C+service+%3D+%22ydb%22%2C+cluster+%3D+%22ydb-ru-prestable%22%7D', + }; + + const result = getLogsLink({ + logging: JSON.stringify(loggingData), + dbName: 'testdb', + }); + + // The URL should contain the dbName in the query parameter + expect(result).toBe( + 'https://logging.url/projects/kikimr/logs?from=now-1h&to=now&query=%7Bproject+%3D+%22kikimr%22%2C+service+%3D+%22ydb%22%2C+cluster+%3D+%22ydb-ru-prestable%22%2C+database+%3D+%22testdb%22%7D', + ); + }); + + test('should handle empty query parameters', () => { + const loggingData = { + url: 'https://logging.url/projects/kikimr/logs?from=now-1h&to=now&query=%7B%7D', + }; + + const result = getLogsLink({ + logging: JSON.stringify(loggingData), + dbName: 'testdb', + }); + + // Should add dbName to empty query + expect(result).toBe( + 'https://logging.url/projects/kikimr/logs?from=now-1h&to=now&query=%7Bdatabase+%3D+%22testdb%22%7D', + ); + }); + + test('should return empty string for invalid data', () => { + expect( + getLogsLink({ + logging: 'invalid json', + dbName: 'testdb', + }), + ).toBe(''); + + expect( + getLogsLink({ + logging: JSON.stringify({}), + dbName: 'testdb', + }), + ).toBe(''); + }); +}); diff --git a/src/utils/logs.ts b/src/utils/logs.ts new file mode 100644 index 000000000..a4356651d --- /dev/null +++ b/src/utils/logs.ts @@ -0,0 +1,36 @@ +interface GetLogsLinkProps { + dbName: string; + logging: string; +} + +export type GetLogsLink = (props: GetLogsLinkProps) => string; + +export function getLogsLink({dbName, logging}: GetLogsLinkProps): string { + try { + const data = JSON.parse(logging); + + if (typeof data === 'object' && 'url' in data) { + const logUrl = data.url; + if (!logUrl) { + return ''; + } + + const url = new URL(logUrl); + + const queryParam = url.searchParams.get('query'); + if (queryParam) { + const decodedQuery = decodeURIComponent(queryParam); + + const queryBetweenBraces = decodedQuery.slice(1, -1); + const witComma = queryBetweenBraces.length > 0; + const updatedQuery = `{${queryBetweenBraces}${witComma ? ', ' : ''}database = "${dbName}"}`; + + url.searchParams.set('query', updatedQuery); + } + + return url.toString(); + } + } catch {} + + return ''; +} diff --git a/src/utils/monitoring.ts b/src/utils/monitoring.ts index 62a427ebb..3b02cec65 100644 --- a/src/utils/monitoring.ts +++ b/src/utils/monitoring.ts @@ -102,7 +102,7 @@ export function parseMonitoringData(monitoring: string): ParsedMonitoringData | interface GetLogsLinkProps { dbName: string; - clusterName: string; + logging: string; } export type GetLogsLink = (props: GetLogsLinkProps) => string;