Skip to content

Commit 1c0e695

Browse files
authored
Merge pull request #2627 from headlamp-k8s/link-preload
frontend: Populate query cache when kubeObject is passed into a Link
2 parents 9928715 + 3b423a4 commit 1c0e695

File tree

3 files changed

+52
-10
lines changed

3 files changed

+52
-10
lines changed

frontend/src/components/common/Link.tsx

+34-5
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
import MuiLink from '@mui/material/Link';
2+
import { useQueryClient } from '@tanstack/react-query';
23
import React from 'react';
34
import { Link as RouterLink } from 'react-router-dom';
5+
import { kubeObjectQueryKey, useEndpoints } from '../../lib/k8s/api/v2/hooks';
46
import { KubeObject } from '../../lib/k8s/KubeObject';
57
import { createRouteURL, RouteURLProps } from '../../lib/router';
68
import { LightTooltip } from './Tooltip';
@@ -28,14 +30,41 @@ export interface LinkObjectProps extends LinkBaseProps {
2830
[prop: string]: any;
2931
}
3032

33+
function KubeObjectLink(props: { kubeObject: KubeObject; [prop: string]: any }) {
34+
const { kubeObject, ...otherProps } = props;
35+
36+
const client = useQueryClient();
37+
const { namespace, name } = kubeObject.metadata;
38+
const endpoint = useEndpoints(kubeObject._class().apiEndpoint.apiInfo, kubeObject.cluster);
39+
40+
return (
41+
<MuiLink
42+
onClick={() => {
43+
const key = kubeObjectQueryKey({
44+
cluster: kubeObject.cluster,
45+
endpoint,
46+
namespace,
47+
name,
48+
});
49+
// prepopulate the query cache with existing object
50+
client.setQueryData(key, kubeObject);
51+
// and invalidate it (mark as stale)
52+
// so that the latest version will be downloaded in the background
53+
client.invalidateQueries({ queryKey: key });
54+
}}
55+
component={RouterLink}
56+
to={kubeObject.getDetailsLink()}
57+
{...otherProps}
58+
>
59+
{props.children || kubeObject!.getName()}
60+
</MuiLink>
61+
);
62+
}
63+
3164
function PureLink(props: React.PropsWithChildren<LinkProps | LinkObjectProps>) {
3265
if ((props as LinkObjectProps).kubeObject) {
3366
const { kubeObject, ...otherProps } = props as LinkObjectProps;
34-
return (
35-
<MuiLink component={RouterLink} to={kubeObject!.getDetailsLink()} {...otherProps}>
36-
{props.children || kubeObject!.getName()}
37-
</MuiLink>
38-
);
67+
return <KubeObjectLink kubeObject={kubeObject!} {...otherProps} />;
3968
}
4069

4170
const { routeName, params = {}, search, state, ...otherProps } = props as LinkProps;

frontend/src/components/workload/Overview.tsx

+2-4
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import Grid from '@mui/material/Grid';
22
import React, { useMemo } from 'react';
33
import { useTranslation } from 'react-i18next';
4-
import { useLocation } from 'react-router-dom';
54
import CronJob from '../../lib/k8s/cronJob';
65
import DaemonSet from '../../lib/k8s/daemonSet';
76
import Deployment from '../../lib/k8s/deployment';
@@ -13,7 +12,7 @@ import { WorkloadClass } from '../../lib/k8s/Workload';
1312
import { Workload } from '../../lib/k8s/Workload';
1413
import { getReadyReplicas, getTotalReplicas } from '../../lib/util';
1514
import Link from '../common/Link';
16-
import { PageGrid, ResourceLink } from '../common/Resource';
15+
import { PageGrid } from '../common/Resource';
1716
import ResourceListView from '../common/Resource/ResourceListView';
1817
import { SectionBox } from '../common/SectionBox';
1918
import { WorkloadCircleChart } from './Charts';
@@ -44,7 +43,6 @@ export default function Overview() {
4443
[pods, deployments, statefulSets, daemonSets, replicaSets, jobs, cronJobs]
4544
);
4645

47-
const location = useLocation();
4846
const { t } = useTranslation('glossary');
4947

5048
function getPods(item: Workload) {
@@ -130,7 +128,7 @@ export default function Overview() {
130128
id: 'name',
131129
label: t('translation|Name'),
132130
getValue: item => item.metadata.name,
133-
render: item => <ResourceLink resource={item} state={{ backLink: { ...location } }} />,
131+
render: item => <Link kubeObject={item} />,
134132
},
135133
'namespace',
136134
'cluster',

frontend/src/lib/k8s/api/v2/hooks.ts

+16-1
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,20 @@ export interface QueryListResponse<DataType, ItemType, ErrorType>
6565
clusterErrors?: Record<string, ApiError | null> | null;
6666
}
6767

68+
export const kubeObjectQueryKey = ({
69+
cluster,
70+
endpoint,
71+
namespace,
72+
name,
73+
queryParams,
74+
}: {
75+
cluster: string;
76+
endpoint?: KubeObjectEndpoint | null;
77+
namespace?: string;
78+
name: string;
79+
queryParams?: QueryParameters;
80+
}) => ['object', cluster, endpoint, namespace ?? '', name, queryParams ?? {}];
81+
6882
/**
6983
* Returns a single KubeObject.
7084
*/
@@ -94,7 +108,8 @@ export function useKubeObject<K extends KubeObject>({
94108
);
95109

96110
const queryKey = useMemo(
97-
() => ['object', cluster, endpoint, namespace ?? '', name, cleanedUpQueryParams],
111+
() =>
112+
kubeObjectQueryKey({ cluster, name, namespace, endpoint, queryParams: cleanedUpQueryParams }),
98113
[endpoint, namespace, name]
99114
);
100115

0 commit comments

Comments
 (0)